-
Notifications
You must be signed in to change notification settings - Fork 775
Expand file tree
/
Copy pathtest_dynamic.py
More file actions
218 lines (152 loc) · 5.83 KB
/
test_dynamic.py
File metadata and controls
218 lines (152 loc) · 5.83 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
# -*- coding: utf-8 -*-
import pytest
from System.Collections.Generic import Dictionary
from System.Dynamic import ExpandoObject
from Python.Test import DynamicMappingObject
from Python.Test import RejectingDeleteDynamicObject
from Python.Test import RejectingSetDynamicObject
from Python.Test import ThrowingDeleteDynamicObject
from Python.Test import ThrowingGetDynamicObject
from Python.Test import ThrowingSetDynamicObject
def _mro_names(obj):
return [f"{t.__module__}.{t.__name__}" for t in type(obj).__mro__]
@pytest.mark.parametrize(
"obj, expected",
[
(DynamicMappingObject(), True),
(ExpandoObject(), True),
(Dictionary[str, int](), False),
],
)
def test_dlr_mixin_presence(obj, expected):
has_mixin = "clr._extras.dlr.DynamicMetaObjectProviderMixin" in _mro_names(obj)
assert has_mixin is expected
@pytest.mark.parametrize("obj", [DynamicMappingObject(), ExpandoObject()])
def test_dynamic_binder(obj):
assert "answer" not in dir(obj)
assert "wrong_answer" not in dir(obj)
setattr(obj, "answer", 42)
obj.wrong_answer = 54
assert obj.answer == 42
assert obj.wrong_answer == 54
assert "answer" in dir(obj)
assert "wrong_answer" in dir(obj)
def test_native_members_are_accessible_and_keep_priority():
obj = DynamicMappingObject()
setattr(obj, "answer", 42)
obj.SetDynamicValue("Multiplier", 999)
# Native field
assert obj.Label == "default"
obj.Label = "changed"
assert obj.Label == "changed"
# Native property takes precedence over dynamic fallback
assert obj.Multiplier == 1
obj.Multiplier = 7
assert obj.Multiplier == 7
# Native method
obj.Multiplier = 3
assert obj.Multiply(5) == 15
def test_dynamic_and_native_members_coexist():
obj = DynamicMappingObject()
setattr(obj, "answer", 42)
obj.Multiplier = 2
assert obj.answer == 42
assert obj.Multiplier == 2
assert obj.Multiply(10) == 20
@pytest.mark.parametrize("obj", [DynamicMappingObject(), ExpandoObject()])
def test_set_and_get_dynamic_property(obj):
"""Test that setting and getting dynamic properties goes through DLR binder."""
# Get initial value (should be None for non-existent property)
assert not hasattr(obj, "MyProp")
# Set a dynamic property to a value
obj.MyProp = 42
assert obj.MyProp == 42
# Set to None and verify it stays None through DLR
obj.MyProp = None
assert obj.MyProp is None
# Set to another value and verify
obj.MyProp = "hello"
assert obj.MyProp == "hello"
def test_update_dynamic_value():
"""Dynamic-only members should use DLR get/set/modify/delete end-to-end."""
obj = DynamicMappingObject()
assert not hasattr(obj, "DynamicOnly")
# Initial set should create a dynamic member
obj.DynamicOnly = "initial"
assert obj.DynamicOnly == "initial"
# Modify flows through TrySetMember
obj.DynamicOnly = "updated"
assert obj.DynamicOnly == "updated"
# Setting None keeps a present member with None value
obj.DynamicOnly = None
assert obj.DynamicOnly is None
# Delete flows through TryDeleteMember
del obj.DynamicOnly
assert "DynamicOnly" not in dir(obj)
assert not hasattr(obj, "DynamicOnly")
def test_dynamic_set_none_updates_managed_store_after_get():
"""Regression: get->set(None)->get must route through DLR and update managed storage."""
obj = DynamicMappingObject()
obj.SetDynamicValue("MyProp", "initial")
x = obj.MyProp
assert x == "initial"
obj.MyProp = None
y = obj.MyProp
assert y is None
assert obj.GetDynamicValue("MyProp") is None
@pytest.mark.parametrize("obj", [DynamicMappingObject(), ExpandoObject()])
def test_dynamic_member_lifecycle(obj):
"""Dynamic members should support set/modify/get/delete via the DLR binder."""
name = "LifecycleMember"
assert not hasattr(obj, name)
setattr(obj, name, 1)
assert getattr(obj, name) == 1
setattr(obj, name, 2)
assert getattr(obj, name) == 2
delattr(obj, name)
assert not hasattr(obj, name)
def test_derive_from_dynamic_class():
class MyMappingObject(DynamicMappingObject):
__namespace__ = "PythonNetTest"
def __init__(self):
self._custom = 0
@property
def custom_property(self):
return self._custom
@custom_property.setter
def custom_property(self, i):
self._custom += i
obj = MyMappingObject()
with pytest.raises(AttributeError):
x = obj.unknown_property
assert obj.custom_property == 0
obj.custom_property = 5
assert obj.custom_property == 5
obj.custom_property = 5
assert obj.custom_property == 10
obj.other_property = None
assert obj.other_property is None
def test_trysetmember_false_raises_attributeerror_instead_of_silent_python_setattr():
obj = RejectingSetDynamicObject()
with pytest.raises(AttributeError):
obj.typoed_name = 42
assert not hasattr(obj, "typoed_name")
def test_trygetmember_exception_is_raised_in_python():
obj = ThrowingGetDynamicObject()
obj.AddDynamicMember("any_key", 1)
with pytest.raises(Exception, match="TryGetMember failed for 'any_key'"):
_ = obj.any_key
def test_trysetmember_exception_is_raised_in_python():
obj = ThrowingSetDynamicObject()
with pytest.raises(Exception, match="TrySetMember failed for 'bad_name'"):
obj.bad_name = 42
def test_trydeletemember_false_raises_attributeerror():
obj = RejectingDeleteDynamicObject()
obj.AddDynamicMember("existing_name", 42)
with pytest.raises(AttributeError):
del obj.missing_name
def test_trydeletemember_exception_is_raised_in_python():
obj = ThrowingDeleteDynamicObject()
obj.bad_name = 42
with pytest.raises(Exception, match="TryDeleteMember failed for 'bad_name'"):
del obj.bad_name