qiskit/test/python/quantum_info/operators/channel/test_linearops.py

263 lines
10 KiB
Python

# This code is part of Qiskit.
#
# (C) Copyright IBM 2017, 2019.
#
# This code is licensed under the Apache License, Version 2.0. You may
# obtain a copy of this license in the LICENSE.txt file in the root directory
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
#
# Any modifications or derivative works of this code must retain this
# copyright notice, and modified files need to carry a notice indicating
# that they have been altered from the originals.
"""Equivalence tests for quantum channel methods."""
import unittest
import numpy as np
from qiskit.quantum_info.operators.operator import Operator
from qiskit.quantum_info.operators.channel.choi import Choi
from qiskit.quantum_info.operators.channel.superop import SuperOp
from qiskit.quantum_info.operators.channel.kraus import Kraus
from qiskit.quantum_info.operators.channel.stinespring import Stinespring
from qiskit.quantum_info.operators.channel.ptm import PTM
from qiskit.quantum_info.operators.channel.chi import Chi
from .channel_test_case import ChannelTestCase
class TestEquivalence(ChannelTestCase):
"""Tests for channel equivalence for linear operations.
This tests that addition, subtraction, multiplication and negation
work for all representations as if they were performed in the SuperOp
representation.s equivalent to performing the same
operations in other representations.
"""
def _compare_add_to_superop(self, rep, dim, samples, unitary=False):
"""Test channel addition is equivalent to SuperOp"""
for _ in range(samples):
if unitary:
mat1 = self.rand_matrix(dim, dim)
mat2 = self.rand_matrix(dim, dim)
sop1 = np.kron(np.conj(mat1), mat1)
sop2 = np.kron(np.conj(mat2), mat2)
else:
sop1 = self.rand_matrix(dim * dim, dim * dim)
sop2 = self.rand_matrix(dim * dim, dim * dim)
target = SuperOp(sop1 + sop2)
channel = SuperOp(rep(SuperOp(sop1))._add(rep(SuperOp(sop2))))
self.assertEqual(channel, target)
def _compare_subtract_to_superop(self, rep, dim, samples, unitary=False):
"""Test channel subtraction is equivalent to SuperOp"""
for _ in range(samples):
if unitary:
mat1 = self.rand_matrix(dim, dim)
mat2 = self.rand_matrix(dim, dim)
sop1 = np.kron(np.conj(mat1), mat1)
sop2 = np.kron(np.conj(mat2), mat2)
else:
sop1 = self.rand_matrix(dim * dim, dim * dim)
sop2 = self.rand_matrix(dim * dim, dim * dim)
target = SuperOp(sop1 - sop2)
channel = SuperOp(rep(SuperOp(sop1))._add(rep(-SuperOp(sop2))))
self.assertEqual(channel, target)
def _compare_subtract_operator_to_superop(self, rep, dim, samples, unitary=False):
"""Test channel addition is equivalent to SuperOp"""
for _ in range(samples):
if unitary:
mat1 = self.rand_matrix(dim, dim)
sop1 = np.kron(np.conj(mat1), mat1)
else:
sop1 = self.rand_matrix(dim * dim, dim * dim)
mat2 = self.rand_matrix(dim, dim)
target = SuperOp(sop1) - SuperOp(Operator(mat2))
channel = SuperOp(rep(SuperOp(sop1)) - Operator(mat2))
self.assertEqual(channel, target)
def _compare_multiply_to_superop(self, rep, dim, samples, unitary=False):
"""Test channel scalar multiplication is equivalent to SuperOp"""
for _ in range(samples):
if unitary:
mat1 = self.rand_matrix(dim, dim)
sop1 = np.kron(np.conj(mat1), mat1)
else:
sop1 = self.rand_matrix(dim * dim, dim * dim)
val = 0.7
target = SuperOp(val * sop1)
channel = SuperOp(rep(SuperOp(sop1))._multiply(val))
self.assertEqual(channel, target)
def _compare_negate_to_superop(self, rep, dim, samples, unitary=False):
"""Test negative channel is equivalent to SuperOp"""
for _ in range(samples):
if unitary:
mat1 = self.rand_matrix(dim, dim)
sop1 = np.kron(np.conj(mat1), mat1)
else:
sop1 = self.rand_matrix(dim * dim, dim * dim)
target = SuperOp(-1 * sop1)
channel = SuperOp(-rep(SuperOp(sop1)))
self.assertEqual(channel, target)
def _check_add_other_reps(self, chan):
"""Check addition works for other representations"""
current_rep = chan.__class__
other_reps = [Operator, Choi, SuperOp, Kraus, Stinespring, Chi, PTM]
for rep in other_reps:
self.assertEqual(current_rep, chan._add(rep(chan)).__class__)
def _check_subtract_other_reps(self, chan):
"""Check subtraction works for other representations"""
current_rep = chan.__class__
other_reps = [Operator, Choi, SuperOp, Kraus, Stinespring, Chi, PTM]
for rep in other_reps:
self.assertEqual(current_rep, chan._add(-rep(chan)).__class__)
def test_choi_add(self):
"""Test addition of Choi matrices is correct."""
self._compare_add_to_superop(Choi, 4, 10)
def test_kraus_add(self):
"""Test addition of Kraus matrices is correct."""
self._compare_add_to_superop(Kraus, 4, 10)
def test_stinespring_add(self):
"""Test addition of Stinespring matrices is correct."""
self._compare_add_to_superop(Stinespring, 4, 10)
def test_chi_add(self):
"""Test addition of Chi matrices is correct."""
self._compare_add_to_superop(Chi, 4, 10)
def test_ptm_add(self):
"""Test addition of PTM matrices is correct."""
self._compare_add_to_superop(PTM, 4, 10)
def test_choi_subtract(self):
"""Test subtraction of Choi matrices is correct."""
self._compare_subtract_to_superop(Choi, 4, 10)
def test_kraus_subtract(self):
"""Test subtraction of Kraus matrices is correct."""
self._compare_subtract_to_superop(Kraus, 4, 10)
def test_stinespring_subtract(self):
"""Test subtraction of Stinespring matrices is correct."""
self._compare_subtract_to_superop(Stinespring, 4, 10)
def test_chi_subtract(self):
"""Test subtraction of Chi matrices is correct."""
self._compare_subtract_to_superop(Chi, 4, 10)
def test_ptm_subtract(self):
"""Test subtraction of PTM matrices is correct."""
self._compare_subtract_to_superop(PTM, 4, 10)
def test_choi_subtract_operator(self):
"""Test subtraction of Operator from Choi is correct."""
self._compare_subtract_operator_to_superop(Choi, 4, 10)
def test_kraus_subtract_operator(self):
"""Test subtraction of Operator from Kraus is correct."""
self._compare_subtract_operator_to_superop(Kraus, 4, 10)
def test_stinespring_subtract_operator(self):
"""Test subtraction of Operator from Stinespring is correct."""
self._compare_subtract_operator_to_superop(Stinespring, 4, 10)
def test_chi_subtract_operator(self):
"""Test subtraction of Operator from Chi is correct."""
self._compare_subtract_operator_to_superop(Chi, 4, 10)
def test_ptm_subtract_operator(self):
"""Test subtraction of Operator from PTM is correct."""
self._compare_subtract_operator_to_superop(PTM, 4, 10)
def test_choi_multiply(self):
"""Test scalar multiplication of Choi matrices is correct."""
self._compare_multiply_to_superop(Choi, 4, 10)
def test_kraus_multiply(self):
"""Test scalar multiplication of Kraus matrices is correct."""
self._compare_multiply_to_superop(Kraus, 4, 10)
def test_stinespring_multiply(self):
"""Test scalar multiplication of Stinespring matrices is correct."""
self._compare_multiply_to_superop(Stinespring, 4, 10)
def test_chi_multiply(self):
"""Test scalar multiplication of Chi matrices is correct."""
self._compare_multiply_to_superop(Chi, 4, 10)
def test_ptm_multiply(self):
"""Test scalar multiplication of PTM matrices is correct."""
self._compare_multiply_to_superop(PTM, 4, 10)
def test_choi_add_other_rep(self):
"""Test addition of Choi matrices is correct."""
chan = Choi(self.choiI)
self._check_add_other_reps(chan)
def test_superop_add_other_rep(self):
"""Test addition of SuperOp matrices is correct."""
chan = SuperOp(self.sopI)
self._check_add_other_reps(chan)
def test_kraus_add_other_rep(self):
"""Test addition of Kraus matrices is correct."""
chan = Kraus(self.UI)
self._check_add_other_reps(chan)
def test_stinespring_add_other_rep(self):
"""Test addition of Stinespring matrices is correct."""
chan = Stinespring(self.UI)
self._check_add_other_reps(chan)
def test_chi_add_other_rep(self):
"""Test addition of Chi matrices is correct."""
chan = Chi(self.chiI)
self._check_add_other_reps(chan)
def test_ptm_add_other_rep(self):
"""Test addition of PTM matrices is correct."""
chan = PTM(self.ptmI)
self._check_add_other_reps(chan)
def test_choi_subtract_other_rep(self):
"""Test subtraction of Choi matrices is correct."""
chan = Choi(self.choiI)
self._check_subtract_other_reps(chan)
def test_superop_subtract_other_rep(self):
"""Test subtraction of SuperOp matrices is correct."""
chan = SuperOp(self.sopI)
self._check_subtract_other_reps(chan)
def test_kraus_subtract_other_rep(self):
"""Test subtraction of Kraus matrices is correct."""
chan = Kraus(self.UI)
self._check_subtract_other_reps(chan)
def test_stinespring_subtract_other_rep(self):
"""Test subtraction of Stinespring matrices is correct."""
chan = Stinespring(self.UI)
self._check_subtract_other_reps(chan)
def test_chi_subtract_other_rep(self):
"""Test subtraction of Chi matrices is correct."""
chan = Chi(self.chiI)
self._check_subtract_other_reps(chan)
def test_ptm_subtract_other_rep(self):
"""Test subtraction of PTM matrices is correct."""
chan = PTM(self.ptmI)
self._check_subtract_other_reps(chan)
if __name__ == "__main__":
unittest.main()