Source code for quantify_scheduler.instrument_coordinator.components.base
# Repository: https://gitlab.com/quantify-os/quantify-scheduler
# Licensed according to the LICENCE file on the main branch
"""Module containing the InstrumentCoordinator interface."""
from __future__ import annotations
from abc import abstractmethod
from typing import TYPE_CHECKING, Any
from qcodes.instrument import Instrument, InstrumentBase, parameter
from qcodes.utils import validators
if TYPE_CHECKING:
    from xarray import Dataset
    from quantify_scheduler.schedules.schedule import CompiledSchedule
[docs]
def instrument_to_component_name(instrument_name: str) -> str:
    """
    Give the name of the instrument coordinator component.
    Parameters
    ----------
    instrument_name
        The name of the instrument.
    Returns
    -------
    :
        The name of the instrument coordinator component.
    """
    return f"ic_{instrument_name}" 
[docs]
class InstrumentCoordinatorComponentBase(Instrument):
    """The InstrumentCoordinator component abstract interface."""
    # NB `_instances` also used by `Instrument` class
[docs]
    _no_gc_instances: dict[str, InstrumentBase] = dict() 
    def __new__(cls, instrument: InstrumentBase) -> InstrumentCoordinatorComponentBase:
        """
        Keeps track of the instances of this class and the instrument.
        NB This is done intentionally to prevent the instances from being garbage collected.
        """
        instance = super().__new__(cls)
        cls._no_gc_instances[f"ic_{instrument.name}"] = instance
        cls._no_gc_instances[instrument.name] = instrument
        return instance
[docs]
    def close(self) -> None:
        """
        Release instances so that garbage collector can claim the objects.
        NB We don't close the instrument because it might be referenced elsewhere.
        """
        self._no_gc_instances.pop(f"ic_{self.instrument_ref()}", None)
        self._no_gc_instances.pop(self.instrument_ref(), None)
        super().close() 
    def __init__(
        self,
        instrument: InstrumentBase,
        **kwargs: Any,  # noqa: ANN401 (complicated subclass overrides)
    ) -> None:
        super().__init__(instrument_to_component_name(instrument.name), **kwargs)
[docs]
        self.instrument_ref = parameter.InstrumentRefParameter(
            "instrument_ref",
            initial_value=instrument.name,
            docstring="A reference of an instrument associated to this component.",
            vals=validators.MultiType(validators.Strings(), validators.Enum(None)),
            instrument=self,
        ) 
[docs]
        self.force_set_parameters = parameter.ManualParameter(
            "force_set_parameters",
            initial_value=False,
            docstring=(
                "A switch to force the setting of a parameter, " + "bypassing the lazy_set utility."
            ),
            vals=validators.Bool(),
            instrument=self,
        ) 
    @property
[docs]
    def instrument(self) -> InstrumentBase:
        """Returns the instrument referenced by `instrument_ref`."""
        return self.instrument_ref.get_instr() 
    @instrument.setter
    def instrument(self, instrument: InstrumentBase) -> None:
        """Sets a new Instrument as reference."""
        self.instrument_ref(instrument.name)
    @property
    @abstractmethod
[docs]
    def is_running(self) -> bool:
        """
        Returns if the InstrumentCoordinator component is running.
        The property ``is_running`` is evaluated each time it is accessed. Example:
        .. code-block::
            while my_instrument_coordinator_component.is_running:
                print('running')
        Returns
        -------
        :
            The components' running state.
        """ 
    @abstractmethod
[docs]
    def start(self) -> None:
        """Starts the InstrumentCoordinator Component.""" 
    @abstractmethod
[docs]
    def stop(self) -> None:
        """Stops the InstrumentCoordinator Component.""" 
    @abstractmethod
[docs]
    def prepare(self, program: Any) -> None:  # noqa: ANN401 (complicated subclass overrides)
        """Initializes the InstrumentCoordinator Component with parameters.""" 
    @abstractmethod
[docs]
    def retrieve_acquisition(self) -> Dataset | None:
        """Gets and returns acquisition data.""" 
    @abstractmethod
[docs]
    def wait_done(self, timeout_sec: int = 10) -> None:
        """
        Wait until the InstrumentCoordinator is done.
        The coordinator is ready when it has stopped running or until it
        has exceeded the amount of time to run.
        The maximum amount of time, in seconds, before it times out is set via the
        timeout_sec parameter.
        Parameters
        ----------
        timeout_sec :
            The maximum amount of time in seconds before a timeout.
        """ 
    @abstractmethod
[docs]
    def get_hardware_log(
        self,
        compiled_schedule: CompiledSchedule,
    ) -> dict | None:
        """Retrieve the hardware logs of the instrument associated to this component."""