Skip to main content

Sampling on Braket's real quantum computers

Here we introduce the Braket backends and some Braket specific features QURI Parts provide

Prerequisite

This section requires topics described in previous sections(Samplers, Sampling estimation and Sampling Backends), so you need to read them before this section.

In this section, we use Amazon Braket as the platform with real quantum computers. In order to use Braket devices provided on AWS, you need to have an AWS account and enable Braket service. Please see Amazon Braket Documentation for details. In this section, instead, we use the local simulator included in Amazon Braket SDK, which does not require an AWS account. The Braket devices provided on AWS and the local simulator have the same interface, you can simply replace them each other.

QURI Parts modules used in this tutorial: quri-parts-circuit, quri-parts-core and quri-parts-braket. You can install them as follows:

!pip install "quri-parts[braket]"
Requirement already satisfied: quri-parts[braket] in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (0.20.3)
Requirement already satisfied: quri-parts-algo in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts[braket]) (0.20.3)
``````output
Collecting quri-parts-braket (from quri-parts[braket])
Using cached quri_parts_braket-0.20.3-py3-none-any.whl.metadata (1.4 kB)
Requirement already satisfied: quri-parts-chem in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts[braket]) (0.20.3)
Requirement already satisfied: quri-parts-circuit in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts[braket]) (0.20.3)
Requirement already satisfied: quri-parts-core in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts[braket]) (0.20.3)
Requirement already satisfied: scipy<2.0.0,>=1.9.1 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts-algo->quri-parts[braket]) (1.11.4)
Requirement already satisfied: typing-extensions<5.0.0,>=4.1.1 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts-algo->quri-parts[braket]) (4.12.2)
``````output
Collecting amazon-braket-schemas<2.0.0,>=1.22.0 (from quri-parts-braket->quri-parts[braket])
Using cached amazon_braket_schemas-1.22.3-py3-none-any.whl.metadata (5.8 kB)
``````output
Collecting amazon-braket-sdk<2.0.0,>=1.25.1 (from quri-parts-braket->quri-parts[braket])
``````output
Downloading amazon_braket_sdk-1.88.2.post0-py3-none-any.whl.metadata (14 kB)
Requirement already satisfied: numpy>=1.22.0 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts-circuit->quri-parts[braket]) (1.26.4)
Requirement already satisfied: quri-parts-rust in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts-circuit->quri-parts[braket]) (0.20.2)
Requirement already satisfied: networkx in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from quri-parts-core->quri-parts[braket]) (3.4.2)
Requirement already satisfied: pydantic>2 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from amazon-braket-schemas<2.0.0,>=1.22.0->quri-parts-braket->quri-parts[braket]) (2.10.2)
``````output
Collecting amazon-braket-default-simulator>=1.26.0 (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
``````output
Downloading amazon_braket_default_simulator-1.26.1-py3-none-any.whl.metadata (6.3 kB)
``````output
Collecting oqpy~=0.3.5 (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached oqpy-0.3.7-py3-none-any.whl.metadata (8.3 kB)
``````output
Collecting backoff (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached backoff-2.2.1-py3-none-any.whl.metadata (14 kB)
``````output
Collecting boltons (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached boltons-24.1.0-py3-none-any.whl.metadata (1.5 kB)
``````output
Collecting boto3>=1.28.53 (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
``````output
Downloading boto3-1.35.73-py3-none-any.whl.metadata (6.7 kB)
``````output
Collecting cloudpickle==2.2.1 (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached cloudpickle-2.2.1-py3-none-any.whl.metadata (6.9 kB)
Requirement already satisfied: nest-asyncio in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket]) (1.6.0)
``````output
Collecting openpulse (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached openpulse-1.0.1-py3-none-any.whl.metadata (2.0 kB)
``````output
Collecting openqasm3 (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
``````output
Using cached openqasm3-1.0.0-py3-none-any.whl.metadata (6.0 kB)
Requirement already satisfied: sympy in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket]) (1.13.3)
``````output
Collecting backports.entry-points-selectable (from amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached backports.entry_points_selectable-1.3.0-py3-none-any.whl.metadata (4.1 kB)
``````output
Collecting opt_einsum (from amazon-braket-default-simulator>=1.26.0->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
``````output
Using cached opt_einsum-3.4.0-py3-none-any.whl.metadata (6.3 kB)
Collecting antlr4-python3-runtime==4.9.2 (from amazon-braket-default-simulator>=1.26.0->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached antlr4_python3_runtime-4.9.2-py3-none-any.whl
``````output
Collecting botocore<1.36.0,>=1.35.73 (from boto3>=1.28.53->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
``````output
Downloading botocore-1.35.73-py3-none-any.whl.metadata (5.7 kB)
``````output
Collecting jmespath<2.0.0,>=0.7.1 (from boto3>=1.28.53->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached jmespath-1.0.1-py3-none-any.whl.metadata (7.6 kB)
``````output
Collecting s3transfer<0.11.0,>=0.10.0 (from boto3>=1.28.53->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached s3transfer-0.10.4-py3-none-any.whl.metadata (1.7 kB)
``````output
Collecting mypy-extensions>=0.2.0 (from oqpy~=0.3.5->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket])
Using cached mypy_extensions-1.0.0-py3-none-any.whl.metadata (1.1 kB)
Requirement already satisfied: annotated-types>=0.6.0 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from pydantic>2->amazon-braket-schemas<2.0.0,>=1.22.0->quri-parts-braket->quri-parts[braket]) (0.7.0)
Requirement already satisfied: pydantic-core==2.27.1 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from pydantic>2->amazon-braket-schemas<2.0.0,>=1.22.0->quri-parts-braket->quri-parts[braket]) (2.27.1)
``````output
Requirement already satisfied: mpmath<1.4,>=1.1.0 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from sympy->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket]) (1.3.0)
Requirement already satisfied: python-dateutil<3.0.0,>=2.1 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from botocore<1.36.0,>=1.35.73->boto3>=1.28.53->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket]) (2.9.0.post0)
Requirement already satisfied: urllib3!=2.2.0,<3,>=1.25.4 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from botocore<1.36.0,>=1.35.73->boto3>=1.28.53->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket]) (2.2.3)
``````output
Requirement already satisfied: six>=1.5 in /home/drokles/.cache/pypoetry/virtualenvs/quri-sdk-notebooks-l1kMe6Lz-py3.11/lib/python3.11/site-packages (from python-dateutil<3.0.0,>=2.1->botocore<1.36.0,>=1.35.73->boto3>=1.28.53->amazon-braket-sdk<2.0.0,>=1.25.1->quri-parts-braket->quri-parts[braket]) (1.16.0)
Using cached quri_parts_braket-0.20.3-py3-none-any.whl (14 kB)
Using cached amazon_braket_schemas-1.22.3-py3-none-any.whl (127 kB)
``````output
Downloading amazon_braket_sdk-1.88.2.post0-py3-none-any.whl (315 kB)
[?25l ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 0.0/315.7 kB ? eta -:--:--
 ━━━━━╺━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 41.0/315.7 kB 1.1 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━╺━━━━━━━━━━━━━━━━━ 174.1/315.7 kB 2.6 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 315.7/315.7 kB 3.2 MB/s eta 0:00:00
[?25hUsing cached cloudpickle-2.2.1-py3-none-any.whl (25 kB)
Downloading amazon_braket_default_simulator-1.26.1-py3-none-any.whl (223 kB)
[?25l ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 0.0/224.0 kB ? eta -:--:--
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━╸━━━━━━━━━━━━━━ 143.4/224.0 kB 5.9 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 224.0/224.0 kB 4.4 MB/s eta 0:00:00
[?25hDownloading boto3-1.35.73-py3-none-any.whl (139 kB)
[?25l ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 0.0/139.2 kB ? eta -:--:--
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━━╺━━━━━━━━━━━━━ 92.2/139.2 kB 3.9 MB/s eta 0:00:01
 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 139.2/139.2 kB 3.5 MB/s eta 0:00:00
[?25hUsing cached oqpy-0.3.7-py3-none-any.whl (36 kB)
Using cached openpulse-1.0.1-py3-none-any.whl (537 kB)
Using cached openqasm3-1.0.0-py3-none-any.whl (539 kB)
``````output
Using cached backoff-2.2.1-py3-none-any.whl (15 kB)
Using cached backports.entry_points_selectable-1.3.0-py3-none-any.whl (6.2 kB)
Using cached boltons-24.1.0-py3-none-any.whl (192 kB)
Downloading botocore-1.35.73-py3-none-any.whl (13.1 MB)
[?25l ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 0.0/13.1 MB ? eta -:--:--
``````output

 ╸━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 0.3/13.1 MB 8.7 MB/s eta 0:00:02
``````output

 ━━╺━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 0.8/13.1 MB 12.0 MB/s eta 0:00:02
 ━━━╸━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 1.2/13.1 MB 12.2 MB/s eta 0:00:01
``````output

 ━━━━━╸━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 1.8/13.1 MB 13.4 MB/s eta 0:00:01
``````output

 ━━━━━━━╺━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 2.4/13.1 MB 14.3 MB/s eta 0:00:01
 ━━━━━━━━━╺━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 3.1/13.1 MB 15.1 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━╸━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 3.8/13.1 MB 15.9 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━╸━━━━━━━━━━━━━━━━━━━━━━━━━━ 4.5/13.1 MB 16.4 MB/s eta 0:00:01
 ━━━━━━━━━━━━━━━╸━━━━━━━━━━━━━━━━━━━━━━━━ 5.1/13.1 MB 16.8 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━╸━━━━━━━━━━━━━━━━━━━━━━ 5.8/13.1 MB 16.9 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━╺━━━━━━━━━━━━━━━━━━━ 6.7/13.1 MB 17.6 MB/s eta 0:00:01
 ━━━━━━━━━━━━━━━━━━━━━━╸━━━━━━━━━━━━━━━━━ 7.5/13.1 MB 18.1 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━╺━━━━━━━━━━━━━━ 8.3/13.1 MB 18.7 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━╺━━━━━━━━━━━ 9.2/13.1 MB 19.2 MB/s eta 0:00:01
 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━╸━━━━━━━━━ 10.0/13.1 MB 19.5 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━╸━━━━━━ 11.1/13.1 MB 21.4 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━╺━━━ 11.9/13.1 MB 22.6 MB/s eta 0:00:01
 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━╸ 13.0/13.1 MB 24.0 MB/s eta 0:00:01
``````output

 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 13.1/13.1 MB 22.9 MB/s eta 0:00:00
[?25hUsing cached jmespath-1.0.1-py3-none-any.whl (20 kB)
Using cached mypy_extensions-1.0.0-py3-none-any.whl (4.7 kB)
Using cached s3transfer-0.10.4-py3-none-any.whl (83 kB)
Using cached opt_einsum-3.4.0-py3-none-any.whl (71 kB)
``````output
Installing collected packages: openqasm3, antlr4-python3-runtime, opt_einsum, mypy-extensions, jmespath, cloudpickle, boltons, backports.entry-points-selectable, backoff, openpulse, botocore, s3transfer, oqpy, amazon-braket-schemas, boto3, amazon-braket-default-simulator, amazon-braket-sdk, quri-parts-braket
``````output
Successfully installed amazon-braket-default-simulator-1.26.1 amazon-braket-schemas-1.22.3 amazon-braket-sdk-1.88.2.post0 antlr4-python3-runtime-4.9.2 backoff-2.2.1 backports.entry-points-selectable-1.3.0 boltons-24.1.0 boto3-1.35.73 botocore-1.35.73 cloudpickle-2.2.1 jmespath-1.0.1 mypy-extensions-1.0.0 openpulse-1.0.1 openqasm3-1.0.0 opt_einsum-3.4.0 oqpy-0.3.7 quri-parts-braket-0.20.3 s3transfer-0.10.4
``````output

[notice] A new release of pip is available: 24.0 -> 24.3.1
[notice] To update, run: pip install --upgrade pip

The BraketSamplingBackend

How to create a SamplingBackend object depends on the used backend. For Braket devices, you can create a BraketSamplingBackend by passing a braket.devices.Device object (provided by Amazon Braket SDK):

from braket.aws import AwsDevice
from braket.devices import LocalSimulator

# A device for QPU provided on AWS
# device = AwsDevice("arn:aws:braket:us-west-1::device/qpu/rigetti/Aspen-M-2")

# A device for the local simulator
device = LocalSimulator()
from quri_parts.braket.backend import BraketSamplingBackend

# Create a SamplingBackend with the device
backend = BraketSamplingBackend(device)

With the sampling backend we just created, we can run the exact codes as in the Sampling Backend and Sampler section of the Sampling backend tutorial.

Qubit mapping

Here, we explain some details you need to know when you use the devices provided by Braket. Following the code in the Sampling Backend tutorial, we consider the following qubit mapping sampling.

from numpy import pi
from quri_parts.circuit import QuantumCircuit
from quri_parts.core.sampling import create_sampler_from_sampling_backend

circuit = QuantumCircuit(4)
circuit.add_X_gate(0)
circuit.add_H_gate(1)
circuit.add_Y_gate(2)
circuit.add_CNOT_gate(1, 2)
circuit.add_RX_gate(3, pi/4)

backend = BraketSamplingBackend(device, qubit_mapping={0: 3, 1: 2, 2: 0, 3: 1})
sampler = create_sampler_from_sampling_backend(backend)
sampling_result = sampler(circuit, 1000)
print(sampling_result)
{5: 427, 11: 84, 3: 434, 13: 55}

The result looks similar to one with no qubit mapping, since the measurement result from the device is mapped backward so that it is interpreted in terms of the original qubit indices.

You may notice that the above mapping is a permutation of the original qubit indices and device qubits with indices larger than 3 are not involved. The reason for choosing such a mapping is to avoid an error of LocalSimulator: the LocalSimulator does not accept non-contiguous qubit indices. On the other hand, the qubit mapping feature of the SamplingBackend accepts such a mapping, as shown below.

When you apply qubit mapping to devices provided on AWS, you will need to enable manual qubit allocation by passing disable_qubit_rewiring=True to the device. You can specify such an argument (i.e. keyword arguments for run method of a braket.devices.Device object) via run_kwargs argument of the BraketSamplingBackend object:

# Commented out because it requires an access to a real device on AWS

# device = AwsDevice("arn:aws:braket:us-west-1::device/qpu/rigetti/Aspen-M-2")
# backend = BraketSamplingBackend(
# device,
# qubit_mapping={0: 10, 1: 13, 2: 17, 3: 21},
# run_kwargs={"disable_qubit_rewiring": True},
# )
# sampler = create_sampler_from_sampling_backend(backend)
# sampling_result = sampler(circuit, 1000)
# print(sampling_result)

Circuit transpilation before execution

The transpilation performed by default depends on the backend; in the case of BraketSamplingBackend, it uses quri_parts.braket.circuit.BraketSetTranspiler for all devices, and also performs some device-specific transpilation defined in quri_parts.braket.backend.transpiler. It is possible to change the former one (device-independent transpilation) by supplying circuit_transpiler argument to BraketSamplingBackend.