Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
A
Acquisition
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
Acquisition
Commits
d123b119
Commit
d123b119
authored
Mar 30, 2015
by
Jason Madden
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Hit 99% test coverage and correct a bug in the __repr__ implementation.
parent
4ffadd23
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
147 additions
and
4 deletions
+147
-4
src/Acquisition/__init__.py
src/Acquisition/__init__.py
+12
-4
src/Acquisition/tests.py
src/Acquisition/tests.py
+135
-0
No files found.
src/Acquisition/__init__.py
View file @
d123b119
...
@@ -348,6 +348,8 @@ class _Wrapper(ExtensionClass.Base):
...
@@ -348,6 +348,8 @@ class _Wrapper(ExtensionClass.Base):
while
isinstance
(
wrapper
.
_obj
,
_Wrapper
)
\
while
isinstance
(
wrapper
.
_obj
,
_Wrapper
)
\
and
(
wrapper
.
_obj
.
_container
is
wrapper
.
_container
.
_obj
):
and
(
wrapper
.
_obj
.
_container
is
wrapper
.
_container
.
_obj
):
# Since we mutate the wrapper as we walk up, we must copy
# Since we mutate the wrapper as we walk up, we must copy
# XXX: This comes from the C implementation. Do we really need to
# copy?
wrapper
=
type
(
wrapper
)(
wrapper
.
_obj
,
wrapper
.
_container
)
wrapper
=
type
(
wrapper
)(
wrapper
.
_obj
,
wrapper
.
_container
)
wrapper
.
_obj
=
wrapper
.
_obj
.
_obj
wrapper
.
_obj
=
wrapper
.
_obj
.
_obj
return
wrapper
return
wrapper
...
@@ -463,11 +465,15 @@ class _Wrapper(ExtensionClass.Base):
...
@@ -463,11 +465,15 @@ class _Wrapper(ExtensionClass.Base):
def
__repr__
(
self
):
def
__repr__
(
self
):
aq_self
=
self
.
_obj
aq_self
=
self
.
_obj
return
type
(
aq_self
).
__repr__
(
aq_self
)
try
:
return
_rebound_method
(
aq_self
.
__repr__
,
self
)()
except
(
AttributeError
,
TypeError
):
return
repr
(
aq_self
)
def
__str__
(
self
):
def
__str__
(
self
):
aq_self
=
self
.
_obj
aq_self
=
self
.
_obj
return
type
(
aq_self
).
__str__
(
aq_self
)
f
=
getattr
(
type
(
aq_self
),
'__str__'
,
object
.
__str__
)
return
_rebound_method
(
f
,
self
)()
__binary_special_methods__
=
[
__binary_special_methods__
=
[
# general numeric
# general numeric
...
@@ -767,7 +773,7 @@ def aq_chain(obj, containment=False):
...
@@ -767,7 +773,7 @@ def aq_chain(obj, containment=False):
def
aq_base
(
obj
):
def
aq_base
(
obj
):
result
=
obj
result
=
obj
while
isinstance
(
result
,
_Wrapper
):
while
isinstance
(
result
,
_Wrapper
):
result
=
result
.
aq_self
result
=
result
.
_obj
return
result
return
result
...
@@ -819,7 +825,9 @@ def aq_inContextOf(self, o, inner=True):
...
@@ -819,7 +825,9 @@ def aq_inContextOf(self, o, inner=True):
if
inner
:
if
inner
:
self
=
aq_inner
(
next
)
self
=
aq_inner
(
next
)
if
self
is
None
:
if
self
is
None
:
# pragma: no cover
# This branch is normally impossible to hit,
# it just mirrors a check in C
break
break
else
:
else
:
self
=
next
self
=
next
...
...
src/Acquisition/tests.py
View file @
d123b119
...
@@ -1659,6 +1659,7 @@ def test_cant_persist_acquisition_wrappers_classic():
...
@@ -1659,6 +1659,7 @@ def test_cant_persist_acquisition_wrappers_classic():
... klass = type(obj)
... klass = type(obj)
... oid = obj._p_oid
... oid = obj._p_oid
... if hasattr(klass, '__getnewargs__'):
... if hasattr(klass, '__getnewargs__'):
... assert klass.__getnewargs__(obj) == () # Coverage, make sure it can be called
... return oid
... return oid
... return 'class_and_oid', klass
... return 'class_and_oid', klass
...
@@ -2570,6 +2571,36 @@ def test___parent__aq_parent_circles():
...
@@ -2570,6 +2571,36 @@ def test___parent__aq_parent_circles():
RuntimeError: Recursion detected in acquisition wrapper
RuntimeError: Recursion detected in acquisition wrapper
"""
"""
if
hasattr
(
Acquisition
.
ImplicitAcquisitionWrapper
,
'_obj'
):
def
test_python_impl_cycle
():
"""
An extra safety belt, specific to the Python implementation
because it's not clear how one could arrive in this situation
naturally.
>>> class Impl(Acquisition.Implicit):
... pass
>>> root = Impl()
>>> root.child = Impl()
>>> child_wrapper = root.child
Now set up the python specific boo-boo:
>>> child_wrapper._obj = child_wrapper
Now nothing works:
>>> child_wrapper.non_existant_attr
Traceback (most recent call last):
...
RuntimeError: Recursion detected in acquisition wrapper
>>> Acquisition.aq_acquire(child_wrapper, 'non_existant_attr')
Traceback (most recent call last):
...
RuntimeError: Recursion detected in acquisition wrapper
"""
def
test_unwrapped_implicit_acquirer_unwraps__parent__
():
def
test_unwrapped_implicit_acquirer_unwraps__parent__
():
"""
"""
...
@@ -2808,6 +2839,110 @@ def test_getitem_setitem_implemented():
...
@@ -2808,6 +2839,110 @@ def test_getitem_setitem_implemented():
{'a': 'b'}
{'a': 'b'}
"""
"""
def
test_wrapped_objects_are_unwrapped_on_set
():
"""
A wrapper is not passed to the base object during `setattr`.
>>> class Impl(Acquisition.Implicit):
... pass
Given two different wrappers:
>>> root = Impl()
>>> child = Impl()
>>> child2 = Impl()
>>> root.child = child
>>> root.child2 = child
If we pass one to the other as an attribute:
>>> root.child.child2 = root.child2
By the time it gets there, it's not wrapped:
>>> type(child.__dict__['child2']) is Impl
True
"""
def
test_wrapper_calls_of_on_non_wrapper
():
"""
The ExtensionClass protocol is respected even for non-Acquisition
objects.
>>> class MyBase(ExtensionClass.Base):
... def __of__(self, other):
... print("Of called")
... return 42
>>> class Impl(Acquisition.Implicit):
... pass
If we have a wrapper around an object that is an extension class,
but not an Acquisition wrapper:
>>> root = Impl()
>>> wrapper = Acquisition.ImplicitAcquisitionWrapper(MyBase(), root)
And access that object itself through a wrapper:
>>> root.child = Impl()
>>> root.child.wrapper = wrapper
The `__of__` protocol is respected implicitly:
>>> root.child.wrapper
Of called
42
Here it is explicitly:
>>> wrapper.__of__(root.child)
Of called
42
"""
def
test_aq_inContextOf_odd_cases
():
"""
The aq_inContextOf function still works in some
artificial cases.
>>> from Acquisition import aq_inContextOf, aq_inner
>>> root = object()
>>> wrapper_around_none = Acquisition.ImplicitAcquisitionWrapper(None,None)
>>> aq_inContextOf(wrapper_around_none, root)
0
If we don't ask for inner objects, the same thing happens in this case:
>>> aq_inContextOf(wrapper_around_none, root, False)
0
Somewhat surprisingly, the `aq_inner` of this wrapper is itself a wrapper:
>>> aq_inner(wrapper_around_none) is None
False
If we manipulate the Python implementation to make this no longer true,
nothing breaks:
>>> setattr(wrapper_around_none, '_obj', None) if hasattr(wrapper_around_none, '_obj') else None
>>> aq_inContextOf(wrapper_around_none, root)
0
>>> wrapper_around_none
None
Following parent pointers in weird circumstances works too:
>>> class WithParent(object):
... __parent__ = None
>>> aq_inContextOf(WithParent(), root)
0
"""
class
TestParent
(
unittest
.
TestCase
):
class
TestParent
(
unittest
.
TestCase
):
def
test_parent_parent_circles
(
self
):
def
test_parent_parent_circles
(
self
):
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment