-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path_base.py
201 lines (170 loc) · 7.34 KB
/
_base.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
"""
Adapta Logging Interface.
"""
# Copyright (c) 2023-2024. ECCO Sneaks & Data
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
import logging
from logging import Handler, StreamHandler
from typing import List, Optional, Dict, final
from adapta.logs._internal_logger import _InternalLogger
from adapta.logs.models import LogLevel
from adapta.logs._internal import MetadataLogger
@final
class SemanticLogger(_InternalLogger):
"""
Proxy for a collection of python loggers that use the same formatting interface.
"""
async def redirect_async(self, tags: Optional[Dict[str, str]] = None, **kwargs):
raise NotImplementedError("Async operations are not supported by this logger")
def redirect(self, tags: Optional[Dict[str, str]] = None, log_source_name: Optional[str] = None, **kwargs):
return self._redirect(logger=self._get_logger(log_source_name), tags=tags)
def start(self) -> None:
pass
def stop(self) -> None:
pass
def __init__(self, fixed_template: Optional[Dict[str, Dict[str, str]]] = None, fixed_template_delimiter=", "):
"""
Creates a new instance of a SemanticLogger
:param fixed_template: Additional template to append to message templates provided via logging methods.
:param fixed_template_delimiter: Optional delimiter to use when appending fixed templates.
"""
super().__init__(fixed_template, fixed_template_delimiter)
self._loggers: Dict[str, logging.Logger] = {}
self._default_log_source = None
self._fixed_template = fixed_template
self._fixed_template_delimiter = fixed_template_delimiter
logging.setLoggerClass(MetadataLogger)
def add_log_source(
self,
*,
log_source_name: str,
min_log_level: LogLevel,
log_handlers: Optional[List[Handler]] = None,
is_default=False,
) -> "SemanticLogger":
"""
Adds a new log source.
:param log_source_name: Name of a source.
:param min_log_level: Minimal log level for this source.
:param log_handlers: Attached log handlers. StreamHandler is used if not provided.
:param is_default: Whether this log source should be used in a `log` method when no log source is explicitly provided.
:return:
"""
new_logger = logging.getLogger(log_source_name)
new_logger.setLevel(min_log_level.value)
if not log_handlers or len(log_handlers) == 0:
log_handlers = [StreamHandler()]
self._loggers.setdefault(log_source_name, new_logger)
if is_default:
self._default_log_source = log_source_name
for log_handler in log_handlers:
new_logger.addHandler(log_handler)
return self
def __getattr__(self, log_source) -> Optional[logging.Logger]:
if log_source in self._loggers:
return self._loggers[log_source]
return None
def _get_logger(self, log_source_name: Optional[str] = None) -> MetadataLogger:
"""
Retrieves a logger by log source name, or a default logger is log source name is not provided.
:param log_source_name: Optional name of a log source.
:return:
"""
assert (
log_source_name or self._default_log_source
), "Argument `log_source` must be provided when no default log source is added. You can add a log source as default by calling `add_log_source(.., is_default=True)`"
if log_source_name:
assert (
log_source_name in self._loggers
), f"{log_source_name} does not have an associated logger. Use add_log_source() to associate a logger with this log source."
return self._loggers[log_source_name or self._default_log_source]
def info(
self,
template: str,
tags: Optional[Dict[str, str]] = None,
log_source_name: Optional[str] = None,
**kwargs,
) -> None:
"""
Sends an INFO level message to configured log sources.
:param template: Message template.
:param tags: Optional message tags.
:param log_source_name: Optional name of a log source, if not using a default.
:param kwargs: Templated arguments (key=value).
:return:
"""
logger = self._get_logger(log_source_name)
self._meta_info(template=template, logger=logger, tags=tags, **kwargs)
def warning(
self,
template: str,
exception: Optional[BaseException] = None,
tags: Optional[Dict[str, str]] = None,
log_source_name: Optional[str] = None,
**kwargs,
) -> None:
"""
Sends a WARNING level message to configured log sources.
:param template: Message template.
:param exception: Exception associated with this warning.
:param tags: Optional message tags.
:param log_source_name: Optional name of a log source, if not using a default.
:param kwargs: Templated arguments (key=value).
:return:
"""
logger = self._get_logger(log_source_name)
self._meta_warning(template=template, logger=logger, tags=tags, exception=exception, **kwargs)
def error(
self,
template: str,
exception: Optional[BaseException] = None,
tags: Optional[Dict[str, str]] = None,
log_source_name: Optional[str] = None,
**kwargs,
) -> None:
"""
Sends an ERROR level message to configured log sources.
:param template: Message template.
:param exception: Exception associated with this error.
:param tags: Optional message tags.
:param log_source_name: Optional name of a log source, if not using a default.
:param kwargs: Templated arguments (key=value).
:return:
"""
logger = self._get_logger(log_source_name)
self._meta_error(template=template, logger=logger, tags=tags, exception=exception, **kwargs)
def debug(
self,
template: str,
exception: Optional[BaseException] = None,
diagnostics: Optional[str] = None, # pylint: disable=R0913
tags: Optional[Dict[str, str]] = None,
log_source_name: Optional[str] = None,
**kwargs,
) -> None:
"""
Sends a DEBUG level message to configured log sources.
:param template: Message template.
:param exception: Exception associated with this error.
:param diagnostics: Optional additional diagnostics info.
:param tags: Optional message tags.
:param log_source_name: Optional name of a log source, if not using a default.
:param kwargs: Templated arguments (key=value).
:return:
"""
logger = self._get_logger(log_source_name)
self._meta_debug(
template=template, logger=logger, tags=tags, exception=exception, diagnostics=diagnostics, **kwargs
)