omlish 0.0.0.dev223__py3-none-any.whl → 0.0.0.dev224__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- omlish/__about__.py +2 -2
- omlish/http/handlers.py +52 -1
- omlish/lite/timing.py +8 -0
- omlish/logs/timing.py +58 -0
- omlish/secrets/tempssl.py +43 -21
- omlish/subprocesses.py +19 -0
- {omlish-0.0.0.dev223.dist-info → omlish-0.0.0.dev224.dist-info}/METADATA +1 -1
- {omlish-0.0.0.dev223.dist-info → omlish-0.0.0.dev224.dist-info}/RECORD +12 -10
- {omlish-0.0.0.dev223.dist-info → omlish-0.0.0.dev224.dist-info}/LICENSE +0 -0
- {omlish-0.0.0.dev223.dist-info → omlish-0.0.0.dev224.dist-info}/WHEEL +0 -0
- {omlish-0.0.0.dev223.dist-info → omlish-0.0.0.dev224.dist-info}/entry_points.txt +0 -0
- {omlish-0.0.0.dev223.dist-info → omlish-0.0.0.dev224.dist-info}/top_level.txt +0 -0
omlish/__about__.py
CHANGED
omlish/http/handlers.py
CHANGED
@@ -63,14 +63,65 @@ class HttpHandler_(abc.ABC): # noqa
|
|
63
63
|
raise NotImplementedError
|
64
64
|
|
65
65
|
|
66
|
+
##
|
67
|
+
|
68
|
+
|
66
69
|
@dc.dataclass(frozen=True)
|
67
70
|
class LoggingHttpHandler(HttpHandler_):
|
68
71
|
handler: HttpHandler
|
69
72
|
log: logging.Logger
|
70
|
-
level: int = logging.
|
73
|
+
level: int = logging.DEBUG
|
71
74
|
|
72
75
|
def __call__(self, req: HttpHandlerRequest) -> HttpHandlerResponse:
|
73
76
|
self.log.log(self.level, '%r', req)
|
74
77
|
resp = self.handler(req)
|
75
78
|
self.log.log(self.level, '%r', resp)
|
76
79
|
return resp
|
80
|
+
|
81
|
+
|
82
|
+
##
|
83
|
+
|
84
|
+
|
85
|
+
@dc.dataclass(frozen=True)
|
86
|
+
class BytesResponseHttpHandler(HttpHandler_):
|
87
|
+
data: bytes
|
88
|
+
|
89
|
+
status: ta.Union[http.HTTPStatus, int] = 200
|
90
|
+
content_type: ta.Optional[str] = 'application/octet-stream'
|
91
|
+
headers: ta.Optional[ta.Mapping[str, str]] = None
|
92
|
+
close_connection: bool = True
|
93
|
+
|
94
|
+
def __call__(self, req: HttpHandlerRequest) -> HttpHandlerResponse:
|
95
|
+
return HttpHandlerResponse(
|
96
|
+
status=self.status,
|
97
|
+
headers={
|
98
|
+
**({'Content-Type': self.content_type} if self.content_type else {}),
|
99
|
+
'Content-Length': str(len(self.data)),
|
100
|
+
**(self.headers or {}),
|
101
|
+
},
|
102
|
+
data=self.data,
|
103
|
+
close_connection=self.close_connection,
|
104
|
+
)
|
105
|
+
|
106
|
+
|
107
|
+
@dc.dataclass(frozen=True)
|
108
|
+
class StringResponseHttpHandler(HttpHandler_):
|
109
|
+
data: str
|
110
|
+
|
111
|
+
status: ta.Union[http.HTTPStatus, int] = 200
|
112
|
+
content_type: ta.Optional[str] = 'text/plain; charset=utf-8'
|
113
|
+
headers: ta.Optional[ta.Mapping[str, str]] = None
|
114
|
+
close_connection: bool = True
|
115
|
+
|
116
|
+
def __call__(self, req: HttpHandlerRequest) -> HttpHandlerResponse:
|
117
|
+
data = self.data.encode('utf-8')
|
118
|
+
return HttpHandlerResponse(
|
119
|
+
status=self.status,
|
120
|
+
headers={
|
121
|
+
**({'Content-Type': self.content_type} if self.content_type else {}),
|
122
|
+
'Content-Length': str(len(data)),
|
123
|
+
**(self.headers or {}),
|
124
|
+
},
|
125
|
+
data=data,
|
126
|
+
close_connection=self.close_connection,
|
127
|
+
)
|
omlish/lite/timing.py
ADDED
omlish/logs/timing.py
ADDED
@@ -0,0 +1,58 @@
|
|
1
|
+
# ruff: noqa: UP006 UP007
|
2
|
+
# @omlish-lite
|
3
|
+
import logging
|
4
|
+
import time
|
5
|
+
import typing as ta
|
6
|
+
|
7
|
+
|
8
|
+
##
|
9
|
+
|
10
|
+
|
11
|
+
class LogTimingContext:
|
12
|
+
DEFAULT_LOG: ta.ClassVar[ta.Optional[logging.Logger]] = None
|
13
|
+
|
14
|
+
class _NOT_SPECIFIED: # noqa
|
15
|
+
def __new__(cls, *args, **kwargs): # noqa
|
16
|
+
raise TypeError
|
17
|
+
|
18
|
+
def __init__(
|
19
|
+
self,
|
20
|
+
description: str,
|
21
|
+
*,
|
22
|
+
log: ta.Union[logging.Logger, ta.Type[_NOT_SPECIFIED], None] = _NOT_SPECIFIED, # noqa
|
23
|
+
level: int = logging.DEBUG,
|
24
|
+
) -> None:
|
25
|
+
super().__init__()
|
26
|
+
|
27
|
+
self._description = description
|
28
|
+
if log is self._NOT_SPECIFIED:
|
29
|
+
log = self.DEFAULT_LOG # noqa
|
30
|
+
self._log: ta.Optional[logging.Logger] = log # type: ignore
|
31
|
+
self._level = level
|
32
|
+
|
33
|
+
def set_description(self, description: str) -> 'LogTimingContext':
|
34
|
+
self._description = description
|
35
|
+
return self
|
36
|
+
|
37
|
+
_begin_time: float
|
38
|
+
_end_time: float
|
39
|
+
|
40
|
+
def __enter__(self) -> 'LogTimingContext':
|
41
|
+
self._begin_time = time.time()
|
42
|
+
|
43
|
+
if self._log is not None:
|
44
|
+
self._log.log(self._level, f'Begin : {self._description}') # noqa
|
45
|
+
|
46
|
+
return self
|
47
|
+
|
48
|
+
def __exit__(self, exc_type, exc_val, exc_tb):
|
49
|
+
self._end_time = time.time()
|
50
|
+
|
51
|
+
if self._log is not None:
|
52
|
+
self._log.log(
|
53
|
+
self._level,
|
54
|
+
f'End : {self._description} - {self._end_time - self._begin_time:0.2f} s elapsed',
|
55
|
+
)
|
56
|
+
|
57
|
+
|
58
|
+
log_timing_context = LogTimingContext
|
omlish/secrets/tempssl.py
CHANGED
@@ -1,10 +1,14 @@
|
|
1
1
|
# @omlish-lite
|
2
2
|
# ruff: noqa: UP006 UP007
|
3
|
+
import dataclasses as dc
|
3
4
|
import os.path
|
4
|
-
import subprocess
|
5
5
|
import tempfile
|
6
6
|
import typing as ta
|
7
7
|
|
8
|
+
from ..lite.cached import cached_nullary
|
9
|
+
from ..subprocesses import SubprocessRun
|
10
|
+
from ..subprocesses import SubprocessRunOutput
|
11
|
+
from ..subprocesses import subprocesses
|
8
12
|
from .ssl import SslCert
|
9
13
|
|
10
14
|
|
@@ -13,11 +17,15 @@ class TempSslCert(ta.NamedTuple):
|
|
13
17
|
temp_dir: str
|
14
18
|
|
15
19
|
|
16
|
-
|
17
|
-
|
20
|
+
@dc.dataclass(frozen=True)
|
21
|
+
class TempSslCertGenerator:
|
22
|
+
@cached_nullary
|
23
|
+
def temp_dir(self) -> str:
|
24
|
+
return tempfile.mkdtemp()
|
18
25
|
|
19
|
-
|
20
|
-
|
26
|
+
@cached_nullary
|
27
|
+
def make_run(self) -> SubprocessRun:
|
28
|
+
return SubprocessRun.of(
|
21
29
|
'openssl',
|
22
30
|
'req',
|
23
31
|
'-x509',
|
@@ -32,19 +40,33 @@ def generate_temp_localhost_ssl_cert() -> TempSslCert:
|
|
32
40
|
|
33
41
|
'-subj', '/CN=localhost',
|
34
42
|
'-addext', 'subjectAltName = DNS:localhost,IP:127.0.0.1',
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
43
|
+
|
44
|
+
cwd=self.temp_dir(),
|
45
|
+
capture_output=True,
|
46
|
+
check=False,
|
47
|
+
)
|
48
|
+
|
49
|
+
def handle_run_output(self, proc: SubprocessRunOutput) -> TempSslCert:
|
50
|
+
if proc.returncode:
|
51
|
+
raise RuntimeError(f'Failed to generate temp ssl cert: {proc.stderr=}')
|
52
|
+
|
53
|
+
key_file = os.path.join(self.temp_dir(), 'key.pem')
|
54
|
+
cert_file = os.path.join(self.temp_dir(), 'cert.pem')
|
55
|
+
for file in [key_file, cert_file]:
|
56
|
+
if not os.path.isfile(file):
|
57
|
+
raise RuntimeError(f'Failed to generate temp ssl cert (file not found): {file}')
|
58
|
+
|
59
|
+
return TempSslCert(
|
60
|
+
SslCert(
|
61
|
+
key_file=key_file,
|
62
|
+
cert_file=cert_file,
|
63
|
+
),
|
64
|
+
temp_dir=self.temp_dir(),
|
65
|
+
)
|
66
|
+
|
67
|
+
def run(self) -> TempSslCert:
|
68
|
+
return self.handle_run_output(subprocesses.run_(self.make_run()))
|
69
|
+
|
70
|
+
|
71
|
+
def generate_temp_localhost_ssl_cert() -> TempSslCert:
|
72
|
+
return TempSslCertGenerator().run()
|
omlish/subprocesses.py
CHANGED
@@ -265,6 +265,25 @@ class SubprocessRun:
|
|
265
265
|
capture_output: ta.Optional[bool] = None
|
266
266
|
kwargs: ta.Optional[ta.Mapping[str, ta.Any]] = None
|
267
267
|
|
268
|
+
@classmethod
|
269
|
+
def of(
|
270
|
+
cls,
|
271
|
+
*cmd: str,
|
272
|
+
input: ta.Any = None, # noqa
|
273
|
+
timeout: ta.Optional[float] = None,
|
274
|
+
check: bool = False,
|
275
|
+
capture_output: ta.Optional[bool] = None,
|
276
|
+
**kwargs: ta.Any,
|
277
|
+
) -> 'SubprocessRun':
|
278
|
+
return cls(
|
279
|
+
cmd=cmd,
|
280
|
+
input=input,
|
281
|
+
timeout=timeout,
|
282
|
+
check=check,
|
283
|
+
capture_output=capture_output,
|
284
|
+
kwargs=kwargs,
|
285
|
+
)
|
286
|
+
|
268
287
|
|
269
288
|
@dc.dataclass(frozen=True)
|
270
289
|
class SubprocessRunOutput(ta.Generic[T]):
|
@@ -1,5 +1,5 @@
|
|
1
1
|
omlish/.manifests.json,sha256=YGmAnUBszmosQQ_7Hh2wwtDiYdYZ4unNKYzOtALuels,7968
|
2
|
-
omlish/__about__.py,sha256=
|
2
|
+
omlish/__about__.py,sha256=S2Ri9mbwlQ-KusjCoVjZBcGHhAJnlWUTzvhVU8dKD9Q,3380
|
3
3
|
omlish/__init__.py,sha256=SsyiITTuK0v74XpKV8dqNaCmjOlan1JZKrHQv5rWKPA,253
|
4
4
|
omlish/c3.py,sha256=ubu7lHwss5V4UznbejAI0qXhXahrU01MysuHOZI9C4U,8116
|
5
5
|
omlish/cached.py,sha256=UI-XTFBwA6YXWJJJeBn-WkwBkfzDjLBBaZf4nIJA9y0,510
|
@@ -11,7 +11,7 @@ omlish/libc.py,sha256=8r7Ejyhttk9ruCfBkxNTrlzir5WPbDE2vmY7VPlceMA,15362
|
|
11
11
|
omlish/outcome.py,sha256=ABIE0zjjTyTNtn-ZqQ_9_mUzLiBQ3sDAyqc9JVD8N2k,7852
|
12
12
|
omlish/runmodule.py,sha256=PWvuAaJ9wQQn6bx9ftEL3_d04DyotNn8dR_twm2pgw0,700
|
13
13
|
omlish/shlex.py,sha256=bsW2XUD8GiMTUTDefJejZ5AyqT1pTgWMPD0BMoF02jE,248
|
14
|
-
omlish/subprocesses.py,sha256=
|
14
|
+
omlish/subprocesses.py,sha256=CKWBnuliVfeIcQjPuIZMMk6uY2bk8LpCW9egMVA059E,13013
|
15
15
|
omlish/sync.py,sha256=QJ79kxmIqDP9SeHDoZAf--DpFIhDQe1jACy8H4N0yZI,2928
|
16
16
|
omlish/algorithm/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
17
17
|
omlish/algorithm/all.py,sha256=FudUHwoaRLNNmqYM3jhP2Yd2BpmYhNBRPaVZzARMoSc,194
|
@@ -304,7 +304,7 @@ omlish/http/consts.py,sha256=7BJ4D1MdIvqBcepkgCfBFHolgTwbOlqsOEiee_IjxOA,2289
|
|
304
304
|
omlish/http/cookies.py,sha256=uuOYlHR6e2SC3GM41V0aozK10nef9tYg83Scqpn5-HM,6351
|
305
305
|
omlish/http/dates.py,sha256=Otgp8wRxPgNGyzx8LFowu1vC4EKJYARCiAwLFncpfHM,2875
|
306
306
|
omlish/http/encodings.py,sha256=w2WoKajpaZnQH8j-IBvk5ZFL2O2pAU_iBvZnkocaTlw,164
|
307
|
-
omlish/http/handlers.py,sha256=
|
307
|
+
omlish/http/handlers.py,sha256=l6JLDuK9XhYvF1IudnPz5hWTZW1dXRjUBSO4Uyck0LE,3365
|
308
308
|
omlish/http/headers.py,sha256=ZMmjrEiYjzo0YTGyK0YsvjdwUazktGqzVVYorY4fd44,5081
|
309
309
|
omlish/http/json.py,sha256=9XwAsl4966Mxrv-1ytyCqhcE6lbBJw-0_tFZzGszgHE,7440
|
310
310
|
omlish/http/jwt.py,sha256=6Rigk1WrJ059DY4jDIKnxjnChWb7aFdermj2AI2DSvk,4346
|
@@ -432,6 +432,7 @@ omlish/lite/resources.py,sha256=YNSmX1Ohck1aoWRs55a-o5ChVbFJIQhtbqE-XwF55Oc,326
|
|
432
432
|
omlish/lite/runtime.py,sha256=XQo408zxTdJdppUZqOWHyeUR50VlCpNIExNGHz4U6O4,459
|
433
433
|
omlish/lite/secrets.py,sha256=3Mz3V2jf__XU9qNHcH56sBSw95L3U2UPL24bjvobG0c,816
|
434
434
|
omlish/lite/strings.py,sha256=QGxT1Yh4oI8ycsfeobxnjEhvDob_GiAKLeIhZwo1j24,1986
|
435
|
+
omlish/lite/timing.py,sha256=aVu3hEDB_jyTF_ryZI7iU-xg4q8CNwqpp9Apfru_iwY,196
|
435
436
|
omlish/lite/types.py,sha256=fP5EMyBdEp2LmDxcHjUDtwAMdR06ISr9lKOL7smWfHM,140
|
436
437
|
omlish/lite/typing.py,sha256=U3-JaEnkDSYxK4tsu_MzUn3RP6qALBe5FXQXpD-licE,1090
|
437
438
|
omlish/logs/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
@@ -447,6 +448,7 @@ omlish/logs/noisy.py,sha256=Ubc-eTH6ZbGYsLfUUi69JAotwuUwzb-SJBeGo_0dIZI,348
|
|
447
448
|
omlish/logs/protocol.py,sha256=NzyCeNBN-fyKpJinhECfjUQSd5MxZLiYbuLCTtW6QUU,4500
|
448
449
|
omlish/logs/proxy.py,sha256=A-ROPUUAlF397qTbEqhel6YhQMstNuXL3Xmts7w9dAo,2347
|
449
450
|
omlish/logs/standard.py,sha256=FbKdF2Z4Na5i2TNwKn0avLJXyICe2JKsPufjvKCHGn0,3162
|
451
|
+
omlish/logs/timing.py,sha256=XrFUHIPT4EHDujLKbGs9fGFMmoM3NEP8xPRaESJr7bQ,1513
|
450
452
|
omlish/logs/utils.py,sha256=mzHrZ9ji75p5A8qR29eUr05CBAHMb8J753MSkID_VaQ,393
|
451
453
|
omlish/manifests/__init__.py,sha256=P2B0dpT8D7l5lJwRGPA92IcQj6oeXfd90X5-q9BJrKg,51
|
452
454
|
omlish/manifests/load.py,sha256=LrWAvBfdzDkFdLuVwfw2RwFvLjxx-rvfkpU9eBsWeIc,5626
|
@@ -515,7 +517,7 @@ omlish/secrets/pwhash.py,sha256=Goktn-swmC6PXqfRBnDrH_Lr42vjckT712UyErPjzkw,4102
|
|
515
517
|
omlish/secrets/secrets.py,sha256=QNgOmRcIRK2fx49bIbBBM2rYbe6IhhLgk8fKvq8guoI,7963
|
516
518
|
omlish/secrets/ssl.py,sha256=TvO1BJeCCBPsOLjO-QH7Q0DC-NS8onfmRxbl4ntOnd8,147
|
517
519
|
omlish/secrets/subprocesses.py,sha256=ffjfbgPbEE_Pwb_87vG4yYR2CGZy3I31mHNCo_0JtHw,1212
|
518
|
-
omlish/secrets/tempssl.py,sha256=
|
520
|
+
omlish/secrets/tempssl.py,sha256=pAmTRFgR23pFRUAIzJrxeJg4JbYFvaVcEtWtvSYyxHw,1932
|
519
521
|
omlish/sockets/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
520
522
|
omlish/sockets/addresses.py,sha256=vbVeQBkzI513H4vRv-JS89QtRbr9U8v5zqkm3oODl_s,1869
|
521
523
|
omlish/sockets/bind.py,sha256=TnG5nm0pnuMxRA02TG2W40RbutrPA6tkOJtbZvBjDWU,8063
|
@@ -665,9 +667,9 @@ omlish/text/indent.py,sha256=YjtJEBYWuk8--b9JU_T6q4yxV85_TR7VEVr5ViRCFwk,1336
|
|
665
667
|
omlish/text/minja.py,sha256=jZC-fp3Xuhx48ppqsf2Sf1pHbC0t8XBB7UpUUoOk2Qw,5751
|
666
668
|
omlish/text/parts.py,sha256=7vPF1aTZdvLVYJ4EwBZVzRSy8XB3YqPd7JwEnNGGAOo,6495
|
667
669
|
omlish/text/random.py,sha256=jNWpqiaKjKyTdMXC-pWAsSC10AAP-cmRRPVhm59ZWLk,194
|
668
|
-
omlish-0.0.0.
|
669
|
-
omlish-0.0.0.
|
670
|
-
omlish-0.0.0.
|
671
|
-
omlish-0.0.0.
|
672
|
-
omlish-0.0.0.
|
673
|
-
omlish-0.0.0.
|
670
|
+
omlish-0.0.0.dev224.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
|
671
|
+
omlish-0.0.0.dev224.dist-info/METADATA,sha256=mQhHJnChb7Igih2af4khc6lOSNpkVqyv8j2xaVk0IpQ,4176
|
672
|
+
omlish-0.0.0.dev224.dist-info/WHEEL,sha256=In9FTNxeP60KnTkGw7wk6mJPYd_dQSjEZmXdBdMCI-8,91
|
673
|
+
omlish-0.0.0.dev224.dist-info/entry_points.txt,sha256=Lt84WvRZJskWCAS7xnQGZIeVWksprtUHj0llrvVmod8,35
|
674
|
+
omlish-0.0.0.dev224.dist-info/top_level.txt,sha256=pePsKdLu7DvtUiecdYXJ78iO80uDNmBlqe-8hOzOmfs,7
|
675
|
+
omlish-0.0.0.dev224.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|