qilisdk 0.1.4__py3-none-any.whl → 0.1.6__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- qilisdk/__init__.py +11 -2
- qilisdk/__init__.pyi +2 -3
- qilisdk/_logging.py +135 -0
- qilisdk/_optionals.py +5 -7
- qilisdk/analog/__init__.py +3 -18
- qilisdk/analog/exceptions.py +2 -4
- qilisdk/analog/hamiltonian.py +455 -110
- qilisdk/analog/linear_schedule.py +121 -0
- qilisdk/analog/schedule.py +275 -79
- qilisdk/{extras → backends}/__init__.py +9 -4
- qilisdk/{common/model.py → backends/__init__.pyi} +3 -1
- qilisdk/backends/backend.py +117 -0
- qilisdk/{extras/cuda → backends}/cuda_backend.py +152 -159
- qilisdk/backends/qutip_backend.py +473 -0
- qilisdk/core/__init__.py +63 -0
- qilisdk/{common → core}/algorithm.py +2 -1
- qilisdk/{extras/qaas/qaas_settings.py → core/exceptions.py} +12 -6
- qilisdk/core/model.py +1034 -0
- qilisdk/core/parameterizable.py +75 -0
- qilisdk/core/qtensor.py +666 -0
- qilisdk/{common → core}/result.py +2 -1
- qilisdk/core/variables.py +1969 -0
- qilisdk/cost_functions/__init__.py +18 -0
- qilisdk/cost_functions/cost_function.py +77 -0
- qilisdk/cost_functions/model_cost_function.py +145 -0
- qilisdk/cost_functions/observable_cost_function.py +109 -0
- qilisdk/digital/__init__.py +3 -22
- qilisdk/digital/ansatz.py +200 -160
- qilisdk/digital/circuit.py +81 -9
- qilisdk/digital/exceptions.py +12 -6
- qilisdk/digital/gates.py +229 -86
- qilisdk/{extras/qaas/qaas_analog_result.py → functionals/__init__.py} +14 -5
- qilisdk/functionals/functional.py +39 -0
- qilisdk/{common/backend.py → functionals/functional_result.py} +3 -1
- qilisdk/functionals/sampling.py +81 -0
- qilisdk/functionals/sampling_result.py +92 -0
- qilisdk/functionals/time_evolution.py +98 -0
- qilisdk/functionals/time_evolution_result.py +84 -0
- qilisdk/functionals/variational_program.py +80 -0
- qilisdk/functionals/variational_program_result.py +69 -0
- qilisdk/logging_config.yaml +16 -0
- qilisdk/{common → optimizers}/__init__.py +1 -1
- qilisdk/optimizers/optimizer.py +39 -0
- qilisdk/{common → optimizers}/optimizer_result.py +3 -12
- qilisdk/{common/optimizer.py → optimizers/scipy_optimizer.py} +10 -28
- qilisdk/settings.py +78 -0
- qilisdk/speqtrum/__init__.py +41 -0
- qilisdk/{extras → speqtrum}/__init__.pyi +3 -3
- qilisdk/speqtrum/experiments/__init__.py +25 -0
- qilisdk/speqtrum/experiments/experiment_functional.py +124 -0
- qilisdk/speqtrum/experiments/experiment_result.py +231 -0
- qilisdk/{extras/qaas → speqtrum}/keyring.py +8 -4
- qilisdk/speqtrum/speqtrum.py +587 -0
- qilisdk/speqtrum/speqtrum_models.py +467 -0
- qilisdk/utils/__init__.py +0 -14
- qilisdk/utils/openqasm2.py +1 -1
- qilisdk/utils/serialization.py +1 -1
- qilisdk/utils/visualization/PlusJakartaSans-SemiBold.ttf +0 -0
- qilisdk/utils/visualization/__init__.py +24 -0
- qilisdk/utils/visualization/circuit_renderers.py +781 -0
- qilisdk/utils/visualization/schedule_renderers.py +166 -0
- qilisdk/utils/visualization/style.py +154 -0
- qilisdk/utils/visualization/themes.py +76 -0
- qilisdk/yaml.py +126 -0
- {qilisdk-0.1.4.dist-info → qilisdk-0.1.6.dist-info}/METADATA +186 -140
- qilisdk-0.1.6.dist-info/RECORD +69 -0
- qilisdk/analog/algorithms.py +0 -111
- qilisdk/analog/analog_backend.py +0 -43
- qilisdk/analog/analog_result.py +0 -114
- qilisdk/analog/quantum_objects.py +0 -596
- qilisdk/digital/digital_algorithm.py +0 -20
- qilisdk/digital/digital_backend.py +0 -90
- qilisdk/digital/digital_result.py +0 -145
- qilisdk/digital/vqe.py +0 -166
- qilisdk/extras/cuda/__init__.py +0 -13
- qilisdk/extras/cuda/cuda_analog_result.py +0 -19
- qilisdk/extras/cuda/cuda_digital_result.py +0 -19
- qilisdk/extras/qaas/__init__.py +0 -13
- qilisdk/extras/qaas/models.py +0 -132
- qilisdk/extras/qaas/qaas_backend.py +0 -255
- qilisdk/extras/qaas/qaas_digital_result.py +0 -20
- qilisdk/extras/qaas/qaas_time_evolution_result.py +0 -20
- qilisdk/extras/qaas/qaas_vqe_result.py +0 -20
- qilisdk-0.1.4.dist-info/RECORD +0 -51
- {qilisdk-0.1.4.dist-info → qilisdk-0.1.6.dist-info}/WHEEL +0 -0
- {qilisdk-0.1.4.dist-info → qilisdk-0.1.6.dist-info}/licenses/LICENCE +0 -0
|
@@ -0,0 +1,467 @@
|
|
|
1
|
+
# Copyright 2025 Qilimanjaro Quantum Tech
|
|
2
|
+
#
|
|
3
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
4
|
+
# you may not use this file except in compliance with the License.
|
|
5
|
+
# You may obtain a copy of the License at
|
|
6
|
+
#
|
|
7
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
8
|
+
#
|
|
9
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
10
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
11
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
12
|
+
# See the License for the specific language governing permissions and
|
|
13
|
+
# limitations under the License.
|
|
14
|
+
# ruff: noqa: ANN001, ANN202, PLR6301
|
|
15
|
+
from email.utils import parsedate_to_datetime
|
|
16
|
+
from enum import Enum
|
|
17
|
+
from typing import Any, Callable, Generic, TypeVar, cast, overload
|
|
18
|
+
|
|
19
|
+
from pydantic import AwareDatetime, BaseModel, ConfigDict, Field, field_serializer, field_validator
|
|
20
|
+
|
|
21
|
+
from qilisdk.functionals import (
|
|
22
|
+
Sampling,
|
|
23
|
+
SamplingResult,
|
|
24
|
+
TimeEvolution,
|
|
25
|
+
TimeEvolutionResult,
|
|
26
|
+
VariationalProgram,
|
|
27
|
+
VariationalProgramResult,
|
|
28
|
+
)
|
|
29
|
+
from qilisdk.functionals.functional_result import FunctionalResult
|
|
30
|
+
from qilisdk.speqtrum.experiments import RabiExperiment, RabiExperimentResult, T1Experiment, T1ExperimentResult
|
|
31
|
+
from qilisdk.utils.serialization import deserialize, serialize
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
class SpeQtrumModel(BaseModel):
|
|
35
|
+
model_config = ConfigDict(validate_by_name=True, validate_by_alias=True, arbitrary_types_allowed=True)
|
|
36
|
+
|
|
37
|
+
|
|
38
|
+
class LoginPayload(BaseModel): ...
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
class Token(SpeQtrumModel):
|
|
42
|
+
"""
|
|
43
|
+
Represents the structure of the login response:
|
|
44
|
+
{
|
|
45
|
+
"accessToken": "...",
|
|
46
|
+
"expiresIn": 123456789,
|
|
47
|
+
"issuedAt": "123456789",
|
|
48
|
+
"refreshToken": "...",
|
|
49
|
+
"tokenType": "bearer"
|
|
50
|
+
}
|
|
51
|
+
"""
|
|
52
|
+
|
|
53
|
+
access_token: str = Field(alias="accessToken")
|
|
54
|
+
expires_in: int = Field(alias="expiresIn")
|
|
55
|
+
issued_at: str = Field(alias="issuedAt")
|
|
56
|
+
refresh_token: str = Field(alias="refreshToken")
|
|
57
|
+
token_type: str = Field(alias="tokenType")
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
class DeviceStatus(str, Enum):
|
|
61
|
+
"""Device status typing for posting"""
|
|
62
|
+
|
|
63
|
+
ONLINE = "online"
|
|
64
|
+
MAINTENANCE = "maintenance"
|
|
65
|
+
OFFLINE = "offline"
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
class DeviceType(str, Enum):
|
|
69
|
+
"""Device type"""
|
|
70
|
+
|
|
71
|
+
QPU_ANALOG = "qpu.analog"
|
|
72
|
+
QPU_DIGITAL = "qpu.digital"
|
|
73
|
+
SIMULATOR = "simulator"
|
|
74
|
+
|
|
75
|
+
|
|
76
|
+
class Device(SpeQtrumModel):
|
|
77
|
+
code: str = Field(...)
|
|
78
|
+
nqubits: int = Field(...)
|
|
79
|
+
name: str = Field(...)
|
|
80
|
+
description: str = Field(...)
|
|
81
|
+
type: DeviceType = Field(...)
|
|
82
|
+
status: DeviceStatus = Field(...)
|
|
83
|
+
|
|
84
|
+
|
|
85
|
+
class ExecuteType(str, Enum):
|
|
86
|
+
SAMPLING = "sampling"
|
|
87
|
+
TIME_EVOLUTION = "time_evolution"
|
|
88
|
+
VARIATIONAL_PROGRAM = "variational_program"
|
|
89
|
+
RABI_EXPERIMENT = "rabi_experiment"
|
|
90
|
+
T1_EXPERIMENT = "t1_experiment"
|
|
91
|
+
|
|
92
|
+
|
|
93
|
+
class SamplingPayload(SpeQtrumModel):
|
|
94
|
+
sampling: Sampling = Field(...)
|
|
95
|
+
|
|
96
|
+
@field_serializer("sampling")
|
|
97
|
+
def _serialize_sampling(self, sampling: Sampling, _info):
|
|
98
|
+
return serialize(sampling)
|
|
99
|
+
|
|
100
|
+
@field_validator("sampling", mode="before")
|
|
101
|
+
def _load_sampling(cls, v):
|
|
102
|
+
if isinstance(v, str):
|
|
103
|
+
return deserialize(v, Sampling)
|
|
104
|
+
return v
|
|
105
|
+
|
|
106
|
+
|
|
107
|
+
class TimeEvolutionPayload(SpeQtrumModel):
|
|
108
|
+
time_evolution: TimeEvolution = Field(...)
|
|
109
|
+
|
|
110
|
+
@field_serializer("time_evolution")
|
|
111
|
+
def _serialize_time_evolution(self, time_evolution: TimeEvolution, _info):
|
|
112
|
+
return serialize(time_evolution)
|
|
113
|
+
|
|
114
|
+
@field_validator("time_evolution", mode="before")
|
|
115
|
+
def _load_time_evolution(cls, v):
|
|
116
|
+
if isinstance(v, str):
|
|
117
|
+
return deserialize(v, TimeEvolution)
|
|
118
|
+
return v
|
|
119
|
+
|
|
120
|
+
|
|
121
|
+
class VariationalProgramPayload(SpeQtrumModel):
|
|
122
|
+
variational_program: VariationalProgram = Field(...)
|
|
123
|
+
|
|
124
|
+
@field_serializer("variational_program")
|
|
125
|
+
def _serialize_variational_program(self, variational_program: VariationalProgram, _info):
|
|
126
|
+
return serialize(variational_program)
|
|
127
|
+
|
|
128
|
+
@field_validator("variational_program", mode="before")
|
|
129
|
+
def _load_variational_program(cls, v):
|
|
130
|
+
if isinstance(v, str):
|
|
131
|
+
return deserialize(v, VariationalProgram)
|
|
132
|
+
return v
|
|
133
|
+
|
|
134
|
+
|
|
135
|
+
class RabiExperimentPayload(SpeQtrumModel):
|
|
136
|
+
rabi_experiment: RabiExperiment = Field(...)
|
|
137
|
+
|
|
138
|
+
@field_serializer("rabi_experiment")
|
|
139
|
+
def _serialize_rabi_experiment(self, rabi_experiment: RabiExperiment, _info):
|
|
140
|
+
return serialize(rabi_experiment)
|
|
141
|
+
|
|
142
|
+
@field_validator("rabi_experiment", mode="before")
|
|
143
|
+
def _load_rabi_experiment(cls, v):
|
|
144
|
+
if isinstance(v, str):
|
|
145
|
+
return deserialize(v, RabiExperiment)
|
|
146
|
+
return v
|
|
147
|
+
|
|
148
|
+
|
|
149
|
+
class T1ExperimentPayload(SpeQtrumModel):
|
|
150
|
+
t1_experiment: T1Experiment = Field(...)
|
|
151
|
+
|
|
152
|
+
@field_serializer("t1_experiment")
|
|
153
|
+
def _serialize_t1_experiment(self, t1_experiment: T1Experiment, _info):
|
|
154
|
+
return serialize(t1_experiment)
|
|
155
|
+
|
|
156
|
+
@field_validator("t1_experiment", mode="before")
|
|
157
|
+
def _load_t1_experiment(cls, v):
|
|
158
|
+
if isinstance(v, str):
|
|
159
|
+
return deserialize(v, T1Experiment)
|
|
160
|
+
return v
|
|
161
|
+
|
|
162
|
+
|
|
163
|
+
class ExecutePayload(SpeQtrumModel):
|
|
164
|
+
type: ExecuteType = Field(...)
|
|
165
|
+
sampling_payload: SamplingPayload | None = None
|
|
166
|
+
time_evolution_payload: TimeEvolutionPayload | None = None
|
|
167
|
+
variational_program_payload: VariationalProgramPayload | None = None
|
|
168
|
+
rabi_experiment_payload: RabiExperimentPayload | None = None
|
|
169
|
+
t1_experiment_payload: T1ExperimentPayload | None = None
|
|
170
|
+
|
|
171
|
+
|
|
172
|
+
class ExecuteResult(SpeQtrumModel):
|
|
173
|
+
type: ExecuteType = Field(...)
|
|
174
|
+
sampling_result: SamplingResult | None = None
|
|
175
|
+
time_evolution_result: TimeEvolutionResult | None = None
|
|
176
|
+
variational_program_result: VariationalProgramResult | None = None
|
|
177
|
+
rabi_experiment_result: RabiExperimentResult | None = None
|
|
178
|
+
t1_experiment_result: T1ExperimentResult | None = None
|
|
179
|
+
|
|
180
|
+
@field_serializer("sampling_result")
|
|
181
|
+
def _serialize_sampling_result(self, sampling_result: SamplingResult, _info):
|
|
182
|
+
return serialize(sampling_result) if sampling_result is not None else None
|
|
183
|
+
|
|
184
|
+
@field_validator("sampling_result", mode="before")
|
|
185
|
+
def _load_sampling_result(cls, v):
|
|
186
|
+
if isinstance(v, str) and v.startswith("!"):
|
|
187
|
+
return deserialize(v, SamplingResult)
|
|
188
|
+
return v
|
|
189
|
+
|
|
190
|
+
@field_serializer("time_evolution_result")
|
|
191
|
+
def _serialize_time_evolution_result(self, time_evolution_result: TimeEvolutionResult, _info):
|
|
192
|
+
return serialize(time_evolution_result) if time_evolution_result is not None else None
|
|
193
|
+
|
|
194
|
+
@field_validator("time_evolution_result", mode="before")
|
|
195
|
+
def _load_time_evolution_result(cls, v):
|
|
196
|
+
if isinstance(v, str) and v.startswith("!"):
|
|
197
|
+
return deserialize(v, TimeEvolutionResult)
|
|
198
|
+
return v
|
|
199
|
+
|
|
200
|
+
@field_serializer("variational_program_result")
|
|
201
|
+
def _serialize_variational_program_result(self, variational_program_result: VariationalProgramResult, _info):
|
|
202
|
+
return serialize(variational_program_result) if variational_program_result is not None else None
|
|
203
|
+
|
|
204
|
+
@field_validator("variational_program_result", mode="before")
|
|
205
|
+
def _load_variational_program_result(cls, v):
|
|
206
|
+
if isinstance(v, str) and v.startswith("!"):
|
|
207
|
+
return deserialize(v, VariationalProgramResult)
|
|
208
|
+
return v
|
|
209
|
+
|
|
210
|
+
@field_serializer("rabi_experiment_result")
|
|
211
|
+
def _serialize_rabi_experiment_result(self, rabi_experiment_result: RabiExperimentResult, _info):
|
|
212
|
+
return serialize(rabi_experiment_result) if rabi_experiment_result is not None else None
|
|
213
|
+
|
|
214
|
+
@field_validator("rabi_experiment_result", mode="before")
|
|
215
|
+
def _load_rabi_experiment_result(cls, v):
|
|
216
|
+
if isinstance(v, str) and v.startswith("!"):
|
|
217
|
+
return deserialize(v, RabiExperimentResult)
|
|
218
|
+
return v
|
|
219
|
+
|
|
220
|
+
@field_serializer("t1_experiment_result")
|
|
221
|
+
def _serialize_t1_experiment_result(self, t1_experiment_result: T1ExperimentResult, _info):
|
|
222
|
+
return serialize(t1_experiment_result) if t1_experiment_result is not None else None
|
|
223
|
+
|
|
224
|
+
@field_validator("t1_experiment_result", mode="before")
|
|
225
|
+
def _load_t1_experiment_result(cls, v):
|
|
226
|
+
if isinstance(v, str) and v.startswith("!"):
|
|
227
|
+
return deserialize(v, T1ExperimentResult)
|
|
228
|
+
return v
|
|
229
|
+
|
|
230
|
+
|
|
231
|
+
ResultT_co = TypeVar("ResultT_co", bound=FunctionalResult, covariant=True)
|
|
232
|
+
VariationalInnerResultT = TypeVar("VariationalInnerResultT", bound=FunctionalResult)
|
|
233
|
+
|
|
234
|
+
|
|
235
|
+
ResultExtractor = Callable[[ExecuteResult], ResultT_co]
|
|
236
|
+
|
|
237
|
+
|
|
238
|
+
# these helpers live outside the models so they can be referenced by default values
|
|
239
|
+
def _require_sampling_result(result: ExecuteResult) -> SamplingResult:
|
|
240
|
+
if result.sampling_result is None:
|
|
241
|
+
raise RuntimeError("SpeQtrum did not return a sampling_result for a sampling execution.")
|
|
242
|
+
return result.sampling_result
|
|
243
|
+
|
|
244
|
+
|
|
245
|
+
def _require_time_evolution_result(result: ExecuteResult) -> TimeEvolutionResult:
|
|
246
|
+
if result.time_evolution_result is None:
|
|
247
|
+
raise RuntimeError("SpeQtrum did not return a time_evolution_result for a time evolution execution.")
|
|
248
|
+
return result.time_evolution_result
|
|
249
|
+
|
|
250
|
+
|
|
251
|
+
def _require_variational_program_result(result: ExecuteResult) -> VariationalProgramResult:
|
|
252
|
+
if result.variational_program_result is None:
|
|
253
|
+
raise RuntimeError("SpeQtrum did not return a variational_program_result for a variational program execution.")
|
|
254
|
+
return result.variational_program_result
|
|
255
|
+
|
|
256
|
+
|
|
257
|
+
def _require_rabi_experiment_result(result: ExecuteResult) -> RabiExperimentResult:
|
|
258
|
+
if result.rabi_experiment_result is None:
|
|
259
|
+
raise RuntimeError("SpeQtrum did not return a rabi_experiment_result for a Rabi experiment execution.")
|
|
260
|
+
return result.rabi_experiment_result
|
|
261
|
+
|
|
262
|
+
|
|
263
|
+
def _require_t1_experiment_result(result: ExecuteResult) -> T1ExperimentResult:
|
|
264
|
+
if result.t1_experiment_result is None:
|
|
265
|
+
raise RuntimeError("SpeQtrum did not return a t1_experiment_result for a T1 experiment execution.")
|
|
266
|
+
return result.t1_experiment_result
|
|
267
|
+
|
|
268
|
+
|
|
269
|
+
def _require_variational_program_result_typed(
|
|
270
|
+
inner_result_type: type[VariationalInnerResultT],
|
|
271
|
+
) -> ResultExtractor[VariationalProgramResult[VariationalInnerResultT]]:
|
|
272
|
+
def _extractor(result: ExecuteResult) -> VariationalProgramResult[VariationalInnerResultT]:
|
|
273
|
+
variational_result = _require_variational_program_result(result)
|
|
274
|
+
optimal_results = variational_result.optimal_execution_results
|
|
275
|
+
if not isinstance(optimal_results, inner_result_type):
|
|
276
|
+
raise RuntimeError(
|
|
277
|
+
"SpeQtrum returned a variational program result whose optimal execution result "
|
|
278
|
+
f"({type(optimal_results).__qualname__}) does not match the expected "
|
|
279
|
+
f"{inner_result_type.__qualname__}."
|
|
280
|
+
)
|
|
281
|
+
return cast("VariationalProgramResult[VariationalInnerResultT]", variational_result)
|
|
282
|
+
|
|
283
|
+
return _extractor
|
|
284
|
+
|
|
285
|
+
|
|
286
|
+
class JobHandle(SpeQtrumModel, Generic[ResultT_co]):
|
|
287
|
+
"""Strongly typed reference to a submitted SpeQtrum job."""
|
|
288
|
+
|
|
289
|
+
id: int
|
|
290
|
+
execute_type: ExecuteType
|
|
291
|
+
extractor: ResultExtractor[ResultT_co] = Field(repr=False, exclude=True)
|
|
292
|
+
|
|
293
|
+
@classmethod
|
|
294
|
+
def sampling(cls, job_id: int) -> "JobHandle[SamplingResult]":
|
|
295
|
+
return cls(id=job_id, execute_type=ExecuteType.SAMPLING, extractor=_require_sampling_result) # type: ignore[return-value, arg-type]
|
|
296
|
+
|
|
297
|
+
@classmethod
|
|
298
|
+
def time_evolution(cls, job_id: int) -> "JobHandle[TimeEvolutionResult]":
|
|
299
|
+
return cls(id=job_id, execute_type=ExecuteType.TIME_EVOLUTION, extractor=_require_time_evolution_result) # type: ignore[return-value, arg-type]
|
|
300
|
+
|
|
301
|
+
@overload
|
|
302
|
+
@classmethod
|
|
303
|
+
def variational_program(cls, job_id: int) -> "JobHandle[VariationalProgramResult]": ...
|
|
304
|
+
|
|
305
|
+
@overload
|
|
306
|
+
@classmethod
|
|
307
|
+
def variational_program(
|
|
308
|
+
cls, job_id: int, *, result_type: type[VariationalInnerResultT]
|
|
309
|
+
) -> "JobHandle[VariationalProgramResult[VariationalInnerResultT]]": ...
|
|
310
|
+
|
|
311
|
+
@classmethod
|
|
312
|
+
def variational_program(
|
|
313
|
+
cls, job_id: int, *, result_type: type[VariationalInnerResultT] | None = None
|
|
314
|
+
) -> "JobHandle[Any]":
|
|
315
|
+
"""Create a variational-program handle for an existing job identifier.
|
|
316
|
+
|
|
317
|
+
Args:
|
|
318
|
+
job_id: Numeric identifier returned by the SpeQtrum service.
|
|
319
|
+
result_type: Optional functional result type expected within the
|
|
320
|
+
variational program payload. When provided the returned handle
|
|
321
|
+
enforces that the optimiser output matches this type.
|
|
322
|
+
|
|
323
|
+
Returns:
|
|
324
|
+
JobHandle: A handle whose ``get_results`` invocation yields a
|
|
325
|
+
``VariationalProgramResult`` preserving the requested inner result
|
|
326
|
+
type when supplied.
|
|
327
|
+
"""
|
|
328
|
+
if result_type is None:
|
|
329
|
+
handle = cls(
|
|
330
|
+
id=job_id,
|
|
331
|
+
execute_type=ExecuteType.VARIATIONAL_PROGRAM,
|
|
332
|
+
extractor=_require_variational_program_result, # type: ignore[arg-type]
|
|
333
|
+
)
|
|
334
|
+
return cast("JobHandle[VariationalProgramResult]", handle)
|
|
335
|
+
|
|
336
|
+
extractor = _require_variational_program_result_typed(result_type)
|
|
337
|
+
handle = cls(id=job_id, execute_type=ExecuteType.VARIATIONAL_PROGRAM, extractor=extractor) # type: ignore[arg-type]
|
|
338
|
+
return cast("JobHandle[VariationalProgramResult[VariationalInnerResultT]]", handle)
|
|
339
|
+
|
|
340
|
+
@classmethod
|
|
341
|
+
def rabi_experiment(cls, job_id: int) -> "JobHandle[RabiExperimentResult]":
|
|
342
|
+
return cls(id=job_id, execute_type=ExecuteType.RABI_EXPERIMENT, extractor=_require_rabi_experiment_result) # type: ignore[return-value, arg-type]
|
|
343
|
+
|
|
344
|
+
@classmethod
|
|
345
|
+
def t1_experiment(cls, job_id: int) -> "JobHandle[T1ExperimentResult]":
|
|
346
|
+
return cls(id=job_id, execute_type=ExecuteType.T1_EXPERIMENT, extractor=_require_t1_experiment_result) # type: ignore[return-value, arg-type]
|
|
347
|
+
|
|
348
|
+
def bind(self, detail: "JobDetail") -> "TypedJobDetail[ResultT_co]":
|
|
349
|
+
"""Attach this handle's typing information to a concrete job detail.
|
|
350
|
+
|
|
351
|
+
Args:
|
|
352
|
+
detail: Un-typed job detail payload returned by the SpeQtrum API.
|
|
353
|
+
|
|
354
|
+
Returns:
|
|
355
|
+
TypedJobDetail: Wrapper exposing ``get_results`` with the typing
|
|
356
|
+
captured when the handle was created.
|
|
357
|
+
"""
|
|
358
|
+
return TypedJobDetail.model_validate(
|
|
359
|
+
{
|
|
360
|
+
**detail.model_dump(),
|
|
361
|
+
"expected_type": self.execute_type,
|
|
362
|
+
"extractor": self.extractor,
|
|
363
|
+
}
|
|
364
|
+
)
|
|
365
|
+
|
|
366
|
+
|
|
367
|
+
class JobStatus(str, Enum):
|
|
368
|
+
"Job has not been submitted to the Lab api"
|
|
369
|
+
|
|
370
|
+
PENDING = "pending"
|
|
371
|
+
"Job has been queued but not yet validated"
|
|
372
|
+
VALIDATING = "validating"
|
|
373
|
+
"Job has been validated and is queued for execution"
|
|
374
|
+
QUEUED = "queued"
|
|
375
|
+
"Job is being executed on the device"
|
|
376
|
+
RUNNING = "running"
|
|
377
|
+
"Job finished successfully"
|
|
378
|
+
COMPLETED = "completed"
|
|
379
|
+
"Job failed due to an error"
|
|
380
|
+
ERROR = "error"
|
|
381
|
+
"Job was cancelled by the user or system"
|
|
382
|
+
CANCELLED = "cancelled"
|
|
383
|
+
"Job failed due to timeout"
|
|
384
|
+
TIMEOUT = "timeout"
|
|
385
|
+
|
|
386
|
+
|
|
387
|
+
class JobType(str, Enum):
|
|
388
|
+
DIGITAL = "digital"
|
|
389
|
+
PULSE = "pulse"
|
|
390
|
+
ANALOG = "analog"
|
|
391
|
+
VARIATIONAL = "variational"
|
|
392
|
+
|
|
393
|
+
|
|
394
|
+
class JobId(SpeQtrumModel):
|
|
395
|
+
"""Handle/reference you normally get back immediately after `POST /execute`."""
|
|
396
|
+
|
|
397
|
+
id: int = Field(...)
|
|
398
|
+
|
|
399
|
+
|
|
400
|
+
class JobInfo(JobId):
|
|
401
|
+
"""
|
|
402
|
+
Light-weight representation suitable for 'list jobs' and polling
|
|
403
|
+
when you do *not* need logs or results.
|
|
404
|
+
"""
|
|
405
|
+
|
|
406
|
+
name: str = Field(...)
|
|
407
|
+
description: str = Field(...)
|
|
408
|
+
device_id: int = Field(...)
|
|
409
|
+
status: JobStatus = Field(...)
|
|
410
|
+
created_at: AwareDatetime = Field(...)
|
|
411
|
+
updated_at: AwareDatetime | None = None
|
|
412
|
+
completed_at: AwareDatetime | None = None
|
|
413
|
+
|
|
414
|
+
@field_validator("created_at", mode="before")
|
|
415
|
+
def _parse_created_at(cls, v):
|
|
416
|
+
return parsedate_to_datetime(v) if isinstance(v, str) else v
|
|
417
|
+
|
|
418
|
+
@field_validator("updated_at", mode="before")
|
|
419
|
+
def _parse_updated_at(cls, v):
|
|
420
|
+
return parsedate_to_datetime(v) if isinstance(v, str) else v
|
|
421
|
+
|
|
422
|
+
@field_validator("completed_at", mode="before")
|
|
423
|
+
def _parse_completed_at(cls, v):
|
|
424
|
+
return parsedate_to_datetime(v) if isinstance(v, str) else v
|
|
425
|
+
|
|
426
|
+
|
|
427
|
+
class JobDetail(JobInfo):
|
|
428
|
+
"""
|
|
429
|
+
Full representation returned by `GET /jobs/{id}` when payload/result/logs
|
|
430
|
+
are requested.
|
|
431
|
+
"""
|
|
432
|
+
|
|
433
|
+
payload: ExecutePayload | None = None
|
|
434
|
+
result: ExecuteResult | None = None
|
|
435
|
+
jobType: JobType | None = None
|
|
436
|
+
logs: str | None = None
|
|
437
|
+
error: str | None = None
|
|
438
|
+
error_logs: str | None = None
|
|
439
|
+
|
|
440
|
+
|
|
441
|
+
class TypedJobDetail(JobDetail, Generic[ResultT_co]):
|
|
442
|
+
"""`JobDetail` subclass that exposes a strongly typed `get_results` method."""
|
|
443
|
+
|
|
444
|
+
expected_type: ExecuteType = Field(repr=False)
|
|
445
|
+
extractor: ResultExtractor[ResultT_co] = Field(repr=False, exclude=True)
|
|
446
|
+
|
|
447
|
+
def get_results(self) -> ResultT_co:
|
|
448
|
+
"""Return the strongly typed execution result.
|
|
449
|
+
|
|
450
|
+
Returns:
|
|
451
|
+
ResultT_co: Result payload associated with the completed job,
|
|
452
|
+
respecting the type information carried by the originating
|
|
453
|
+
``JobHandle``.
|
|
454
|
+
|
|
455
|
+
Raises:
|
|
456
|
+
RuntimeError: If SpeQtrum has not populated the result payload or
|
|
457
|
+
the execute type disagrees with the handle.
|
|
458
|
+
"""
|
|
459
|
+
if self.result is None:
|
|
460
|
+
raise RuntimeError("The job completed without a result payload; inspect `error` or `logs` for details.")
|
|
461
|
+
|
|
462
|
+
if self.result.type != self.expected_type:
|
|
463
|
+
raise RuntimeError(
|
|
464
|
+
f"Expected a result of type '{self.expected_type.value}' but received '{self.result.type.value}'."
|
|
465
|
+
)
|
|
466
|
+
|
|
467
|
+
return self.extractor(self.result)
|
qilisdk/utils/__init__.py
CHANGED
|
@@ -11,17 +11,3 @@
|
|
|
11
11
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
12
12
|
# See the License for the specific language governing permissions and
|
|
13
13
|
# limitations under the License.
|
|
14
|
-
|
|
15
|
-
from .openqasm2 import from_qasm2, from_qasm2_file, to_qasm2, to_qasm2_file
|
|
16
|
-
from .serialization import deserialize, deserialize_from, serialize, serialize_to
|
|
17
|
-
|
|
18
|
-
__all__ = [
|
|
19
|
-
"deserialize",
|
|
20
|
-
"deserialize_from",
|
|
21
|
-
"from_qasm2",
|
|
22
|
-
"from_qasm2_file",
|
|
23
|
-
"serialize",
|
|
24
|
-
"serialize_to",
|
|
25
|
-
"to_qasm2",
|
|
26
|
-
"to_qasm2_file",
|
|
27
|
-
]
|
qilisdk/utils/openqasm2.py
CHANGED
|
@@ -70,7 +70,7 @@ def to_qasm2(circuit: Circuit) -> str:
|
|
|
70
70
|
# Format parameter string, if any.
|
|
71
71
|
param_str = ""
|
|
72
72
|
if gate.is_parameterized:
|
|
73
|
-
parameters = ", ".join(str(p) for p in gate.
|
|
73
|
+
parameters = ", ".join(str(p) for p in gate.get_parameter_values())
|
|
74
74
|
param_str = f"({parameters})"
|
|
75
75
|
# Format qubit operands.
|
|
76
76
|
qubit_str = ", ".join(f"q[{q}]" for q in gate.qubits)
|
qilisdk/utils/serialization.py
CHANGED
|
Binary file
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
# Copyright 2025 Qilimanjaro Quantum Tech
|
|
2
|
+
#
|
|
3
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
4
|
+
# you may not use this file except in compliance with the License.
|
|
5
|
+
# You may obtain a copy of the License at
|
|
6
|
+
#
|
|
7
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
|
8
|
+
#
|
|
9
|
+
# Unless required by applicable law or agreed to in writing, software
|
|
10
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
11
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
12
|
+
# See the License for the specific language governing permissions and
|
|
13
|
+
# limitations under the License.
|
|
14
|
+
|
|
15
|
+
from .style import CircuitStyle, ScheduleStyle
|
|
16
|
+
from .themes import Theme, dark, light
|
|
17
|
+
|
|
18
|
+
__all__ = [
|
|
19
|
+
"CircuitStyle",
|
|
20
|
+
"ScheduleStyle",
|
|
21
|
+
"Theme",
|
|
22
|
+
"dark",
|
|
23
|
+
"light",
|
|
24
|
+
]
|