Source code for qleet.interface.metric_spec

"""This module houses the ways to specify a metric, and sample solutions to compute it's value.

Metrics are a useful abstraction which given the state of the circuit compute some
classical value which we need to interpret or plot.
"""

import abc
import typing

import numpy as np
import sympy
import warnings

warnings.filterwarnings("ignore")
import tensorflow_quantum as tfq
import cirq

from ..interface.circuit import CircuitDescriptor


[docs]class MetricSpecifier(abc.ABC): """Class to specify classical metrics which are a function of the sampled quantum state. Examples would be an arbitrary cost function, Mean Squared Error for some regression task, size of the max cut, etc. This is an abstract class, all metrics should be it's subclasses How the metric is computed is left for the user to decide, it can be from the actual samples drawn from the circuit, or from the state vector, or from the density matrix. """ def __init__(self, default_call_mode: str = "samples") -> None: """Creates the metric specifier object. :type default_call_mode: str :param default_call_mode: Which function to use by default for computing metric """ self.__mode_to_function_map = { "samples": self.from_samples_vector, "state_vector": self.from_state_vector, "density_matrix": self.from_density_matrix, } assert default_call_mode in ["samples", "state_vector", "density_matrix"] self.default_call_mode = default_call_mode self.default_call_function = self.__mode_to_function_map[default_call_mode]
[docs] def from_circuit( self, circuit_descriptor: CircuitDescriptor, parameters: typing.Union[np.ndarray, typing.List], mode: str = "samples", ) -> float: """Computes the value of the metric from the circuit, by using the default mode or metric computation. :type circuit_descriptor: CircuitDescriptor :param circuit_descriptor: The provided circuit :type parameters: List or Numpy array :param parameters: List of values of the parameters to sample the circuit at :type mode: str :param mode: From what to compute the metric, samples, state_vector, or density_matrix :return: The value of the metric at those parameters :rtype: float :raises NotImplementedError: if required mode of evaluating metric wasn't implemented :raises ValueError: if the mode specified wasn't valid """ if mode == "samples": samples = sample_solutions( circuit=circuit_descriptor.cirq_circuit, param_symbols=circuit_descriptor.parameters, param_values=parameters, ) return self.from_samples_vector(samples) elif mode == "state_vector": raise NotImplementedError elif mode == "density_matrix": raise NotImplementedError else: raise ValueError( "Provided mode should be one of [samples, state_vector, density_matrix]" )
[docs] @abc.abstractmethod def from_state_vector(self, state_vector: np.ndarray) -> float: """Returns the value of the loss function given the state vector of the state prepared from the circuit. :type state_vector: np.ndarray, 1-D of shape (2^n,) :param state_vector: State vector of state prepared by circuit :return: value of the loss function :rtype: float """ raise NotImplementedError
[docs] @abc.abstractmethod def from_density_matrix(self, density_matrix: np.ndarray) -> float: """Returns the value of the loss function given the density matrix of the state prepared from the circuit using the noise model provided. :type density_matrix: np.ndarray, 2-D of shape (2^n, 2^n) :param density_matrix: Vector of samples drawn from the circuit :return: value of the loss function :rtype: float """ raise NotImplementedError
[docs] @abc.abstractmethod def from_samples_vector(self, samples_vector: np.ndarray) -> float: """Returns the value of the loss function from one set of measurements sampled from the circuit. :type samples_vector: np.ndarray, 1-D of shape (n,) :param samples_vector: Vector of samples drawn from the circuit :return: value of the loss function :rtype: float """ raise NotImplementedError
[docs]def sample_solutions( circuit: cirq.Circuit, param_symbols: typing.List[sympy.Symbol], param_values: typing.Iterable, samples: int = 1000, ) -> np.ndarray: """Get the computed cuts for a given ansatz :type circuit: cirq.Circuit :param circuit: Circuit to be sampled :type param_symbols: List of sympy.Symbols :param param_symbols: The symbols of model parameters :type param_values: List of floats :param param_values: The value of model parameters to sample at, 1-D vector :type samples: int :param samples: Number of times to sample the resulting quantum state :return: 2-D matrix, n_samples rows of boolean vectors showing the cut :rtype: np.array """ output = tfq.layers.Sample()( circuit, symbol_names=param_symbols, symbol_values=[param_values], repetitions=samples, ) return output.numpy()[0]