from __future__ import annotations
import math
from collections import Counter
from enum import Enum
from itertools import product
from typing import Dict, List, Tuple, Union, cast
import numpy as np
from matplotlib.pyplot import subplots
from numpy.random import Generator
from numpy.typing import NDArray
from pytket import Circuit, Qubit
from pytket.circuit import OpType
from pytket.pauli import Pauli, QubitPauliString, QubitPauliTensor
from scipy.linalg import fractional_matrix_power # type: ignore
from .qermit_pauli import QermitPauli
Direction = Enum("Direction", ["forward", "backward"])
[docs]
class ErrorDistribution:
"""
Model of a Pauli error channel. Contains utilities to analyse and
sample from distributions of errors.
Attributes:
distribution: Dictionary mapping a string of Pauli errors to the
probability that they occur.
rng: Randomness generator.
"""
distribution: Dict[Tuple[Pauli, ...], float]
rng: Generator
def __init__(
self,
distribution: Dict[Tuple[Pauli, ...], float],
rng: Generator = np.random.default_rng(),
):
"""
:param distribution: Dictionary mapping a string of Pauli errors
to the probability that they occur.
:param rng: Randomness generator, defaults to np.random.default_rng()
:raises Exception: Raised if error probabilities sum to greater than 1.
"""
if sum(distribution.values()) > 1:
if not math.isclose(sum(distribution.values()), 1):
raise Exception(
f"Probabilities sum to {sum(distribution.values())}"
+ " but should be less than or equal to 1."
)
# If the given distribution is empty then no
# noise will be acted.
if distribution == {}:
pass
# If it it not empty then we check that the number
# of qubits in each of the errors match.
else:
n_qubits = len(list(distribution.keys())[0])
if not all(len(error) == n_qubits for error in distribution.keys()):
raise Exception("Errors must all act on the same number of qubits.")
self.distribution = distribution
self.rng = rng
@property
def identity_error_rate(self) -> float:
"""The rate at which no error occurs.
:return: Rate at which no error occurs.
Calculated as 1 minus the total error rate of
error in this distribution.
"""
return 1 - sum(self.distribution.values())
[docs]
def to_ptm(self) -> Tuple[NDArray, Dict[Tuple[Pauli, ...], int]]:
"""Convert error distribution to Pauli Transfer Matrix (PTM) form.
:return: PTM of error distribution and Pauli index dictionary.
The Pauli index dictionary maps Pauli errors to their
index in the PTM
"""
# Initialise an empty PTM and index dictionary
# of the appropriate size.
ptm = np.zeros((4**self.n_qubits, 4**self.n_qubits))
pauli_index = {
pauli: index
for index, pauli in enumerate(
product({Pauli.I, Pauli.X, Pauli.Y, Pauli.Z}, repeat=self.n_qubits)
)
}
# For each pauli, calculate the corresponding
# PTM entry as a sum pf error weights multiplied by +/-1
# Depending on commutation relations.
for pauli_tuple, index in pauli_index.items():
pauli = QermitPauli(
QubitPauliTensor(
string=QubitPauliString(
paulis=list(pauli_tuple),
qubits=[Qubit(i) for i in range(self.n_qubits)],
),
coeff=1,
)
)
# Can add the identity error rate.
# This will not come up in the following for loop
# as the error distribution does not save
# the rate at which no errors occur.
ptm[index][index] += self.identity_error_rate
for error, error_rate in self.distribution.items():
error_pauli = QermitPauli(
QubitPauliTensor(
string=QubitPauliString(
paulis=list(error),
qubits=[Qubit(i) for i in range(self.n_qubits)],
),
coeff=1,
)
)
commute_coeff = (
1
if pauli.qubit_pauli_tensor.commutes_with(
error_pauli.qubit_pauli_tensor
)
else -1
)
ptm[index][index] += error_rate * commute_coeff
# Some checks that the form of the PTM is correct.
identity = tuple(Pauli.I for _ in range(self.n_qubits))
if not abs(ptm[pauli_index[identity]][pauli_index[identity]] - 1.0) < 10 ** (
-6
):
raise Exception(
"The identity entry of the PTM is incorrect. "
+ "This is a fault in Qermit. "
+ "Please report this as an issue."
)
if not self == ErrorDistribution.from_ptm(ptm=ptm, pauli_index=pauli_index):
raise Exception(
"From PTM does not match to PTM. "
+ "This is a fault in Qermit. "
+ "Please report this as an issue."
)
return ptm, pauli_index
[docs]
@classmethod
def from_ptm(
cls, ptm: NDArray, pauli_index: Dict[Tuple[Pauli, ...], int]
) -> ErrorDistribution:
"""Convert a Pauli Transfer Matrix (PTM) to an error distribution.
:param ptm: Pauli Transfer Matrix to convert. Should be a 4^n by 4^n matrix
where n is the number of qubits.
:param pauli_index: A dictionary mapping Pauli errors to
their index in the PTM.
:return: The converted error distribution.
"""
if ptm.ndim != 2:
raise Exception(
f"This given matrix is not has dimension {ptm.ndim} "
+ "but should have dimension 2."
)
if ptm.shape[0] != ptm.shape[1]:
raise Exception(
"The dimensions of the given PTM are "
+ f"{ptm.shape[0]} and {ptm.shape[1]} "
+ "but they should match."
)
n_qubit = math.log(ptm.shape[0], 4)
if n_qubit % 1 != 0.0:
raise Exception(
"The given PTM should have a dimension of the form 4^n "
+ "where n is the number of qubits."
)
if not np.array_equal(ptm, np.diag(np.diag(ptm))):
raise Exception("The given PTM is not diagonal as it should be.")
# calculate the error rates by solving simultaneous
# linear equations. In particular the matrix to invert
# is the matrix of commutation values.
commutation_matrix = np.zeros(ptm.shape)
for pauli_one_tuple, index_one in pauli_index.items():
pauli_one = QermitPauli(
QubitPauliTensor(
string=QubitPauliString(
paulis=list(pauli_one_tuple),
qubits=[Qubit(i) for i in range(len(pauli_one_tuple))],
),
coeff=1,
)
)
for pauli_two_tuple, index_two in pauli_index.items():
pauli_two = QermitPauli(
QubitPauliTensor(
string=QubitPauliString(
paulis=list(pauli_two_tuple),
qubits=[Qubit(i) for i in range(len(pauli_two_tuple))],
),
coeff=1,
)
)
commutation_matrix[index_one][index_two] = (
1
if pauli_one.qubit_pauli_tensor.commutes_with(
pauli_two.qubit_pauli_tensor
)
else -1
)
error_rate_list = np.matmul(ptm.diagonal(), np.linalg.inv(commutation_matrix))
distribution = {
error: error_rate_list[index]
for error, index in pauli_index.items()
if (error_rate_list[index] > 10 ** (-6))
and error != tuple(Pauli.I for _ in range(int(n_qubit)))
}
return cls(distribution=distribution)
@property
def n_qubits(self) -> int:
"""The number of qubits this error distribution acts on."""
return len(list(self.distribution.keys())[0])
def __eq__(self, other: object) -> bool:
"""Check equality of two instances of ErrorDistribution by ensuring
that all keys in distribution match, and that the probabilities are
close for each value.
:param other: Instance of ErrorDistribution to be compared against.
:return: True if two instances are equal, false otherwise.
"""
if not isinstance(other, ErrorDistribution):
return False
# Check all pauli error in this distributions are the same.
if set(self.distribution.keys()) != set(other.distribution.keys()):
return False
# Check all probabilities are close.
if not all(
math.isclose(
self.distribution[error], other.distribution[error], abs_tol=0.01
)
for error in self.distribution.keys()
):
return False
# Otherwise they are equal.
return True
def __str__(self) -> str:
"""Generates string representation of error distribution.
:return: String representation of error distribution.
"""
return "".join(f"{key}:{value} \n" for key, value in self.distribution.items())
[docs]
@classmethod
def mixture(cls, distribution_list: List[ErrorDistribution]) -> ErrorDistribution:
"""Generates the distribution corresponding to the mixture of a
list of distributions.
:param distribution_list: List of instances of ErrorDistribution.
:return: Mixture distribution.
"""
return cls(
distribution={
error: sum(
distribution.distribution.get(error, 0)
for distribution in distribution_list
)
/ len(distribution_list)
for error in set(
error
for distribution in distribution_list
for error in distribution.distribution
)
}
)
[docs]
def order(self, reverse: bool = True):
"""Reorders the distribution dictionary based on probabilities.
:param reverse: Order from high to low, defaults to True
"""
self.distribution = {
error: probability
for error, probability in sorted(
self.distribution.items(), key=lambda x: x[1], reverse=reverse
)
}
[docs]
def reset_rng(self, rng: Generator):
"""Reset randomness generator.
:param rng: Randomness generator.
"""
self.rng = rng
[docs]
def to_dict(self) -> List[Dict[str, Union[List[int], float]]]:
"""Produces json serialisable representation of ErrorDistribution.
:return: Json serialisable representation of ErrorDistribution.
"""
return [
{
"op_list": [op.value for op in op_list],
"noise_level": noise_level,
}
for op_list, noise_level in self.distribution.items()
]
[docs]
@classmethod
def from_dict(
cls, distribution_dict: List[Dict[str, Union[List[int], float]]]
) -> ErrorDistribution:
"""Generates ErrorDistribution from json serialisable representation.
:param distribution_dict: List of dictionaries, each of which map
a property of the distribution to its value.
:return: ErrorDistribution created from serialised representation.
"""
return cls(
distribution={
tuple(Pauli(op) for op in cast(List[int], noise_op["op_list"])): cast(
float, noise_op["noise_level"]
)
for noise_op in distribution_dict
}
)
[docs]
def sample(self) -> Union[Tuple[Pauli, ...], None]:
"""Draw sample from distribution.
:return: Either one of the pauli strings in the support of the
distribution, or None. None can be returned if the total proability
of the distribution not 1, and should be interpreted as the
the unspecified support.
"""
return_val = self.rng.uniform(0, 1)
total = 0.0
for error, prob in self.distribution.items():
total += prob
if total >= return_val:
return error
return None
[docs]
def plot(self):
"""
Generates plot of distribution.
"""
fig, ax = subplots()
to_plot = {key: value for key, value in self.distribution.items()}
ax.bar(range(len(to_plot)), list(to_plot.values()), align="center")
ax.set_xticks(
ticks=range(len(to_plot)),
labels=list(
"".join(tuple(op.name for op in op_tuple))
for op_tuple in to_plot.keys()
),
)
return fig
[docs]
def scale(self, scaling_factor: float) -> ErrorDistribution:
"""Scale the error rates of this error distribution.
This is done by converting the error distribution to a PTM,
scaling that matrix appropriately, and converting back to a
new error distribution.
:param scaling_factor: The factor by which the noise should be scaled.
:return: A new error distribution with the noise scaled.
"""
ptm, pauli_index = self.to_ptm()
scaled_ptm = fractional_matrix_power(ptm, scaling_factor)
return ErrorDistribution.from_ptm(ptm=scaled_ptm, pauli_index=pauli_index)
class LogicalErrorDistribution:
"""
Class for managing distributions of logical errors from a noisy
simulation of a circuit. This differs from ErrorDistribution in that
the errors are pauli strings rather than tuples of Paulis. That is to
say that the errors are fixed to qubits.
Attributes:
pauli_error_counter: Counts number of pauli errors in distribution
"""
pauli_error_counter: Counter[QermitPauli]
def __init__(self, pauli_error_counter: Counter[QermitPauli], **kwargs):
"""Initialisation method. Stores pauli error counter.
:param pauli_error_counter: Counter of pauli errors.
:key total: The total number of shots taken when measuring the
errors. By default this will be taken to be the total
number of errors.
"""
self.pauli_error_counter = pauli_error_counter
self.total = kwargs.get("total", sum(self.pauli_error_counter.values()))
@property
def distribution(self) -> Dict[QubitPauliString, float]:
"""Probability distribution equivalent to counts distribution.
:return: Dictionary mapping QubitPauliString to probability that
that error occurs.
"""
distribution: Dict[QubitPauliString, float] = {}
for stab, count in dict(self.pauli_error_counter).items():
# Note that the phase is ignored here
pauli_string = stab.qubit_pauli_tensor.string
distribution[pauli_string] = (
distribution.get(pauli_string, 0) + count / self.total
)
return distribution
def post_select(self, qubit_list: List[Qubit]) -> LogicalErrorDistribution:
"""Post select based on the given qubits. In particular remove the
the given qubits, and the shots with measurable errors on those qubits.
:param qubit_list: List of qubits to be post selected on.
:return: New LogicalErrorDistribution with given qubits removed,
and those shots where there are measurable errors on those
qubits removed.
"""
# the number of error free shots.
total = self.total - sum(self.pauli_error_counter.values())
distribution: Counter[QermitPauli] = Counter()
for pauli_error, count in self.pauli_error_counter.items():
if not pauli_error.is_measureable(qubit_list):
distribution[pauli_error.reduce_qubits(qubit_list)] += count
total += count
return LogicalErrorDistribution(
pauli_error_counter=distribution,
total=total,
)
[docs]
class NoiseModel:
"""
Module for managing and executing a circuit noise model. In particular
error models are assigned to each gate, and logical errors from
circuits can be sampled.
Attributes:
noise_model: Mapping from gates to the error model which corresponds
to that gate.
"""
noise_model: Dict[OpType, ErrorDistribution]
def __init__(self, noise_model: Dict[OpType, ErrorDistribution]):
"""
:param noise_model: Map from gates to their error models.
"""
self.noise_model = noise_model
[docs]
def scale(self, scaling_factor: float) -> NoiseModel:
"""Generate new error model where all error rates have been scaled by
the given scaling factor.
:param scaling_factor: Factor by which to scale the error rates.
:return: New noise model with scaled error rates.
"""
return NoiseModel(
noise_model={
op_type: error_distribution.scale(scaling_factor=scaling_factor)
for op_type, error_distribution in self.noise_model.items()
}
)
[docs]
def reset_rng(self, rng: Generator):
"""Reset randomness generator.
:param rng: Randomness generator to be reset to.
"""
for distribution in self.noise_model.values():
distribution.reset_rng(rng=rng)
[docs]
def plot(self):
"""Generates plot of noise model."""
fig_list = []
for noisy_gate, distribution in self.noise_model.items():
fig = distribution.plot()
ax = fig.axes[0]
ax.set_title(noisy_gate.name)
fig_list.append(fig)
return fig_list
def __eq__(self, other: object) -> bool:
"""Checks equality by checking all gates in the two noise models match,
and that the noise models of each gate match.
:param other: Noise model to be compared against.
:return: True if equivalent, false otherwise.
"""
if not isinstance(other, NoiseModel):
return False
if not (sorted(self.noise_model.keys()) == sorted(other.noise_model.keys())):
return False
if not all(
self.noise_model[op] == other.noise_model[op]
for op in self.noise_model.keys()
):
return False
return True
[docs]
def to_dict(self) -> Dict[str, List[Dict[str, Union[List[int], float]]]]:
"""Json serialisable object representing noise model.
:return: Json serialisable object representing noise model.
"""
return {
op.name: distribution.to_dict()
for op, distribution in self.noise_model.items()
}
[docs]
@classmethod
def from_dict(
cls, noise_model_dict: Dict[str, List[Dict[str, Union[List[int], float]]]]
) -> NoiseModel:
"""Convert JSON serialised version of noise model back to an instance
of NoiseModel.
:param noise_model_dict: JSON serialised version of NoiseModel
:return: Instance of noise model corresponding to JSON serialised
version.
"""
return cls(
noise_model={
OpType.from_name(op): ErrorDistribution.from_dict(error_distribution)
for op, error_distribution in noise_model_dict.items()
}
)
@property
def noisy_gates(self) -> List[OpType]:
"""List of OpTypes with noise.
:return: List of OpTypes with noise.
"""
return list(self.noise_model.keys())
[docs]
def get_error_distribution(self, optype: OpType) -> ErrorDistribution:
"""Recovers error model corresponding to particular OpType.
:param optype: OpType for which noise model should be retrieved.
:return: Error model corresponding to particular OpType.
"""
return self.noise_model[optype]
[docs]
def get_effective_pre_error_distribution(
self,
cliff_circ: Circuit,
n_rand: int = 1000,
**kwargs,
) -> LogicalErrorDistribution:
"""Retrieve the effective noise model of a given circuit. This is to
say, repeatedly generate circuits with coherent noise added at random.
Push all errors to the front of the circuit. Return a counter of the
errors which have been pushed to the front.
:param cliff_circ: Circuit to be simulated. This should be a Clifford
circuit.
:param n_rand: Number of random circuit instances, defaults to 1000
:return: Resulting distribution of errors.
"""
error_counter = self.counter_propagate(
cliff_circ=cliff_circ,
n_counts=n_rand,
direction=Direction.backward,
)
return LogicalErrorDistribution(error_counter, total=n_rand)
[docs]
def counter_propagate(
self, cliff_circ: Circuit, n_counts: int, **kwargs
) -> Counter[QermitPauli]:
"""Generate random noisy instances of the given circuit and propagate
the noise to create a counter of logical errors. Note that
kwargs are passed onto `random_propagate`.
:param cliff_circ: Circuit to be simulated. This should be a Clifford
circuit.
:param n_counts: Number of random instances.
:return: Counter of logical errors.
"""
error_counter: Counter[QermitPauli] = Counter()
# TODO: There is some time wasted here, if for example there is
# no error in back_propagate_random_error. There may be a saving to
# be made here if there errors are sampled before the back
# propagation occurs?
for _ in range(n_counts):
pauli_error = self.random_propagate(cliff_circ, **kwargs)
# Check if the error is the identity.
if pauli_error.qubit_pauli_tensor.string != QubitPauliString():
error_counter.update([pauli_error])
return error_counter
[docs]
def random_propagate(
self,
cliff_circ: Circuit,
direction: Direction = Direction.backward,
) -> QermitPauli:
"""Generate a random noisy instance of the given circuit and
propagate the noise forward or backward to recover the logical error.
:param cliff_circ: Circuit to be simulated. This should be a Clifford
circuit.
:param direction: Direction in which noise should be propagated,
defaults to 'backward'
:raises Exception: Raised if direction is invalid.
:return: Resulting logical error.
"""
pauli_error = QermitPauli(
QubitPauliTensor(
string=QubitPauliString(
map={qubit: Pauli.I for qubit in cliff_circ.qubits}
),
coeff=1,
)
)
# Commands are ordered in reverse or original order depending on which
# way the noise is being pushed.
if direction == Direction.backward:
command_list = list(reversed(cliff_circ.get_commands()))
elif direction == Direction.forward:
command_list = cliff_circ.get_commands()
else:
raise Exception(
"Direction must be Direction.backward or Direction.forward."
)
# For each command in the circuit, add an error as appropriate, and
# push the total error through the command.
for command in command_list:
if command.op.type in [OpType.Measure, OpType.Barrier]:
continue
if direction == Direction.forward:
# Apply gate to total error.
pauli_error.apply_gate(
op=command.op,
qubits=cast(List[Qubit], command.args),
)
# Add noise operation if appropriate.
if command.op.type in self.noisy_gates:
error_distribution = self.get_error_distribution(optype=command.op.type)
error = error_distribution.sample()
if error is not None:
for pauli, qubit in zip(error, command.args):
if direction == Direction.backward:
pauli_error.pre_apply_pauli(
pauli=pauli, qubit=cast(Qubit, qubit)
)
elif direction == Direction.forward:
pauli_error.post_apply_pauli(
pauli=pauli, qubit=cast(Qubit, qubit)
)
else:
raise Exception(
"Direction must be Direction.backward or Direction.forward. "
)
if direction == Direction.backward:
# Note that here we wish to pull the pauli back through the gate,
# which has the same effect on the pauli as pushing through the
# dagger.
pauli_error.apply_gate(
op=command.op.dagger,
qubits=cast(List[Qubit], command.args),
)
return pauli_error