omlish 0.0.0.dev118__py3-none-any.whl → 0.0.0.dev119__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/lite/journald.py +163 -0
- omlish/lite/logs.py +6 -2
- omlish/logs/_abc.py +53 -0
- omlish/logs/handlers.py +1 -1
- omlish/testing/pytest/plugins/pydevd.py +6 -6
- {omlish-0.0.0.dev118.dist-info → omlish-0.0.0.dev119.dist-info}/METADATA +1 -1
- {omlish-0.0.0.dev118.dist-info → omlish-0.0.0.dev119.dist-info}/RECORD +12 -11
- {omlish-0.0.0.dev118.dist-info → omlish-0.0.0.dev119.dist-info}/LICENSE +0 -0
- {omlish-0.0.0.dev118.dist-info → omlish-0.0.0.dev119.dist-info}/WHEEL +0 -0
- {omlish-0.0.0.dev118.dist-info → omlish-0.0.0.dev119.dist-info}/entry_points.txt +0 -0
- {omlish-0.0.0.dev118.dist-info → omlish-0.0.0.dev119.dist-info}/top_level.txt +0 -0
omlish/__about__.py
CHANGED
omlish/lite/journald.py
ADDED
@@ -0,0 +1,163 @@
|
|
1
|
+
# ruff: noqa: UP007 UP012
|
2
|
+
import ctypes as ct
|
3
|
+
import logging
|
4
|
+
import sys
|
5
|
+
import syslog
|
6
|
+
import threading
|
7
|
+
import typing as ta
|
8
|
+
|
9
|
+
from .cached import cached_nullary
|
10
|
+
|
11
|
+
|
12
|
+
##
|
13
|
+
|
14
|
+
|
15
|
+
class sd_iovec(ct.Structure): # noqa
|
16
|
+
pass
|
17
|
+
|
18
|
+
|
19
|
+
sd_iovec._fields_ = [
|
20
|
+
('iov_base', ct.c_void_p), # Pointer to data.
|
21
|
+
('iov_len', ct.c_size_t), # Length of data.
|
22
|
+
]
|
23
|
+
|
24
|
+
|
25
|
+
##
|
26
|
+
|
27
|
+
|
28
|
+
@cached_nullary
|
29
|
+
def sd_libsystemd() -> ta.Any:
|
30
|
+
lib = ct.CDLL('libsystemd.so.0')
|
31
|
+
|
32
|
+
lib.sd_journal_sendv = lib['sd_journal_sendv'] # type: ignore
|
33
|
+
lib.sd_journal_sendv.restype = ct.c_int
|
34
|
+
lib.sd_journal_sendv.argtypes = [ct.POINTER(sd_iovec), ct.c_int]
|
35
|
+
|
36
|
+
return lib
|
37
|
+
|
38
|
+
|
39
|
+
@cached_nullary
|
40
|
+
def sd_try_libsystemd() -> ta.Optional[ta.Any]:
|
41
|
+
try:
|
42
|
+
return sd_libsystemd()
|
43
|
+
except OSError: # noqa
|
44
|
+
return None
|
45
|
+
|
46
|
+
|
47
|
+
##
|
48
|
+
|
49
|
+
|
50
|
+
def sd_journald_send(**fields: str) -> int:
|
51
|
+
lib = sd_libsystemd()
|
52
|
+
|
53
|
+
msgs = [
|
54
|
+
f'{k.upper()}={v}\0'.encode('utf-8')
|
55
|
+
for k, v in fields.items()
|
56
|
+
]
|
57
|
+
|
58
|
+
vec = (sd_iovec * len(msgs))()
|
59
|
+
cl = (ct.c_char_p * len(msgs))() # noqa
|
60
|
+
for i in range(len(msgs)):
|
61
|
+
vec[i].iov_base = ct.cast(ct.c_char_p(msgs[i]), ct.c_void_p)
|
62
|
+
vec[i].iov_len = len(msgs[i]) - 1
|
63
|
+
|
64
|
+
return lib.sd_journal_sendv(vec, len(msgs))
|
65
|
+
|
66
|
+
|
67
|
+
##
|
68
|
+
|
69
|
+
|
70
|
+
SD_LOG_LEVEL_MAP: ta.Mapping[int, int] = {
|
71
|
+
logging.FATAL: syslog.LOG_EMERG, # system is unusable
|
72
|
+
# LOG_ALERT ? # action must be taken immediately
|
73
|
+
logging.CRITICAL: syslog.LOG_CRIT,
|
74
|
+
logging.ERROR: syslog.LOG_ERR,
|
75
|
+
logging.WARNING: syslog.LOG_WARNING,
|
76
|
+
# LOG_NOTICE ? # normal but significant condition
|
77
|
+
logging.INFO: syslog.LOG_INFO,
|
78
|
+
logging.DEBUG: syslog.LOG_DEBUG,
|
79
|
+
}
|
80
|
+
|
81
|
+
|
82
|
+
class JournaldLogHandler(logging.Handler):
|
83
|
+
"""
|
84
|
+
TODO:
|
85
|
+
- fallback handler for when this barfs
|
86
|
+
"""
|
87
|
+
|
88
|
+
def __init__(
|
89
|
+
self,
|
90
|
+
*,
|
91
|
+
use_formatter_output: bool = False,
|
92
|
+
) -> None:
|
93
|
+
super().__init__()
|
94
|
+
|
95
|
+
sd_libsystemd()
|
96
|
+
|
97
|
+
self._use_formatter_output = use_formatter_output
|
98
|
+
|
99
|
+
#
|
100
|
+
|
101
|
+
EXTRA_RECORD_ATTRS_BY_JOURNALD_FIELD: ta.ClassVar[ta.Mapping[str, str]] = {
|
102
|
+
'name': 'name',
|
103
|
+
'module': 'module',
|
104
|
+
'exception': 'exc_text',
|
105
|
+
'thread_name': 'threadName',
|
106
|
+
'task_name': 'taskName',
|
107
|
+
}
|
108
|
+
|
109
|
+
def make_fields(self, record: logging.LogRecord) -> ta.Mapping[str, str]:
|
110
|
+
formatter_message = self.format(record)
|
111
|
+
if self._use_formatter_output:
|
112
|
+
message = formatter_message
|
113
|
+
else:
|
114
|
+
message = record.message
|
115
|
+
|
116
|
+
fields: dict[str, str] = {
|
117
|
+
'message': message,
|
118
|
+
'priority': str(SD_LOG_LEVEL_MAP[record.levelno]),
|
119
|
+
'tid': str(threading.get_ident()),
|
120
|
+
}
|
121
|
+
|
122
|
+
if (pathname := record.pathname) is not None:
|
123
|
+
fields['code_file'] = pathname
|
124
|
+
if (lineno := record.lineno) is not None:
|
125
|
+
fields['code_lineno'] = str(lineno)
|
126
|
+
if (func_name := record.funcName) is not None:
|
127
|
+
fields['code_func'] = func_name
|
128
|
+
|
129
|
+
for f, a in self.EXTRA_RECORD_ATTRS_BY_JOURNALD_FIELD.items():
|
130
|
+
if (v := getattr(record, a, None)) is not None:
|
131
|
+
fields[f] = str(v)
|
132
|
+
|
133
|
+
return fields
|
134
|
+
|
135
|
+
#
|
136
|
+
|
137
|
+
def emit(self, record: logging.LogRecord) -> None:
|
138
|
+
try:
|
139
|
+
fields = self.make_fields(record)
|
140
|
+
|
141
|
+
if rc := sd_journald_send(**fields):
|
142
|
+
raise RuntimeError(f'{self.__class__.__name__}.emit failed: {rc=}') # noqa
|
143
|
+
|
144
|
+
except RecursionError: # See issue 36272
|
145
|
+
raise
|
146
|
+
|
147
|
+
except Exception: # noqa
|
148
|
+
self.handleError(record)
|
149
|
+
|
150
|
+
|
151
|
+
def journald_log_handler_factory(
|
152
|
+
*,
|
153
|
+
no_tty_check: bool = False,
|
154
|
+
no_fallback: bool = False,
|
155
|
+
) -> logging.Handler:
|
156
|
+
if (
|
157
|
+
sys.platform == 'linux' and
|
158
|
+
(no_tty_check or not sys.stderr.isatty()) and
|
159
|
+
(no_fallback or sd_try_libsystemd() is not None)
|
160
|
+
):
|
161
|
+
return JournaldLogHandler()
|
162
|
+
|
163
|
+
return logging.StreamHandler()
|
omlish/lite/logs.py
CHANGED
@@ -91,7 +91,7 @@ class StandardLogFormatter(logging.Formatter):
|
|
91
91
|
if datefmt:
|
92
92
|
return ct.strftime(datefmt) # noqa
|
93
93
|
else:
|
94
|
-
t = ct.strftime(
|
94
|
+
t = ct.strftime('%Y-%m-%d %H:%M:%S')
|
95
95
|
return '%s.%03d' % (t, record.msecs)
|
96
96
|
|
97
97
|
|
@@ -228,6 +228,7 @@ def configure_standard_logging(
|
|
228
228
|
json: bool = False,
|
229
229
|
target: ta.Optional[logging.Logger] = None,
|
230
230
|
force: bool = False,
|
231
|
+
handler_factory: ta.Optional[ta.Callable[[], logging.Handler]] = None,
|
231
232
|
) -> ta.Optional[StandardLogHandler]:
|
232
233
|
with _locking_logging_module_lock():
|
233
234
|
if target is None:
|
@@ -241,7 +242,10 @@ def configure_standard_logging(
|
|
241
242
|
|
242
243
|
#
|
243
244
|
|
244
|
-
|
245
|
+
if handler_factory is not None:
|
246
|
+
handler = handler_factory()
|
247
|
+
else:
|
248
|
+
handler = logging.StreamHandler()
|
245
249
|
|
246
250
|
#
|
247
251
|
|
omlish/logs/_abc.py
CHANGED
@@ -16,27 +16,80 @@ ExceptionInfo: ta.TypeAlias = tuple[type[BaseException], BaseException, types.Tr
|
|
16
16
|
|
17
17
|
|
18
18
|
class LogRecord:
|
19
|
+
"""https://docs.python.org/3/library/logging.html#logrecord-attributes"""
|
20
|
+
|
21
|
+
# Name of the logger used to log the call.
|
19
22
|
name: str
|
23
|
+
|
24
|
+
# Human-readable time when the LogRecord was created. By default this is of the form '2003-07-08 16:49:45,896' (the
|
25
|
+
# numbers after the comma are millisecond portion of the time).
|
26
|
+
asctime: str
|
27
|
+
|
28
|
+
# The logged message, computed as msg % args. This is set when Formatter.format() is invoked.
|
29
|
+
message: str
|
30
|
+
|
31
|
+
# The format string passed in the original logging call. Merged with args to produce message, or an arbitrary object
|
32
|
+
# (see Using arbitrary objects as messages).
|
20
33
|
msg: str
|
34
|
+
|
35
|
+
# The tuple of arguments merged into msg to produce message, or a dict whose values are used for the merge (when
|
36
|
+
# there is only one argument, and it is a dictionary).
|
21
37
|
args: tuple
|
38
|
+
|
39
|
+
# Text logging level for the message ('DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL').
|
22
40
|
levelname: str
|
41
|
+
|
42
|
+
# # Numeric logging level for the message (DEBUG, INFO, WARNING, ERROR, CRITICAL).
|
23
43
|
levelno: Level
|
44
|
+
|
45
|
+
# Full pathname of the source file where the logging call was issued (if available).
|
24
46
|
pathname: str
|
47
|
+
|
48
|
+
# Filename portion of pathname.
|
25
49
|
filename: str
|
50
|
+
|
51
|
+
# Module (name portion of filename).
|
26
52
|
module: str
|
53
|
+
|
54
|
+
# Exception tuple (à la sys.exc_info) or, if no exception has occurred, None.
|
27
55
|
exc_info: ExceptionInfo | None
|
56
|
+
|
28
57
|
exc_text: str | None
|
58
|
+
|
59
|
+
# Stack frame information (where available) from the bottom of the stack in the current thread, up to and including
|
60
|
+
# the stack frame of the logging call which resulted in the creation of this record.
|
29
61
|
stack_info: str | None
|
62
|
+
|
63
|
+
# Source line number where the logging call was issued (if available).
|
30
64
|
lineno: int
|
65
|
+
|
66
|
+
# Name of function containing the logging call.
|
31
67
|
funcName: str
|
68
|
+
|
69
|
+
# Time when the LogRecord was created (as returned by time.time_ns() / 1e9).
|
32
70
|
created: float
|
71
|
+
|
72
|
+
# Millisecond portion of the time when the LogRecord was created.
|
33
73
|
msecs: float
|
74
|
+
|
75
|
+
# Time in milliseconds when the LogRecord was created, relative to the time the logging module was loaded.
|
34
76
|
relativeCreated: float
|
77
|
+
|
78
|
+
# Thread ID (if available).
|
35
79
|
thread: int
|
80
|
+
|
81
|
+
# Thread name (if available).
|
36
82
|
threadName: str
|
83
|
+
|
84
|
+
# Process name (if available).
|
37
85
|
processName: str
|
86
|
+
|
87
|
+
# Process ID (if available).
|
38
88
|
process: int
|
39
89
|
|
90
|
+
# asyncio.Task name (if available).
|
91
|
+
taskName: str
|
92
|
+
|
40
93
|
|
41
94
|
##
|
42
95
|
|
omlish/logs/handlers.py
CHANGED
@@ -16,9 +16,9 @@ class PydevdPlugin:
|
|
16
16
|
# if (dbg := opd.get_global_debugger()) is not None:
|
17
17
|
# dbg.set_unit_tests_debugging_mode()
|
18
18
|
|
19
|
-
def pytest_exception_interact(self, node, call, report):
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
19
|
+
# def pytest_exception_interact(self, node, call, report):
|
20
|
+
# if opd.get_setup() is not None:
|
21
|
+
# if not node.session.config.option.no_pydevd:
|
22
|
+
# opd.debug_unhandled_exception(call.excinfo._excinfo) # noqa
|
23
|
+
#
|
24
|
+
# return report
|
@@ -1,5 +1,5 @@
|
|
1
1
|
omlish/.manifests.json,sha256=CxGnj-UiRPlZgmgWoovDWrOnqpSEmBy_kqA7cdfSA3w,1431
|
2
|
-
omlish/__about__.py,sha256=
|
2
|
+
omlish/__about__.py,sha256=cKBa887pzxYkOLw1HFwEiT4g4dP8AxuJQrGNkTmiju8,3352
|
3
3
|
omlish/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
4
4
|
omlish/argparse.py,sha256=cqKGAqcxuxv_s62z0gq29L9KAvg_3-_rFvXKjVpRJjo,8126
|
5
5
|
omlish/c3.py,sha256=4vogWgwPb8TbNS2KkZxpoWbwjj7MuHG2lQG-hdtkvjI,8062
|
@@ -303,8 +303,9 @@ omlish/lite/check.py,sha256=ouJme9tkzWXKymm_xZDK4mngdYSkxDrok6CSegvf-1w,1015
|
|
303
303
|
omlish/lite/contextmanagers.py,sha256=_n6a9xhn06BD8H6A_SDtcipMrSBpzBqcxI0Ob2juomM,1226
|
304
304
|
omlish/lite/docker.py,sha256=3IVZZtIm7-UdB2SwArmN_MosTva1_KifyYp3YWjODbE,337
|
305
305
|
omlish/lite/io.py,sha256=lcpI1cS_Kn90tvYMg8ZWkSlYloS4RFqXCk-rKyclhdg,3148
|
306
|
+
omlish/lite/journald.py,sha256=3nfahFbTrdrfp9txhtto6JYAyrus2kcAFtviqdm3qAo,3949
|
306
307
|
omlish/lite/json.py,sha256=7-02Ny4fq-6YAu5ynvqoijhuYXWpLmfCI19GUeZnb1c,740
|
307
|
-
omlish/lite/logs.py,sha256=
|
308
|
+
omlish/lite/logs.py,sha256=tw7mbQslkyo9LopfgQnj0tYiqJ9TDNiw7D07aF7Dm2g,6176
|
308
309
|
omlish/lite/marshal.py,sha256=SyYMsJ-TaGO9FX7LykBB0WtqsqetX9eLBotPiz3M_xg,9478
|
309
310
|
omlish/lite/pidfile.py,sha256=PRSDOAXmNkNwxh-Vwif0Nrs8RAmWroiNhLKIbdjwzBc,1723
|
310
311
|
omlish/lite/reflect.py,sha256=9QYJwdINraq1JNMEgvoqeSlVvRRgOXpxAkpgX8EgRXc,1307
|
@@ -313,10 +314,10 @@ omlish/lite/secrets.py,sha256=3Mz3V2jf__XU9qNHcH56sBSw95L3U2UPL24bjvobG0c,816
|
|
313
314
|
omlish/lite/strings.py,sha256=QURcE4-1pKVW8eT_5VCJpXaHDWR2dW2pYOChTJnZDiQ,1504
|
314
315
|
omlish/lite/subprocesses.py,sha256=_YwUpvfaC2pV5TMC9-Ivuw1Ao-YxteD3a1NQwGERft4,3380
|
315
316
|
omlish/logs/__init__.py,sha256=FbOyAW-lGH8gyBlSVArwljdYAU6RnwZLI5LwAfuNnrk,438
|
316
|
-
omlish/logs/_abc.py,sha256=
|
317
|
+
omlish/logs/_abc.py,sha256=rWySJcr1vatu-AR1EYtODRhi-TjFaixqUzXeWg1c0GA,8006
|
317
318
|
omlish/logs/configs.py,sha256=EE0jlNaXJbGnM7V-y4xS5VwyTBSTzFzc0BYaVjg0JmA,1283
|
318
319
|
omlish/logs/formatters.py,sha256=q79nMnR2mRIStPyGrydQHpYTXgC5HHptt8lH3W2Wwbs,671
|
319
|
-
omlish/logs/handlers.py,sha256=
|
320
|
+
omlish/logs/handlers.py,sha256=UpzUf3kWBBzWOnrtljoZsLjISw3Ix-ePz3Nsmp6lRgE,255
|
320
321
|
omlish/logs/noisy.py,sha256=Ubc-eTH6ZbGYsLfUUi69JAotwuUwzb-SJBeGo_0dIZI,348
|
321
322
|
omlish/logs/utils.py,sha256=MgGovbP0zUrZ3FGD3qYNQWn-l0jy0Y0bStcQvv5BOmQ,391
|
322
323
|
omlish/marshal/__init__.py,sha256=iVA7n31L08Bdub6HKPvYOXVvDhk2CMA6rPeKDL_u1to,2298
|
@@ -457,7 +458,7 @@ omlish/testing/pytest/plugins/asyncs.py,sha256=SV6oKCy50CGkzLGYX-CT4MfWNqsrH8ONE
|
|
457
458
|
omlish/testing/pytest/plugins/depskip.py,sha256=xithY-OMtjwhv8mcRNkv-WI_PSQtHldQ8H1s60MIXkk,2673
|
458
459
|
omlish/testing/pytest/plugins/logging.py,sha256=1zs6Xe54wiaSjabCviaFXwKkoN97CKm3mA5mEoUeJGs,380
|
459
460
|
omlish/testing/pytest/plugins/managermarks.py,sha256=AP3ty-QB-8O5DkulwUOudBlUOvXMHhBfNyY-0yCmejk,1520
|
460
|
-
omlish/testing/pytest/plugins/pydevd.py,sha256=
|
461
|
+
omlish/testing/pytest/plugins/pydevd.py,sha256=AXtN83M39ZKJ4VH3MJEhvPnAmYzD5u1r8ehz-0om50Q,842
|
461
462
|
omlish/testing/pytest/plugins/repeat.py,sha256=flSQzE9GFOWksVKz-mUGnpxJpv3yRqn1G4K8pW7JHs0,498
|
462
463
|
omlish/testing/pytest/plugins/skips.py,sha256=EoZDg1uWccgbAegmzqI85c7RliycD1e2J4Y7vfDRhwM,1041
|
463
464
|
omlish/testing/pytest/plugins/spacing.py,sha256=JQQhi9q3c523Ro1a_K_9RGAb7HotiO74N8bYX2VESFE,707
|
@@ -470,9 +471,9 @@ omlish/text/glyphsplit.py,sha256=Ug-dPRO7x-OrNNr8g1y6DotSZ2KH0S-VcOmUobwa4B0,329
|
|
470
471
|
omlish/text/indent.py,sha256=6Jj6TFY9unaPa4xPzrnZemJ-fHsV53IamP93XGjSUHs,1274
|
471
472
|
omlish/text/parts.py,sha256=7vPF1aTZdvLVYJ4EwBZVzRSy8XB3YqPd7JwEnNGGAOo,6495
|
472
473
|
omlish/text/random.py,sha256=jNWpqiaKjKyTdMXC-pWAsSC10AAP-cmRRPVhm59ZWLk,194
|
473
|
-
omlish-0.0.0.
|
474
|
-
omlish-0.0.0.
|
475
|
-
omlish-0.0.0.
|
476
|
-
omlish-0.0.0.
|
477
|
-
omlish-0.0.0.
|
478
|
-
omlish-0.0.0.
|
474
|
+
omlish-0.0.0.dev119.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
|
475
|
+
omlish-0.0.0.dev119.dist-info/METADATA,sha256=sMg30UlhW1GLNw8MkOhVqMaaiObB2IXCn22KIqaMpWU,4000
|
476
|
+
omlish-0.0.0.dev119.dist-info/WHEEL,sha256=R06PA3UVYHThwHvxuRWMqaGcr-PuniXahwjmQRFMEkY,91
|
477
|
+
omlish-0.0.0.dev119.dist-info/entry_points.txt,sha256=Lt84WvRZJskWCAS7xnQGZIeVWksprtUHj0llrvVmod8,35
|
478
|
+
omlish-0.0.0.dev119.dist-info/top_level.txt,sha256=pePsKdLu7DvtUiecdYXJ78iO80uDNmBlqe-8hOzOmfs,7
|
479
|
+
omlish-0.0.0.dev119.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|