omlish 0.0.0.dev161__py3-none-any.whl → 0.0.0.dev163__py3-none-any.whl

Sign up to get free protection for your applications and to get access to all the features.
Files changed (45) hide show
  1. omlish/.manifests.json +228 -0
  2. omlish/__about__.py +2 -2
  3. omlish/asyncs/bluelet/LICENSE +6 -0
  4. omlish/asyncs/bluelet/__init__.py +0 -0
  5. omlish/asyncs/bluelet/all.py +67 -0
  6. omlish/asyncs/bluelet/api.py +23 -0
  7. omlish/asyncs/bluelet/core.py +178 -0
  8. omlish/asyncs/bluelet/events.py +78 -0
  9. omlish/asyncs/bluelet/files.py +80 -0
  10. omlish/asyncs/bluelet/runner.py +416 -0
  11. omlish/asyncs/bluelet/sockets.py +214 -0
  12. omlish/codecs/__init__.py +69 -0
  13. omlish/codecs/base.py +102 -0
  14. omlish/codecs/bytes.py +119 -0
  15. omlish/codecs/chain.py +23 -0
  16. omlish/codecs/funcs.py +28 -0
  17. omlish/codecs/registry.py +139 -0
  18. omlish/codecs/standard.py +4 -0
  19. omlish/codecs/text.py +217 -0
  20. omlish/formats/cbor.py +31 -0
  21. omlish/formats/codecs.py +93 -0
  22. omlish/formats/json/codecs.py +33 -0
  23. omlish/formats/json5.py +31 -0
  24. omlish/formats/pickle.py +31 -0
  25. omlish/formats/toml.py +17 -0
  26. omlish/formats/yaml.py +18 -0
  27. omlish/io/compress/brotli.py +15 -1
  28. omlish/io/compress/bz2.py +14 -0
  29. omlish/io/compress/codecs.py +58 -0
  30. omlish/io/compress/gzip.py +11 -0
  31. omlish/io/compress/lz4.py +14 -0
  32. omlish/io/compress/lzma.py +14 -0
  33. omlish/io/compress/snappy.py +14 -0
  34. omlish/io/compress/zlib.py +14 -0
  35. omlish/io/compress/zstd.py +14 -0
  36. omlish/lang/__init__.py +1 -0
  37. omlish/lang/functions.py +11 -0
  38. omlish/lite/inject.py +16 -29
  39. omlish/manifests/load.py +44 -6
  40. {omlish-0.0.0.dev161.dist-info → omlish-0.0.0.dev163.dist-info}/METADATA +1 -1
  41. {omlish-0.0.0.dev161.dist-info → omlish-0.0.0.dev163.dist-info}/RECORD +45 -21
  42. {omlish-0.0.0.dev161.dist-info → omlish-0.0.0.dev163.dist-info}/LICENSE +0 -0
  43. {omlish-0.0.0.dev161.dist-info → omlish-0.0.0.dev163.dist-info}/WHEEL +0 -0
  44. {omlish-0.0.0.dev161.dist-info → omlish-0.0.0.dev163.dist-info}/entry_points.txt +0 -0
  45. {omlish-0.0.0.dev161.dist-info → omlish-0.0.0.dev163.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,416 @@
1
+ # ruff: noqa: UP006 UP007
2
+ # @omlish-lite
3
+ # Based on bluelet ( https://github.com/sampsyo/bluelet ) by Adrian Sampson, original license:
4
+ # THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
5
+ # WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
6
+ # COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
7
+ # OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
8
+ """
9
+ TODO:
10
+ - use fdio
11
+ - wrap coros in Tasks :|
12
+ - (unit)tests lol
13
+ - * subprocesses
14
+ - canceling
15
+ - timeouts
16
+ - task groups
17
+ - gather
18
+ - locks / semaphores / events / etc
19
+ - rename Coro to Bluelet?
20
+ - shutdown
21
+ - ensure resource cleanup
22
+ - run_thread? whatever?
23
+
24
+ Subprocs:
25
+ - https://github.com/python/cpython/issues/120804 - GH-120804: Remove get_child_watcher and set_child_watcher from
26
+ asyncio
27
+ - https://github.com/python/cpython/pull/17063/files bpo-38692: Add os.pidfd_open
28
+ - clone PidfdChildWatcher + ThreadedChildWatcher
29
+ """
30
+ import collections
31
+ import dataclasses as dc
32
+ import errno
33
+ import logging
34
+ import select
35
+ import sys
36
+ import time
37
+ import traceback
38
+ import types
39
+ import typing as ta
40
+ import weakref
41
+
42
+ from .core import BlueletCoro
43
+ from .core import BlueletExcInfo
44
+ from .core import CoreBlueletEvent
45
+ from .core import DelegationBlueletEvent
46
+ from .core import ExceptionBlueletEvent
47
+ from .core import JoinBlueletEvent
48
+ from .core import KillBlueletEvent
49
+ from .core import ReturnBlueletEvent
50
+ from .core import SleepBlueletEvent
51
+ from .core import SpawnBlueletEvent
52
+ from .core import ValueBlueletEvent
53
+ from .core import _BlueletAwaitableDriver
54
+ from .events import BlueletEvent
55
+ from .events import BlueletWaitable
56
+ from .events import WaitableBlueletEvent
57
+
58
+
59
+ ##
60
+
61
+
62
+ class BlueletCoroException(Exception): # noqa
63
+ def __init__(self, coro: BlueletCoro, exc_info: BlueletExcInfo) -> None:
64
+ super().__init__()
65
+ self.coro = coro
66
+ self.exc_info = exc_info
67
+
68
+ @staticmethod
69
+ def _exc_info() -> BlueletExcInfo:
70
+ return sys.exc_info() # type: ignore
71
+
72
+ @staticmethod
73
+ def _reraise(typ: ta.Type[BaseException], exc: BaseException, tb: types.TracebackType) -> ta.NoReturn: # noqa
74
+ raise exc.with_traceback(tb)
75
+
76
+ def reraise(self) -> ta.NoReturn:
77
+ self._reraise(self.exc_info[0], self.exc_info[1], self.exc_info[2])
78
+
79
+
80
+ ##
81
+
82
+
83
+ def _bluelet_event_select(
84
+ events: ta.Iterable[BlueletEvent],
85
+ *,
86
+ log: ta.Optional[logging.Logger] = None,
87
+ ) -> ta.Set[WaitableBlueletEvent]:
88
+ """
89
+ Perform a select() over all the Events provided, returning the ones ready to be fired. Only WaitableEvents
90
+ (including SleepEvents) matter here; all other events are ignored (and thus postponed).
91
+ """
92
+
93
+ waitable_to_event: ta.Dict[ta.Tuple[str, BlueletWaitable], WaitableBlueletEvent] = {}
94
+ rlist: ta.List[BlueletWaitable] = []
95
+ wlist: ta.List[BlueletWaitable] = []
96
+ xlist: ta.List[BlueletWaitable] = []
97
+ earliest_wakeup: ta.Optional[float] = None
98
+
99
+ # Gather waitables and wakeup times.
100
+ for event in events:
101
+ if isinstance(event, SleepBlueletEvent):
102
+ if not earliest_wakeup:
103
+ earliest_wakeup = event.wakeup_time
104
+ else:
105
+ earliest_wakeup = min(earliest_wakeup, event.wakeup_time)
106
+
107
+ elif isinstance(event, WaitableBlueletEvent):
108
+ ew = event.waitables()
109
+ rlist.extend(ew.r)
110
+ wlist.extend(ew.w)
111
+ xlist.extend(ew.x)
112
+ for waitable in ew.r:
113
+ waitable_to_event[('r', waitable)] = event
114
+ for waitable in ew.w:
115
+ waitable_to_event[('w', waitable)] = event
116
+ for waitable in ew.x:
117
+ waitable_to_event[('x', waitable)] = event
118
+
119
+ # If we have a any sleeping coros, determine how long to sleep.
120
+ if earliest_wakeup:
121
+ timeout = max(earliest_wakeup - time.time(), 0.)
122
+ else:
123
+ timeout = None
124
+
125
+ # Perform select() if we have any waitables.
126
+ if rlist or wlist or xlist:
127
+ if log:
128
+ log.debug('_bluelet_event_select: +select: %r %r %r %r', rlist, wlist, xlist, timeout)
129
+ rready, wready, xready = select.select(rlist, wlist, xlist, timeout)
130
+ if log:
131
+ log.debug('_bluelet_event_select: -select: %r %r %r', rready, wready, xready)
132
+
133
+ else:
134
+ rready, wready, xready = [], [], []
135
+ if timeout:
136
+ if log:
137
+ log.debug('_bluelet_event_select: sleep: %r', timeout)
138
+ time.sleep(timeout)
139
+
140
+ # Gather ready events corresponding to the ready waitables.
141
+ ready_events: ta.Set[WaitableBlueletEvent] = set()
142
+ for ready in rready:
143
+ ready_events.add(waitable_to_event[('r', ready)])
144
+ for ready in wready:
145
+ ready_events.add(waitable_to_event[('w', ready)])
146
+ for ready in xready:
147
+ ready_events.add(waitable_to_event[('x', ready)])
148
+
149
+ # Gather any finished sleeps.
150
+ for event in events:
151
+ if isinstance(event, SleepBlueletEvent) and not event.time_left():
152
+ ready_events.add(event)
153
+
154
+ return ready_events
155
+
156
+
157
+ ##
158
+
159
+
160
+ class _SuspendedBlueletEvent(CoreBlueletEvent):
161
+ pass
162
+
163
+
164
+ _BLUELET_SUSPENDED = _SuspendedBlueletEvent() # Special sentinel placeholder for suspended coros.
165
+
166
+
167
+ @dc.dataclass(frozen=True, eq=False)
168
+ class _DelegatedBlueletEvent(CoreBlueletEvent):
169
+ """Placeholder indicating that a coro has delegated execution to a different coro."""
170
+
171
+ child: BlueletCoro
172
+
173
+
174
+ class _BlueletRunner:
175
+ """
176
+ Schedules a coroutine, running it to completion. This encapsulates the Bluelet scheduler, which the root coroutine
177
+ can add to by spawning new coroutines.
178
+ """
179
+
180
+ def __init__(
181
+ self,
182
+ root_coro: BlueletCoro,
183
+ *,
184
+ log: ta.Optional[logging.Logger] = None,
185
+ ) -> None:
186
+ super().__init__()
187
+
188
+ self._root_coro = root_coro
189
+ self._log = log
190
+
191
+ # The "coros" dictionary keeps track of all the currently-executing and suspended coroutines. It maps
192
+ # coroutines to their currently "blocking" event. The event value may be SUSPENDED if the coroutine is waiting
193
+ # on some other condition: namely, a delegated coroutine or a joined coroutine. In this case, the coroutine
194
+ # should *also* appear as a value in one of the below dictionaries `delegators` or `joiners`.
195
+ self._coros: ta.Dict[BlueletCoro, BlueletEvent] = {self._root_coro: ValueBlueletEvent(None)}
196
+
197
+ # Maps child coroutines to delegating parents.
198
+ self._delegators: ta.Dict[BlueletCoro, BlueletCoro] = {}
199
+
200
+ # Maps child coroutines to joining (exit-waiting) parents.
201
+ self._joiners: ta.MutableMapping[BlueletCoro, ta.List[BlueletCoro]] = collections.defaultdict(list)
202
+
203
+ # History of spawned coroutines for joining of already completed coroutines.
204
+ self._history: ta.MutableMapping[BlueletCoro, ta.Optional[BlueletEvent]] = \
205
+ weakref.WeakKeyDictionary({self._root_coro: None})
206
+
207
+ def _complete_coro(self, coro: BlueletCoro, return_value: ta.Any) -> None:
208
+ """
209
+ Remove a coroutine from the scheduling pool, awaking delegators and joiners as necessary and returning the
210
+ specified value to any delegating parent.
211
+ """
212
+
213
+ del self._coros[coro]
214
+
215
+ # Resume delegator.
216
+ if coro in self._delegators:
217
+ self._coros[self._delegators[coro]] = ValueBlueletEvent(return_value)
218
+ del self._delegators[coro]
219
+
220
+ # Resume joiners.
221
+ if coro in self._joiners:
222
+ for parent in self._joiners[coro]:
223
+ self._coros[parent] = ValueBlueletEvent(None)
224
+ del self._joiners[coro]
225
+
226
+ def _advance_coro(self, coro: BlueletCoro, value: ta.Any, is_exc: bool = False) -> None:
227
+ """
228
+ After an event is fired, run a given coroutine associated with it in the coros dict until it yields again. If
229
+ the coroutine exits, then the coro is removed from the pool. If the coroutine raises an exception, it is
230
+ reraised in a CoroException. If is_exc is True, then the value must be an exc_info tuple and the exception is
231
+ thrown into the coroutine.
232
+ """
233
+
234
+ try:
235
+ if is_exc:
236
+ next_event = coro.throw(*value)
237
+ else:
238
+ next_event = coro.send(value)
239
+
240
+ except StopIteration:
241
+ # Coro is done.
242
+ self._complete_coro(coro, None)
243
+
244
+ except BaseException: # noqa
245
+ # Coro raised some other exception.
246
+ del self._coros[coro]
247
+ # Note: Don't use `raise from` as this should support 3.8.
248
+ raise BlueletCoroException(coro, BlueletCoroException._exc_info()) # noqa
249
+
250
+ else:
251
+ if isinstance(next_event, ta.Generator):
252
+ # Automatically invoke sub-coroutines. (Shorthand for explicit bluelet.call().)
253
+ next_event = DelegationBlueletEvent(next_event)
254
+
255
+ if isinstance(next_event, types.CoroutineType): # type: ignore[unreachable]
256
+ next_event = DelegationBlueletEvent(_BlueletAwaitableDriver(next_event)()) # type: ignore[unreachable]
257
+
258
+ if not isinstance(next_event, BlueletEvent):
259
+ raise TypeError(next_event)
260
+
261
+ self._coros[coro] = next_event
262
+
263
+ def _kill_coro(self, coro: BlueletCoro) -> None:
264
+ """Unschedule this coro and its (recursive) delegates."""
265
+
266
+ # Collect all coroutines in the delegation stack.
267
+ coros = [coro]
268
+ while isinstance((cur := self._coros[coro]), _DelegatedBlueletEvent):
269
+ coro = cur.child # noqa
270
+ coros.append(coro)
271
+
272
+ # Complete each coroutine from the top to the bottom of the stack.
273
+ for coro in reversed(coros):
274
+ self._complete_coro(coro, None)
275
+
276
+ def close(self) -> None:
277
+ # If any coros still remain, kill them.
278
+ for coro in self._coros:
279
+ coro.close()
280
+
281
+ self._coros.clear()
282
+
283
+ def _handle_core_event(self, coro: BlueletCoro, event: CoreBlueletEvent) -> bool:
284
+ if self._log:
285
+ self._log.debug(f'{self.__class__.__name__}._handle_core_event: %r %r', coro, event)
286
+
287
+ if isinstance(event, SpawnBlueletEvent):
288
+ sc = ta.cast(BlueletCoro, event.spawned) # FIXME
289
+ self._coros[sc] = ValueBlueletEvent(None) # Spawn.
290
+ self._history[sc] = None # Record in history.
291
+ self._advance_coro(coro, None)
292
+ return True
293
+
294
+ elif isinstance(event, ValueBlueletEvent):
295
+ self._advance_coro(coro, event.value)
296
+ return True
297
+
298
+ elif isinstance(event, ExceptionBlueletEvent):
299
+ self._advance_coro(coro, event.exc_info, True)
300
+ return True
301
+
302
+ elif isinstance(event, DelegationBlueletEvent):
303
+ self._coros[coro] = _DelegatedBlueletEvent(event.spawned) # Suspend.
304
+ self._coros[event.spawned] = ValueBlueletEvent(None) # Spawn.
305
+ self._history[event.spawned] = None # Record in history.
306
+ self._delegators[event.spawned] = coro
307
+ return True
308
+
309
+ elif isinstance(event, ReturnBlueletEvent):
310
+ # Coro is done.
311
+ self._complete_coro(coro, event.value)
312
+ return True
313
+
314
+ elif isinstance(event, JoinBlueletEvent):
315
+ if event.child not in self._coros and event.child in self._history:
316
+ self._coros[coro] = ValueBlueletEvent(None)
317
+ else:
318
+ self._coros[coro] = _BLUELET_SUSPENDED # Suspend.
319
+ self._joiners[event.child].append(coro)
320
+ return True
321
+
322
+ elif isinstance(event, KillBlueletEvent):
323
+ self._coros[coro] = ValueBlueletEvent(None)
324
+ self._kill_coro(event.child)
325
+ return True
326
+
327
+ elif isinstance(event, (_DelegatedBlueletEvent, _SuspendedBlueletEvent)):
328
+ return False
329
+
330
+ else:
331
+ raise TypeError(event)
332
+
333
+ def _step(self) -> ta.Optional[BlueletCoroException]:
334
+ if self._log:
335
+ self._log.debug(f'{self.__class__.__name__}._step') # Noqa
336
+
337
+ try:
338
+ # Look for events that can be run immediately. Continue running immediate events until nothing is ready.
339
+ while True:
340
+ have_ready = False
341
+ for coro, event in list(self._coros.items()):
342
+ if isinstance(event, CoreBlueletEvent) and not isinstance(event, SleepBlueletEvent):
343
+ have_ready |= self._handle_core_event(coro, event)
344
+ elif isinstance(event, WaitableBlueletEvent):
345
+ pass
346
+ else:
347
+ raise TypeError(f'Unknown event type: {event}') # noqa
348
+
349
+ # Only start the select when nothing else is ready.
350
+ if not have_ready:
351
+ break
352
+
353
+ # Wait and fire.
354
+ event2coro = {v: k for k, v in self._coros.items()}
355
+ for event in _bluelet_event_select(self._coros.values()):
356
+ # Run the IO operation, but catch socket errors.
357
+ try:
358
+ value = event.fire()
359
+ except OSError as exc:
360
+ if isinstance(exc.args, tuple) and exc.args[0] == errno.EPIPE:
361
+ # Broken pipe. Remote host disconnected.
362
+ pass
363
+ elif isinstance(exc.args, tuple) and exc.args[0] == errno.ECONNRESET:
364
+ # Connection was reset by peer.
365
+ pass
366
+ else:
367
+ traceback.print_exc()
368
+ # Abort the coroutine.
369
+ self._coros[event2coro[event]] = ReturnBlueletEvent(None)
370
+ else:
371
+ self._advance_coro(event2coro[event], value)
372
+
373
+ except BlueletCoroException as te:
374
+ if self._log and self._log.isEnabledFor(logging.DEBUG):
375
+ self._log.exception(f'{self.__class__.__name__}._step')
376
+
377
+ # Exception raised from inside a coro.
378
+ event = ExceptionBlueletEvent(te.exc_info)
379
+ if te.coro in self._delegators:
380
+ # The coro is a delegate. Raise exception in its delegator.
381
+ self._coros[self._delegators[te.coro]] = event
382
+ del self._delegators[te.coro]
383
+ else:
384
+ # The coro is root-level. Raise in client code.
385
+ return te
386
+
387
+ except BaseException: # noqa
388
+ ei = BlueletCoroException._exc_info() # noqa
389
+
390
+ if self._log and self._log.isEnabledFor(logging.DEBUG):
391
+ self._log.exception(f'{self.__class__.__name__}._step')
392
+
393
+ # For instance, KeyboardInterrupt during select(). Raise into root coro and terminate others.
394
+ self._coros = {self._root_coro: ExceptionBlueletEvent(ei)} # noqa
395
+
396
+ return None
397
+
398
+ def run(self) -> None:
399
+ # Continue advancing coros until root coro exits.
400
+ exit_ce: BlueletCoroException | None = None
401
+ while self._coros:
402
+ exit_ce = self._step()
403
+
404
+ self.close()
405
+
406
+ # If we're exiting with an exception, raise it in the client.
407
+ if exit_ce:
408
+ exit_ce.reraise()
409
+
410
+
411
+ ##
412
+
413
+
414
+ class _RunnerBlueletApi:
415
+ def run(self, root_coro: BlueletCoro) -> None:
416
+ _BlueletRunner(root_coro).run()
@@ -0,0 +1,214 @@
1
+ # ruff: noqa: UP006 UP007
2
+ # @omlish-lite
3
+ # Based on bluelet ( https://github.com/sampsyo/bluelet ) by Adrian Sampson, original license:
4
+ # THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
5
+ # WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
6
+ # COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
7
+ # OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
8
+ import abc
9
+ import dataclasses as dc
10
+ import socket
11
+ import typing as ta
12
+
13
+ from .core import BlueletCoro
14
+ from .core import ReturnBlueletEvent
15
+ from .core import ValueBlueletEvent
16
+ from .core import _CoreBlueletApi
17
+ from .events import BlueletEvent
18
+ from .events import BlueletWaitables
19
+ from .events import WaitableBlueletEvent
20
+
21
+
22
+ ##
23
+
24
+
25
+ class SocketClosedBlueletError(Exception):
26
+ pass
27
+
28
+
29
+ class BlueletListener:
30
+ """A socket wrapper object for listening sockets."""
31
+
32
+ def __init__(self, host: str, port: int) -> None:
33
+ """Create a listening socket on the given hostname and port."""
34
+
35
+ super().__init__()
36
+ self._closed = False
37
+ self.host = host
38
+ self.port = port
39
+
40
+ self.sock = sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
41
+ sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
42
+ sock.bind((host, port))
43
+ sock.listen(5)
44
+
45
+ def accept(self) -> 'AcceptBlueletEvent':
46
+ """
47
+ An event that waits for a connection on the listening socket. When a connection is made, the event returns a
48
+ Connection object.
49
+ """
50
+
51
+ if self._closed:
52
+ raise SocketClosedBlueletError
53
+ return AcceptBlueletEvent(self)
54
+
55
+ def close(self) -> None:
56
+ """Immediately close the listening socket. (Not an event.)"""
57
+
58
+ self._closed = True
59
+ self.sock.close()
60
+
61
+
62
+ class BlueletConnection:
63
+ """A socket wrapper object for connected sockets."""
64
+
65
+ def __init__(self, sock: socket.socket, addr: ta.Tuple[str, int]) -> None:
66
+ super().__init__()
67
+ self.sock = sock
68
+ self.addr = addr
69
+ self._buf = bytearray()
70
+ self._closed: bool = False
71
+
72
+ def close(self) -> None:
73
+ """Close the connection."""
74
+
75
+ self._closed = True
76
+ self.sock.close()
77
+
78
+ def recv(self, size: int) -> BlueletEvent:
79
+ """Read at most size bytes of data from the socket."""
80
+
81
+ if self._closed:
82
+ raise SocketClosedBlueletError
83
+
84
+ if self._buf:
85
+ # We already have data read previously.
86
+ out = self._buf[:size]
87
+ self._buf = self._buf[size:]
88
+ return ValueBlueletEvent(bytes(out))
89
+ else:
90
+ return ReceiveBlueletEvent(self, size)
91
+
92
+ def send(self, data: bytes) -> BlueletEvent:
93
+ """Sends data on the socket, returning the number of bytes successfully sent."""
94
+
95
+ if self._closed:
96
+ raise SocketClosedBlueletError
97
+ return SendBlueletEvent(self, data)
98
+
99
+ def sendall(self, data: bytes) -> BlueletEvent:
100
+ """Send all of data on the socket."""
101
+
102
+ if self._closed:
103
+ raise SocketClosedBlueletError
104
+ return SendBlueletEvent(self, data, True)
105
+
106
+ def readline(self, terminator: bytes = b'\n', bufsize: int = 1024) -> BlueletCoro:
107
+ """Reads a line (delimited by terminator) from the socket."""
108
+
109
+ if self._closed:
110
+ raise SocketClosedBlueletError
111
+
112
+ while True:
113
+ if terminator in self._buf:
114
+ line, self._buf = self._buf.split(terminator, 1)
115
+ line += terminator
116
+ yield ReturnBlueletEvent(bytes(line))
117
+ break
118
+
119
+ if (data := (yield ReceiveBlueletEvent(self, bufsize))):
120
+ self._buf += data
121
+ else:
122
+ line = self._buf
123
+ self._buf = bytearray()
124
+ yield ReturnBlueletEvent(bytes(line))
125
+ break
126
+
127
+
128
+ ##
129
+
130
+
131
+ class SocketBlueletEvent(BlueletEvent, abc.ABC): # noqa
132
+ pass
133
+
134
+
135
+ @dc.dataclass(frozen=True, eq=False)
136
+ class AcceptBlueletEvent(WaitableBlueletEvent, SocketBlueletEvent):
137
+ """An event for Listener objects (listening sockets) that suspends execution until the socket gets a connection."""
138
+
139
+ listener: BlueletListener
140
+
141
+ def waitables(self) -> BlueletWaitables:
142
+ return BlueletWaitables(r=[self.listener.sock])
143
+
144
+ def fire(self) -> BlueletConnection:
145
+ sock, addr = self.listener.sock.accept()
146
+ return BlueletConnection(sock, addr)
147
+
148
+
149
+ @dc.dataclass(frozen=True, eq=False)
150
+ class ReceiveBlueletEvent(WaitableBlueletEvent, SocketBlueletEvent):
151
+ """An event for Connection objects (connected sockets) for asynchronously reading data."""
152
+
153
+ conn: BlueletConnection
154
+ bufsize: int
155
+
156
+ def waitables(self) -> BlueletWaitables:
157
+ return BlueletWaitables(r=[self.conn.sock])
158
+
159
+ def fire(self) -> bytes:
160
+ return self.conn.sock.recv(self.bufsize)
161
+
162
+
163
+ @dc.dataclass(frozen=True, eq=False)
164
+ class SendBlueletEvent(WaitableBlueletEvent, SocketBlueletEvent):
165
+ """An event for Connection objects (connected sockets) for asynchronously writing data."""
166
+
167
+ conn: BlueletConnection
168
+ data: bytes
169
+ sendall: bool = False
170
+
171
+ def waitables(self) -> BlueletWaitables:
172
+ return BlueletWaitables(w=[self.conn.sock])
173
+
174
+ def fire(self) -> ta.Optional[int]:
175
+ if self.sendall:
176
+ self.conn.sock.sendall(self.data)
177
+ return None
178
+ else:
179
+ return self.conn.sock.send(self.data)
180
+
181
+
182
+ ##
183
+
184
+
185
+ class _SocketsBlueletApi(_CoreBlueletApi):
186
+ def connect(self, host: str, port: int) -> BlueletEvent:
187
+ """Event: connect to a network address and return a Connection object for communicating on the socket."""
188
+
189
+ addr = (host, port)
190
+ sock = socket.create_connection(addr)
191
+ return ValueBlueletEvent(BlueletConnection(sock, addr))
192
+
193
+ def server(self, host: str, port: int, func) -> BlueletCoro:
194
+ """
195
+ A coroutine that runs a network server. Host and port specify the listening address. func should be a coroutine
196
+ that takes a single parameter, a Connection object. The coroutine is invoked for every incoming connection on
197
+ the listening socket.
198
+ """
199
+
200
+ def handler(conn):
201
+ try:
202
+ yield func(conn)
203
+ finally:
204
+ conn.close()
205
+
206
+ listener = BlueletListener(host, port)
207
+ try:
208
+ while True:
209
+ conn = yield listener.accept()
210
+ yield self.spawn(handler(conn))
211
+ except KeyboardInterrupt:
212
+ pass
213
+ finally:
214
+ listener.close()
@@ -0,0 +1,69 @@
1
+ from .base import ( # noqa
2
+ EagerCodec,
3
+ IncrementalCodec,
4
+ ComboCodec,
5
+
6
+ check_codec_name,
7
+
8
+ Codec,
9
+
10
+ LazyLoadedCodec,
11
+ )
12
+
13
+ from .bytes import ( # noqa
14
+ ASCII85,
15
+ BASE16,
16
+ BASE32,
17
+ BASE64,
18
+ BASE85,
19
+ BASE32_HEX,
20
+ BASE64_HEX,
21
+ BASE64_URLSAFE,
22
+ HEX,
23
+ )
24
+
25
+ from .chain import ( # noqa
26
+ ChainEagerCodec,
27
+
28
+ chain,
29
+ )
30
+
31
+ from .funcs import ( # noqa
32
+ FnPairEagerCodec,
33
+ )
34
+
35
+ from .registry import ( # noqa
36
+ CodecRegistry,
37
+
38
+ REGISTRY,
39
+ register,
40
+ lookup,
41
+
42
+ encode,
43
+ decode,
44
+ )
45
+
46
+ from .standard import ( # noqa
47
+ STANDARD_CODECS,
48
+ )
49
+
50
+ from .text import ( # noqa
51
+ TextEncodingErrors,
52
+ TextEncodingOptions,
53
+
54
+ TextEncodingComboCodec,
55
+
56
+ TextEncodingCodec,
57
+
58
+ ASCII,
59
+ LATIN1,
60
+ UTF32,
61
+ UTF32BE,
62
+ UTF32LE,
63
+ UTF16,
64
+ UTF16BE,
65
+ UTF16LE,
66
+ UTF7,
67
+ UTF8,
68
+ UTF8SIG,
69
+ )