# Repository: https://gitlab.com/quantify-os/quantify-core
# Licensed according to the LICENCE file on the main branch
"""Helpers for performing experiments."""
from __future__ import annotations
import warnings
from typing import Any, Optional, Union, Dict, List, Literal
import numpy as np
from deepdiff import DeepDiff
from qcodes.instrument import Instrument, InstrumentChannel
from qcodes.parameters import Parameter
from quantify_core.data.handling import get_latest_tuid, load_snapshot
from quantify_core.data.types import TUID
from quantify_core.visualization.pyqt_plotmon import PlotMonitor_pyqt
# pylint: disable=broad-except
[docs]
def load_settings_onto_instrument(
instrument: Union[Instrument, InstrumentChannel, Parameter],
tuid: TUID | None = None,
datadir: str = None,
exception_handling: Literal["raise", "warn"] = "raise",
) -> None:
"""
Loads settings from a previous experiment onto a current
:class:`~qcodes.instrument.Instrument`, or any of its submodules or
parameters. This information is loaded from the 'snapshot.json' file in
the provided experiment directory.
Parameters
----------
instrument :
the :class:`~qcodes.instrument.Instrument`,
:class:`~qcodes.instrument.InstrumentChannel` or
:class:`~qcodes.parameters.Parameter` to be configured.
tuid : :class:`~quantify_core.data.types.TUID`
the TUID of the experiment. If None use latest TUID.
datadir : str
path of the data directory. If `None`, uses `get_datadir()` to
determine the data directory.
exception_handling:
desired behaviour if error occurs when trying to get parameter:
raise exception or give warning.
Raises
------
ValueError
if the provided instrument has no match in the loaded snapshot.
"""
if tuid is None:
tuid = get_latest_tuid()
instruments = load_snapshot(tuid, datadir)["instruments"]
instruments_numpy_array = load_snapshot(tuid, datadir, list_to_ndarray=True)[
"instruments"
]
def _try_to_set_par_safe(
instr_mod: Union[Instrument, InstrumentChannel],
parname: str,
value: Any,
value_np: Any,
):
"""Tries to set a parameter and emits a warning if not successful."""
# do not try to set parameters that are not settable
if not "set" in dir(instr_mod.parameters[parname]):
return
# do not set to None if value is already None. If there is a runtime
# error when getting the parameter, do not try to set the parameter.
try:
get_val = instr_mod.parameters[parname]()
except Exception as exc:
if exception_handling == "raise":
raise exc
warnings.warn(
f"Could not get value of {parname} parameter due to '{exc}'. "
"We will not try to set this parameter."
)
return
if get_val is None and value is None:
return
# Make sure the parameter is actually a settable
try:
if isinstance(get_val, np.ndarray):
instr_mod.set(parname, value_np)
else:
instr_mod.set(parname, value)
except (RuntimeError, KeyError, ValueError, TypeError) as exc:
warnings.warn(
f'Parameter "{parname}" of "{instr_mod.name}" '
f'could not be set to "{value}" due to error:\n{exc}'
)
parents = get_all_parents(instrument)
# Find the snapshot for this instrument, submodule or parameter
for parent in parents:
if isinstance(parent, Instrument):
if parent.name not in instruments:
raise ValueError(
f'Instrument "{parent.name}" not found in snapshot {datadir}:{tuid}'
)
instr_mod_snap = instruments[parent.name]
instr_mod_snap_numpy_array = instruments_numpy_array[parent.name]
if isinstance(parent, InstrumentChannel):
if parent._short_name not in instr_mod_snap["submodules"]:
raise ValueError(
f'Submodule "{parent.name}" not found in snapshot {datadir}:{tuid}'
)
instr_mod_snap = instr_mod_snap["submodules"][parent._short_name]
instr_mod_snap_numpy_array = instr_mod_snap_numpy_array["submodules"][
parent._short_name
]
# If we are only setting one parameter, try to set this immediately
if isinstance(parent, Parameter):
if parent.name not in instr_mod_snap["parameters"]:
address = ".".join([p._short_name for p in parents])
raise ValueError(
f'Parameter "{address}" not found in snapshot {datadir}:{tuid}'
)
value = instr_mod_snap["parameters"][parent.name]["value"]
value_np = instr_mod_snap_numpy_array["parameters"][parent.name]["value"]
_try_to_set_par_safe(parents[-2], parent.name, value, value_np)
return
def _set_params_instr_mod(
instr_mod_snap: Dict,
instr_mod_snap_np: Dict,
instr_mod: Union[Instrument, InstrumentChannel],
):
"""
Private function to set parameters and recursively set parameters of submodules.
"""
# iterate over top-level parameters
for parname, par in instr_mod_snap["parameters"].items():
# Check that the parameter exists in this instrument
if parname in instr_mod.parameters:
value = par["value"]
value_np = instr_mod_snap_np["parameters"][parname]["value"]
_try_to_set_par_safe(instr_mod, parname, value, value_np)
else:
warnings.warn(
f"Could not set parameter {parname} in {instr_mod.name}. "
f"{instr_mod.name} does not possess a parameter named {parname}."
)
# recursively call this function for all submodules
if "submodules" in instr_mod_snap.keys():
for module_name, module_snap in instr_mod_snap["submodules"].items():
submodule = instr_mod.submodules[module_name]
module_snap_np = instr_mod_snap_np["submodules"][module_name]
_set_params_instr_mod(
instr_mod_snap=module_snap,
instr_mod_snap_np=module_snap_np,
instr_mod=submodule,
)
# set the top-level parameters and then recursively set parameters of submodules
_set_params_instr_mod(
instr_mod_snap=instr_mod_snap,
instr_mod_snap_np=instr_mod_snap_numpy_array,
instr_mod=instrument,
)
[docs]
def compare_snapshots(old_snapshot: dict, new_snapshot: dict) -> dict:
"""
Generate a diff between two quantify snapshots, showing which qcodes parameters
have changed.
This function only considers changes in numerical values of quantities and
ignores type changes, such as :code:`numpy.float64` to :code:`float`, for
example.
We also only consider the values of qcodes parameters, and not metadata like
timestamps, which always change from snapshot to snapshot.
Parameters
----------
old_snapshot:
The original snapshot to be compared
new_snapshot:
The new snapshot to be compared
Return
------
:
A dictionary summarising the differences between the two snapshots
"""
return DeepDiff(
old_snapshot,
new_snapshot,
exclude_regex_paths=[r"\['ts'\]", r"\['raw_value'\]", r"\['IDN'\]"],
ignore_numeric_type_changes=True,
)
[docs]
def get_all_parents(instr_mod: Union[Instrument, InstrumentChannel, Parameter]) -> List:
"""
Get a list of all the parent submodules and instruments of a given QCodes
instrument, submodule or parameter.
Parameters
-----------
instr_mod:
The QCodes instrument, submodule or parameter whose parents we wish to find
Returns
-------
:
A list of all the parents of that object (and the object itself)
"""
if hasattr(instr_mod, "_parent"):
parents = get_all_parents(instr_mod._parent)
parents.append(instr_mod)
elif hasattr(instr_mod, "_instrument"):
parents = get_all_parents(instr_mod._instrument)
parents.append(instr_mod)
else:
parents = [instr_mod]
return parents
[docs]
def create_plotmon_from_historical(
tuid: Optional[TUID] = None, label: str = ""
) -> PlotMonitor_pyqt:
"""
Creates a plotmon using the dataset of the provided experiment denoted by the tuid
in the datadir.
Loads the data and draws any required figures.
NB Creating a new plotmon can be slow. Consider using
:func:`!PlotMonitor_pyqt.tuids_extra` to visualize dataset in the same plotmon.
Parameters
----------
tuid
the TUID of the experiment.
label
if the `tuid` is not provided, as label will be used to search for the latest
dataset.
Returns
-------
:
the plotmon
"""
# avoid creating a plotmon with the same name
if tuid is None:
tuid = get_latest_tuid(contains=label)
tuid_str = str(tuid)
# python attributes can't start with a number
name = "_" + tuid_str
# hyphens not allowed in instrument names for qcodes v0.34 and up
name = name.replace("-", "_")
plotmon = PlotMonitor_pyqt(name)
plotmon.tuids_append(tuid)
plotmon.update() # make sure everything is drawn
return plotmon