TKET Backend
tutorial¶
Download this notebook - backends_example.ipynb
This example shows how to use pytket
to execute quantum circuits on both simulators and real devices, and how to interpret the results. As tket is designed to be platform-agnostic, we have unified the interfaces of different providers as much as possible into the Backend
class for maximum portability of code.
For the full list of supported backends see the pytket extensions index page.
In this notebook we will focus on the Aer, IBMQ and ProjectQ backends.
To get started, we must install the core pytket package and the subpackages required to interface with the desired providers. We will also need the QubitOperator
class from openfermion
to construct operators for a later example. To get everything run the following in shell:
pip install pytket pytket-qiskit pytket-projectq openfermion
First, import the backends that we will be demonstrating.
from pytket.extensions.qiskit import (
AerStateBackend,
AerBackend,
AerUnitaryBackend,
IBMQBackend,
IBMQEmulatorBackend,
)
from pytket.extensions.projectq import ProjectQBackend
We are also going to be making a circuit to run on these backends, so import the Circuit
class.
from pytket import Circuit, Qubit
Below we generate a circuit which will produce a Bell state, assuming the qubits are all initialised in the |0> state:
circ = Circuit(2)
circ.H(0)
circ.CX(0, 1)
As a sanity check, we will use the AerStateBackend
to verify that circ
does actually produce a Bell state.
To submit a circuit for excution on a backend we can use process_circuit
with appropriate arguments. If we have multiple circuits to excecute, we can use process_circuits
(note the plural), which will attempt to batch up the circuits if possible. Both methods return a ResultHandle
object per submitted Circuit
which you can use with result retrieval methods to get the result type you want (as long as that result type is supported by the backend).
Calling get_state
will return a numpy
array corresponding to the statevector.
This style of usage is used consistently in the pytket
backends.
aer_state_b = AerStateBackend()
state_handle = aer_state_b.process_circuit(circ)
statevector = aer_state_b.get_result(state_handle).get_state()
print(statevector)
As we can see, the output state vector \(\lvert \psi_{\mathrm{circ}}\rangle\) is \((\lvert00\rangle + \lvert11\rangle)/\sqrt2\).
This is a symmetric state. For non-symmetric states, we default to an ILO-BE format (increasing lexicographic order of (qu)bit ids, big-endian), but an alternative convention can be specified when retrieving results from backends. See the docs for the BasisOrder
enum for more information.
A lesser-used simulator available through Qiskit Aer is their unitary simulator. This will be somewhat more expensive to run, but returns the full unitary matrix for the provided circuit. This is useful in the design of small subcircuits that will be used multiple times within other larger circuits - statevector simulators will only test that they act correctly on the \(\lvert 0 \rangle^{\otimes n}\) state, which is not enough to guarantee the circuit’s correctness.
The AerUnitaryBackend
provides a convenient access point for this simulator for use with pytket
circuits. The unitary of the circuit can be retrieved from backends that support it using the BackendResult.get_unitary
interface. In this example, we chose to use Backend.run_circuit
, which is equivalent to calling process_circuit
followed by get_result
.
aer_unitary_b = AerUnitaryBackend()
result = aer_unitary_b.run_circuit(circ)
print(result.get_unitary())
Note that state vector and unitary simulations are also available in pytket directly. In general, we recommend you use these unless you require another Backend explicitly.
statevector = circ.get_statevector()
unitary = circ.get_unitary()
Now suppose we want to measure this Bell state to get some actual results out, so let’s append some Measure
gates to the circuit. The Circuit
class has the measure_all
utility function which appends Measure
gates on every qubit. All of these results will be written to the default classical register (‘c’). This function will automatically add the classical bits to the circuit if they are not already there.
circ.measure_all()
We can get some measured counts out from the AerBackend
, which is an interface to the Qiskit Aer QASM simulator. Suppose we would like to get 10 shots out (10 repeats of the circuit and measurement). We can seed the simulator’s random-number generator in order to make the results reproducible, using an optional keyword argument to process_circuit
.
aer_b = AerBackend()
handle = aer_b.process_circuit(circ, n_shots=10, seed=1)
counts = aer_b.get_result(handle).get_counts()
print(counts)
What happens if we simulate some noise in our imagined device, using the Qiskit Aer noise model?
To investigate this, we will require an import from Qiskit. For more information about noise modelling using Qiskit Aer, see the qiskit_aer documentation documentation.
from qiskit_aer.noise import NoiseModel
my_noise_model = NoiseModel()
readout_error = 0.2
for q in range(2):
my_noise_model.add_readout_error(
[[1 - readout_error, readout_error], [readout_error, 1 - readout_error]], [q]
)
This simple noise model gives a 20% chance that, upon measurement, a qubit that would otherwise have been measured as \(0\) would instead be measured as \(1\), and vice versa. Let’s see what our shot table looks like with this model:
noisy_aer_b = AerBackend(my_noise_model)
noisy_handle = noisy_aer_b.process_circuit(circ, n_shots=10, seed=1, valid_check=False)
noisy_counts = noisy_aer_b.get_result(noisy_handle).get_counts()
print(noisy_counts)
We now have some spurious \(01\) and \(10\) measurements, which could never happen when measuring a Bell state on a noiseless device.
The AerBackend
class can accept any Qiskit noise model.
All backends expose a generic get_result
method which takes a ResultHandle
and returns the respective result in the form of a BackendResult
object. This object may hold measured results in the form of shots or counts, or an exact statevector from simulation. Measured results are stored as OutcomeArray
objects, which compresses measured bit values into 8-bit integers. We can extract the bitwise values using to_readouts
.
Instead of an assumed ILO or DLO convention, we can use this object to request only the Bit
measurements we want, in the order we want. Let’s try reversing the bits of the noisy results.
backend_result = noisy_aer_b.get_result(noisy_handle)
bits = circ.bits
outcomes = backend_result.get_counts([bits[1], bits[0]])
print(outcomes)
BackendResult
objects can be natively serialized to and deserialized from a dictionary. This dictionary can be immediately dumped to json
for storing results.
from pytket.backends.backendresult import BackendResult
result_dict = backend_result.to_dict()
print(result_dict)
print(BackendResult.from_dict(result_dict).get_counts())
The last simulator we will demonstrate is the ProjectQBackend
. ProjectQ offers fast simulation of quantum circuits with built-in support for fast expectation values from operators. The ProjectQBackend
exposes this functionality to take in OpenFermion QubitOperator
instances. These are convertible to and from QubitPauliOperator
instances in Pytket.
Note: ProjectQ can also produce statevectors in the style of AerStateBackend
, and similarly Aer backends can calculate expectation values directly, consult the relevant documentation to see more.
Let’s create an OpenFermion QubitOperator
object and a new circuit:
import openfermion as of
hamiltonian = 0.5 * of.QubitOperator("X0 X2") + 0.3 * of.QubitOperator("Z0")
circ2 = Circuit(3)
circ2.Y(0)
circ2.H(1)
circ2.Rx(0.3, 2)
We convert the OpenFermion Hamiltonian into a pytket QubitPauliOperator:
from pytket.pauli import Pauli, QubitPauliString
from pytket.utils.operators import QubitPauliOperator
pauli_sym = {"I": Pauli.I, "X": Pauli.X, "Y": Pauli.Y, "Z": Pauli.Z}
def qps_from_openfermion(paulis):
"""Convert OpenFermion tensor of Paulis to pytket QubitPauliString."""
qlist = []
plist = []
for q, p in paulis:
qlist.append(Qubit(q))
plist.append(pauli_sym[p])
return QubitPauliString(qlist, plist)
def qpo_from_openfermion(openf_op):
"""Convert OpenFermion QubitOperator to pytket QubitPauliOperator."""
tk_op = dict()
for term, coeff in openf_op.terms.items():
string = qps_from_openfermion(term)
tk_op[string] = coeff
return QubitPauliOperator(tk_op)
hamiltonian_op = qpo_from_openfermion(hamiltonian)
Now we can create a ProjectQBackend
instance and feed it our circuit and QubitOperator
:
projectq_b = ProjectQBackend()
expectation = projectq_b.get_operator_expectation_value(circ2, hamiltonian_op)
print(expectation)
The last leg of this tour includes running a pytket circuit on an actual quantum computer. To do this, you will need an IBM quantum experience account and have your credentials stored on your computer. See https://quantum-computing.ibm.com to make an account and view available devices and their specs.
Physical devices have much stronger constraints on the form of admissible circuits than simulators. They tend to support a minimal gate set, have restricted connectivity between qubits for two-qubit gates, and can have limited support for classical control flow or conditional gates. This is where we can invoke the tket compiler passes to transform our desired circuit into one that is suitable for the backend.
To check our code works correctly, we can use the IBMQEmulatorBackend
to run our code exactly as if it were going to run on a real device, but just execute on a simulator (with a basic noise model adapted from the reported device properties).
Let’s create an IBMQEmulatorBackend
for the ibmq_manila
device and check if our circuit is valid to be run.
ibmq_b_emu = IBMQEmulatorBackend("ibmq_manila")
ibmq_b_emu.valid_circuit(circ)
It looks like we need to compile this circuit to be compatible with the device. To simplify this procedure, we provide minimal compilation passes designed for each backend (the default_compilation_pass()
method) which will guarantee compatibility with the device. These may still fail if the input circuit has too many qubits or unsupported usage of conditional gates. The default passes can have their degree of optimisation by changing an integer parameter (optimisation levels 0, 1, 2), and they can be easily composed with any of tket’s other optimisation passes for better performance.
For convenience, we also wrap up this pass into the get_compiled_circuit
method if you just want to compile a single circuit.
compiled_circ = ibmq_b_emu.get_compiled_circuit(circ)
Let’s create a backend for running on the actual device and check our compiled circuit is valid for this backend too.
ibmq_b = IBMQBackend("ibmq_manila")
ibmq_b.valid_circuit(compiled_circ)
We are now good to run this circuit on the device. After submitting, we can use the handle to check on the status of the job, so that we know when results are ready to be retrieved. The circuit_status
method works for all backends, and returns a CircuitStatus
object. If we just run get_result
straight away, the backend will wait for results to complete, blocking any other code from running.
In this notebook we will use the emulated backend ibmq_b_emu
to illustrate, but the workflow is the same as for the real backend ibmq_b
(except that the latter will typically take much longer because of the size of the queue).
quantum_handle = ibmq_b_emu.process_circuit(compiled_circ, n_shots=10)
print(ibmq_b_emu.circuit_status(quantum_handle))
quantum_counts = ibmq_b_emu.get_result(quantum_handle).get_counts()
print(quantum_counts)
These are from an actual device, so it’s impossible to perfectly predict what the results will be. However, because of the problem of noise, it would be unsurprising to find a few \(01\) or \(10\) results in the table. The circuit is very short, so it should be fairly close to the ideal result.
The devices available through the IBM Q Experience serve jobs one at a time from their respective queues, so a large amount of experiment time can be taken up by waiting for your jobs to reach the front of the queue. pytket
allows circuits to be submitted to any backend in a single batch using the process_circuits
method. For the IBMQBackend
, this will collate the circuits into as few jobs as possible which will all be sent off into the queue for the device. The method returns a ResultHandle
per submitted circuit, in the order of submission.
circuits = []
for i in range(5):
c = Circuit(2)
c.Rx(0.2 * i, 0).CX(0, 1)
c.measure_all()
circuits.append(ibmq_b_emu.get_compiled_circuit(c))
handles = ibmq_b_emu.process_circuits(circuits, n_shots=100)
print(handles)
We can now retrieve the results and process them. As we measured each circuit in the \(Z\)-basis, we can obtain the expectation value for the \(ZZ\) operator immediately from these measurement results. We can calculate this using the expectation_from_counts
utility method in pytket
.
from pytket.utils import expectation_from_counts
for handle in handles:
counts = ibmq_b_emu.get_result(handle).get_counts()
exp_val = expectation_from_counts(counts)
print(exp_val)
A ResultHandle
can be easily stored in its string representaton and later reconstructed using the from_str
method. For example, we could do something like this:
from pytket.backends import ResultHandle
c = Circuit(2).Rx(0.5, 0).CX(0, 1).measure_all()
c = ibmq_b_emu.get_compiled_circuit(c)
handle = ibmq_b_emu.process_circuit(c, n_shots=10)
handlestring = str(handle)
print(handlestring)
# ... later ...
oldhandle = ResultHandle.from_str(handlestring)
print(ibmq_b_emu.get_result(oldhandle).get_counts())
For backends which support persistent handles (e.g. IBMQBackend
, QuantinuumBackend
, BraketBackend
and AQTBackend
) you can even stop your python session and use your result handles in a separate script to retrive results when they are ready, by storing the handle strings. For experiments with long queue times, this enables separate job submission and retrieval. Use Backend.persistent_handles
to check whether a backend supports this feature.
All backends will also cache all results obtained in the current python session, so you can use the ResultHandle
to retrieve the results many times if you need to reuse the results. Over a long experiment, this can consume a large amount of RAM, so we recommend removing results from the cache when you are done with them. A simple way to achieve this is by calling Backend.empty_cache
(e.g. at the end of each loop of a variational algorithm), or removing individual results with Backend.pop_result
.
The backends in pytket
are designed to be as similar to one another as possible. The example above using physical devices can be run entirely on a simulator by swapping out the IBMQBackend
constructor for any other backend supporting shot outputs (e.g. AerBackend
, ProjectQBackend
, ForestBackend
), or passing it the name of a different device. Furthermore, using pytket it is simple to convert between handling shot tables, counts maps and statevectors.
For more information on backends and other pytket
features, read our documentation or see the other pytket examples.