Source code for qbraid.runtime.native.device

# Copyright 2025 qBraid
#
# 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.

# pylint: disable=arguments-differ,too-many-arguments

"""
Module defining QbraidDevice class

"""

from __future__ import annotations

from typing import TYPE_CHECKING, Any, overload

from qbraid_core.services.runtime import QuantumRuntimeClient
from qbraid_core.services.runtime.schemas import JobRequest, Program

from qbraid._logging import logger
from qbraid.runtime.device import QuantumDevice
from qbraid.runtime.group import get_active_group, get_active_group_session
from qbraid.runtime.noise import NoiseModel

from .job import QbraidJob

if TYPE_CHECKING:
    import qbraid_core.services.runtime

    import qbraid.runtime


[docs] class QbraidDevice(QuantumDevice): """Class to represent a qBraid device."""
[docs] def __init__( self, profile: qbraid.runtime.TargetProfile, client: qbraid_core.services.runtime.QuantumRuntimeClient | None = None, **kwargs, ): """Create a new QbraidDevice object.""" super().__init__(profile=profile, **kwargs) self._client = client or QuantumRuntimeClient()
@property def client(self) -> QuantumRuntimeClient: """Return the QuantumClient object.""" return self._client def __str__(self): """String representation of the QbraidDevice object.""" return f"{self.__class__.__name__}('{self.id}')" def status(self) -> qbraid.runtime.DeviceStatus: """Return device status.""" device_data = self.client.get_device(self.id) return device_data.status def queue_depth(self) -> int: """Return the number of jobs in the queue for the backend""" device_data = self.client.get_device(self.id) return device_data.queueDepth or 0 def _resolve_noise_model(self, noise_model: NoiseModel | str) -> str: """Verify given noise model is supported by device and map to string representation.""" if self.profile.noise_models is None: raise ValueError("Noise models are not supported by this device.") if isinstance(noise_model, NoiseModel): noise_model = noise_model.value elif not isinstance(noise_model, str): raise ValueError( f"Invalid type for noise model: {type(noise_model)}. Expected str or NoiseModel." ) if noise_model not in self.profile.noise_models: raise ValueError(f"Noise model '{noise_model}' not supported by device.") return self.profile.noise_models.get(noise_model).name @overload def submit( self, run_input: Program, shots: int | None = None, name: str | None = None, tags: dict[str, str | int | bool] | None = None, runtime_options: dict[str, Any] | None = None, as_batch: bool = False, ) -> QbraidJob: ... @overload def submit( self, run_input: list[Program], shots: int | None = None, name: str | None = None, tags: dict[str, str | int | bool] | None = None, runtime_options: dict[str, Any] | None = None, as_batch: bool = False, ) -> list[QbraidJob]: ... @overload def submit( self, run_input: list[Program], shots: int | None = None, name: str | None = None, tags: dict[str, str | int | bool] | None = None, runtime_options: dict[str, Any] | None = None, as_batch: bool = True, ) -> QbraidJob: ... def submit( self, run_input: Program | list[Program], shots: int | None = None, name: str | None = None, tags: dict[str, str | int | bool] | None = None, runtime_options: dict[str, Any] | None = None, as_batch: bool = False, ) -> QbraidJob | list[QbraidJob]: """Submit a program to the device. If an active GroupJobSession context exists, the group QRN is automatically included in the job request and submitted jobs are registered with the session. Args: run_input: A single program or a list of programs to submit to the device. shots: The number of shots to run the program(s). name: The name of the job. tags: A dictionary of tags to add to the job. runtime_options: A dictionary of runtime options to pass to the device. as_batch: When True, submit all programs as a single batch job (one API call, one QRN, one status). Returns a single QbraidJob. Check QbraidDevice.profile.batch_job_support to verify if batch jobs are supported by this device. """ tags = tags or {} runtime_options = runtime_options or {} noise_model: NoiseModel | str | None = runtime_options.pop("noise_model", None) # Read group context group_job_qrn = get_active_group() session = get_active_group_session() if group_job_qrn else None if noise_model: runtime_options["noiseModel"] = self._resolve_noise_model(noise_model) if as_batch: if not self.profile.get("batch_job_support"): raise ValueError("Batch jobs are not supported by this device.") if not isinstance(run_input, list): raise ValueError("Batch jobs require a list of programs.") # Wrap so the loop iterates once: [Program] for single, [list[Program]] # for batch (sends the full list as one API call → 1 QRN, N circuits). is_single_input = as_batch or not isinstance(run_input, list) run_input = [run_input] if is_single_input else run_input logger.debug( "Submitting %s to device '%s' (group: %s)", "batch job" if as_batch else f"{len(run_input)} job(s)", self.id, group_job_qrn, ) jobs = [] for program in run_input: job_request = JobRequest( deviceQrn=self.id, program=program, shots=shots, name=name, tags=tags, runtimeOptions=runtime_options, groupJobQrn=group_job_qrn, ) job_data = self.client.create_job(job_request) job = QbraidJob(job_id=job_data.jobQrn, device=self, client=self.client) jobs.append(job) if session is not None: session._register_job(job) return jobs[0] if is_single_input else jobs