Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[WIP] Redesign capabilities dictionary #781

Merged
merged 53 commits into from
Sep 14, 2020
Merged
Show file tree
Hide file tree
Changes from 23 commits
Commits
Show all changes
53 commits
Select commit Hold shift + click to select a range
c365ff5
prototype
mariaschuld Aug 13, 2020
d0b2435
backup
mariaschuld Aug 17, 2020
fae2c75
backup
mariaschuld Aug 17, 2020
b118948
finish prototype
mariaschuld Aug 17, 2020
8f8c05a
backup
mariaschuld Aug 17, 2020
48915e5
Merge branch 'master' into capabilities_redesign
mariaschuld Aug 24, 2020
511710e
make values boolean
mariaschuld Aug 24, 2020
ffdf851
backup
mariaschuld Aug 25, 2020
36e2300
update dictionary keys
mariaschuld Aug 25, 2020
d48d568
backup
mariaschuld Aug 26, 2020
db44760
fix bugs in tensor tests, set tol higher for sampled tests
mariaschuld Aug 27, 2020
a433006
add noisy capability
mariaschuld Aug 27, 2020
0251a3e
improve developer information
mariaschuld Aug 27, 2020
1bf5d3b
remove sampled from tensor device, and add analytic attribute
mariaschuld Aug 27, 2020
1a2b9cd
fix all tests for base devices
mariaschuld Aug 27, 2020
37050ee
polish docstring some more
mariaschuld Aug 27, 2020
a3c8166
reverse changing of tolerance
mariaschuld Aug 27, 2020
12e7efd
fix general tests
mariaschuld Aug 27, 2020
909a775
fix some more tests
mariaschuld Aug 27, 2020
ec2ec7a
Merge branch 'master' into capabilities_redesign
mariaschuld Aug 27, 2020
45498d9
black
mariaschuld Aug 27, 2020
455bb89
fix code factor
mariaschuld Aug 27, 2020
e5b0b18
black again
mariaschuld Aug 27, 2020
0a6c28c
add capabilities tests
mariaschuld Aug 27, 2020
9a3376a
update black version and rerun black
mariaschuld Aug 27, 2020
106c719
Update pennylane/_device.py
mariaschuld Aug 27, 2020
2f0ba49
Update pennylane/_device.py
mariaschuld Aug 27, 2020
a7159b6
Update pennylane/devices/tests/test_gates.py
mariaschuld Aug 27, 2020
b168253
rewrite core to check for old and new keys
mariaschuld Aug 27, 2020
a3acad7
Merge branch 'capabilities_redesign' of github.com:XanaduAI/pennylane…
mariaschuld Aug 27, 2020
aba06fe
add test for reversible diff
mariaschuld Aug 27, 2020
f22737a
Merge branch 'master' into capabilities_redesign
mariaschuld Aug 27, 2020
09a983a
fix isinstance
mariaschuld Aug 27, 2020
227d4d4
Merge branch 'capabilities_redesign' of github.com:XanaduAI/pennylane…
mariaschuld Aug 27, 2020
9b277a9
black needs to be applied again
mariaschuld Aug 27, 2020
0b78eeb
Merge branch 'master' of github.com:XanaduAI/pennylane into capabilit…
mariaschuld Sep 5, 2020
5d01545
polish
mariaschuld Sep 5, 2020
48c4bae
fix test bug
mariaschuld Sep 5, 2020
34c4bfe
Merge branch 'master' into capabilities_redesign
antalszava Sep 8, 2020
b9b5a60
Update doc/development/plugins.rst
mariaschuld Sep 9, 2020
0cc5b10
remove unused capabilities
mariaschuld Sep 9, 2020
21b9726
remove accidentally commited files
mariaschuld Sep 9, 2020
3c7b9eb
fix test
mariaschuld Sep 9, 2020
f9ee3d0
polish docstring
mariaschuld Sep 9, 2020
1d8cef9
remove default keys from Device, make better capability device tests,…
mariaschuld Sep 10, 2020
fc86431
black
mariaschuld Sep 10, 2020
6cecd28
fix codefactor issues
mariaschuld Sep 10, 2020
5b0925a
black
mariaschuld Sep 10, 2020
35413ec
Merge branch 'master' into capabilities_redesign
mariaschuld Sep 14, 2020
1b4e97c
Merge branch 'master' into capabilities_redesign
mariaschuld Sep 14, 2020
b5aa8b2
implement Tom's code review
mariaschuld Sep 14, 2020
73be770
fix test
mariaschuld Sep 14, 2020
d658563
fix test again
mariaschuld Sep 14, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
26 changes: 23 additions & 3 deletions doc/development/plugins.rst
Original file line number Diff line number Diff line change
Expand Up @@ -101,15 +101,35 @@ as well as potential further capabilities, by providing the following class attr
conversion between the two conventions takes place automatically
by the plugin device.

