Source code for pyquil.api._wavefunction_simulator

##############################################################################
# Copyright 2018 Rigetti Computing
#
#    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.
##############################################################################
from typing import Optional, Union, cast

import numpy as np
from qcs_sdk import QCSClient, qvm
from qcs_sdk.qvm import QVMOptions

from pyquil.api import MemoryMap
from pyquil.api._qvm import (
    validate_noise_probabilities,
)
from pyquil.gates import MOVE
from pyquil.paulis import PauliSum, PauliTerm
from pyquil.quil import Program
from pyquil.quilatom import MemoryReference
from pyquil.wavefunction import Wavefunction


[docs] class WavefunctionSimulator: def __init__( self, *, gate_noise: Optional[tuple[float, float, float]] = None, measurement_noise: Optional[tuple[float, float, float]] = None, random_seed: Optional[int] = None, timeout: float = 10.0, client_configuration: Optional[QCSClient] = None, ) -> None: """Return a simulator that propagates a wavefunction representation of a quantum state. :param gate_noise: A tuple of three numbers [Px, Py, Pz] indicating the probability of an X, Y, or Z gate getting applied to each qubit after a gate application or reset. :param measurement_noise: A tuple of three numbers [Px, Py, Pz] indicating the probability of an X, Y, or Z gate getting applied before a measurement. :param random_seed: A seed for the simulator's random number generators. Either None (for an automatically generated seed) or a non-negative integer. :param timeout: Time limit for requests, in seconds. :param client_configuration: Optional client configuration. If none is provided, a default one will be loaded. """ validate_noise_probabilities(gate_noise) validate_noise_probabilities(measurement_noise) self.gate_noise = gate_noise self.measurement_noise = measurement_noise if random_seed is None: self.random_seed = None elif isinstance(random_seed, int) and random_seed >= 0: self.random_seed = random_seed else: raise TypeError("random_seed should be None or a non-negative int") self.timeout = timeout self._client = client_configuration or QCSClient.load() self._qvm_client = qvm.QVMClient.new_http(self._client.qvm_url)
[docs] def wavefunction(self, quil_program: Program, memory_map: Optional[MemoryMap] = None) -> Wavefunction: """Simulate a Quil program and return the wavefunction. .. note:: If your program contains measurements or noisy gates, this method may not do what you want. If the execution of ``quil_program`` is **non-deterministic** then the final wavefunction only represents a stochastically generated sample and the wavefunctions returned by *different* ``wavefunction`` calls *will generally be different*. :param quil_program: A Quil program. :param memory_map: An assignment of classical registers to values, representing an initial state for the QAM's classical memory. This is expected to be of type dict[str, list[Union[int, float]]], where the keys are memory region names and the values are arrays of initialization data. :return: A Wavefunction object representing the state of the QVM. """ if memory_map is not None: quil_program = self.augment_program_with_memory_values(quil_program, memory_map) request = qvm.api.WavefunctionRequest( quil_program.out(calibrations=False), self.measurement_noise, self.gate_noise, self.random_seed, ) wavefunction = bytes( qvm.api.get_wavefunction(request, self._qvm_client, options=QVMOptions(timeout_seconds=self.timeout)) ) return Wavefunction.from_bit_packed_string(wavefunction)
[docs] def expectation( self, prep_prog: Program, pauli_terms: Union[PauliSum, list[PauliTerm]], memory_map: Optional[dict[str, list[Union[int, float]]]] = None, ) -> Union[float, np.ndarray]: """Calculate the expectation value of Pauli operators given a state prepared by prep_program. If ``pauli_terms`` is a ``PauliSum`` then the returned value is a single ``float``, otherwise the returned value is an array of values, one for each ``PauliTerm`` in the list. .. note:: If your program contains measurements or noisy gates, this method may not do what you want. If the execution of ``quil_program`` is **non-deterministic** then the final wavefunction from which the expectation value is calculated only represents a stochastically generated sample and the wavefunctions returned by *different* ``wavefunction`` calls *will generally be different*. :param prep_prog: A program that prepares the state on which we measure the expectation. :param pauli_terms: A Pauli representation of a quantum operator. :param memory_map: An assignment of classical registers to values, representing an initial state for the QAM's classical memory. This is expected to be of type dict[str, list[Union[int, float]]], where the keys are memory region names and the values are arrays of initialization data. :return: Either a float or array floats depending on ``pauli_terms``. """ is_pauli_sum = False if isinstance(pauli_terms, PauliSum): progs, coeffs = pauli_terms.get_programs() is_pauli_sum = True else: coeffs = np.array([pt.coefficient for pt in pauli_terms]) progs = [pt.program for pt in pauli_terms] if memory_map is not None: prep_prog = self.augment_program_with_memory_values(prep_prog, memory_map) request = qvm.api.ExpectationRequest(prep_prog.out(), [prog.out() for prog in progs]) expectations = qvm.api.measure_expectation( request, self._qvm_client, options=QVMOptions(timeout_seconds=self.timeout) ) bare_results = np.asarray(expectations) results = coeffs * bare_results if is_pauli_sum: return np.sum(results) # type: ignore return results # type: ignore
[docs] def run_and_measure( self, quil_program: Program, qubits: Optional[list[int]] = None, trials: int = 1, memory_map: Optional[MemoryMap] = None, ) -> np.ndarray: """Run a Quil program once to determine the final wavefunction, and measure multiple times. Alternatively, consider using ``wavefunction`` and calling ``sample_bitstrings`` on the resulting object. For a large wavefunction and a low-medium number of trials, use this function. On the other hand, if you're sampling a small system many times you might want to use ``Wavefunction.sample_bitstrings``. .. note:: If your program contains measurements or noisy gates, this method may not do what you want. If the execution of ``quil_program`` is **non-deterministic** then the final wavefunction from which the returned bitstrings are sampled itself only represents a stochastically generated sample and the outcomes sampled from *different* ``run_and_measure`` calls *generally sample different bitstring distributions*. :param quil_program: The program to run and measure :param qubits: An optional list of qubits to measure. The order of this list is respected in the returned bitstrings. If not provided, all qubits used in the program will be measured and returned in their sorted order. :param int trials: Number of times to sample from the prepared wavefunction. :param memory_map: An assignment of classical registers to values, representing an initial state for the QAM's classical memory. This is expected to be of type dict[str, list[Union[int, float]]], where the keys are memory region names and the values are arrays of initialization data. :return: An array of measurement results (0 or 1) of shape (trials, len(qubits)) """ if qubits is None: qubits = sorted(cast(set[int], quil_program.get_qubits(indices=True))) if memory_map is not None: quil_program = self.augment_program_with_memory_values(quil_program, memory_map) request = qvm.api.MultishotMeasureRequest( quil_program.out(), trials, qubits, ) measured_qubits = qvm.api.run_and_measure( request, client=self._qvm_client, options=QVMOptions(timeout_seconds=self.timeout) ) return np.asarray(measured_qubits)
[docs] @staticmethod def augment_program_with_memory_values( quil_program: Program, memory_map: MemoryMap, ) -> Program: p = Program() # we stupidly allowed memory_map to be of type dict[MemoryReference, Any], whereas qc.run # takes a memory initialization argument of type dict[str, list[Union[int, float]]. until # we are in a position to remove this, we support both styles of input. if len(memory_map.keys()) == 0: return quil_program elif isinstance(list(memory_map.keys())[0], str): for name, arr in memory_map.items(): for index, value in enumerate(arr): p += MOVE(MemoryReference(name, offset=index), value) else: raise TypeError("Bad memory_map type; expected dict[str, list[Union[int, float]]].") p += quil_program return p