Source code for qilisdk.backends.backend
# Copyright 2025 Qilimanjaro Quantum Tech
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from __future__ import annotations
from abc import ABC
from typing import TYPE_CHECKING, Callable, TypeVar, cast, overload
from qilisdk.functionals.functional_result import FunctionalResult
from qilisdk.functionals.sampling import Sampling
from qilisdk.functionals.time_evolution import TimeEvolution
from qilisdk.functionals.variational_program import VariationalProgram
from qilisdk.functionals.variational_program_result import VariationalProgramResult
if TYPE_CHECKING:
from qilisdk.functionals.functional import Functional, PrimitiveFunctional
from qilisdk.functionals.sampling_result import SamplingResult
from qilisdk.functionals.time_evolution_result import TimeEvolutionResult
[docs]
TResult = TypeVar("TResult", bound=FunctionalResult)
[docs]
class Backend(ABC):
def __init__(self) -> None:
self._handlers: dict[type[Functional], Callable[[Functional], FunctionalResult]] = {
Sampling: lambda f: self._execute_sampling(cast("Sampling", f)),
TimeEvolution: lambda f: self._execute_time_evolution(cast("TimeEvolution", f)),
VariationalProgram: lambda f: self._execute_variational_program(cast("VariationalProgram", f)),
}
@overload
[docs]
def execute(self, functional: Sampling) -> SamplingResult: ...
@overload
def execute(self, functional: TimeEvolution) -> TimeEvolutionResult: ...
@overload
def execute(self, functional: VariationalProgram[Sampling]) -> VariationalProgramResult[SamplingResult]: ...
@overload
def execute(
self, functional: VariationalProgram[TimeEvolution]
) -> VariationalProgramResult[TimeEvolutionResult]: ...
@overload
def execute(self, functional: PrimitiveFunctional[TResult]) -> TResult: ...
def execute(self, functional: Functional) -> FunctionalResult:
try:
handler = self._handlers[type(functional)]
except KeyError as exc:
raise NotImplementedError(
f"{type(self).__qualname__} does not support {type(functional).__qualname__}"
) from exc
return handler(functional)
def _execute_sampling(self, functional: Sampling) -> SamplingResult:
raise NotImplementedError(f"{type(self).__qualname__} has no Sampling implementation")
def _execute_time_evolution(self, functional: TimeEvolution) -> TimeEvolutionResult:
raise NotImplementedError(f"{type(self).__qualname__} has no TimeEvolution implementation")
def _execute_variational_program(
self, functional: VariationalProgram[PrimitiveFunctional[TResult]]
) -> VariationalProgramResult[TResult]:
"""Optimize a Parameterized Program (:class:`~qilisdk.functionals.variational_program.VariationalProgram`)
and returns the optimal parameters and results.
Args:
functional (VariationalProgram): The variational program to be optimized.
Returns:
ParameterizedProgramResults: The final optimizer and functional results.
Raises:
ValueError: If the functional is not parameterized.
"""
def evaluate_sample(parameters: list[float]) -> float:
param_names = functional.functional.get_parameter_names()
param_dict = {param_names[i]: param for i, param in enumerate(parameters)}
err = functional.check_parameter_constraints(param_dict)
if err > 0:
return err
functional.functional.set_parameters(param_dict)
results = self.execute(functional.functional)
final_results = functional.cost_function.compute_cost(results)
if isinstance(final_results, float):
return final_results
if isinstance(final_results, complex) and final_results.imag == 0:
return final_results.real
raise ValueError(f"Unsupported result type {type(final_results)}.")
if len(functional.functional.get_parameters()) == 0:
raise ValueError("Functional provided is not parameterized.")
optimizer_result = functional.optimizer.optimize(
cost_function=evaluate_sample,
init_parameters=list(functional.functional.get_parameters().values()),
bounds=list(functional.functional.get_parameter_bounds().values()),
store_intermediate_results=functional.store_intermediate_results,
)
param_names = functional.functional.get_parameter_names()
functional.functional.set_parameters(
{param_names[i]: param for i, param in enumerate(optimizer_result.optimal_parameters)}
)
optimal_results: TResult = cast("TResult", self.execute(functional.functional))
return VariationalProgramResult(optimizer_result=optimizer_result, result=optimal_results)