-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlogging_config.py
227 lines (211 loc) · 9.54 KB
/
logging_config.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
import logging
import logging.config
from annotated_logger import AnnotatedAdapter, AnnotatedFilter, AnnotatedLogger
from annotated_logger.plugins import BasePlugin, RenamerPlugin, RuntimeAnnotationsPlugin
# This logging config creates 4 loggers:
# * A logger for "annotated_logger.logging_config", which logs all messages as json and
# also logs errors as plain text. This is an example of how to log to multiple places.
# * A logger for "annotated_logger.logging_config_weird", which logs all messages at
# info and up. It has a different namespace (_weird instead of .weird) and so has
# isolated annotations.
# * A logger for "annotated_logger.logging_config.long", which logs all messages at info
# as text with a note added. This logger allows it's logs to propagate up and so the
# "annotated_logger.logging_config" logger will also log these messages in it's json
# format without the note from this logger.
# * A logger for "annotated_logger.logging_config.logger", which logs all messages as
# json. This logger does not propagate so that the "annotated_logger.logging_config"
# logger doesn't also log these messages. This logger is used by a non annotated
# method, but defines a filter that is annotated with the base annotations defined
# in `AnnotatedLogger(...`. This is an example of how to add annotations to external
# logs such as django. Note, the annotations this logger receives are based on the
# annotations passed in to the `AnnotatedLogger` invocation with the config,
# The second invocation for "weird" has different annotations. You should be able
# to have multiple of these with different annotations by invoking `AnnotatedLogger`
# multiple times and including `disable_existing_loggers` in the later configs.
# You can also provide custom annotations here if you wish to override the
# annotations from the annotated logger.
#
# Note: When creating multiple loggers, especially when doing so in different
# files/configs keep in mind that names should be unique or they will override
# eachother leaving you with a very confusing mess.
# If you want to see how to more easily merge settings into the default logging
# dict this package uses see the `actions.py` example.
LOGGING = {
"version": 1,
"disable_existing_loggers": False,
"filters": {
"logging_config.logger_filter": {
"annotated_filter": True,
# You can override the annotations that would be provided like so
# But, if you want to do that you are likely better off
# using a filter not associated with an AnnotatedLogger
# like the `logging_config.logger_filter_parens` below
"annotations": {"config_based_filter": True},
},
"logging_config.logger_filter_parens": {
"()": AnnotatedFilter,
"annotations": {"decorated": False, "class_based_filter": True},
"class_annotations": {},
"plugins": [
BasePlugin(),
RuntimeAnnotationsPlugin({"custom_runtime": lambda _record: True}),
],
},
},
"handlers": {
"logging_config.annotated_handler": {
"class": "logging.StreamHandler",
"formatter": "logging_config.annotated_formatter",
},
"logging_config.logger_handler": {
"class": "logging.StreamHandler",
# Note that this handler is specifically attached to
# `annotated_logger.logging_config.logger` which we intend to use only as a
# logger not generated by a decorator. If you add filters to a handler that
# will be invoked when logging from a logger generated by the decorator
# AKA, logging via the `annotated_logger` parameter, these filters will
# be applied *after* the filters that are dynamically generated/updated
# and so can override any annotations that share a name.
# In this case, if a function annotated `custom_runtime` or one of the
# other fields set in the filters above, that value would be overridden
# by the value in the filter set in the config.
"filters": [
"logging_config.logger_filter",
"logging_config.logger_filter_parens",
],
"formatter": "logging_config.annotated_formatter",
},
"logging_config.long_handler": {
"class": "logging.StreamHandler",
"formatter": "logging_config.long_formatter",
},
"logging_config.error_handler": {
"class": "logging.StreamHandler",
"level": "ERROR",
"formatter": "logging_config.error_formatter",
},
"logging_config.weird_handler": {
"class": "logging.StreamHandler",
"formatter": "logging_config.weird_formatter",
},
},
"formatters": {
"logging_config.annotated_formatter": {
"class": "pythonjsonlogger.json.JsonFormatter",
# Note that this format string uses `time` which is set by the renamer
# plugin. It also has `lvl` which is there strictly to test our fallback
# to using `levelno` in the mocks to determine level.
"format": "{time} {lvl} {name} {runtime} {message}",
"style": "{",
},
"logging_config.error_formatter": {
"format": "{level} {message}",
"style": "{",
},
"logging_config.long_formatter": {
"format": "{lvl} Long message, may be split {message}",
# 3.12 added support for defaults in dict configs
# With that we can add the format and defaults below
# for a more realistic example. Not all of the messages
# in the method we set to use this are long enough to be split,
# so, some of them don't have the message_part(s) fields.
# "format": "{level} {message_part}/{message_parts} {message}", # noqa: ERA001 E501
# "defaults": {"message_part": 1, "message_parts": 1}, # noqa: ERA001
"style": "{",
},
"logging_config.weird_formatter": {
"class": "pythonjsonlogger.json.JsonFormatter",
"format": "{time} {lvl} {name} {message}",
"style": "{",
},
},
"loggers": {
"annotated_logger.logging_config": {
"level": "DEBUG",
"handlers": [
"logging_config.annotated_handler",
"logging_config.error_handler",
],
"propagate": True,
},
"annotated_logger.logging_config_weird": {
"level": "INFO",
"handlers": ["logging_config.weird_handler"],
"propagate": True,
},
"annotated_logger.logging_config.long": {
"level": "INFO",
"handlers": ["logging_config.long_handler"],
"propagate": True,
},
"annotated_logger.logging_config.logger": {
"handlers": ["logging_config.logger_handler"],
"propagate": False,
},
},
}
def runtime(_record: logging.LogRecord) -> str:
"""Return the string every time."""
return "this function is called every time"
annotated_logger = AnnotatedLogger(
annotations={"hostname": "my-host"},
# This is deprecated, use the RuntimeAnnotationsPlugin instead.
# This param is kept for backwards compatibility and creates a
# RuntimeAnnotationsPlugin instead.
# This is left as an example and to provide test coverage.
plugins=[
RenamerPlugin(time="created", lvl="levelname"),
RuntimeAnnotationsPlugin({"runtime": runtime}),
],
log_level=logging.DEBUG,
max_length=200,
name="annotated_logger.logging_config",
config=LOGGING,
)
annotate_logs = annotated_logger.annotate_logs
weird_annotated_logger = AnnotatedLogger(
annotations={"weird": True},
plugins=[RenamerPlugin(time="created", lvl="levelname")],
log_level=logging.INFO,
name="annotated_logger.logging_config_weird",
config=False,
)
weird_annotate_logs = weird_annotated_logger.annotate_logs
logger = logging.getLogger("annotated_logger.logging_config.logger")
logger.setLevel("DEBUG")
def make_some_logs() -> None:
"""Log messages using a native logging logger."""
logger.debug("this is debug")
logger.info("this is info")
logger.warning("this is warning")
logger.error("this is error")
@annotate_logs(_typing_requested=True, _typing_self=False)
def make_some_annotated_logs(annotated_logger: AnnotatedAdapter) -> None:
"""Log messages using the provided annotated_logger."""
annotated_logger.debug("this is debug")
annotated_logger.info("this is info")
annotated_logger.warning("this is warning")
annotated_logger.error("this is error")
@weird_annotate_logs(
_typing_requested=True,
_typing_self=False,
)
def make_some_weird_logs(annotated_logger: AnnotatedAdapter) -> None:
"""Log messages using the provided annotated_logger."""
annotated_logger.debug("this is debug")
annotated_logger.info("this is info")
annotated_logger.warning("this is warning")
annotated_logger.error("this is error")
@annotate_logs(
_typing_requested=True,
_typing_self=False,
logger_name="annotated_logger.logging_config.long",
success_info=False,
)
def log_really_long_message(annotated_logger: AnnotatedAdapter) -> None:
"""Log a message that is so long it will get split."""
message = "1" * 200 + "2" * 200 + "3333"
annotated_logger.info(message)
annotated_logger.info("4" * 200)
annotated_logger.info("5" * 201)
annotated_logger.info("6" * 199)