From 363180d1f4c7ee7afd2dbb1c2cdd3b9555940fd2 Mon Sep 17 00:00:00 2001 From: anivegesana Date: Fri, 6 May 2022 19:50:35 -0700 Subject: [PATCH 1/2] Fix class cell in `__new__` for metaclasses --- dill/_dill.py | 8 +++++++- tests/test_classdef.py | 27 +++++++++++++++++++++++++++ 2 files changed, 34 insertions(+), 1 deletion(-) diff --git a/dill/_dill.py b/dill/_dill.py index 40c01fc5..1c3caaed 100644 --- a/dill/_dill.py +++ b/dill/_dill.py @@ -1589,7 +1589,13 @@ def save_cell(pickler, obj): log.info("# Ce3") return if is_dill(pickler, child=True): - postproc = next(iter(pickler._postproc.values()), None) + if id(f) in pickler._postproc: + # Already seen. Add to its postprocessing. + postproc = pickler._postproc[id(f)] + else: + # Haven't seen it. Add to the highest possible object and set its + # value as late as possible to prevent cycle. + postproc = next(iter(pickler._postproc.values()), None) if postproc is not None: log.info("Ce2: %s" % obj) # _CELL_REF is defined in _shims.py to support older versions of diff --git a/tests/test_classdef.py b/tests/test_classdef.py index ca6ab6ad..09b36187 100644 --- a/tests/test_classdef.py +++ b/tests/test_classdef.py @@ -216,6 +216,32 @@ def test_slots(): assert dill.pickles(Y.y) assert dill.copy(y).y == value +def test_metaclass(): + class metaclass_with_new(type): + def __new__(mcls, name, bases, ns, **kwds): + cls = super().__new__(mcls, name, bases, ns, **kwds) + assert mcls is not None + assert cls.method(mcls) + return cls + def method(cls, mcls): + return isinstance(cls, mcls) + + if dill._dill.PY3: + l = locals() + exec("""class subclass_with_new(metaclass=metaclass_with_new): + def __new__(cls): + self = super().__new__(cls) + return self""", None, l) + subclass_with_new = l['subclass_with_new'] + else: + class subclass_with_new: + __metaclass__ = metaclass_with_new + def __new__(cls): + self = super().__new__(cls) + return self + + assert dill.copy(subclass_with_new()) + if __name__ == '__main__': test_class_instances() @@ -227,3 +253,4 @@ def test_slots(): test_array_subclass() test_method_decorator() test_slots() + test_metaclass() From 77355a5c471adea587c554344d53b6e03f510e4d Mon Sep 17 00:00:00 2001 From: anivegesana Date: Sat, 7 May 2022 09:21:12 -0700 Subject: [PATCH 2/2] Python 3 super --- tests/test_classdef.py | 29 +++++++++++++++++++---------- 1 file changed, 19 insertions(+), 10 deletions(-) diff --git a/tests/test_classdef.py b/tests/test_classdef.py index 09b36187..e23bef0a 100644 --- a/tests/test_classdef.py +++ b/tests/test_classdef.py @@ -217,16 +217,16 @@ def test_slots(): assert dill.copy(y).y == value def test_metaclass(): - class metaclass_with_new(type): - def __new__(mcls, name, bases, ns, **kwds): - cls = super().__new__(mcls, name, bases, ns, **kwds) - assert mcls is not None - assert cls.method(mcls) - return cls - def method(cls, mcls): - return isinstance(cls, mcls) - if dill._dill.PY3: + class metaclass_with_new(type): + def __new__(mcls, name, bases, ns, **kwds): + cls = super().__new__(mcls, name, bases, ns, **kwds) + assert mcls is not None + assert cls.method(mcls) + return cls + def method(cls, mcls): + return isinstance(cls, mcls) + l = locals() exec("""class subclass_with_new(metaclass=metaclass_with_new): def __new__(cls): @@ -234,10 +234,19 @@ def __new__(cls): return self""", None, l) subclass_with_new = l['subclass_with_new'] else: + class metaclass_with_new(type): + def __new__(mcls, name, bases, ns, **kwds): + cls = super(mcls, metaclass_with_new).__new__(mcls, name, bases, ns, **kwds) + assert mcls is not None + assert cls.method(mcls) + return cls + def method(cls, mcls): + return isinstance(cls, mcls) + class subclass_with_new: __metaclass__ = metaclass_with_new def __new__(cls): - self = super().__new__(cls) + self = super(subclass_with_new, cls).__new__(cls) return self assert dill.copy(subclass_with_new())