Robot Framework
logger.py
Go to the documentation of this file.
1 # Copyright 2008-2015 Nokia Networks
2 # Copyright 2016- Robot Framework Foundation
3 #
4 # Licensed under the Apache License, Version 2.0 (the "License");
5 # you may not use this file except in compliance with the License.
6 # You may obtain a copy of the License at
7 #
8 # http://www.apache.org/licenses/LICENSE-2.0
9 #
10 # Unless required by applicable law or agreed to in writing, software
11 # distributed under the License is distributed on an "AS IS" BASIS,
12 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 # See the License for the specific language governing permissions and
14 # limitations under the License.
15 
16 from contextlib import contextmanager
17 import os
18 
19 from robot.errors import DataError
20 
21 from .console import ConsoleOutput
22 from .filelogger import FileLogger
23 from .loggerhelper import AbstractLogger, AbstractLoggerProxy
24 from .stdoutlogsplitter import StdoutLogSplitter
25 
26 
27 
36 
37  def __init__(self, register_console_logger=True):
38  self._console_logger_console_logger = None
39  self._syslog_syslog = None
40  self._xml_logger_xml_logger = None
41  self._listeners_listeners = None
42  self._library_listeners_library_listeners = None
43  self._other_loggers_other_loggers = []
44  self._message_cache_message_cache = []
45  self._log_message_cache_log_message_cache = None
46  self._started_keywords_started_keywords = 0
47  self._error_occurred_error_occurred = False
48  self._error_listener_error_listener = None
49  self._prev_log_message_handlers_prev_log_message_handlers = []
50  self._enabled_enabled = 0
51  self._cache_only_cache_only = False
52  if register_console_logger:
53  self.register_console_loggerregister_console_logger()
54 
55  @property
56  start_loggers = property
57 
58  def start_loggers(self):
59  loggers = [self._console_logger_console_logger, self._syslog_syslog, self._xml_logger_xml_logger,
60  self._listeners_listeners, self._library_listeners_library_listeners]
61  return [logger for logger in self._other_loggers_other_loggers + loggers if logger]
62 
63  @property
64  end_loggers = property
65 
66  def end_loggers(self):
67  loggers = [self._listeners_listeners, self._library_listeners_library_listeners,
68  self._console_logger_console_logger, self._syslog_syslog, self._xml_logger_xml_logger]
69  return [logger for logger in loggers + self._other_loggers_other_loggers if logger]
70 
71  def __iter__(self):
72  return iter(self.end_loggersend_loggersend_loggers)
73 
74  def __enter__(self):
75  if not self._enabled_enabled:
76  self.register_syslogregister_syslog()
77  self._enabled_enabled += 1
78 
79  def __exit__(self, *exc_info):
80  self._enabled_enabled -= 1
81  if not self._enabled_enabled:
82  self.closeclose()
83 
84  def register_console_logger(self, type='verbose', width=78, colors='AUTO',
85  markers='AUTO', stdout=None, stderr=None):
86  logger = ConsoleOutput(type, width, colors, markers, stdout, stderr)
87  self._console_logger_console_logger = self._wrap_and_relay_wrap_and_relay(logger)
88 
89  def _wrap_and_relay(self, logger):
90  logger = LoggerProxy(logger)
91  self._relay_cached_messages_relay_cached_messages(logger)
92  return logger
93 
94  def _relay_cached_messages(self, logger):
95  if self._message_cache_message_cache:
96  for msg in self._message_cache_message_cache[:]:
97  logger.message(msg)
98 
100  self._console_logger_console_logger = None
101 
102  def register_syslog(self, path=None, level='INFO'):
103  if not path:
104  path = os.environ.get('ROBOT_SYSLOG_FILE', 'NONE')
105  level = os.environ.get('ROBOT_SYSLOG_LEVEL', level)
106  if path.upper() == 'NONE':
107  return
108  try:
109  syslog = FileLogger(path, level)
110  except DataError as err:
111  self.errorerror("Opening syslog file '%s' failed: %s" % (path, err.message))
112  else:
113  self._syslog_syslog = self._wrap_and_relay_wrap_and_relay(syslog)
114 
115  def register_xml_logger(self, logger):
116  self._xml_logger_xml_logger = self._wrap_and_relay_wrap_and_relay(logger)
117 
119  self._xml_logger_xml_logger = None
120 
121  def register_listeners(self, listeners, library_listeners):
122  self._listeners_listeners = listeners
123  self._library_listeners_library_listeners = library_listeners
124  if listeners:
125  self._relay_cached_messages_relay_cached_messages(listeners)
126 
127  def register_logger(self, *loggers):
128  for logger in loggers:
129  logger = self._wrap_and_relay_wrap_and_relay(logger)
130  self._other_loggers_other_loggers.append(logger)
131 
132  def unregister_logger(self, *loggers):
133  for logger in loggers:
134  self._other_loggers_other_loggers = [proxy for proxy in self._other_loggers_other_loggers
135  if proxy.logger is not logger]
136 
138  self._message_cache_message_cache = None
139 
140  def register_error_listener(self, listener):
141  self._error_listener_error_listener = listener
142  if self._error_occurred_error_occurred:
143  listener()
144 
145 
146  def message(self, msg):
147  if not self._cache_only_cache_only:
148  for logger in self:
149  logger.message(msg)
150  if self._message_cache_message_cache is not None:
151  self._message_cache_message_cache.append(msg)
152  if msg.level == 'ERROR':
153  self._error_occurred_error_occurred = True
154  if self._error_listener_error_listener:
155  self._error_listener_error_listener()
156 
157  @property
158  @contextmanager
159  cache_only = property
160 
161  def cache_only(self):
162  self._cache_only_cache_only = True
163  try:
164  yield
165  finally:
166  self._cache_only_cache_only = False
167 
168  @property
169  @contextmanager
170  delayed_logging = property
171 
172  def delayed_logging(self):
173  prev_cache = self._log_message_cache_log_message_cache
174  self._log_message_cache_log_message_cache = []
175  try:
176  yield
177  finally:
178  messages = self._log_message_cache_log_message_cache
179  self._log_message_cache_log_message_cache = prev_cache
180  for msg in messages or ():
181  self._log_message_log_message(msg, no_cache=True)
182 
183 
186  def _log_message(self, msg, no_cache=False):
187  if self._log_message_cache_log_message_cache is not None and not no_cache:
188  msg.resolve_delayed_message()
189  self._log_message_cache_log_message_cache.append(msg)
190  return
191  for logger in self:
192  logger.log_message(msg)
193  if msg.level in ('WARN', 'ERROR'):
194  self.messagemessagemessage(msg)
195 
196  log_message = message
197 
198  def log_output(self, output):
199  for msg in StdoutLogSplitter(output):
200  self.log_messagelog_messagelog_message(msg)
201 
203  self._prev_log_message_handlers_prev_log_message_handlers.append(self.log_messagelog_messagelog_message)
204  self.log_messagelog_messagelog_message = self.messagemessagemessage
205 
207  self.log_messagelog_messagelog_message = self._prev_log_message_handlers_prev_log_message_handlers.pop()
208 
209  def start_suite(self, suite):
210  for logger in self.start_loggersstart_loggersstart_loggers:
211  logger.start_suite(suite)
212 
213  def end_suite(self, suite):
214  for logger in self.end_loggersend_loggersend_loggers:
215  logger.end_suite(suite)
216 
217  def start_test(self, test):
218  for logger in self.start_loggersstart_loggersstart_loggers:
219  logger.start_test(test)
220 
221  def end_test(self, test):
222  for logger in self.end_loggersend_loggersend_loggers:
223  logger.end_test(test)
224 
225  def start_keyword(self, keyword):
226  # TODO: Could _prev_log_message_handlers be used also here?
227  self._started_keywords_started_keywords += 1
228  self.log_messagelog_messagelog_message = self._log_message_log_message
229  for logger in self.start_loggersstart_loggersstart_loggers:
230  logger.start_keyword(keyword)
231 
232  def end_keyword(self, keyword):
233  self._started_keywords_started_keywords -= 1
234  for logger in self.end_loggersend_loggersend_loggers:
235  logger.end_keyword(keyword)
236  if not self._started_keywords_started_keywords:
237  self.log_messagelog_messagelog_message = self.messagemessagemessage
238 
239  def imported(self, import_type, name, **attrs):
240  for logger in self:
241  logger.imported(import_type, name, attrs)
242 
243 
244  def output_file(self, file_type, path):
245  for logger in self:
246  logger.output_file(file_type, path)
247 
248  def close(self):
249  for logger in self:
250  logger.close()
251  self.__init____init____init__(register_console_logger=False)
252 
253 
255 
258  _methods = ('start_suite', 'end_suite', 'start_test', 'end_test',
259  'start_keyword', 'end_keyword', 'message', 'log_message',
260  'imported', 'output_file', 'close')
261 
262 
265  _start_keyword_methods = {
266  'For': 'start_for',
267  'ForIteration': 'start_for_iteration',
268  'While': 'start_while',
269  'WhileIteration': 'start_while_iteration',
270  'If': 'start_if',
271  'IfBranch': 'start_if_branch',
272  'Try': 'start_try',
273  'TryBranch': 'start_try_branch',
274  'Return': 'start_return',
275  'Continue': 'start_continue',
276  'Break': 'start_break',
277  }
278 
281  _end_keyword_methods = {
282  'For': 'end_for',
283  'ForIteration': 'end_for_iteration',
284  'While': 'end_while',
285  'WhileIteration': 'end_while_iteration',
286  'If': 'end_if',
287  'IfBranch': 'end_if_branch',
288  'Try': 'end_try',
289  'TryBranch': 'end_try_branch',
290  'Return': 'end_return',
291  'Continue': 'end_continue',
292  'Break': 'end_break',
293  }
294 
295  def start_keyword(self, kw):
296  # Dispatch start_keyword calls to more precise methods when logger
297  # implements them. This horrible hack is needed because internal logger
298  # knows only about keywords. It should be rewritten.
299  name = self._start_keyword_methods_start_keyword_methods.get(type(kw.result).__name__)
300  if name and hasattr(self.loggerlogger, name):
301  method = getattr(self.loggerlogger, name)
302  else:
303  method = self.loggerlogger.start_keyword
304  method(kw)
305 
306  def end_keyword(self, kw):
307  # See start_keyword comment for explanation of this horrible hack.
308  name = self._end_keyword_methods_end_keyword_methods.get(type(kw.result).__name__)
309  if name and hasattr(self.loggerlogger, name):
310  method = getattr(self.loggerlogger, name)
311  else:
312  method = self.loggerlogger.end_keyword
313  method(kw)
314 
315 
316 LOGGER = Logger()
def start_keyword(self, kw)
Definition: logger.py:295
dictionary _start_keyword_methods
Definition: logger.py:265
A global logger proxy to delegating messages to registered loggers.
Definition: logger.py:35
def register_syslog(self, path=None, level='INFO')
Definition: logger.py:102
def end_suite(self, suite)
Definition: logger.py:213
def __init__(self, register_console_logger=True)
Definition: logger.py:37
def register_listeners(self, listeners, library_listeners)
Definition: logger.py:121
def __exit__(self, *exc_info)
Definition: logger.py:79
def unregister_console_logger(self)
Definition: logger.py:99
def start_loggers(self)
Definition: logger.py:58
def _relay_cached_messages(self, logger)
Definition: logger.py:94
def start_keyword(self, keyword)
Definition: logger.py:225
def enable_library_import_logging(self)
Definition: logger.py:202
def disable_library_import_logging(self)
Definition: logger.py:206
def _log_message(self, msg, no_cache=False)
Log messages written (mainly) by libraries.
Definition: logger.py:186
def end_test(self, test)
Definition: logger.py:221
def register_logger(self, *loggers)
Definition: logger.py:127
def register_error_listener(self, listener)
Definition: logger.py:140
def unregister_logger(self, *loggers)
Definition: logger.py:132
def imported(self, import_type, name, **attrs)
Definition: logger.py:239
def _wrap_and_relay(self, logger)
Definition: logger.py:89
def disable_message_cache(self)
Definition: logger.py:137
def register_console_logger(self, type='verbose', width=78, colors='AUTO', markers='AUTO', stdout=None, stderr=None)
Definition: logger.py:85
def log_output(self, output)
Definition: logger.py:198
def output_file(self, file_type, path)
Finished output, report, log, debug, or xunit file.
Definition: logger.py:244
def end_keyword(self, keyword)
Definition: logger.py:232
def register_xml_logger(self, logger)
Definition: logger.py:115
def unregister_xml_logger(self)
Definition: logger.py:118
def start_test(self, test)
Definition: logger.py:217
def start_suite(self, suite)
Definition: logger.py:209
def message(self, msg)
Messages about what the framework is doing, warnings, errors, ...
Definition: logger.py:146
def __init__(self, level='TRACE')
Definition: loggerhelper.py:35
Splits messages logged through stdout (or stderr) into Message objects.
def ConsoleOutput(type='verbose', width=78, colors='AUTO', markers='AUTO', stdout=None, stderr=None)
Definition: __init__.py:24