QInfer/python-qinfer

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

Summary

Maintainability
C
1 day
Test Coverage
#!/usr/bin/python
# -*- coding: utf-8 -*-
##
# models.py: Likelihood models for quantum 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
from __future__ import unicode_literals

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

from builtins import range, map

from qinfer import FiniteOutcomeModel

import numpy as np

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

## DESIGN NOTES ##############################################################

"""
Bases are always assumed to have exactly one traceful element— in particular,
the zeroth basis element.
"""

## 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 TomographyModel(FiniteOutcomeModel):
    r"""
    Model for tomographically learning a quantum state using
    two-outcome positive-operator valued measures (POVMs).

    :param TomographyBasis basis: Basis used in representing
        states as model parameter vectors.
    :param bool allow_subnormalized: If `False`, states
        :math:`\rho` are constrained during resampling such
        that :math:`\Tr(\rho) = 1`. 
    """

    def __init__(self, basis, allow_subnormalized=False):
        self._dim = basis.dim
        self._basis = basis
        self._allow_subnormalied = allow_subnormalized
        super(TomographyModel, self).__init__()

    @property
    def dim(self):
        """
        Dimension of the Hilbert space on which density
        operators learned by this model act.

        :type: `int`
        """
        return self._dim
    @property
    def basis(self):
        """
        Basis used in converting between :class:`~qutip.Qobj` and
        model parameter vector representations of states.

        :type: `TomographyBasis`
        """
        return self._basis        

    @property
    def n_modelparams(self):
        return self._dim ** 2

    @property
    def modelparam_names(self):
        return list(map(
            r'\langle\!\langle{} | \rho\rangle\!\rangle'.format,
            self.basis.labels
        ))

    @property
    def is_n_outcomes_constant(self):
        return True

    @property
    def expparams_dtype(self):
        return [
            (str('meas'), float, self._dim ** 2)
        ]

    def n_outcomes(self, expparams):
        return 2

    def are_models_valid(self, modelparams):
        # This is wrong, but is wrong for the sake of speed.
        # As a future improvement, validity checking needs to
        # be enabled as a non-default option.
        return np.ones((modelparams.shape[0],), dtype=bool)

    def canonicalize(self, modelparams):
        """
        Truncates negative eigenvalues and from each
        state represented by a tensor of model parameter
        vectors, and renormalizes as appropriate.

        :param np.ndarray modelparams: Array of shape
            ``(n_states, dim**2)`` containing model parameter
            representations of each of ``n_states`` different
            states.
        :return: The same model parameter tensor with all
            states truncated to be positive operators. If
            :attr:`~TomographyModel.allow_subnormalized` is
            `False`, all states are also renormalized to trace
            one. 
        """
        modelparams = np.apply_along_axis(self.trunc_neg_eigs, 1, modelparams)
        # Renormalizes particles if allow_subnormalized=False.
        if not self._allow_subnormalied:
            modelparams = self.renormalize(modelparams)

        return modelparams

    def trunc_neg_eigs(self, particle):
        """
        Given a state represented as a model parameter vector,
        returns a model parameter vector representing the same
        state with any negative eigenvalues set to zero.

        :param np.ndarray particle: Vector of length ``(dim ** 2, )``
            representing a state.
        :return: The same state with any negative eigenvalues
            set to zero.
        """
        arr = np.tensordot(particle, self._basis.data.conj(), 1)
        w, v = np.linalg.eig(arr)
        if np.all(w >= 0):
            return particle
        else:
            w[w < 0] = 0
            new_arr = np.dot(v * w, v.conj().T)
            new_particle = np.real(np.dot(self._basis.flat(), new_arr.flatten()))
            assert new_particle[0] > 0
            return new_particle

    def renormalize(self, modelparams):
        """
        Renormalizes one or more states represented as model
        parameter vectors, such that each state has trace 1.

        :param np.ndarray modelparams: Array of shape ``(n_states,
            dim ** 2)`` representing one or more states as 
            model parameter vectors.
        :return: The same state, normalized to trace one.
        """
        # The 0th basis element (identity) should have
        # a value 1 / sqrt{dim}, since the trace of that basis
        # element is fixed to be sqrt{dim} by convention.
        norm = modelparams[:, 0] * np.sqrt(self._dim)
        assert not np.sum(norm == 0)
        return modelparams / norm[:, None]

    def likelihood(self, outcomes, modelparams, expparams):
        super(TomographyModel, self).likelihood(outcomes, modelparams, expparams)

        pr1 = np.empty((modelparams.shape[0], expparams.shape[0]))

        pr1[:, :] = np.einsum(
            'ei,mi->me',
            # This should be the Hermitian conjugate, but since
            # expparams['meas'] is real (that is, since the measurement)
            # is Hermitian, then that's not needed here.
            expparams['meas'],
            modelparams
        )
        np.clip(pr1, 0, 1, out=pr1)

        return FiniteOutcomeModel.pr0_to_likelihood_array(outcomes, 1 - pr1)

class DiffusiveTomographyModel(TomographyModel):
    @property
    def n_modelparams(self):
        return super(DiffusiveTomographyModel, self).n_modelparams + 1

    @property
    def expparams_dtype(self):
        return super(DiffusiveTomographyModel, self).expparams_dtype + [
            (str('t'), float)
        ]
    

    @property
    def modelparam_names(self):
        return super(DiffusiveTomographyModel, self).modelparam_names + [r'\epsilon']

    def are_models_valid(self, modelparams):
        return np.logical_and(
            super(DiffusiveTomographyModel, self).are_models_valid(modelparams),
            modelparams[:, -1] > 0
        )

    def canonicalize(self, modelparams):
        return np.concatenate([
            super(DiffusiveTomographyModel, self).canonicalize(modelparams[:, :-1]),
            modelparams[:, -1, None]
        ], axis=1)
    def likelihood(self, outcomes, modelparams, expparams):
        return super(DiffusiveTomographyModel, self).likelihood(outcomes, modelparams[:, :-1], expparams)

    def update_timestep(self, modelparams, expparams):
        # modelparams: [n_m, d² + 1]
        # expparams:   [n_e,]

        # eps:         [n_m, 1] * [n_e] → [n_m, n_e, 1]
        eps = (modelparams[:, -1, None] * np.sqrt(expparams['t']))[:, :, None]
        # steps:       [n_m, n_e, 1] * [n_m, 1, d²]
        steps = eps * np.random.randn(*modelparams[:, None, :].shape)
        steps[:, :, [0, -1]] = 0

        raw_modelparams = modelparams[:, None, :] + steps
        # raw_modelparams[:, :, :-1] = np.apply_along_axis(self.trunc_neg_eigs, 2, raw_modelparams[:, :, :-1])
        for idx_experiment in range(len(expparams)):
            raw_modelparams[:, idx_experiment, :] = self.canonicalize(raw_modelparams[:, idx_experiment, :])
        return raw_modelparams.transpose((0, 2, 1))