From b683c25714676ea20e72053fa2e40610b3b97081 Mon Sep 17 00:00:00 2001 From: James Stronz Date: Mon, 18 Nov 2019 23:12:51 -0600 Subject: [PATCH] Refine changes in core/protocol and core/netref related to debugging a potential fix for #355 --- rpyc/core/netref.py | 57 ++++++++++++++----------------------------- rpyc/core/protocol.py | 13 +++++----- 2 files changed, 25 insertions(+), 45 deletions(-) diff --git a/rpyc/core/netref.py b/rpyc/core/netref.py index 90dc2c1c..a65f6ccd 100644 --- a/rpyc/core/netref.py +++ b/rpyc/core/netref.py @@ -40,8 +40,6 @@ ] """a list of types considered built-in (shared between connections)""" -_builtin_type_name_pack = get_id_pack(type)[0] - try: BaseException except NameError: @@ -129,13 +127,6 @@ class BaseNetref(with_metaclass(NetrefMetaclass, object)): __slots__ = ["____conn__", "____id_pack__", "__weakref__", "____refcount__"] def __init__(self, conn, id_pack): - """ - if id_pack[0] == '__builtin__.list': - print('#'*120) - print('ctor of', id_pack) - print('#'*120) - import traceback - traceback.print_stack()""" self.____conn__ = conn self.____id_pack__ = id_pack self.____refcount__ = 1 @@ -153,11 +144,8 @@ def __getattribute__(self, name): if name in LOCAL_ATTRS: if name == "__class__": cls = object.__getattribute__(self, "__class__") - print('GA getattr', self.____id_pack__, cls) - # if self.____id_pack__[0] == '__builtin__.list': import pdb; pdb.set_trace() if cls is None: cls = self.__getattr__("__class__") - print('GA', self.____id_pack__, cls) return cls elif name == "__doc__": return self.__getattr__("__doc__") @@ -231,8 +219,6 @@ def __reduce_ex__(self, proto): return pickle.loads, (syncreq(self, consts.HANDLE_PICKLE, proto),) def __instancecheck__(self, other): - print(type(self), self) - print(type(other), other) # support for checking cached instances across connections if isinstance(other, BaseNetref): if self.____id_pack__[2] != 0: @@ -288,6 +274,7 @@ def method(_self, *args, **kwargs): method.__doc__ = doc return method + class NetrefClass(object): # TODO add slots def __init__(self, class_obj): @@ -320,38 +307,30 @@ def class_factory(id_pack, methods, class_descriptor=None): """ ns = {"__slots__": (), "__class__": None} name_pack = id_pack[0] - bases = (BaseNetref,) class_descriptor = None - if name_pack is not None: # attempt to resolve against builtins and sys.modules - if name_pack in _normalized_builtin_types: - _builtin_class = _normalized_builtin_types.get(name_pack) - ns["__class__"] = _builtin_class - class_descriptor = NetrefClass(ns["__class__"]) - if ns["__class__"] is None: - _module = None - __next_module_name = name_pack - cursor = len(name_pack) - while cursor != -1: - _module = sys.modules.get(name_pack[:cursor]) - if _module is None: - cursor = name_pack.rfind('.') - continue - _class_name = name_pack[cursor + 1:] - _class = getattr(_module, _class_name, None) - if _class is not None and hasattr(_class, '__class__'): - class_descriptor = NetrefClass(_class) - break + if name_pack is not None: + # attempt to resolve __class__ using sys.modules (i.e. builtins and imported modules) + _module = None + cursor = len(name_pack) + while cursor != -1: + _module = sys.modules.get(name_pack[:cursor]) + if _module is None: + cursor = name_pack.rfind('.') + continue + _class_name = name_pack[cursor + 1:] + _class = getattr(_module, _class_name, None) + if _class is not None and hasattr(_class, '__class__'): + class_descriptor = NetrefClass(_class) + break ns['__class__'] = class_descriptor - netref_name = class_descriptor.owner.__name__ if class_descriptor is not None and id_pack[2] == 0 else name_pack - - # if name_pack == '__builtin__.list' and id_pack[2] !=0: import pdb; pdb.set_trace() - + netref_name = class_descriptor.owner.__name__ if class_descriptor is not None and id_pack[2] == 0 else name_pack + # create methods that must perform a syncreq for name, doc in methods: name = str(name) # IronPython issue #10 # only create methods that wont shadow BaseNetref during merge for mro if name not in LOCAL_ATTRS: # i.e. `name != __class__` ns[name] = _make_method(name, doc) - return type(netref_name, bases, ns) + return type(netref_name, (BaseNetref,), ns) for _builtin in _builtin_types: diff --git a/rpyc/core/protocol.py b/rpyc/core/protocol.py index cb25ee02..e4ed6518 100644 --- a/rpyc/core/protocol.py +++ b/rpyc/core/protocol.py @@ -306,17 +306,18 @@ def _unbox(self, package): # boxing def _netref_factory(self, id_pack): # boxing """id_pack is for remote, so when class id fails to directly match """ cls = None - if id_pack[2] == 0: - if id_pack[0] in netref.builtin_classes_cache: - cls = netref.builtin_classes_cache[id_pack[0]] - elif id_pack[1] in self._netref_classes_cache: # - cls = self._netref_classes_cache[id_pack[1]] + if id_pack[2] == 0 and id_pack in self._netref_classes_cache: + cls = self._netref_classes_cache[id_pack] + elif id_pack[0] in netref.builtin_classes_cache: + cls = netref.builtin_classes_cache[id_pack[0]] if cls is None: # in the future, it could see if a sys.module cache/lookup hits first cls_methods = self.sync_request(consts.HANDLE_INSPECT, id_pack) cls = netref.class_factory(id_pack, cls_methods) if id_pack[2] == 0: - self._netref_classes_cache[id_pack[1]] = cls + # only use cached netrefs for classes + # ... instance caching after gc of a proxy will take some mental gymnastics + self._netref_classes_cache[id_pack] = cls return cls(self, id_pack) def _dispatch_request(self, seq, raw_args): # dispatch