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 12 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
18 changes: 15 additions & 3 deletions doc/development/plugins.rst
Original file line number Diff line number Diff line change
Expand Up @@ -101,15 +101,27 @@ 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:
* :attr:`.Device.capabilities()`: a function overriding the corresponding base class method,
which updates the ``_capabilities`` dictionary containing information about the properties of
the device. These properties are queried by PennyLane core and/or external apps. Examples are:

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

* ``'inverse_operations'`` (*bool*): ``True`` if the device supports
* ``'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_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_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 in remote.

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

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

Expand Down
39 changes: 32 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,
Copy link
Contributor Author

Choose a reason for hiding this comment

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

In two minds about this one: we could only define it in the devices that actually have passthru functionality (so that absence of the key signifies that the device does not)...

'supports_reversible_diff': False,
'supports_exact': False,
'supports_sampled': False,
'supports_inverse_operations': False,
'supports_tensor_observables': False,
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
'provides_jacobian': False,
'executes_in_remote': False,
'takes_fixed_number_of_wires': False,
'performs_noisy_computation': False,
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
}
"""The capabilities dictionary stores the properties of a device. Devices can add their
own custom properties and overwrite existing ones by overwriting the ``capabilities`` class method."""
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Not sure if one would use docstrings for class attributes?


_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 overwrite or add capabilities should overwrite this method with

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,7 @@ 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)

return operation in self.operations

Expand Down Expand Up @@ -468,7 +493,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 +509,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 +526,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
12 changes: 11 additions & 1 deletion pennylane/beta/devices/default_tensor.py
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,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 @@ -165,6 +164,17 @@ def __init__(self, wires, shots=1000, representation="exact", contraction_method
self._contraction_method = contraction_method
self.reset()

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

def reset(self):
"""Reset the device."""
self._clear_network_data()
Expand Down
15 changes: 9 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,15 @@ 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,
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
passthru_interface='tf',
)
return capabilities

def reset(self):
self.res = None
self.variables = []
Expand Down
2 changes: 2 additions & 0 deletions pennylane/devices/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,3 +32,5 @@
"""
from .default_qubit import DefaultQubit
from .default_gaussian import DefaultGaussian
from .default_qubit_tf import DefaultQubitTF
from .default_qubit_autograd import DefaultQubitAutograd
mariaschuld marked this conversation as resolved.
Show resolved Hide resolved
12 changes: 10 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,16 @@ 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
10 changes: 9 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,15 @@ 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,
Copy link
Contributor

Choose a reason for hiding this comment

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

How come this would not be defined among the capabilities in Device?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

It is set to False in Device...

Copy link
Contributor

Choose a reason for hiding this comment

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

Oh not sure how I missed that, my bad!

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 understanding, how do we know if something supports reversible diff?

supports_inverse_operations=True,
)
return capabilities

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

Expand Down
15 changes: 9 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,15 @@ class DefaultQubitAutograd(DefaultQubit):
_conj = staticmethod(np.conj)
_imag = staticmethod(np.imag)

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

@staticmethod
def _scatter(indices, array, new_dimensions):
new_array = np.zeros(new_dimensions, dtype=array.dtype.type)
Expand Down
15 changes: 9 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,15 @@ class DefaultQubitTF(DefaultQubit):
_conj = staticmethod(tf.math.conj)
_imag = staticmethod(tf.math.conj)

@classmethod
def capabilities(cls):
capabilities = super().capabilities().copy()
capabilities.update(
provides_jacobian=False,
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 @@ -29,7 +29,7 @@
# Tolerance for analytic tests
TOL = 1e-6
# Tolerance for non-analytic tests
TOL_STOCHASTIC = 0.05
TOL_STOCHASTIC = 0.1
# Number of shots to call the devices with
N_SHOTS = 10000
# List of all devices that are included in PennyLane
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