* :attr:`.Device._capabilities`: a dictionary containing information about the capabilities of
the device. Keys currently supported include:
* :func:`.Device.capabilities`: A class method which returns the dictionary of capabilities of a device. A
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

image

Just wondering if it would make sense to update the subsection title to make it more capability focused, e.g., "Defining capabilities"

new device should override this method to retrieve the parent classes' capabilities dictionary, make a copy
and update and/or add capabilities before returning the copy.

Examples of capabilities are:

* ``'model'`` (*str*): either ``'qubit'`` or ``'CV'``.

* ``'inverse_operations'`` (*bool*): ``True`` if the device supports
* ``'supports_exact'`` (*bool*): ``True`` if the device is a simulator that returns analytic results.

* ``'supports_sampled'`` (*bool*): ``True`` if outputs are estimates computed from samples of measurement outcomes.

* ``'supports_inverse_operations'`` (*bool*): ``True`` if the device supports
applying the inverse of operations. Operations which should be inverted
have the property ``operation.inverse == True``.

* ``'supports_tensor_observables'`` (*bool*): ``True`` if the device supports observables composed from tensor
products such as ``PauliZ(wires=0) @ PauliZ(wires=1)``.

* ``'executes_in_remote'`` (*bool*): ``True`` if the device runs computations on remote servers or devices.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For my own understanding: what would be a use case for executes_in_remote?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For example when devices are advertised on our homepage, and their capabilities automatically searched through...

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Got it. Not quite sure if it's best to add an entry in the code itself for documentation purposes (just because it gives the impression that specific logic would follow), although I agree that this seems to be a brief enough solution.


* ``'takes_fixed_number_of_wires'`` (*bool*): ``True`` if the device's number of wires are fixed.

* ``'runs_noisy_computations'`` (*bool*): ``True`` if the device's results are stochastic due to
physical or simulated noise.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if the device's results are stochastic due to physical

Would that mean that hardware devices would have this attribute be set to True?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, and some simulators...


Capabilities are queried by PennyLane core to make decisions on how to best run computations,
and by external apps build on top of the device ecosystem.
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved

Adding arguments to your device
--------------------------------

Expand Down
41 changes: 34 additions & 7 deletions pennylane/_device.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,22 @@ class Device(abc.ABC):
"""

# pylint: disable=too-many-public-methods
_capabilities = {} #: dict[str->*]: plugin capabilities
_capabilities = {
"model": None,
"passthru_interface": None,
"supports_reversible_diff": False,
"supports_exact": False,
"supports_sampled": False,
"supports_inverse_operations": False,
"supports_tensor_observables": False,
"provides_jacobian": False,
"executes_in_remote": False,
"takes_fixed_number_of_wires": False,
"performs_noisy_computation": False,
}
"""The capabilities dictionary stores the properties of a device. Devices can add their
own custom properties and overwrite existing ones by overriding the ``capabilities()`` method."""
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved

_circuits = {} #: dict[str->Circuit]: circuit templates associated with this API class
_asarray = staticmethod(np.asarray)

Expand Down Expand Up @@ -213,9 +228,19 @@ def map_wires(self, wires):

@classmethod
def capabilities(cls):
"""Get the other capabilities of the plugin.
"""Get the capabilities of this device class.

Inheriting classes that change or add capabilities must override this method, for example via

Measurements, batching etc.
.. code-block::
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(
new_capability=...,
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
)
return capabilities

Returns:
dict[str->*]: results
Expand Down Expand Up @@ -413,7 +438,9 @@ def supports_operation(self, operation):
if operation.endswith(Operation.string_for_inverse):
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
return operation[
: -len(Operation.string_for_inverse)
] in self.operations and self.capabilities().get("inverse_operations", False)
] in self.operations and self.capabilities().get(
"supports_inverse_operations", False
)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Woah, black left this looking really weird

Copy link
Member

@josh146 josh146 Aug 27, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Slight tangent, but I've always thought the way we do inverse operations is a bit weird. Really, there are only two gates that need the plugin to implement inverse; the non-Hermitian, non-parametrized S and T gates. Every single other gate is either its own inverse (PauliZ, Hadamard), or the inverse is a function of the arguments (RX(-x), QubitUnitary(U.conj().T)).

I've always wondered if we could treat inverses like decompositions:

  • PL-Core asks the device to implement an inverse of a gate.

  • If the device doesn't support the inverse natively, PL fallsback to the above parameter transform, or if unknown, simply raises an error.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Having said that, I don't think it's priority to fix, but would be a nice to have.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice idea indeed!


return operation in self.operations

Expand Down Expand Up @@ -468,7 +495,7 @@ def check_validity(self, queue, observables):
operation_name = o.name

if o.inverse:
if not self.capabilities().get("inverse_operations", False):
if not self.capabilities().get("supports_inverse_operations", False):
raise DeviceError(
"The inverse of gates are not supported on device {}".format(
self.short_name
Expand All @@ -484,7 +511,7 @@ def check_validity(self, queue, observables):
for o in observables:

if isinstance(o, Tensor):
if not self.capabilities().get("tensor_observables", False):
if not self.capabilities().get("supports_tensor_observables", False):
raise DeviceError(
"Tensor observables not supported on device {}".format(self.short_name)
)
Expand All @@ -501,7 +528,7 @@ def check_validity(self, queue, observables):
observable_name = o.name

if issubclass(o.__class__, Operation) and o.inverse:
if not self.capabilities().get("inverse_operations", False):
if not self.capabilities().get("supports_inverse_operations", False):
raise DeviceError(
"The inverse of gates are not supported on device {}".format(
self.short_name
Expand Down
32 changes: 7 additions & 25 deletions pennylane/_qubit_device.py
Original file line number Diff line number Diff line change
Expand Up @@ -111,32 +111,14 @@ def __init__(self, wires=1, shots=1000, analytic=True):

@classmethod
def capabilities(cls):
"""Get the capabilities of the plugin.

