CheeseSignal 1.1.0__py2.py3-none-any.whl → 1.2.0__py2.py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- CheeseSignal/__init__.py +1 -360
- CheeseSignal/signal.py +359 -0
- {cheesesignal-1.1.0.dist-info → cheesesignal-1.2.0.dist-info}/METADATA +2 -2
- cheesesignal-1.2.0.dist-info/RECORD +6 -0
- {cheesesignal-1.1.0.dist-info → cheesesignal-1.2.0.dist-info}/WHEEL +1 -1
- cheesesignal-1.1.0.dist-info/RECORD +0 -5
- {cheesesignal-1.1.0.dist-info → cheesesignal-1.2.0.dist-info}/licenses/LICENSE +0 -0
CheeseSignal/__init__.py
CHANGED
|
@@ -1,360 +1 @@
|
|
|
1
|
-
from
|
|
2
|
-
|
|
3
|
-
if TYPE_CHECKING:
|
|
4
|
-
class Signal:
|
|
5
|
-
...
|
|
6
|
-
|
|
7
|
-
class Receiver:
|
|
8
|
-
def __init__(self, signal: 'Signal', fn: Callable, *, expected_receive_num: int, auto_remove: bool):
|
|
9
|
-
'''
|
|
10
|
-
- Args
|
|
11
|
-
|
|
12
|
-
- expected_receive_num: 期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
13
|
-
|
|
14
|
-
- auto_remove: 是否自动删除响应次数超出期望次数的接收器。
|
|
15
|
-
'''
|
|
16
|
-
|
|
17
|
-
self._signal: 'Signal' = signal
|
|
18
|
-
self.fn: Callable = fn
|
|
19
|
-
self._expected_receive_num: int = expected_receive_num
|
|
20
|
-
self._auto_remove: bool = auto_remove
|
|
21
|
-
self._total_receive_num: int = 0
|
|
22
|
-
self._active: bool = True
|
|
23
|
-
|
|
24
|
-
def reset(self):
|
|
25
|
-
'''
|
|
26
|
-
重统计置数据。
|
|
27
|
-
|
|
28
|
-
在有期望信号接受次数的情况下,`auto_remove is False`的接收器会重新开始计数并接收信号。
|
|
29
|
-
'''
|
|
30
|
-
|
|
31
|
-
self._total_receive_num = 0
|
|
32
|
-
|
|
33
|
-
@property
|
|
34
|
-
def expected_receive_num(self) -> int:
|
|
35
|
-
'''
|
|
36
|
-
期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
37
|
-
|
|
38
|
-
设置值小于`total_receive_num`且`auto_remove is True`,则会立刻删除。
|
|
39
|
-
'''
|
|
40
|
-
|
|
41
|
-
return self._expected_receive_num
|
|
42
|
-
|
|
43
|
-
@expected_receive_num.setter
|
|
44
|
-
def expected_receive_num(self, value: int):
|
|
45
|
-
self._expected_receive_num = value
|
|
46
|
-
|
|
47
|
-
if self._auto_remove and self.is_expired:
|
|
48
|
-
self._signal.receivers.remove(self)
|
|
49
|
-
|
|
50
|
-
@property
|
|
51
|
-
def auto_remove(self) -> bool:
|
|
52
|
-
'''
|
|
53
|
-
是否自动删除响应次数超出期望次数的接收器。
|
|
54
|
-
|
|
55
|
-
设置为`True`时若该receiver过期,则会立刻删除。
|
|
56
|
-
'''
|
|
57
|
-
|
|
58
|
-
return self._auto_remove
|
|
59
|
-
|
|
60
|
-
@auto_remove.setter
|
|
61
|
-
def auto_remove(self, value: bool):
|
|
62
|
-
self._auto_remove = value
|
|
63
|
-
|
|
64
|
-
if self._auto_remove and self.is_expired:
|
|
65
|
-
self._signal.receivers.remove(self)
|
|
66
|
-
|
|
67
|
-
@property
|
|
68
|
-
def active(self) -> bool:
|
|
69
|
-
'''
|
|
70
|
-
是否激活;未激活将忽略信号。
|
|
71
|
-
'''
|
|
72
|
-
|
|
73
|
-
return self._active
|
|
74
|
-
|
|
75
|
-
@active.setter
|
|
76
|
-
def active(self, value: bool):
|
|
77
|
-
self._active = value
|
|
78
|
-
|
|
79
|
-
@property
|
|
80
|
-
def total_receive_num(self) -> int:
|
|
81
|
-
'''
|
|
82
|
-
【只读】 总计信号接受次数。
|
|
83
|
-
'''
|
|
84
|
-
|
|
85
|
-
return self._total_receive_num
|
|
86
|
-
|
|
87
|
-
@property
|
|
88
|
-
def remaining_receive_num(self) -> int:
|
|
89
|
-
'''
|
|
90
|
-
【只读】 剩余的期望信号接受次数;返回为-1代表无期望信号接受次数。
|
|
91
|
-
'''
|
|
92
|
-
|
|
93
|
-
if not self.expected_receive_num:
|
|
94
|
-
return -1
|
|
95
|
-
return self.expected_receive_num - self.total_receive_num
|
|
96
|
-
|
|
97
|
-
@property
|
|
98
|
-
def is_expired(self) -> bool:
|
|
99
|
-
'''
|
|
100
|
-
【只读】 是否过期。
|
|
101
|
-
'''
|
|
102
|
-
|
|
103
|
-
return not self.is_unexpired
|
|
104
|
-
|
|
105
|
-
@property
|
|
106
|
-
def is_unexpired(self) -> bool:
|
|
107
|
-
'''
|
|
108
|
-
【只读】 是否未过期。
|
|
109
|
-
'''
|
|
110
|
-
|
|
111
|
-
if self.remaining_receive_num == -1:
|
|
112
|
-
return True
|
|
113
|
-
return True if self.remaining_receive_num else False
|
|
114
|
-
|
|
115
|
-
class Signal:
|
|
116
|
-
def __init__(self):
|
|
117
|
-
self._receivers: List[Receiver] = []
|
|
118
|
-
self._total_send_num: int = 0
|
|
119
|
-
|
|
120
|
-
@overload
|
|
121
|
-
def connect(self, fn: Callable, *, expected_receive_num: int = 0, auto_remove: bool = False):
|
|
122
|
-
'''
|
|
123
|
-
通过函数注册响应函数。
|
|
124
|
-
|
|
125
|
-
```python
|
|
126
|
-
from CheeseSignal import Signal
|
|
127
|
-
|
|
128
|
-
def receiver(*args, **kwargs):
|
|
129
|
-
...
|
|
130
|
-
|
|
131
|
-
signal = Signal()
|
|
132
|
-
signal.connect(receiver)
|
|
133
|
-
```
|
|
134
|
-
|
|
135
|
-
- Args
|
|
136
|
-
|
|
137
|
-
- expected_receive_num: 期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
138
|
-
|
|
139
|
-
- auto_remove: 是否自动删除响应次数超出期望次数的接收器。
|
|
140
|
-
|
|
141
|
-
- Raise
|
|
142
|
-
|
|
143
|
-
- ValueError: 已有重复的函数接收器。
|
|
144
|
-
'''
|
|
145
|
-
|
|
146
|
-
@overload
|
|
147
|
-
def connect(self, *, expected_receive_num: int = 0, auto_remove: bool = False):
|
|
148
|
-
'''
|
|
149
|
-
通过装饰器注册响应函数。
|
|
150
|
-
|
|
151
|
-
```python
|
|
152
|
-
from CheeseSignal import Signal
|
|
153
|
-
|
|
154
|
-
signal = Signal()
|
|
155
|
-
|
|
156
|
-
@signal.connect()
|
|
157
|
-
def receiver(*args, **kwargs):
|
|
158
|
-
...
|
|
159
|
-
```
|
|
160
|
-
|
|
161
|
-
- Args
|
|
162
|
-
|
|
163
|
-
- expected_receive_num: 期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
164
|
-
|
|
165
|
-
- auto_remove: 是否自动删除响应次数超出期望次数的接收器。
|
|
166
|
-
|
|
167
|
-
- Raise
|
|
168
|
-
|
|
169
|
-
- ValueError: 已有重复的函数接收器。
|
|
170
|
-
'''
|
|
171
|
-
|
|
172
|
-
def connect(self, arg1: Callable | None = None, *, expected_receive_num: int = 0, auto_remove: bool = False):
|
|
173
|
-
if not arg1:
|
|
174
|
-
def wrapper(fn):
|
|
175
|
-
self._connect(fn, expected_receive_num = expected_receive_num, auto_remove = auto_remove)
|
|
176
|
-
return fn
|
|
177
|
-
return wrapper
|
|
178
|
-
|
|
179
|
-
self._connect(arg1, expected_receive_num = expected_receive_num, auto_remove = auto_remove)
|
|
180
|
-
|
|
181
|
-
def _connect(self, fn: Callable, *, expected_receive_num: int, auto_remove: bool):
|
|
182
|
-
for receiver in self.receivers:
|
|
183
|
-
if receiver.fn == fn:
|
|
184
|
-
raise ValueError('已有重复的函数接收器')
|
|
185
|
-
|
|
186
|
-
self.receivers.append(Receiver(self, fn, expected_receive_num = expected_receive_num, auto_remove = auto_remove))
|
|
187
|
-
|
|
188
|
-
def send(self, *args, **kwargs) -> List[Any]:
|
|
189
|
-
'''
|
|
190
|
-
发送信号。
|
|
191
|
-
|
|
192
|
-
```python
|
|
193
|
-
from CheeseSignal import Signal
|
|
194
|
-
|
|
195
|
-
signal = Signal()
|
|
196
|
-
signal.send('data1', 'data2', **{
|
|
197
|
-
'key1': 'value1',
|
|
198
|
-
'key2': 'value2'
|
|
199
|
-
})
|
|
200
|
-
```
|
|
201
|
-
'''
|
|
202
|
-
|
|
203
|
-
self._total_send_num += 1
|
|
204
|
-
|
|
205
|
-
results = []
|
|
206
|
-
for receiver in self.receivers[:]:
|
|
207
|
-
if receiver.active and receiver.is_unexpired:
|
|
208
|
-
receiver._total_receive_num += 1
|
|
209
|
-
results.append(receiver.fn(*args, **kwargs))
|
|
210
|
-
|
|
211
|
-
if receiver.is_expired:
|
|
212
|
-
self.receivers.remove(receiver)
|
|
213
|
-
return results
|
|
214
|
-
|
|
215
|
-
async def async_send(self, *args, **kwargs) -> List[Any]:
|
|
216
|
-
'''
|
|
217
|
-
在协程环境中发送信号,并请保证所有接收函数都是协程函数。
|
|
218
|
-
|
|
219
|
-
```python
|
|
220
|
-
import asyncio
|
|
221
|
-
|
|
222
|
-
from CheeseSignal import Signal
|
|
223
|
-
|
|
224
|
-
async def run_asyncio():
|
|
225
|
-
signal = Signal()
|
|
226
|
-
await signal.async_send('data1', 'data2', **{
|
|
227
|
-
'key1': 'value1',
|
|
228
|
-
'key2': 'value2'
|
|
229
|
-
})
|
|
230
|
-
|
|
231
|
-
asyncio.run(run_asyncio())
|
|
232
|
-
```
|
|
233
|
-
'''
|
|
234
|
-
|
|
235
|
-
self._total_send_num += 1
|
|
236
|
-
|
|
237
|
-
results = []
|
|
238
|
-
for receiver in self.receivers[:]:
|
|
239
|
-
if receiver.active and receiver.is_unexpired:
|
|
240
|
-
receiver._total_receive_num += 1
|
|
241
|
-
results.append(await receiver.fn(*args, **kwargs))
|
|
242
|
-
|
|
243
|
-
if receiver.is_expired:
|
|
244
|
-
self.receivers.remove(receiver)
|
|
245
|
-
return results
|
|
246
|
-
|
|
247
|
-
def disconnect(self, fn: Callable):
|
|
248
|
-
'''
|
|
249
|
-
断开接收器。
|
|
250
|
-
|
|
251
|
-
```python
|
|
252
|
-
from CheeseSignal import Signal
|
|
253
|
-
|
|
254
|
-
def receiver(*args, **kwargs):
|
|
255
|
-
...
|
|
256
|
-
|
|
257
|
-
signal = Signal()
|
|
258
|
-
signal.connect(receiver)
|
|
259
|
-
signal.disconnect(receiver)
|
|
260
|
-
```
|
|
261
|
-
|
|
262
|
-
- Raise
|
|
263
|
-
|
|
264
|
-
- ValueError: 未找到该函数的接收器。
|
|
265
|
-
'''
|
|
266
|
-
|
|
267
|
-
for receiver in self.receivers:
|
|
268
|
-
if receiver.fn == fn:
|
|
269
|
-
self.receivers.remove(receiver)
|
|
270
|
-
return
|
|
271
|
-
|
|
272
|
-
raise ValueError('未找到该函数的接收器')
|
|
273
|
-
|
|
274
|
-
def reset(self):
|
|
275
|
-
'''
|
|
276
|
-
重置统计数据;所有的接收器的统计数据也会同步清空。
|
|
277
|
-
'''
|
|
278
|
-
|
|
279
|
-
self._total_send_num = 0
|
|
280
|
-
for receiver in self.receivers:
|
|
281
|
-
receiver.reset()
|
|
282
|
-
|
|
283
|
-
def disconnect_all(self):
|
|
284
|
-
'''
|
|
285
|
-
断开所有接收器。
|
|
286
|
-
'''
|
|
287
|
-
|
|
288
|
-
self._receivers = []
|
|
289
|
-
|
|
290
|
-
def get_receiver(self, fn: Callable) -> Receiver:
|
|
291
|
-
'''
|
|
292
|
-
获取接收器。
|
|
293
|
-
|
|
294
|
-
```python
|
|
295
|
-
from CheeseSignal import Signal
|
|
296
|
-
|
|
297
|
-
def receiver(*args, **kwargs):
|
|
298
|
-
...
|
|
299
|
-
|
|
300
|
-
signal = Signal()
|
|
301
|
-
signal.connect(receiver)
|
|
302
|
-
|
|
303
|
-
print(signal.get_receiver(receiver))
|
|
304
|
-
```
|
|
305
|
-
|
|
306
|
-
- Raise
|
|
307
|
-
|
|
308
|
-
- ValueError: 未找到该函数的接收器。
|
|
309
|
-
'''
|
|
310
|
-
|
|
311
|
-
for receiver in self.receivers:
|
|
312
|
-
if receiver.fn == fn:
|
|
313
|
-
return receiver
|
|
314
|
-
|
|
315
|
-
raise ValueError('未找到该函数的接收器')
|
|
316
|
-
|
|
317
|
-
def index(self, fn: Callable) -> int:
|
|
318
|
-
'''
|
|
319
|
-
获取接收器的顺序位置。
|
|
320
|
-
|
|
321
|
-
```python
|
|
322
|
-
from CheeseSignal import Signal
|
|
323
|
-
|
|
324
|
-
def receiver(*args, **kwargs):
|
|
325
|
-
...
|
|
326
|
-
|
|
327
|
-
signal = Signal()
|
|
328
|
-
signal.connect(receiver)
|
|
329
|
-
|
|
330
|
-
print(signal.index(receiver))
|
|
331
|
-
```
|
|
332
|
-
|
|
333
|
-
- Raise
|
|
334
|
-
|
|
335
|
-
- ValueError: 未找到该函数的接收器。
|
|
336
|
-
'''
|
|
337
|
-
|
|
338
|
-
i = 0
|
|
339
|
-
for receiver in self.receivers:
|
|
340
|
-
if receiver.fn == fn:
|
|
341
|
-
return i
|
|
342
|
-
i += 1
|
|
343
|
-
|
|
344
|
-
raise ValueError('未找到该函数的接收器')
|
|
345
|
-
|
|
346
|
-
@property
|
|
347
|
-
def receivers(self) -> List[Receiver]:
|
|
348
|
-
'''
|
|
349
|
-
【只读】 接收器的列表,按注册顺序排序。
|
|
350
|
-
'''
|
|
351
|
-
|
|
352
|
-
return self._receivers
|
|
353
|
-
|
|
354
|
-
@property
|
|
355
|
-
def total_send_num(self) -> int:
|
|
356
|
-
'''
|
|
357
|
-
【只读】 总计信号发送次数。
|
|
358
|
-
'''
|
|
359
|
-
|
|
360
|
-
return self._total_send_num
|
|
1
|
+
from CheeseSignal.signal import Receiver, Signal
|
CheeseSignal/signal.py
ADDED
|
@@ -0,0 +1,359 @@
|
|
|
1
|
+
import asyncio
|
|
2
|
+
from typing import List, Callable, overload, Any, TYPE_CHECKING, Literal, Tuple
|
|
3
|
+
|
|
4
|
+
if TYPE_CHECKING:
|
|
5
|
+
class Signal:
|
|
6
|
+
...
|
|
7
|
+
|
|
8
|
+
class Receiver:
|
|
9
|
+
def __init__(self, signal: 'Signal', fn: Callable, *, expected_receive_num: int, auto_remove: bool, runType: Literal['ORDERED', 'CONCURRENT', 'NO_WAIT'] = 'ORDERED'):
|
|
10
|
+
'''
|
|
11
|
+
- Args
|
|
12
|
+
- expected_receive_num: 期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
13
|
+
|
|
14
|
+
- auto_remove: 是否自动删除响应次数超出期望次数的接收器。
|
|
15
|
+
'''
|
|
16
|
+
|
|
17
|
+
self._signal: 'Signal' = signal
|
|
18
|
+
self.fn: Callable = fn
|
|
19
|
+
self._expected_receive_num: int = expected_receive_num
|
|
20
|
+
self._auto_remove: bool = auto_remove
|
|
21
|
+
self._total_receive_num: int = 0
|
|
22
|
+
self._active: bool = True
|
|
23
|
+
self._runType: Literal['ORDERED', 'CONCURRENT', 'NO_WAIT'] = runType
|
|
24
|
+
|
|
25
|
+
def reset(self):
|
|
26
|
+
'''
|
|
27
|
+
重统计置数据。
|
|
28
|
+
|
|
29
|
+
在有期望信号接受次数的情况下,`auto_remove is False`的接收器会重新开始计数并接收信号。
|
|
30
|
+
'''
|
|
31
|
+
|
|
32
|
+
self._total_receive_num = 0
|
|
33
|
+
|
|
34
|
+
@property
|
|
35
|
+
def expected_receive_num(self) -> int:
|
|
36
|
+
'''
|
|
37
|
+
期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
38
|
+
|
|
39
|
+
设置值小于`total_receive_num`且`auto_remove is True`,则会立刻删除。
|
|
40
|
+
'''
|
|
41
|
+
|
|
42
|
+
return self._expected_receive_num
|
|
43
|
+
|
|
44
|
+
@expected_receive_num.setter
|
|
45
|
+
def expected_receive_num(self, value: int):
|
|
46
|
+
self._expected_receive_num = value
|
|
47
|
+
|
|
48
|
+
if self._auto_remove and self.is_expired:
|
|
49
|
+
self._signal.receivers.remove(self)
|
|
50
|
+
|
|
51
|
+
@property
|
|
52
|
+
def auto_remove(self) -> bool:
|
|
53
|
+
'''
|
|
54
|
+
是否自动删除响应次数超出期望次数的接收器。
|
|
55
|
+
|
|
56
|
+
设置为`True`时若该receiver过期,则会立刻删除。
|
|
57
|
+
'''
|
|
58
|
+
|
|
59
|
+
return self._auto_remove
|
|
60
|
+
|
|
61
|
+
@auto_remove.setter
|
|
62
|
+
def auto_remove(self, value: bool):
|
|
63
|
+
self._auto_remove = value
|
|
64
|
+
|
|
65
|
+
if self._auto_remove and self.is_expired:
|
|
66
|
+
self._signal.receivers.remove(self)
|
|
67
|
+
|
|
68
|
+
@property
|
|
69
|
+
def active(self) -> bool:
|
|
70
|
+
'''
|
|
71
|
+
是否激活;未激活将忽略信号。
|
|
72
|
+
'''
|
|
73
|
+
|
|
74
|
+
return self._active
|
|
75
|
+
|
|
76
|
+
@active.setter
|
|
77
|
+
def active(self, value: bool):
|
|
78
|
+
self._active = value
|
|
79
|
+
|
|
80
|
+
@property
|
|
81
|
+
def total_receive_num(self) -> int:
|
|
82
|
+
'''
|
|
83
|
+
【只读】 总计信号接受次数。
|
|
84
|
+
'''
|
|
85
|
+
|
|
86
|
+
return self._total_receive_num
|
|
87
|
+
|
|
88
|
+
@property
|
|
89
|
+
def remaining_receive_num(self) -> int:
|
|
90
|
+
'''
|
|
91
|
+
【只读】 剩余的期望信号接受次数;返回为-1代表无期望信号接受次数。
|
|
92
|
+
'''
|
|
93
|
+
|
|
94
|
+
if not self.expected_receive_num:
|
|
95
|
+
return -1
|
|
96
|
+
return self.expected_receive_num - self.total_receive_num
|
|
97
|
+
|
|
98
|
+
@property
|
|
99
|
+
def is_expired(self) -> bool:
|
|
100
|
+
'''
|
|
101
|
+
【只读】 是否过期。
|
|
102
|
+
'''
|
|
103
|
+
|
|
104
|
+
return not self.is_unexpired
|
|
105
|
+
|
|
106
|
+
@property
|
|
107
|
+
def is_unexpired(self) -> bool:
|
|
108
|
+
'''
|
|
109
|
+
【只读】 是否未过期。
|
|
110
|
+
'''
|
|
111
|
+
|
|
112
|
+
if self.remaining_receive_num == -1:
|
|
113
|
+
return True
|
|
114
|
+
return True if self.remaining_receive_num else False
|
|
115
|
+
|
|
116
|
+
@property
|
|
117
|
+
def runType(self) -> Literal['ORDERED', 'CONCURRENT', 'NO_WAIT']:
|
|
118
|
+
'''
|
|
119
|
+
【只读】 运行的方式,仅在`async_send`时有效。
|
|
120
|
+
|
|
121
|
+
- ORDERED: 按顺序执行,返回结果。
|
|
122
|
+
|
|
123
|
+
- CONCURRENT: 并行执行,返回结果。
|
|
124
|
+
|
|
125
|
+
- NO_WAIT: 并行执行,不阻塞代码。
|
|
126
|
+
'''
|
|
127
|
+
|
|
128
|
+
return self._runType
|
|
129
|
+
|
|
130
|
+
@runType.setter
|
|
131
|
+
def runType(self, value: Literal['ORDERED', 'CONCURRENT', 'NO_WAIT']):
|
|
132
|
+
self._runType = value
|
|
133
|
+
|
|
134
|
+
class Signal:
|
|
135
|
+
def __init__(self):
|
|
136
|
+
self._receivers: List[Receiver] = []
|
|
137
|
+
self._total_send_num: int = 0
|
|
138
|
+
|
|
139
|
+
@overload
|
|
140
|
+
def connect(self, fn: Callable, *, expected_receive_num: int = 0, auto_remove: bool = False, runType: Literal['ORDERED', 'CONCURRENT', 'NO_WAIT'] = 'ORDERED'):
|
|
141
|
+
'''
|
|
142
|
+
通过函数注册响应函数。
|
|
143
|
+
|
|
144
|
+
>>> from CheeseSignal import Signal
|
|
145
|
+
>>>
|
|
146
|
+
>>> def receiver(*args, **kwargs):
|
|
147
|
+
... ...
|
|
148
|
+
>>>
|
|
149
|
+
>>> signal = Signal()
|
|
150
|
+
>>> signal.connect(receiver)
|
|
151
|
+
|
|
152
|
+
- Args
|
|
153
|
+
- expected_receive_num: 期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
154
|
+
|
|
155
|
+
- auto_remove: 是否自动删除响应次数超出期望次数的接收器。
|
|
156
|
+
|
|
157
|
+
- Raise
|
|
158
|
+
- ValueError: 已有重复的函数接收器。
|
|
159
|
+
'''
|
|
160
|
+
|
|
161
|
+
@overload
|
|
162
|
+
def connect(self, *, expected_receive_num: int = 0, auto_remove: bool = False):
|
|
163
|
+
'''
|
|
164
|
+
通过装饰器注册响应函数。
|
|
165
|
+
|
|
166
|
+
>>> from CheeseSignal import Signal
|
|
167
|
+
>>>
|
|
168
|
+
>>> signal = Signal()
|
|
169
|
+
>>>
|
|
170
|
+
>>> @signal.connect()
|
|
171
|
+
>>> def receiver(*args, **kwargs):
|
|
172
|
+
... ...
|
|
173
|
+
|
|
174
|
+
- Args
|
|
175
|
+
- expected_receive_num: 期望接受信号的次数,超过该次数则不再响应信号;0为无限次。
|
|
176
|
+
|
|
177
|
+
- auto_remove: 是否自动删除响应次数超出期望次数的接收器。
|
|
178
|
+
|
|
179
|
+
- Raise
|
|
180
|
+
- ValueError: 已有重复的函数接收器。
|
|
181
|
+
'''
|
|
182
|
+
|
|
183
|
+
def connect(self, arg1: Callable | None = None, *, expected_receive_num: int = 0, auto_remove: bool = False, runType: Literal['ORDERED', 'CONCURRENT', 'NO_WAIT'] = 'ORDERED'):
|
|
184
|
+
if not arg1:
|
|
185
|
+
def wrapper(fn):
|
|
186
|
+
self._connect(fn, expected_receive_num = expected_receive_num, auto_remove = auto_remove, runType = runType)
|
|
187
|
+
return fn
|
|
188
|
+
return wrapper
|
|
189
|
+
|
|
190
|
+
self._connect(arg1, expected_receive_num = expected_receive_num, auto_remove = auto_remove, runType = runType)
|
|
191
|
+
|
|
192
|
+
def _connect(self, fn: Callable, *, expected_receive_num: int, auto_remove: bool, runType: Literal['ORDERED', 'CONCURRENT', 'NO_WAIT']):
|
|
193
|
+
for receiver in self.receivers:
|
|
194
|
+
if receiver.fn == fn:
|
|
195
|
+
raise ValueError('已有重复的函数接收器')
|
|
196
|
+
|
|
197
|
+
self.receivers.append(Receiver(self, fn, expected_receive_num = expected_receive_num, auto_remove = auto_remove, runType = runType))
|
|
198
|
+
|
|
199
|
+
def send(self, *args, **kwargs) -> List[Any]:
|
|
200
|
+
'''
|
|
201
|
+
发送信号。
|
|
202
|
+
|
|
203
|
+
>>> from CheeseSignal import Signal
|
|
204
|
+
>>>
|
|
205
|
+
>>> signal = Signal()
|
|
206
|
+
>>> signal.send('data1', 'data2', **{
|
|
207
|
+
... 'key1': 'value1',
|
|
208
|
+
... 'key2': 'value2'
|
|
209
|
+
>>> })
|
|
210
|
+
'''
|
|
211
|
+
|
|
212
|
+
self._total_send_num += 1
|
|
213
|
+
|
|
214
|
+
results = []
|
|
215
|
+
for receiver in self.receivers[:]:
|
|
216
|
+
if receiver.active and receiver.is_unexpired:
|
|
217
|
+
receiver._total_receive_num += 1
|
|
218
|
+
results.append(receiver.fn(*args, **kwargs))
|
|
219
|
+
|
|
220
|
+
if receiver.is_expired:
|
|
221
|
+
self.receivers.remove(receiver)
|
|
222
|
+
return results
|
|
223
|
+
|
|
224
|
+
async def async_send(self, *args, **kwargs) -> Tuple[List[Any], List[Any]]:
|
|
225
|
+
'''
|
|
226
|
+
在协程环境中发送信号,并请保证所有接收函数都是协程函数。
|
|
227
|
+
|
|
228
|
+
>>> import asyncio
|
|
229
|
+
>>>
|
|
230
|
+
>>> from CheeseSignal import Signal
|
|
231
|
+
>>>
|
|
232
|
+
>>> async def run_asyncio():
|
|
233
|
+
... signal = Signal()
|
|
234
|
+
... await signal.async_send('data1', 'data2', **{
|
|
235
|
+
... 'key1': 'value1',
|
|
236
|
+
... 'key2': 'value2'
|
|
237
|
+
... })
|
|
238
|
+
>>>
|
|
239
|
+
>>> asyncio.run(run_asyncio())
|
|
240
|
+
|
|
241
|
+
- Returns
|
|
242
|
+
返回长度为2的元祖,分别为ORDERED类型和CONCURRENT类型的结果。
|
|
243
|
+
'''
|
|
244
|
+
|
|
245
|
+
self._total_send_num += 1
|
|
246
|
+
receivers = [receiver for receiver in self.receivers.copy() if receiver.active and receiver.is_unexpired]
|
|
247
|
+
|
|
248
|
+
[asyncio.create_task(receiver.fn(*args, **kwargs)) for receiver in receivers if receiver.runType == 'NO_WAIT']
|
|
249
|
+
concurrent_tasks = [receiver.fn(*args, **kwargs) for receiver in receivers if receiver.runType == 'CONCURRENT']
|
|
250
|
+
concurrent_results = await asyncio.gather(*concurrent_tasks) if concurrent_tasks else []
|
|
251
|
+
ordered_results = [await receiver.fn(*args, **kwargs) for receiver in receivers if receiver.runType == 'ORDERED']
|
|
252
|
+
|
|
253
|
+
return concurrent_results, ordered_results
|
|
254
|
+
|
|
255
|
+
def disconnect(self, fn: Callable):
|
|
256
|
+
'''
|
|
257
|
+
断开接收器。
|
|
258
|
+
|
|
259
|
+
>>> from CheeseSignal import Signal
|
|
260
|
+
>>>
|
|
261
|
+
>>> def receiver(*args, **kwargs):
|
|
262
|
+
... ...
|
|
263
|
+
>>>
|
|
264
|
+
>>> signal = Signal()
|
|
265
|
+
>>> signal.connect(receiver)
|
|
266
|
+
>>> signal.disconnect(receiver)
|
|
267
|
+
|
|
268
|
+
- Raise
|
|
269
|
+
- ValueError: 未找到该函数的接收器。
|
|
270
|
+
'''
|
|
271
|
+
|
|
272
|
+
for receiver in self.receivers:
|
|
273
|
+
if receiver.fn == fn:
|
|
274
|
+
self.receivers.remove(receiver)
|
|
275
|
+
return
|
|
276
|
+
|
|
277
|
+
raise ValueError('未找到该函数的接收器')
|
|
278
|
+
|
|
279
|
+
def reset(self):
|
|
280
|
+
'''
|
|
281
|
+
重置统计数据;所有的接收器的统计数据也会同步清空。
|
|
282
|
+
'''
|
|
283
|
+
|
|
284
|
+
self._total_send_num = 0
|
|
285
|
+
for receiver in self.receivers:
|
|
286
|
+
receiver.reset()
|
|
287
|
+
|
|
288
|
+
def disconnect_all(self):
|
|
289
|
+
'''
|
|
290
|
+
断开所有接收器。
|
|
291
|
+
'''
|
|
292
|
+
|
|
293
|
+
self._receivers = []
|
|
294
|
+
|
|
295
|
+
def get_receiver(self, fn: Callable) -> Receiver:
|
|
296
|
+
'''
|
|
297
|
+
获取接收器。
|
|
298
|
+
|
|
299
|
+
>>> from CheeseSignal import Signal
|
|
300
|
+
>>>
|
|
301
|
+
>>> def receiver(*args, **kwargs):
|
|
302
|
+
... ...
|
|
303
|
+
>>>
|
|
304
|
+
>>> signal = Signal()
|
|
305
|
+
>>> signal.connect(receiver)
|
|
306
|
+
>>>
|
|
307
|
+
>>> print(signal.get_receiver(receiver))
|
|
308
|
+
|
|
309
|
+
- Raise
|
|
310
|
+
- ValueError: 未找到该函数的接收器。
|
|
311
|
+
'''
|
|
312
|
+
|
|
313
|
+
for receiver in self.receivers:
|
|
314
|
+
if receiver.fn == fn:
|
|
315
|
+
return receiver
|
|
316
|
+
|
|
317
|
+
raise ValueError('未找到该函数的接收器')
|
|
318
|
+
|
|
319
|
+
def index(self, fn: Callable) -> int:
|
|
320
|
+
'''
|
|
321
|
+
获取接收器的顺序位置。
|
|
322
|
+
|
|
323
|
+
>>> from CheeseSignal import Signal
|
|
324
|
+
>>>
|
|
325
|
+
>>> def receiver(*args, **kwargs):
|
|
326
|
+
... ...
|
|
327
|
+
>>>
|
|
328
|
+
>>> signal = Signal()
|
|
329
|
+
>>> signal.connect(receiver)
|
|
330
|
+
>>>
|
|
331
|
+
>>> print(signal.index(receiver))
|
|
332
|
+
|
|
333
|
+
- Raise
|
|
334
|
+
- ValueError: 未找到该函数的接收器。
|
|
335
|
+
'''
|
|
336
|
+
|
|
337
|
+
i = 0
|
|
338
|
+
for receiver in self.receivers:
|
|
339
|
+
if receiver.fn == fn:
|
|
340
|
+
return i
|
|
341
|
+
i += 1
|
|
342
|
+
|
|
343
|
+
raise ValueError('未找到该函数的接收器')
|
|
344
|
+
|
|
345
|
+
@property
|
|
346
|
+
def receivers(self) -> List[Receiver]:
|
|
347
|
+
'''
|
|
348
|
+
【只读】 接收器的列表,按注册顺序排序。
|
|
349
|
+
'''
|
|
350
|
+
|
|
351
|
+
return self._receivers
|
|
352
|
+
|
|
353
|
+
@property
|
|
354
|
+
def total_send_num(self) -> int:
|
|
355
|
+
'''
|
|
356
|
+
【只读】 总计信号发送次数。
|
|
357
|
+
'''
|
|
358
|
+
|
|
359
|
+
return self._total_send_num
|
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
Metadata-Version: 2.3
|
|
2
2
|
Name: CheeseSignal
|
|
3
|
-
Version: 1.
|
|
3
|
+
Version: 1.2.0
|
|
4
4
|
Summary: 一款简约的信号系统。
|
|
5
5
|
Project-URL: Source, https://github.com/CheeseUnknown/CheeseSignal
|
|
6
6
|
Author-email: Cheese Unknown <cheese@cheese.ren>
|
|
7
7
|
License-File: LICENSE
|
|
8
|
-
Keywords:
|
|
8
|
+
Keywords: Signal
|
|
9
9
|
Description-Content-Type: text/markdown
|
|
10
10
|
|
|
11
11
|
# **CheeseSignal**
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
CheeseSignal/__init__.py,sha256=--LvG_YyRo9N3ubHlhunm4FiiMD60hM2LsdB9uiprPM,49
|
|
2
|
+
CheeseSignal/signal.py,sha256=sxPyXtRIxgt23uTxn2SQjoF1QSIeRZxqz1MGwzij7to,10842
|
|
3
|
+
cheesesignal-1.2.0.dist-info/METADATA,sha256=09TA7w_cQcne48pQILYiOrSl-5K0sjat4KVUvo7jvuI,5141
|
|
4
|
+
cheesesignal-1.2.0.dist-info/WHEEL,sha256=fl6v0VwpzfGBVsGtkAkhILUlJxROXbA3HvRL6Fe3140,105
|
|
5
|
+
cheesesignal-1.2.0.dist-info/licenses/LICENSE,sha256=5vFb3i4UDlskszJ3jGPh8bXrM_axJfDRRuvLu1M3bIs,1070
|
|
6
|
+
cheesesignal-1.2.0.dist-info/RECORD,,
|
|
@@ -1,5 +0,0 @@
|
|
|
1
|
-
CheeseSignal/__init__.py,sha256=UNrN77xeVfj6YkPgPKjgZo9DjodL_uJ_mW6EjTKrLHM,9397
|
|
2
|
-
cheesesignal-1.1.0.dist-info/METADATA,sha256=XgScoe7qZ45H2-zhXWupDgI-YohJmqNLpDwx2kwf1ic,5146
|
|
3
|
-
cheesesignal-1.1.0.dist-info/WHEEL,sha256=cDcbFFSNXOE-241I5PFuLkIYfR_FM7WTlPEi33njInY,105
|
|
4
|
-
cheesesignal-1.1.0.dist-info/licenses/LICENSE,sha256=5vFb3i4UDlskszJ3jGPh8bXrM_axJfDRRuvLu1M3bIs,1070
|
|
5
|
-
cheesesignal-1.1.0.dist-info/RECORD,,
|
|
File without changes
|