From 64461631f239ba988b171f6ad46762712a5dcb9c Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Thu, 29 Aug 2024 19:42:27 +0200 Subject: [PATCH 1/9] feat: implemented raw for M gate --- src/qibo/gates/abstract.py | 15 ++++++++++++++- src/qibo/gates/measurements.py | 24 +++++++++++++++++------- src/qibo/measurements.py | 9 +++++++-- 3 files changed, 38 insertions(+), 10 deletions(-) diff --git a/src/qibo/gates/abstract.py b/src/qibo/gates/abstract.py index b829392995..4ed3eadb32 100644 --- a/src/qibo/gates/abstract.py +++ b/src/qibo/gates/abstract.py @@ -14,7 +14,18 @@ "_target_qubits", "_control_qubits", ] -REQUIRED_FIELDS_INIT_KWARGS = ["theta", "phi", "lam", "phi0", "phi1"] +REQUIRED_FIELDS_INIT_KWARGS = [ + "theta", + "phi", + "lam", + "phi0", + "phi1", + "register_name", + "collapse", + "basis", + "p0", + "p1", +] class Gate: @@ -107,6 +118,8 @@ def from_dict(raw: dict): raise ValueError(f"Unknown gate {raw['_class']}") gate = cls(*raw["init_args"], **raw["init_kwargs"]) + if raw["_class"] == "M" and raw["samples"] is not None: + gate.result.register_samples(self.backend.cast(raw["samples"], int)) try: return gate.controlled_by(*raw["_control_qubits"]) except RuntimeError as e: diff --git a/src/qibo/gates/measurements.py b/src/qibo/gates/measurements.py index 34e1ca4f1a..bf7af782c7 100644 --- a/src/qibo/gates/measurements.py +++ b/src/qibo/gates/measurements.py @@ -70,6 +70,7 @@ def __init__( self.init_kwargs = { "register_name": register_name, "collapse": collapse, + "basis": self.basis_gates, "p0": p0, "p1": p1, } @@ -102,6 +103,22 @@ def __init__( if gate is not None: self.basis.append(gate) + @property + def raw(self) -> dict: + """Serialize to dictionary. + + The values used in the serialization should be compatible with a + JSON dump (or any other one supporting a minimal set of scalar + types). Though the specific implementation is up to the specific + gate. + """ + encoded_simple = super().raw + encoded_simple.pop("_control_qubits") + basis = [g.__name__ for g in encoded_simple["init_kwargs"]["basis"]] + encoded_simple["init_kwargs"]["basis"] = basis + encoded_simple.update(self.result.raw) + return encoded_simple + @staticmethod def _get_bitflip_tuple( qubits: Tuple[int, ...], probs: "ProbsType" @@ -204,13 +221,6 @@ def apply_clifford(self, backend, state, nqubits): self.result.add_shot_from_sample(sample[0]) return state - def to_json(self): - """Serializes the measurement gate to json.""" - encoding = json.loads(super().to_json()) - encoding.pop("_control_qubits") - encoding.update({"basis": [g.__name__ for g in self.basis_gates]}) - return json.dumps(encoding) - @classmethod def load(cls, payload): """Constructs a measurement gate starting from a json serialized diff --git a/src/qibo/measurements.py b/src/qibo/measurements.py index 7fae1fd695..1f6de9f893 100644 --- a/src/qibo/measurements.py +++ b/src/qibo/measurements.py @@ -96,6 +96,11 @@ def __repr__(self): nshots = self.nshots return f"MeasurementResult(qubits={qubits}, nshots={nshots})" + @property + def raw(self) -> dict: + samples = self._samples.tolist() if self.has_samples() else self._samples + return {"samples": samples} + def add_shot(self, probs): qubits = sorted(self.measurement_gate.target_qubits) shot = self.backend.sample_shots(probs, 1) @@ -117,12 +122,12 @@ def add_shot_from_sample(self, sample): def has_samples(self): return self._samples is not None - def register_samples(self, samples, backend=None): + def register_samples(self, samples): """Register samples array to the ``MeasurementResult`` object.""" self._samples = samples self.nshots = len(samples) - def register_frequencies(self, frequencies, backend=None): + def register_frequencies(self, frequencies): """Register frequencies to the ``MeasurementResult`` object.""" self._frequencies = frequencies self.nshots = sum(frequencies.values()) From fe42b66dea9737121aba15f5866b13b0ea4d0d41 Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Fri, 30 Aug 2024 10:19:34 +0200 Subject: [PATCH 2/9] fix: moving cast inside register_samples --- src/qibo/gates/abstract.py | 2 +- src/qibo/measurements.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/qibo/gates/abstract.py b/src/qibo/gates/abstract.py index 4ed3eadb32..e00733b72f 100644 --- a/src/qibo/gates/abstract.py +++ b/src/qibo/gates/abstract.py @@ -119,7 +119,7 @@ def from_dict(raw: dict): gate = cls(*raw["init_args"], **raw["init_kwargs"]) if raw["_class"] == "M" and raw["samples"] is not None: - gate.result.register_samples(self.backend.cast(raw["samples"], int)) + gate.result.register_samples(raw["samples"]) try: return gate.controlled_by(*raw["_control_qubits"]) except RuntimeError as e: diff --git a/src/qibo/measurements.py b/src/qibo/measurements.py index 1f6de9f893..4c602ef17a 100644 --- a/src/qibo/measurements.py +++ b/src/qibo/measurements.py @@ -124,7 +124,7 @@ def has_samples(self): def register_samples(self, samples): """Register samples array to the ``MeasurementResult`` object.""" - self._samples = samples + self._samples = self.backend.cast(samples, int) self.nshots = len(samples) def register_frequencies(self, frequencies): From 5679978699b3b1783af6e799069391d7d8fe6b08 Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Fri, 30 Aug 2024 14:25:05 +0200 Subject: [PATCH 3/9] feat: accepting string as basis + replaced the load function --- src/qibo/backends/clifford.py | 2 +- src/qibo/gates/measurements.py | 51 +++++++++++++------------------ src/qibo/measurements.py | 6 ++-- src/qibo/quantum_info/clifford.py | 2 +- src/qibo/result.py | 6 ++-- 5 files changed, 30 insertions(+), 37 deletions(-) diff --git a/src/qibo/backends/clifford.py b/src/qibo/backends/clifford.py index 5fd2d1cd6d..1ba071cd40 100644 --- a/src/qibo/backends/clifford.py +++ b/src/qibo/backends/clifford.py @@ -259,7 +259,7 @@ def execute_circuit_repeated(self, circuit, nshots: int = 1000, initial_state=No samples = self.np.vstack(samples) for meas in circuit.measurements: - meas.result.register_samples(samples[:, meas.target_qubits], self) + meas.result.register_samples(samples[:, meas.target_qubits]) result = Clifford( self.zero_state(circuit.nqubits), diff --git a/src/qibo/gates/measurements.py b/src/qibo/gates/measurements.py index bf7af782c7..4b63a58cb9 100644 --- a/src/qibo/gates/measurements.py +++ b/src/qibo/gates/measurements.py @@ -1,5 +1,5 @@ import json -from typing import Dict, Optional, Tuple +from typing import Dict, Optional, Tuple, Union from qibo import gates from qibo.config import raise_error @@ -23,11 +23,13 @@ class M(Gate): performed. Can be used only for single shot measurements. If ``True`` the collapsed state vector is returned. If ``False`` the measurement result is returned. - basis (:class:`qibo.gates.Gate`, list): Basis to measure. - Can be a qibo gate or a callable that accepts a qubit, - for example: ``lambda q: gates.RX(q, 0.2)`` - or a list of these, if a different basis will be used for each - measurement qubit. + basis (:class:`qibo.gates.Gate`, str, list): Basis to measure. + Can be either: + - a qibo gate + - the string representing the gate + - a callable that accepts a qubit, for example: ``lambda q: gates.RX(q, 0.2)`` + - a list of the above, if a different basis will be used for each + measurement qubit. Default is Z. p0 (dict): Optional bitflip probability map. Can be: A dictionary that maps each measured qubit to the probability @@ -46,7 +48,7 @@ def __init__( *q, register_name: Optional[str] = None, collapse: bool = False, - basis: Gate = Z, + basis: Union[Gate, str] = Z, p0: Optional["ProbsType"] = None, p1: Optional["ProbsType"] = None, ): @@ -61,16 +63,24 @@ def __init__( # relevant for experiments only self.pulses = None # saving basis for __repr__ ans save to file + to_gate = lambda x: getattr(gates, x) if isinstance(x, str) else x if not isinstance(basis, list): - self.basis_gates = len(q) * [basis] + self.basis_gates = len(q) * [to_gate(basis)] + basis = len(self.target_qubits) * [basis] + elif len(basis) != len(self.target_qubits): + raise_error( + ValueError, + f"Given basis list has length {len(basis)} while " + f"we are measuring {len(self.target_qubits)} qubits.", + ) else: - self.basis_gates = basis + self.basis_gates = [to_gate(g) for g in basis] self.init_args = q self.init_kwargs = { "register_name": register_name, "collapse": collapse, - "basis": self.basis_gates, + "basis": [g.__name__ for g in self.basis_gates], "p0": p0, "p1": p1, } @@ -89,16 +99,8 @@ def __init__( # list of gates that will be added to the circuit before the # measurement, in order to rotate to the given basis - if not isinstance(basis, list): - basis = len(self.target_qubits) * [basis] - elif len(basis) != len(self.target_qubits): - raise_error( - ValueError, - f"Given basis list has length {len(basis)} while " - f"we are measuring {len(self.target_qubits)} qubits.", - ) self.basis = [] - for qubit, basis_cls in zip(self.target_qubits, basis): + for qubit, basis_cls in zip(self.target_qubits, self.basis_gates): gate = basis_cls(qubit).basis_rotation() if gate is not None: self.basis.append(gate) @@ -113,9 +115,6 @@ def raw(self) -> dict: gate. """ encoded_simple = super().raw - encoded_simple.pop("_control_qubits") - basis = [g.__name__ for g in encoded_simple["init_kwargs"]["basis"]] - encoded_simple["init_kwargs"]["basis"] = basis encoded_simple.update(self.result.raw) return encoded_simple @@ -226,13 +225,7 @@ def load(cls, payload): """Constructs a measurement gate starting from a json serialized one.""" args = json.loads(payload) - # drop general serialization data, unused in this specialized loader - for key in ("name", "init_args", "_class"): - args.pop(key) - qubits = args.pop("_target_qubits") - args["basis"] = [getattr(gates, g) for g in args["basis"]] - args.update(args.pop("init_kwargs")) - return cls(*qubits, **args) + return cls.from_dict(args) # Overload on_qubits to copy also gate.result, controlled by can be removed for measurements def on_qubits(self, qubit_map) -> "Gate": diff --git a/src/qibo/measurements.py b/src/qibo/measurements.py index 4c602ef17a..2409f8a134 100644 --- a/src/qibo/measurements.py +++ b/src/qibo/measurements.py @@ -80,8 +80,10 @@ class MeasurementResult: """ def __init__(self, gate, nshots=0, backend=None): + from qibo.backends import _check_backend + self.measurement_gate = gate - self.backend = backend + self.backend = _check_backend(backend) self.nshots = nshots self.circuit = None @@ -124,7 +126,7 @@ def has_samples(self): def register_samples(self, samples): """Register samples array to the ``MeasurementResult`` object.""" - self._samples = self.backend.cast(samples, int) + self._samples = self.backend.cast(samples, self.backend.np.int64) self.nshots = len(samples) def register_frequencies(self, frequencies): diff --git a/src/qibo/quantum_info/clifford.py b/src/qibo/quantum_info/clifford.py index 11d1b4e9ec..000aa6bcd0 100644 --- a/src/qibo/quantum_info/clifford.py +++ b/src/qibo/quantum_info/clifford.py @@ -269,7 +269,7 @@ def samples(self, binary: bool = True, registers: bool = False): self._samples = self._backend.cast(samples, dtype="int32") for gate in self.measurements: rqubits = tuple(qubit_map.get(q) for q in gate.target_qubits) - gate.result.register_samples(self._samples[:, rqubits], self._backend) + gate.result.register_samples(self._samples[:, rqubits]) if registers: return { diff --git a/src/qibo/result.py b/src/qibo/result.py index f9b101ac9e..b2fa8a95fd 100644 --- a/src/qibo/result.py +++ b/src/qibo/result.py @@ -244,7 +244,7 @@ def frequencies(self, binary: bool = True, registers: bool = False): if int(bitstring[qubit_map.get(q)]): idx += 2 ** (len(rqubits) - i - 1) rfreqs[idx] += freq - gate.result.register_frequencies(rfreqs, self.backend) + gate.result.register_frequencies(rfreqs) else: self._frequencies = self.backend.calculate_frequencies( self.samples(binary=False) @@ -356,9 +356,7 @@ def samples(self, binary: bool = True, registers: bool = False): self._samples = samples for gate in self.measurements: rqubits = tuple(qubit_map.get(q) for q in gate.target_qubits) - gate.result.register_samples( - self._samples[:, rqubits], self.backend - ) + gate.result.register_samples(self._samples[:, rqubits]) if registers: return { From 78f65d4019e0f0a9c6b0166bdebc5778a9ca36da Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Fri, 30 Aug 2024 18:12:59 +0200 Subject: [PATCH 4/9] feat: made nshots a property of measurement result --- src/qibo/measurements.py | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/src/qibo/measurements.py b/src/qibo/measurements.py index 2409f8a134..88133cb20a 100644 --- a/src/qibo/measurements.py +++ b/src/qibo/measurements.py @@ -84,7 +84,6 @@ def __init__(self, gate, nshots=0, backend=None): self.measurement_gate = gate self.backend = _check_backend(backend) - self.nshots = nshots self.circuit = None self._samples = None @@ -103,6 +102,13 @@ def raw(self) -> dict: samples = self._samples.tolist() if self.has_samples() else self._samples return {"samples": samples} + @property + def nshots(self) -> int: + if self.has_samples(): + return len(self._samples) + elif self._frequencies is not None: + return sum(self._frequencies.values()) + def add_shot(self, probs): qubits = sorted(self.measurement_gate.target_qubits) shot = self.backend.sample_shots(probs, 1) @@ -111,7 +117,6 @@ def add_shot(self, probs): self._samples.append(bshot[0]) else: self._samples = [bshot[0]] - self.nshots += 1 return shot def add_shot_from_sample(self, sample): @@ -119,7 +124,6 @@ def add_shot_from_sample(self, sample): self._samples.append(sample) else: self._samples = [sample] - self.nshots += 1 def has_samples(self): return self._samples is not None @@ -127,12 +131,10 @@ def has_samples(self): def register_samples(self, samples): """Register samples array to the ``MeasurementResult`` object.""" self._samples = self.backend.cast(samples, self.backend.np.int64) - self.nshots = len(samples) def register_frequencies(self, frequencies): """Register frequencies to the ``MeasurementResult`` object.""" self._frequencies = frequencies - self.nshots = sum(frequencies.values()) def reset(self): """Remove all registered samples and frequencies.""" From 21da5d0eb23ea1d6b0d2b358dfe4523a56bf3a8e Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Fri, 30 Aug 2024 18:35:05 +0200 Subject: [PATCH 5/9] feat: removed backend from the measurement result attributes --- src/qibo/gates/measurements.py | 4 ++-- src/qibo/measurements.py | 31 +++++++++++++++++++------------ tests/test_states.py | 6 +++--- 3 files changed, 24 insertions(+), 17 deletions(-) diff --git a/src/qibo/gates/measurements.py b/src/qibo/gates/measurements.py index 4b63a58cb9..6c911c343a 100644 --- a/src/qibo/gates/measurements.py +++ b/src/qibo/gates/measurements.py @@ -194,7 +194,7 @@ def apply(self, backend, state, nqubits): qubits = sorted(self.target_qubits) # measure and get result probs = backend.calculate_probabilities(state, qubits, nqubits) - shot = self.result.add_shot(probs) + shot = self.result.add_shot(probs, backend=backend) # collapse state return backend.collapse_state(state, qubits, shot, nqubits) @@ -206,7 +206,7 @@ def apply_density_matrix(self, backend, state, nqubits): qubits = sorted(self.target_qubits) # measure and get result probs = backend.calculate_probabilities_density_matrix(state, qubits, nqubits) - shot = self.result.add_shot(probs) + shot = self.result.add_shot(probs, backend=backend) # collapse state return backend.collapse_density_matrix(state, qubits, shot, nqubits) diff --git a/src/qibo/measurements.py b/src/qibo/measurements.py index 88133cb20a..3c8f6d2231 100644 --- a/src/qibo/measurements.py +++ b/src/qibo/measurements.py @@ -7,6 +7,13 @@ from qibo.config import raise_error +def _check_backend(backend): + """This is only needed due to the circular import with qibo.backends.""" + from qibo.backends import _check_backend + + return _check_backend(backend) + + def frequencies_to_binary(frequencies, nqubits): return collections.Counter( {"{:b}".format(k).zfill(nqubits): v for k, v in frequencies.items()} @@ -79,11 +86,8 @@ class MeasurementResult: to use for calculations. """ - def __init__(self, gate, nshots=0, backend=None): - from qibo.backends import _check_backend - + def __init__(self, gate): self.measurement_gate = gate - self.backend = _check_backend(backend) self.circuit = None self._samples = None @@ -109,10 +113,11 @@ def nshots(self) -> int: elif self._frequencies is not None: return sum(self._frequencies.values()) - def add_shot(self, probs): + def add_shot(self, probs, backend=None): + backend = _check_backend(backend) qubits = sorted(self.measurement_gate.target_qubits) - shot = self.backend.sample_shots(probs, 1) - bshot = self.backend.samples_to_binary(shot, len(qubits)) + shot = backend.sample_shots(probs, 1) + bshot = backend.samples_to_binary(shot, len(qubits)) if self._samples: self._samples.append(bshot[0]) else: @@ -130,7 +135,7 @@ def has_samples(self): def register_samples(self, samples): """Register samples array to the ``MeasurementResult`` object.""" - self._samples = self.backend.cast(samples, self.backend.np.int64) + self._samples = samples def register_frequencies(self, frequencies): """Register frequencies to the ``MeasurementResult`` object.""" @@ -153,7 +158,7 @@ def symbols(self): return self._symbols - def samples(self, binary=True, registers=False): + def samples(self, binary=True, registers=False, backend=None): """Returns raw measurement samples. Args: @@ -168,6 +173,7 @@ def samples(self, binary=True, registers=False): samples are returned in decimal form as a tensor of shape `(nshots,)`. """ + backend = _check_backend(backend) if self._samples is None: if self.circuit is None: raise_error( @@ -181,9 +187,9 @@ def samples(self, binary=True, registers=False): return self._samples qubits = self.measurement_gate.target_qubits - return self.backend.samples_to_decimal(self._samples, len(qubits)) + return backend.samples_to_decimal(self._samples, len(qubits)) - def frequencies(self, binary=True, registers=False): + def frequencies(self, binary=True, registers=False, backend=None): """Returns the frequencies of measured samples. Args: @@ -201,8 +207,9 @@ def frequencies(self, binary=True, registers=False): If `binary` is `False` the keys of the `Counter` are integers. """ + backend = _check_backend(backend) if self._frequencies is None: - self._frequencies = self.backend.calculate_frequencies( + self._frequencies = backend.calculate_frequencies( self.samples(binary=False) ) if binary: diff --git a/tests/test_states.py b/tests/test_states.py index fc0cd512d2..9ce556485b 100644 --- a/tests/test_states.py +++ b/tests/test_states.py @@ -8,12 +8,12 @@ def test_measurement_result_repr(): - result = MeasurementResult(gates.M(0), nshots=10) - assert str(result) == "MeasurementResult(qubits=(0,), nshots=10)" + result = MeasurementResult(gates.M(0)) + assert str(result) == "MeasurementResult(qubits=(0,), nshots=None)" def test_measurement_result_error(): - result = MeasurementResult(gates.M(0), nshots=10) + result = MeasurementResult(gates.M(0)) with pytest.raises(RuntimeError): samples = result.samples() From 66332782ba85274aea8b63fbbec6507ab39562a5 Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Fri, 30 Aug 2024 18:52:40 +0200 Subject: [PATCH 6/9] feat: changed raw of M --- src/qibo/gates/abstract.py | 2 +- src/qibo/gates/measurements.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/qibo/gates/abstract.py b/src/qibo/gates/abstract.py index e00733b72f..d8bfc9dd21 100644 --- a/src/qibo/gates/abstract.py +++ b/src/qibo/gates/abstract.py @@ -119,7 +119,7 @@ def from_dict(raw: dict): gate = cls(*raw["init_args"], **raw["init_kwargs"]) if raw["_class"] == "M" and raw["samples"] is not None: - gate.result.register_samples(raw["samples"]) + gate.result.register_samples(raw["measurement_result"]["samples"]) try: return gate.controlled_by(*raw["_control_qubits"]) except RuntimeError as e: diff --git a/src/qibo/gates/measurements.py b/src/qibo/gates/measurements.py index 6c911c343a..7e1559e9d9 100644 --- a/src/qibo/gates/measurements.py +++ b/src/qibo/gates/measurements.py @@ -115,7 +115,7 @@ def raw(self) -> dict: gate. """ encoded_simple = super().raw - encoded_simple.update(self.result.raw) + encoded_simple.update({"measurement_result": self.result.raw}) return encoded_simple @staticmethod From aa574459557bcc78c98a6be6e41595fd99ee6795 Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Fri, 30 Aug 2024 19:11:45 +0200 Subject: [PATCH 7/9] fix: tiny fix --- src/qibo/gates/abstract.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/qibo/gates/abstract.py b/src/qibo/gates/abstract.py index d8bfc9dd21..250c309934 100644 --- a/src/qibo/gates/abstract.py +++ b/src/qibo/gates/abstract.py @@ -118,7 +118,7 @@ def from_dict(raw: dict): raise ValueError(f"Unknown gate {raw['_class']}") gate = cls(*raw["init_args"], **raw["init_kwargs"]) - if raw["_class"] == "M" and raw["samples"] is not None: + if raw["_class"] == "M" and raw["measurement_result"]["samples"] is not None: gate.result.register_samples(raw["measurement_result"]["samples"]) try: return gate.controlled_by(*raw["_control_qubits"]) From 8df290787613df62998a5dc8a4a94eb3a4fd7194 Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Sat, 31 Aug 2024 11:26:57 +0200 Subject: [PATCH 8/9] test: added test for nshots --- tests/test_measurements.py | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/tests/test_measurements.py b/tests/test_measurements.py index 050d2c1970..55fb4b665f 100644 --- a/tests/test_measurements.py +++ b/tests/test_measurements.py @@ -6,6 +6,7 @@ import pytest from qibo import gates, models +from qibo.measurements import MeasurementResult def assert_result( @@ -473,3 +474,23 @@ def test_measurementsymbol_pickling(backend): assert symbol.index == new_symbol.index assert symbol.name == new_symbol.name backend.assert_allclose(symbol.result.samples(), new_symbol.result.samples()) + + +def test_measurementresult_nshots(backend): + gate = gates.M(*range(3)) + result = MeasurementResult(gate) + # nshots starting from samples + nshots = 10 + samples = backend.cast( + [[i % 2, i % 2, i % 2] for i in range(nshots)], backend.np.int64 + ) + result.register_samples(samples) + assert result.nshots == nshots + # nshots starting from frequencies + result = MeasurementResult(gate) + states, counts = np.unique(samples, axis=0, return_counts=True) + to_str = lambda x: [str(item) for item in x] + states = ["".join(to_str(s)) for s in states.tolist()] + freq = dict(zip(states, counts.tolist())) + result.register_frequencies(freq) + assert result.nshots == nshots From 7864590905c676d043f5ede50567529ba5ddc267 Mon Sep 17 00:00:00 2001 From: BrunoLiegiBastonLiegi Date: Sat, 31 Aug 2024 11:55:01 +0200 Subject: [PATCH 9/9] test: added test for M serialization --- tests/test_measurements.py | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/tests/test_measurements.py b/tests/test_measurements.py index 55fb4b665f..979cb843b6 100644 --- a/tests/test_measurements.py +++ b/tests/test_measurements.py @@ -1,5 +1,6 @@ """Test circuit result measurements and measurement gate and as part of circuit.""" +import json import pickle import numpy as np @@ -494,3 +495,21 @@ def test_measurementresult_nshots(backend): freq = dict(zip(states, counts.tolist())) result.register_frequencies(freq) assert result.nshots == nshots + + +def test_measurement_serialization(backend): + kwargs = { + "register_name": "test", + "collapse": False, + "basis": ["Z", "X", "Y"], + "p0": 0.1, + "p1": 0.2, + } + gate = gates.M(*range(3), **kwargs) + samples = backend.cast(np.random.randint(2, size=(100, 3)), backend.np.int64) + gate.result.register_samples(samples) + dump = gate.to_json() + load = gates.M.from_dict(json.loads(dump)) + for k, v in kwargs.items(): + assert load.init_kwargs[k] == v + backend.assert_allclose(samples, load.result.samples())