Capabilities include:

* ``"model"`` (*str*): either ``"qubit"`` or ``"CV"``.

* ``"inverse_operations"`` (*bool*): ``True`` if the device supports
applying the inverse of operations. Operations which should be inverted
have ``operation.inverse == True``.

* ``"tensor_observables" (*bool*): ``True`` if the device supports
expectation values/variance/samples of :class:`~.Tensor` observables.

The qubit device class has built-in support for tensor observables. As a
result, devices that inherit from this class automatically
have the following items in their capabilities
dictionary:

* ``"model": "qubit"``
* ``"tensor_observables": True``

Returns:
dict[str->*]: results
"""
capabilities = cls._capabilities
capabilities.update(model="qubit", tensor_observables=True)
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
capabilities = super().capabilities().copy()
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
capabilities.update(
model="qubit",
supports_exact=True,
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
supports_sampled=True,
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
supports_tensor_observables=True,
)
return capabilities

def reset(self):
Expand Down
14 changes: 13 additions & 1 deletion pennylane/beta/devices/default_tensor.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@
r"""
Experimental simulator plugin based on tensor network contractions
"""
# pylint: disable=too-many-instance-attributes
import warnings
from itertools import product

Expand Down Expand Up @@ -101,7 +102,6 @@ class DefaultTensor(Device):
pennylane_requires = "0.12"
version = "0.12.0"
author = "Xanadu Inc."
_capabilities = {"model": "qubit", "tensor_observables": True}

_operation_map = {
"BasisState": None,
Expand Down Expand Up @@ -164,6 +164,18 @@ def __init__(self, wires, shots=1000, representation="exact", contraction_method
self._rep = representation
self._contraction_method = contraction_method
self.reset()
self.analytic = False
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(
model="qubit",
supports_exact=True,
supports_tensor_observables=True,
supports_inverse_operations=True,
)
return capabilities
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved

def reset(self):
"""Reset the device."""
Expand Down
14 changes: 8 additions & 6 deletions pennylane/beta/devices/default_tensor_tf.py
Original file line number Diff line number Diff line change
Expand Up @@ -115,12 +115,6 @@ class DefaultTensorTF(DefaultTensor):
# pylint: disable=too-many-instance-attributes
name = "PennyLane TensorNetwork (TensorFlow) simulator plugin"
short_name = "default.tensor.tf"
_capabilities = {
"model": "qubit",
"tensor_observables": True,
"provides_jacobian": True,
"passthru_interface": "tf",
}

_operation_map = copy.copy(DefaultTensor._operation_map)
_operation_map.update(
Expand Down Expand Up @@ -174,6 +168,14 @@ def __init__(self, wires, shots=1000, representation="exact", contraction_method
contraction_method=contraction_method,
)

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(
provides_jacobian=True, passthru_interface="tf",
)
return capabilities

def reset(self):
self.res = None
self.variables = []
Expand Down
3 changes: 3 additions & 0 deletions pennylane/devices/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,5 +30,8 @@
autograd_ops
tests
"""
# DefaultQubitTF and DefaultQubitAutograd not imported here since this
# would lead to an automatic import of tensorflow and autograd, which are
# not PennyLane core dependencies
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
from .default_qubit import DefaultQubit
from .default_gaussian import DefaultGaussian
10 changes: 8 additions & 2 deletions pennylane/devices/default_gaussian.py
Original file line number Diff line number Diff line change
Expand Up @@ -656,8 +656,6 @@ class DefaultGaussian(Device):
version = "0.12.0"
author = "Xanadu Inc."

