QInfer/python-qinfer

View on GitHub
src/qinfer/tomography/expdesign.py

Summary

Maintainability
D
1 day
Test Coverage
#!/usr/bin/python
# -*- coding: utf-8 -*-
##
# expdesign.py: Experiment design heuristics specialized for state and
#     process tomography.
##
# © 2017, Chris Ferrie (csferrie@gmail.com) and
#         Christopher Granade (cgranade@cgranade.com).
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
#     1. Redistributions of source code must retain the above copyright
#        notice, this list of conditions and the following disclaimer.
#
#     2. Redistributions in binary form must reproduce the above copyright
#        notice, this list of conditions and the following disclaimer in the
#        documentation and/or other materials provided with the distribution.
#
#     3. Neither the name of the copyright holder nor the names of its
#        contributors may be used to endorse or promote products derived from
#        this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
##

# TODO: docstrings!
# TODO: unit tests!

## DOCSTRING #################################################################

"""
"""

## FEATURES ##################################################################

from __future__ import absolute_import
from __future__ import division

## IMPORTS ###################################################################

from builtins import range

from qinfer import Heuristic
from qinfer.tomography.bases import pauli_basis

import numpy as np

from abc import abstractmethod

# Since the rest of QInfer does not require QuTiP,
# we need to import it in a way that we don't propagate exceptions if QuTiP
# is missing or is too early a version.
from qinfer.utils import get_qutip_module
qt = get_qutip_module('3.2')

## EXPORTS ###################################################################
# TODO

## FUNCTIONS #################################################################

# TODO: document, contribute to QuTiP?
def heisenberg_weyl_operators(d=2):
    w = np.exp(2 * np.pi * 1j / d)
    X = qt.Qobj([
        qt.basis(d, (idx + 1) % d).data.todense().view(np.ndarray)[:, 0] for idx in range(d)
    ])
    Z = qt.Qobj(np.diag(w ** np.arange(d)))
    
    return [X**i * Z**j for i in range(d) for j in range(d)]

## CLASSES ####################################################################

class StateTomographyHeuristic(Heuristic):
    # TODO: docstring
    # for when I write that docstring, basis is provided so that this can
    # be used to define measurement parts of the process tomography experiments.
    def __init__(self, updater, basis=None, other_fields=None):     
        self._up = updater
        self._other_fields = {} if other_fields is None else other_fields
        self._dim = updater.model.base_model.dim
        self._basis = updater.model.base_model.basis if basis is None else basis
        
    def __call__(self):
        expparams = np.zeros((1,), dtype=self._up.model.expparams_dtype)
        expparams['meas'][0, :] = self._basis.state_to_modelparams(
            self._next_measurement()
        )

        for field, value in self._other_fields.items():
                expparams[field] = value
        
        return expparams
    
    @abstractmethod
    def _next_measurement(self):
        pass
    

class RandomStabilizerStateHeuristic(StateTomographyHeuristic):
    """
    Randomly chooses rank-1 projectors onto a stabilizer state.
    """
    
    # This heuristic isn't terribly *efficient*, but the heuristic
    # gets called far less than the likelihood itself, so that shouldn't
    # dominate.
    
    def __init__(self, updater, basis=None, other_fields=None):
        super(RandomStabilizerStateHeuristic, self).__init__(
            updater, basis, other_fields
        )
    
        self._hw_group = heisenberg_weyl_operators(self._basis.dim)
    
    def _next_measurement(self):
        return qt.ket2dm(
            np.random.choice(np.random.choice(
                self._hw_group
            ).eigenstates()[1])
        )


class RandomPauliHeuristic(StateTomographyHeuristic):
    """
    Randomly chooses a Pauli measurement. Defined for qubits only.
    """
    # This heuristic is likewise a bit silly, as it draws a
    # random Pauli then decomposes it into what is almost
    # certainly a Pauli basis. We do this, however, in the interest
    # of generality. Someone could have used a different basis
    # to define their tomography model, after all.
    def __init__(self, updater, basis=None, other_fields=None):
        super(RandomPauliHeuristic, self).__init__(
            updater, basis, other_fields
        )
        
        nq = int(np.log2(self._dim))
        if 2**nq != self._dim:
            raise ValueError("RandomPauliHeuristic is defined only for qubits.")
        
        self._nq = nq
        self._pauli_basis = pauli_basis(nq)
        
    def _next_measurement(self):
        # Remember, the basis elements are normalized to 1 / sqrt(d).
        return (
            self._pauli_basis[0] +
            self._pauli_basis[1 + np.random.choice(len(self._pauli_basis) - 1)]
        ) * np.sqrt(self._dim) / 2


class ProcessTomographyHeuristic(Heuristic):
    def __init__(self, updater, basis, other_fields=None):     
        self._up = updater
        self._other_fields = {} if other_fields is None else other_fields
        self._dim = updater.model.base_model.dim
        self._basis = basis
        self._channel_basis = updater.model.base_model.basis
        
    def __call__(self):
        expparams = np.zeros((1,), dtype=self._up.model.expparams_dtype)
        expparams['meas'][0, :] = self._channel_basis.state_to_modelparams(
            # By PBT (Apx A), we need to multiply the state by a factor of
            # D for this to work. Here, however, dim means the channel,
            # so we need a square root.
            np.sqrt(self._dim) * qt.tensor(*self._next_prepmeas())
        )

        for field, value in self._other_fields.items():
                expparams[field] = value
        
        return expparams
    
    @abstractmethod
    def _next_prepmeas(self):
        """
        Implementing subclasses should return a tuple ``(preparation, measurement)``
        of two Qobjs, normalized in the 1- and ∞-norms, respectively.
        """
        pass
    
class ProductHeuristic(ProcessTomographyHeuristic):
    """
    Takes two heuristic classes, one for preparations
    and one for measurements, then returns a sample from
    each. The preparation heuristic is assumed to return only
    trace-1 Hermitian operators.
    """
    
    def __init__(self,
            updater, basis, prep_heuristic_class,
            meas_heuristic_class, other_fields=None
    ):
        super(ProductHeuristic, self).__init__(updater, basis, other_fields)
        self._ph = prep_heuristic_class(updater, basis=basis)
        self._mh = meas_heuristic_class(updater, basis=basis)
        
    def _next_prepmeas(self):
        prep = self._ph._next_measurement().unit()
        meas = self._mh._next_measurement()
        
        return prep, meas
    
class BestOfKMetaheuristic(Heuristic):
    """
    Draws :math:`k` different state or tomography
    measurements, then selects the one that has the largest
    expected value under the action of the covariance superoperator
    for the current posterior.
    """
    def __init__(self, updater, base_heuristic, k=3, other_fields=None):
        self._up = updater
        self._base_heuristic = base_heuristic
        self._k = k
        # TODO: consolidate this other_fields madness into
        #       a base heuristic class, not even just a tomography one!!
        self._other_fields = {} if other_fields is None else other_fields
        
    def __call__(self):
        expparams = np.array([
            self._base_heuristic()[0] for _ in range(self._k)
        ], dtype=self._up.model.expparams_dtype)
        meas = expparams['meas']
        cov_expectations = np.einsum('ei,ij,ej->e',
            meas, self._up.est_covariance_mtx(), meas
        )
        
        expparams = expparams[np.argmax(cov_expectations), None]

        for field, value in self._other_fields.items():
                expparams[field] = value
                
        return expparams