forked from reflex-dev/reflex
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtest_event.py
393 lines (324 loc) ยท 13.2 KB
/
test_event.py
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
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
import json
from typing import List
import pytest
from reflex import event
from reflex.event import Event, EventHandler, EventSpec, call_event_handler, fix_events
from reflex.state import BaseState
from reflex.utils import format
from reflex.vars import Var
def make_var(value) -> Var:
"""Make a variable.
Args:
value: The value of the var.
Returns:
The var.
"""
var = Var.create(value)
assert var is not None
return var
def test_create_event():
"""Test creating an event."""
event = Event(token="token", name="state.do_thing", payload={"arg": "value"})
assert event.token == "token"
assert event.name == "state.do_thing"
assert event.payload == {"arg": "value"}
def test_call_event_handler():
"""Test that calling an event handler creates an event spec."""
def test_fn():
pass
test_fn.__qualname__ = "test_fn"
def test_fn_with_args(_, arg1, arg2):
pass
test_fn_with_args.__qualname__ = "test_fn_with_args"
handler = EventHandler(fn=test_fn)
event_spec = handler()
assert event_spec.handler == handler
assert event_spec.args == ()
assert format.format_event(event_spec) == 'Event("test_fn", {})'
handler = EventHandler(fn=test_fn_with_args)
event_spec = handler(make_var("first"), make_var("second"))
# Test passing vars as args.
assert event_spec.handler == handler
assert event_spec.args[0][0].equals(Var.create_safe("arg1"))
assert event_spec.args[0][1].equals(Var.create_safe("first"))
assert event_spec.args[1][0].equals(Var.create_safe("arg2"))
assert event_spec.args[1][1].equals(Var.create_safe("second"))
assert (
format.format_event(event_spec)
== 'Event("test_fn_with_args", {arg1:first,arg2:second})'
)
# Passing args as strings should format differently.
event_spec = handler("first", "second") # type: ignore
assert (
format.format_event(event_spec)
== 'Event("test_fn_with_args", {arg1:`first`,arg2:`second`})'
)
first, second = 123, "456"
handler = EventHandler(fn=test_fn_with_args)
event_spec = handler(first, second) # type: ignore
assert (
format.format_event(event_spec)
== 'Event("test_fn_with_args", {arg1:123,arg2:`456`})'
)
assert event_spec.handler == handler
assert event_spec.args[0][0].equals(Var.create_safe("arg1"))
assert event_spec.args[0][1].equals(Var.create_safe(first))
assert event_spec.args[1][0].equals(Var.create_safe("arg2"))
assert event_spec.args[1][1].equals(Var.create_safe(second))
handler = EventHandler(fn=test_fn_with_args)
with pytest.raises(TypeError):
handler(test_fn) # type: ignore
def test_call_event_handler_partial():
"""Calling an EventHandler with incomplete args returns an EventSpec that can be extended."""
def test_fn_with_args(_, arg1, arg2):
pass
test_fn_with_args.__qualname__ = "test_fn_with_args"
def spec(a2: str) -> List[str]:
return [a2]
handler = EventHandler(fn=test_fn_with_args)
event_spec = handler(make_var("first"))
event_spec2 = call_event_handler(event_spec, spec)
assert event_spec.handler == handler
assert len(event_spec.args) == 1
assert event_spec.args[0][0].equals(Var.create_safe("arg1"))
assert event_spec.args[0][1].equals(Var.create_safe("first"))
assert format.format_event(event_spec) == 'Event("test_fn_with_args", {arg1:first})'
assert event_spec2 is not event_spec
assert event_spec2.handler == handler
assert len(event_spec2.args) == 2
assert event_spec2.args[0][0].equals(Var.create_safe("arg1"))
assert event_spec2.args[0][1].equals(Var.create_safe("first"))
assert event_spec2.args[1][0].equals(Var.create_safe("arg2"))
assert event_spec2.args[1][1].equals(Var.create_safe("_a2"))
assert (
format.format_event(event_spec2)
== 'Event("test_fn_with_args", {arg1:first,arg2:_a2})'
)
@pytest.mark.parametrize(
("arg1", "arg2"),
(
(1, 2),
(1, "2"),
({"a": 1}, {"b": 2}),
),
)
def test_fix_events(arg1, arg2):
"""Test that chaining an event handler with args formats the payload correctly.
Args:
arg1: The first arg passed to the handler.
arg2: The second arg passed to the handler.
"""
def test_fn_with_args(_, arg1, arg2):
pass
test_fn_with_args.__qualname__ = "test_fn_with_args"
handler = EventHandler(fn=test_fn_with_args)
event_spec = handler(arg1, arg2)
event = fix_events([event_spec], token="foo")[0]
assert event.name == test_fn_with_args.__qualname__
assert event.token == "foo"
assert event.payload == {"arg1": arg1, "arg2": arg2}
@pytest.mark.parametrize(
"input,output",
[
(
("/path", None, None),
'Event("_redirect", {path:`/path`,external:false,replace:false})',
),
(
("/path", True, None),
'Event("_redirect", {path:`/path`,external:true,replace:false})',
),
(
("/path", False, None),
'Event("_redirect", {path:`/path`,external:false,replace:false})',
),
(
(Var.create_safe("path"), None, None),
'Event("_redirect", {path:path,external:false,replace:false})',
),
(
("/path", None, True),
'Event("_redirect", {path:`/path`,external:false,replace:true})',
),
(
("/path", True, True),
'Event("_redirect", {path:`/path`,external:true,replace:true})',
),
],
)
def test_event_redirect(input, output):
"""Test the event redirect function.
Args:
input: The input for running the test.
output: The expected output to validate the test.
"""
path, external, replace = input
kwargs = {}
if external is not None:
kwargs["external"] = external
if replace is not None:
kwargs["replace"] = replace
spec = event.redirect(path, **kwargs)
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_redirect"
# this asserts need comment about what it's testing (they fail with Var as input)
# assert spec.args[0][0].equals(Var.create_safe("path"))
# assert spec.args[0][1].equals(Var.create_safe("/path"))
assert format.format_event(spec) == output
def test_event_console_log():
"""Test the event console log function."""
spec = event.console_log("message")
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_console"
assert spec.args[0][0].equals(Var.create_safe("message"))
assert spec.args[0][1].equals(Var.create_safe("message"))
assert format.format_event(spec) == 'Event("_console", {message:`message`})'
spec = event.console_log(Var.create_safe("message"))
assert format.format_event(spec) == 'Event("_console", {message:message})'
def test_event_window_alert():
"""Test the event window alert function."""
spec = event.window_alert("message")
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_alert"
assert spec.args[0][0].equals(Var.create_safe("message"))
assert spec.args[0][1].equals(Var.create_safe("message"))
assert format.format_event(spec) == 'Event("_alert", {message:`message`})'
spec = event.window_alert(Var.create_safe("message"))
assert format.format_event(spec) == 'Event("_alert", {message:message})'
def test_set_focus():
"""Test the event set focus function."""
spec = event.set_focus("input1")
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_set_focus"
assert spec.args[0][0].equals(Var.create_safe("ref"))
assert spec.args[0][1].equals(Var.create_safe("ref_input1"))
assert format.format_event(spec) == 'Event("_set_focus", {ref:`ref_input1`})'
spec = event.set_focus("input1")
assert format.format_event(spec) == 'Event("_set_focus", {ref:`ref_input1`})'
def test_set_value():
"""Test the event window alert function."""
spec = event.set_value("input1", "")
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_set_value"
assert spec.args[0][0].equals(Var.create_safe("ref"))
assert spec.args[0][1].equals(Var.create_safe("ref_input1"))
assert spec.args[1][0].equals(Var.create_safe("value"))
assert spec.args[1][1].equals(Var.create_safe(""))
assert (
format.format_event(spec) == 'Event("_set_value", {ref:`ref_input1`,value:``})'
)
spec = event.set_value("input1", Var.create_safe("message"))
assert (
format.format_event(spec)
== 'Event("_set_value", {ref:`ref_input1`,value:message})'
)
def test_remove_cookie():
"""Test the event remove_cookie."""
spec = event.remove_cookie("testkey")
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_remove_cookie"
assert spec.args[0][0].equals(Var.create_safe("key"))
assert spec.args[0][1].equals(Var.create_safe("testkey"))
assert spec.args[1][0].equals(Var.create_safe("options"))
assert spec.args[1][1].equals(Var.create_safe({"path": "/"}))
assert (
format.format_event(spec)
== 'Event("_remove_cookie", {key:`testkey`,options:{"path": "/"}})'
)
def test_remove_cookie_with_options():
"""Test the event remove_cookie with options."""
options = {
"path": "/foo",
"domain": "example.com",
"secure": True,
"sameSite": "strict",
}
spec = event.remove_cookie("testkey", options)
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_remove_cookie"
assert spec.args[0][0].equals(Var.create_safe("key"))
assert spec.args[0][1].equals(Var.create_safe("testkey"))
assert spec.args[1][0].equals(Var.create_safe("options"))
assert spec.args[1][1].equals(Var.create_safe(options))
assert (
format.format_event(spec)
== f'Event("_remove_cookie", {{key:`testkey`,options:{json.dumps(options)}}})'
)
def test_clear_local_storage():
"""Test the event clear_local_storage."""
spec = event.clear_local_storage()
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_clear_local_storage"
assert not spec.args
assert format.format_event(spec) == 'Event("_clear_local_storage", {})'
def test_remove_local_storage():
"""Test the event remove_local_storage."""
spec = event.remove_local_storage("testkey")
assert isinstance(spec, EventSpec)
assert spec.handler.fn.__qualname__ == "_remove_local_storage"
assert spec.args[0][0].equals(Var.create_safe("key"))
assert spec.args[0][1].equals(Var.create_safe("testkey"))
assert (
format.format_event(spec) == 'Event("_remove_local_storage", {key:`testkey`})'
)
def test_event_actions():
"""Test DOM event actions, like stopPropagation and preventDefault."""
# EventHandler
handler = EventHandler(fn=lambda: None)
assert not handler.event_actions
sp_handler = handler.stop_propagation
assert handler is not sp_handler
assert sp_handler.event_actions == {"stopPropagation": True}
pd_handler = handler.prevent_default
assert handler is not pd_handler
assert pd_handler.event_actions == {"preventDefault": True}
both_handler = sp_handler.prevent_default
assert both_handler is not sp_handler
assert both_handler.event_actions == {
"stopPropagation": True,
"preventDefault": True,
}
throttle_handler = handler.throttle(300)
assert handler is not throttle_handler
assert throttle_handler.event_actions == {"throttle": 300}
debounce_handler = handler.debounce(300)
assert handler is not debounce_handler
assert debounce_handler.event_actions == {"debounce": 300}
all_handler = handler.stop_propagation.prevent_default.throttle(200).debounce(100)
assert handler is not all_handler
assert all_handler.event_actions == {
"stopPropagation": True,
"preventDefault": True,
"throttle": 200,
"debounce": 100,
}
assert not handler.event_actions
# Convert to EventSpec should carry event actions
sp_handler2 = handler.stop_propagation.throttle(200)
spec = sp_handler2()
assert spec.event_actions == {"stopPropagation": True, "throttle": 200}
assert spec.event_actions == sp_handler2.event_actions
assert spec.event_actions is not sp_handler2.event_actions
# But it should be a copy!
assert spec.event_actions is not sp_handler2.event_actions
spec2 = spec.prevent_default
assert spec is not spec2
assert spec2.event_actions == {
"stopPropagation": True,
"preventDefault": True,
"throttle": 200,
}
assert spec2.event_actions != spec.event_actions
# The original handler should still not be touched.
assert not handler.event_actions
def test_event_actions_on_state():
class EventActionState(BaseState):
def handler(self):
pass
handler = EventActionState.handler
assert isinstance(handler, EventHandler)
assert not handler.event_actions
sp_handler = EventActionState.handler.stop_propagation
assert sp_handler.event_actions == {"stopPropagation": True}
# should NOT affect other references to the handler
assert not handler.event_actions