_capabilities = {"model": "cv"}
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved

_operation_map = {
"Beamsplitter": beamsplitter,
"ControlledAddition": controlled_addition,
Expand Down Expand Up @@ -695,6 +693,14 @@ def __init__(self, wires, *, shots=1000, hbar=2, analytic=True):

self.reset()

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(
model="cv", supports_exact=True, supports_sampling=True,
)
return capabilities

def pre_apply(self):
self.reset()

Expand Down
9 changes: 8 additions & 1 deletion pennylane/devices/default_qubit.py
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,6 @@ class DefaultQubit(QubitDevice):
pennylane_requires = "0.12"
version = "0.12.0"
author = "Xanadu Inc."
_capabilities = {"inverse_operations": True, "reversible_diff": True}

operations = {
"BasisState",
Expand Down Expand Up @@ -158,6 +157,14 @@ def _get_unitary_matrix(self, unitary): # pylint: disable=no-self-use

return unitary.matrix

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(
supports_reversible_diff=True, supports_inverse_operations=True,
)
return capabilities

def _create_basis_state(self, index):
"""Return a computational basis state over all wires.

Expand Down
12 changes: 6 additions & 6 deletions pennylane/devices/default_qubit_autograd.py
Original file line number Diff line number Diff line change
Expand Up @@ -74,12 +74,6 @@ class DefaultQubitAutograd(DefaultQubit):
name = "Default qubit (Autograd) PennyLane plugin"
short_name = "default.qubit.autograd"

_capabilities = {
"model": "qubit",
"provides_jacobian": False,
"passthru_interface": "autograd",
}

parametric_ops = {
"PhaseShift": autograd_ops.PhaseShift,
"RX": autograd_ops.RX,
Expand Down Expand Up @@ -107,6 +101,12 @@ class DefaultQubitAutograd(DefaultQubit):
_conj = staticmethod(np.conj)
_imag = staticmethod(np.imag)

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(passthru_interface="autograd",)
return capabilities

@staticmethod
def _scatter(indices, array, new_dimensions):
new_array = np.zeros(new_dimensions, dtype=array.dtype.type)
Expand Down
12 changes: 6 additions & 6 deletions pennylane/devices/default_qubit_tf.py
Original file line number Diff line number Diff line change
Expand Up @@ -124,12 +124,6 @@ class DefaultQubitTF(DefaultQubit):
name = "Default qubit (TensorFlow) PennyLane plugin"
short_name = "default.qubit.tf"

_capabilities = {
"model": "qubit",
"provides_jacobian": False,
"passthru_interface": "tf",
}

parametric_ops = {
"PhaseShift": tf_ops.PhaseShift,
"RX": tf_ops.RX,
Expand Down Expand Up @@ -157,6 +151,12 @@ class DefaultQubitTF(DefaultQubit):
_conj = staticmethod(tf.math.conj)
_imag = staticmethod(tf.math.conj)

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(passthru_interface="tf",)
return capabilities

@staticmethod
def _scatter(indices, array, new_dimensions):
indices = np.expand_dims(indices, 1)
Expand Down
4 changes: 2 additions & 2 deletions pennylane/devices/tests/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@
# List of all devices that are included in PennyLane
LIST_CORE_DEVICES = {"default.qubit", "default.qubit.tf", "default.qubit.autograd"}
# TODO: add beta devices "default.tensor", "default.tensor.tf", which currently
# do not have an "analytic" attribute.
# do not return probabilities.


@pytest.fixture(scope="function")
Expand Down Expand Up @@ -104,7 +104,7 @@ def _device(wires):
)

capabilities = dev.capabilities()
if "model" not in capabilities or not capabilities["model"] == "qubit":
if capabilities.get("model", None) != "qubit":
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
# exit the tests if device based on cv model (currently not supported)
pytest.exit("The device test suite currently only runs on qubit-based devices.")

Expand Down
Loading