from __future__ import annotations
from typing import Any, Callable, Collection, Optional, TypeVar, Union
import numpy as np
import numpy.typing as npt
from scipy.optimize import OptimizeResult
from scipy.optimize import minimize as scipy_minimize
from sympy import Expr
from typeguard import typechecked
from mpqp.core.circuit import QCircuit
from mpqp.execution.devices import AvailableDevice
from mpqp.execution.runner import _run_single # pyright: ignore[reportPrivateUsage]
from mpqp.execution.vqa.optimizer import Optimizer
T1 = TypeVar("T1")
T2 = TypeVar("T2")
OptimizerInput = Union[list[float], npt.NDArray[np.float32]]
OptimizableFunc = Callable[[OptimizerInput], float]
OptimizerOptions = dict[str, Any]
OptimizerCallable = Callable[
[OptimizableFunc, Optional[OptimizerInput], Optional[OptimizerOptions]],
tuple[float, OptimizerInput],
]
# TODO: all those functions with almost or exactly the same signature look like
# a code smell to me.
# TODO: test the minimizer options
def _maps(l1: Collection[T1], l2: Collection[T2]) -> dict[T1, T2]:
"""Does like zip, but with a dictionary instead of a list of tuples"""
if len(l1) != len(l2):
ValueError(
f"Length of the two collections are not equal ({len(l1)} and {len(l2)})."
)
return {e1: e2 for e1, e2 in zip(l1, l2)}
[docs]@typechecked
def minimize(
optimizable: QCircuit | OptimizableFunc,
method: Optimizer | OptimizerCallable,
device: Optional[AvailableDevice] = None,
init_params: Optional[OptimizerInput] = None,
nb_params: Optional[int] = None,
optimizer_options: Optional[dict[str, Any]] = None,
) -> tuple[float, OptimizerInput]:
"""This function runs an optimization on the parameters of the circuit, in order to
minimize the measured expectation value of observables associated with the given circuit.
Note that this means that the latter should contain an ``ExpectationMeasure``.
Args:
optimizable: Either the circuit, containing symbols and an expectation
measure, or the evaluation function.
method: The method used to optimize most of those methods come from
``scipy``. If the choices offered in this package are not
covering your needs, you can define your own optimizer. This should be
a function taking as input a function representing the circuit, with
as many inputs as the circuit has parameters, and any optional
initialization parameters, and returning the optimal value reached
and the parameters used to reach this value.
device: The device on which the circuit should be run.
init_params: The optional initialization parameters (the value
attributed to the symbols in the first loop of the optimizer).
nb_params: Number of variables to input in ``optimizable``. It is only
useful if ``optimizable`` is a Callable and if ``init_params`` was
not given. If not this argument is not taken into account.
optimizer_options: Options used to configure the VQA optimizer (maximum
iterations, convergence threshold, etc...). These options are passed
as is to the minimizer.
Returns:
The optimal value reached and the parameters corresponding to this value.
Examples:
>>> alpha, beta = symbols("α β")
>>> circuit = QCircuit([
... H(0),
... Rx(alpha, 1),
... CNOT(1,0),
... Rz(beta, 0),
... ExpectationMeasure(
... Observable(np.diag([1,2,-3,4])),
... [0,1],
... shots=0,
... ),
... ])
>>> minimize(
... circuit,
... Optimizer.BFGS,
... ATOSDevice.MYQLM_PYLINALG,
... optimizer_options={"maxiter":50},
... )
(-0.9999999999999996, array([0., 0.]))
>>> def cost_func(params):
... run_res = run(
... circuit,
... ATOSDevice.MYQLM_PYLINALG,
... {alpha: params[0], beta: params[1]}
... )
... return 1 - run_res.expectation_value ** 2
>>> minimize(
... cost_func,
... Optimizer.BFGS,
... nb_params=2,
... optimizer_options={"maxiter":50},
... )
(8.881784197001252e-16, array([0., 0.]))
"""
if isinstance(optimizable, QCircuit):
if device is None:
raise ValueError("A device is needed to optimize a circuit")
optimizer = _minimize_remote if device.is_remote() else _minimize_local
return optimizer(optimizable, method, device, init_params, nb_params)
else:
# TODO: find a way to know if the job is remote or local from the function
return _minimize_local(
optimizable, method, device, init_params, nb_params, optimizer_options
)
@typechecked
def _minimize_remote(
optimizable: QCircuit | OptimizableFunc,
method: Optimizer | OptimizerCallable,
device: Optional[AvailableDevice] = None,
init_params: Optional[OptimizerInput] = None,
nb_params: Optional[int] = None,
optimizer_options: Optional[dict[str, Any]] = None,
) -> tuple[float, OptimizerInput]:
"""This function runs an optimization on the parameters of the circuit, to
minimize the expectation value of the measure of the circuit by it's
observables. Note that this means that the circuit should contain an
expectation measure.
Args:
optimizable: Either the circuit, containing symbols and an expectation
measure, or the evaluation function.
method: The method used to optimize most of those methods come from
either scipy or cma. If the choice offered in this package are not
covering your needs, you can define your own optimizer. It should be
a function taking as input a function representing the circuit, with
as many inputs as the circuit has parameters, as well as optional
initialization parameters, and returning the optimal value reached
and the parameters used to reach this value.
device: The device on which the circuit should be run.
init_params: The optional initialization parameters (the value
attributed to the symbols in the first loop of the optimizer).
nb_params: number of variables to input in ``optimizable``. It is only
useful if ``optimizable`` is a Callable and if ``init_params`` was
not given. If not this argument is not taken into account.
optimizer_options: Options used to configure the VQA optimizer (maximum
iterations, convergence threshold, etc...). These options are passed
as is to the minimizer.
Returns:
The optimal value reached and the parameters used to reach this value.
TODO to implement on QLM first
"""
raise NotImplementedError()
@typechecked
def _minimize_local(
optimizable: QCircuit | OptimizableFunc,
method: Optimizer | OptimizerCallable,
device: Optional[AvailableDevice] = None,
init_params: Optional[OptimizerInput] = None,
nb_params: Optional[int] = None,
optimizer_options: Optional[dict[str, Any]] = None,
) -> tuple[float, OptimizerInput]:
"""This function runs an optimization on the parameters of the circuit, to
minimize the expectation value of the measure of the circuit by it's
observables. Note that this means that the circuit should contain an
expectation measure.
Args:
optimizable: Either the circuit, containing symbols and an expectation
measure, or the evaluation function.
method: The method used to optimize most of those methods come from
either scipy or cma. If the choice offered in this package are not
covering your needs, you can define your own optimizer. It should be
a function taking as input a function representing the circuit, with
as many inputs as the circuit has parameters, as well as optional
initialization parameters, and returning the optimal value reached
and the parameters used to reach this value.
device: The device on which the circuit should be run.
init_params: The optional initialization parameters (the value
attributed to the symbols in the first loop of the optimizer).
nb_params: number of variables to input in ``optimizable``. It is only
useful if ``optimizable`` is a Callable and if ``init_params`` was
not given. If not this argument is not taken into account.
optimizer_options: Options used to configure the VQA optimizer (maximum
iterations, convergence threshold, etc...). These options are passed
as is to the minimizer.
Returns:
the optimal value reached and the parameters used to reach this value.
"""
if isinstance(optimizable, QCircuit):
if device is None:
raise ValueError("A device is needed to optimize a circuit")
return _minimize_local_circ(
optimizable, device, method, init_params, optimizer_options
)
else:
return _minimize_local_func(
optimizable, method, init_params, nb_params, optimizer_options
)
@typechecked
def _minimize_local_circ(
circ: QCircuit,
device: AvailableDevice,
method: Optimizer | OptimizerCallable,
init_params: Optional[OptimizerInput] = None,
optimizer_options: Optional[dict[str, Any]] = None,
) -> tuple[float, OptimizerInput]:
"""This function runs an optimization on the parameters of the circuit, to
minimize the expectation value of the measure of the circuit by it's
observables. Note that this means that the circuit should contain an
expectation measure!
Args:
circ: Either the circuit, containing symbols and an expectation measure.
method: The method used to optimize most of those methods come from
either scipy or cma. If the choice offered in this package are not
covering your needs, you can define your own optimizer. It should be
a function taking as input a function representing the circuit, with
as many inputs as the circuit has parameters, as well as optional
initialization parameters, and returning the optimal value reached
and the parameters used to reach this value.
device: The device on which the circuit should be run.
init_params: The optional initialization parameters (the value
attributed to the symbols in the first loop of the optimizer).
optimizer_options: Options used to configure the VQA optimizer (maximum
iterations, convergence threshold, etc...). These options are passed
as is to the minimizer.
Returns:
The optimal value reached and the parameters used to reach this value.
"""
# The sympy `free_symbols` method returns in fact sets of Basic, which
# are theoretically different from Expr, but in our case the difference
# is not relevant.
# TODO: bellow might be a bug, check why we need this type ignore
variables: set[Expr] = circ.variables() # pyright: ignore[reportAssignmentType]
def eval_circ(params: OptimizerInput):
# pyright is bad with abstract numeric types:
# "float" is incompatible with "Complex"
return _run_single(
circ,
device,
_maps(variables, params), # pyright: ignore[reportArgumentType]
).expectation_value
return _minimize_local_func(
eval_circ, method, init_params, len(variables), optimizer_options
)
@typechecked
def _minimize_local_func(
eval_func: OptimizableFunc,
method: Optimizer | OptimizerCallable,
init_params: Optional[OptimizerInput] = None,
nb_params: Optional[int] = None,
optimizer_options: Optional[OptimizerOptions] = None,
) -> tuple[float, OptimizerInput]:
"""This function runs an optimization on the parameters of the circuit, to
minimize the expectation value of the measure of the circuit by it's
observables. Note that this means that the circuit should contain an
expectation measure!
Args:
eval_func: Evaluation function.
method: The method used to optimize most of those methods come from
either scipy or cma. If the choice offered in this package are not
covering your needs, you can define your own optimizer. It should be
a function taking as input a function representing the circuit, with
as many inputs as the circuit has parameters, as well as optional
initialization parameters, and returning the optimal value reached
and the parameters used to reach this value.
init_params: The optional initialization parameters (the value
attributed to the symbols in the first loop of the optimizer).
nb_params: number of variables to input in ``optimizable``. It is only
useful if ``init_params`` was not given. If not this argument is not
taken into account.
optimizer_options: Options used to configure the VQA optimizer (maximum
iterations, convergence threshold, etc...). These options are passed
as is to the minimizer.
Returns:
The optimal value reached and the parameters used to reach this value.
"""
if init_params is None:
if nb_params is None:
raise ValueError(
"Please provide either a set of initialization parameters or "
"the number of parameters expected by the function."
)
else:
init_params = [0.0] * nb_params
if isinstance(method, Optimizer):
res: OptimizeResult = scipy_minimize(
eval_func,
x0=np.array(init_params),
method=method.name.lower(),
options=optimizer_options,
)
return res.fun, res.x
else:
return method(eval_func, init_params, optimizer_options)