Source code for pennylane.transforms.insert_ops

# Copyright 2021 Xanadu Quantum Technologies Inc.

# 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.
"""
Provides transforms for inserting operations into quantum circuits.
"""
from collections.abc import Sequence
from types import FunctionType
from typing import Type, Union

import pennylane as qml
from pennylane.operation import Operation
from pennylane.ops.op_math import Adjoint
from pennylane.tape import QuantumScript, QuantumScriptBatch
from pennylane.transforms import transform
from pennylane.typing import PostprocessingFn

# pylint: disable=too-many-branches


def _check_position(position):
    """Checks the position argument to determine if an operation or list of operations was provided."""
    not_op = False
    req_ops = False
    if isinstance(position, list):
        req_ops = position.copy()
        for operation in req_ops:
            try:
                if Operation not in operation.__bases__:
                    not_op = True
            except AttributeError:
                not_op = True
    elif not isinstance(position, list):
        try:
            if Operation in position.__bases__:
                req_ops = [position]
            else:
                not_op = True
        except AttributeError:
            not_op = True
    return not_op, req_ops


[docs]@transform def insert( tape: QuantumScript, op: Union[callable, Type[Operation]], op_args: Union[tuple, float], position: Union[str, list, Type[Operation]] = "all", before: bool = False, ) -> tuple[QuantumScriptBatch, PostprocessingFn]: """Insert an operation into specified points in an input circuit. Circuits passed through this transform will be updated to have the operation, specified by the ``op`` argument, added according to the positioning specified in the ``position`` argument. Only single qubit operations are permitted to be inserted. The type of ``op`` can be either a single operation or a quantum function acting on a single wire. A quantum function can be used to specify a sequence of operations acting on a single qubit (see the usage details for more information). Args: tape (QNode or QuantumTape or Callable or pennylane.devices.Device): the input circuit to be transformed. op (callable or Type[Operation]): the single-qubit operation, or sequence of operations acting on a single qubit, to be inserted into the circuit op_args (tuple or float): the arguments fed to the operation, either as a tuple or a single float position (str or PennyLane operation or list of operations): Specification of where to add the operation. Should be one of: ``"all"`` to add the operation after all gates (except state preparations); ``"start"`` to add the operation to all wires at the start of the circuit (but after state preparations); ``"end"`` to add the operation to all wires at the end of the circuit; list of operations to add the operation before or after depending on ``before``. before (bool): Whether to add the operation before the given operation(s) in ``position``. Default is ``False`` and the operation is inserted after. Returns: qnode (QNode) or quantum function (Callable) or tuple[List[.QuantumTape], function] or device (pennylane.devices.Device): The transformed circuit as described in :func:`qml.transform <pennylane.transform>`. Raises: ValueError: if a single operation acting on multiple wires is passed to ``op`` ValueError: if the requested ``position`` argument is not ``'start'``, ``'end'`` or ``'all'`` OR PennyLane Operation **Example:** The following QNode can be transformed to add noise to the circuit: .. code-block:: python3 from functools import partial dev = qml.device("default.mixed", wires=2) @partial(qml.transforms.insert, op=qml.AmplitudeDamping, op_args=0.2, position="end") @qml.qnode(dev) def f(w, x, y, z): qml.RX(w, wires=0) qml.RY(x, wires=1) qml.CNOT(wires=[0, 1]) qml.RY(y, wires=0) qml.RX(z, wires=1) return qml.expval(qml.Z(0) @ qml.Z(1)) Executions of this circuit will differ from the noise-free value: >>> f(0.9, 0.4, 0.5, 0.6) tensor(0.754847, requires_grad=True) >>> print(qml.draw(f)(0.9, 0.4, 0.5, 0.6)) 0: ──RX(0.90)─╭●──RY(0.50)──AmplitudeDamping(0.20)─┤ ╭<Z@Z> 1: ──RY(0.40)─╰X──RX(0.60)──AmplitudeDamping(0.20)─┤ ╰<Z@Z> .. details:: :title: Usage Details **Specifying the operation as a quantum function:** Instead of specifying ``op`` as a single :class:`~.Operation`, we can instead define a quantum function. For example: .. code-block:: python3 def op(x, y, wires): qml.RX(x, wires=wires) qml.PhaseShift(y, wires=wires) This operation can be inserted into the following circuit: .. code-block:: python3 dev = qml.device("default.qubit", wires=2) @qml.qnode(dev) @partial(qml.transforms.insert, op=op, op_args=[0.2, 0.3], position="end") def f(w, x, y, z): qml.RX(w, wires=0) qml.RY(x, wires=1) qml.CNOT(wires=[0, 1]) qml.RY(y, wires=0) qml.RX(z, wires=1) return qml.expval(qml.Z(0) @ qml.Z(1)) To check this, let's print out the circuit: >>> print(qml.draw(f)(0.9, 0.4, 0.5, 0.6)) 0: ──RX(0.90)─╭●──RY(0.50)──RX(0.20)──Rϕ(0.30)─┤ ╭<Z@Z> 1: ──RY(0.40)─╰X──RX(0.60)──RX(0.20)──Rϕ(0.30)─┤ ╰<Z@Z> **Transforming tapes:** Consider the following tape: .. code-block:: python3 ops = [ qml.RX(0.9, wires=0), qml.RY(0.4, wires=1), qml.CNOT((0,1)), qml.RY(0.5, wires=0), qml.RX(0.6, wires=1) ] measurements = [qml.expval(qml.Z(0) @ qml.Z(1))] tape = qml.tape.QuantumTape(ops, measurements) We can add the :class:`~.AmplitudeDamping` channel to the end of the circuit using: >>> from pennylane.transforms import insert >>> [noisy_tape], _ = insert(tape, qml.AmplitudeDamping, 0.05, position="end") >>> print(qml.drawer.tape_text(noisy_tape, decimals=2)) 0: ──RX(0.90)─╭●──RY(0.50)──AmplitudeDamping(0.05)─┤ ╭<Z@Z> 1: ──RY(0.40)─╰X──RX(0.60)──AmplitudeDamping(0.05)─┤ ╰<Z@Z> **Transforming devices:** Consider the following QNode: .. code-block:: python3 dev = qml.device("default.mixed", wires=2) def f(w, x, y, z): qml.RX(w, wires=0) qml.RY(x, wires=1) qml.CNOT(wires=[0, 1]) qml.RY(y, wires=0) qml.RX(z, wires=1) return qml.expval(qml.Z(0) @ qml.Z(1)) qnode = qml.QNode(f, dev) Execution of the circuit on ``dev`` will be noise-free: >>> qnode(0.9, 0.4, 0.5, 0.6) tensor(0.86243536, requires_grad=True) However, noise can be easily added to the device: >>> dev_noisy = qml.transforms.insert(dev, qml.AmplitudeDamping, 0.2) >>> qnode_noisy = qml.QNode(f, dev_noisy) >>> qnode_noisy(0.9, 0.4, 0.5, 0.6) tensor(0.72945434, requires_grad=True) """ # decompose templates and their adjoints to fix a bug in the tutorial_error_mitigation demo def stop_at(obj): if not isinstance(obj, qml.operation.Operator): return True if not obj.has_decomposition: return True return not (hasattr(qml.templates, obj.name) or isinstance(obj, Adjoint)) error_type = (qml.operation.DecompositionUndefinedError,) decompose = qml.devices.preprocess.decompose [tape], _ = decompose(tape, stopping_condition=stop_at, name="insert", error=error_type) if not isinstance(op, FunctionType) and op.num_wires != 1: raise ValueError("Only single-qubit operations can be inserted into the circuit") not_op, req_ops = _check_position(position) if position not in ("start", "end", "all") and not_op: raise ValueError( "Position must be either 'start', 'end', or 'all' (default) OR a PennyLane operation or list of operations." ) if not isinstance(op_args, Sequence): op_args = [op_args] new_operations = [] for prep_op in tape.operations[: tape.num_preps]: new_operations.append(prep_op) if position == "start": for w in tape.wires: sub_tape = qml.tape.make_qscript(op)(*op_args, wires=w) new_operations.extend(sub_tape.operations) for circuit_op in tape.operations[tape.num_preps :]: if not before: new_operations.append(circuit_op) if position == "all": for w in circuit_op.wires: sub_tape = qml.tape.make_qscript(op)(*op_args, wires=w) new_operations.extend(sub_tape.operations) if req_ops: for operation in req_ops: if operation == type(circuit_op): for w in circuit_op.wires: sub_tape = qml.tape.make_qscript(op)(*op_args, wires=w) new_operations.extend(sub_tape.operations) if before: new_operations.append(circuit_op) if position == "end": for w in tape.wires: sub_tape = qml.tape.make_qscript(op)(*op_args, wires=w) new_operations.extend(sub_tape.operations) new_tape = type(tape)(new_operations, tape.measurements, shots=tape.shots) def null_postprocessing(results): """A postprocesing function returned by a transform that only converts the batch of results into a result for a single ``QuantumTape``. """ return results[0] return [new_tape], null_postprocessing