Skip to content

Commit

Permalink
Makes unpickling work properly using the Application Registry
Browse files Browse the repository at this point in the history
Close #1175
  • Loading branch information
hgrecco committed Sep 22, 2020
1 parent ffc05dc commit c84d3f4
Show file tree
Hide file tree
Showing 5 changed files with 32 additions and 7 deletions.
18 changes: 18 additions & 0 deletions pint/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,24 @@ def _unpickle(cls, *args):
return cls(*args)


def _unpickle_quantity(cls, *args):
"""Rebuild quantity upon unpickling using the application registry.
"""
return _unpickle(_APP_REGISTRY.Quantity, *args)


def _unpickle_unit(cls, *args):
"""Rebuild unit upon unpickling using the application registry.
"""
return _unpickle(_APP_REGISTRY.Unit, *args)


def _unpickle_measurement(cls, *args):
"""Rebuild measurement upon unpickling using the application registry.
"""
return _unpickle(_APP_REGISTRY.Measurement, *args)


def set_application_registry(registry):
"""Set the application registry, which is used for unpickling operations
and when invoking pint.Quantity or pint.Unit directly.
Expand Down
4 changes: 2 additions & 2 deletions pint/measurement.py
Original file line number Diff line number Diff line change
Expand Up @@ -70,9 +70,9 @@ def rel(self):

def __reduce__(self):
# See notes in Quantity.__reduce__
from . import _unpickle
from . import _unpickle_measurement

return _unpickle, (Measurement, self.magnitude, self._units)
return _unpickle_measurement, (Measurement, self.magnitude, self._units)

def __repr__(self):
return "<Measurement({}, {}, {})>".format(
Expand Down
4 changes: 2 additions & 2 deletions pint/quantity.py
Original file line number Diff line number Diff line change
Expand Up @@ -192,11 +192,11 @@ def __reduce__(self):
"""Allow pickling quantities. Since UnitRegistries are not pickled, upon
unpickling the new object is always attached to the application registry.
"""
from . import _unpickle
from . import _unpickle_quantity

# Note: type(self) would be a mistake as subclasses built by
# build_quantity_class can't be pickled
return _unpickle, (Quantity, self.magnitude, self._units)
return _unpickle_quantity, (Quantity, self.magnitude, self._units)

def __new__(cls, value, units=None):
if is_upcast_type(type(value)):
Expand Down
9 changes: 8 additions & 1 deletion pint/testsuite/test_issues.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

import pytest

from pint import Context, DimensionalityError, UnitRegistry
from pint import Context, DimensionalityError, UnitRegistry, get_application_registry
from pint.compat import np
from pint.testsuite import QuantityTestCase, helpers
from pint.unit import UnitsContainer
Expand Down Expand Up @@ -775,6 +775,13 @@ def test_issue_1136(self):
with pytest.raises(DimensionalityError):
2 ** ureg.Quantity([2, 3], "m")

def test_issue1175(self):
import pickle
foo1 = get_application_registry().Quantity(1, 's')
foo2 = pickle.loads(pickle.dumps(foo1))
self.assertIsInstance(foo1, foo2.__class__)
self.assertIsInstance(foo2, foo1.__class__)


if np is not None:

Expand Down
4 changes: 2 additions & 2 deletions pint/unit.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,9 +28,9 @@ class Unit(PrettyIPython, SharedRegistryObject):

def __reduce__(self):
# See notes in Quantity.__reduce__
from . import _unpickle
from . import _unpickle_unit

return _unpickle, (Unit, self._units)
return _unpickle_unit, (Unit, self._units)

def __init__(self, units):
super().__init__()
Expand Down

0 comments on commit c84d3f4

Please sign in to comment.