146
188
if size <= 4 << 20:
148
190
old_fname = trace_fname + '.old'
149
from osutils import rename
150
rename(trace_fname, old_fname)
191
osutils.rename(trace_fname, old_fname)
155
def open_tracefile(tracefilename=None):
156
# Messages are always written to here, so that we have some
157
# information if something goes wrong. In a future version this
158
# file will be removed on successful completion.
159
global _file_handler, _bzr_log_file, _bzr_log_filename
162
if tracefilename is None:
196
def _get_bzr_log_filename():
197
bzr_log = os.environ.get('BZR_LOG')
200
home = os.environ.get('BZR_HOME')
163
202
if sys.platform == 'win32':
164
203
from bzrlib import win32utils
165
204
home = win32utils.get_home_location()
167
206
home = os.path.expanduser('~')
168
_bzr_log_filename = os.path.join(home, '.bzr.log')
170
_bzr_log_filename = tracefilename
172
_bzr_log_filename = os.path.expanduser(_bzr_log_filename)
207
return os.path.join(home, '.bzr.log')
211
"""Open the .bzr.log trace file.
213
If the log is more than a particular length, the old file is renamed to
214
.bzr.log.old and a new file is started. Otherwise, we append to the
217
This sets the global _bzr_log_filename.
219
global _bzr_log_filename
221
def _open_or_create_log_file(filename):
222
"""Open existing log file, or create with ownership and permissions
224
It inherits the ownership and permissions (masked by umask) from
225
the containing directory to cope better with being run under sudo
226
with $HOME still set to the user's homedir.
228
flags = os.O_WRONLY | os.O_APPEND | osutils.O_TEXT
231
fd = os.open(filename, flags)
234
if e.errno != errno.ENOENT:
237
fd = os.open(filename, flags | os.O_CREAT | os.O_EXCL, 0666)
239
if e.errno != errno.EEXIST:
242
osutils.copy_ownership_from_path(filename)
244
return os.fdopen(fd, 'at', 0) # unbuffered
247
_bzr_log_filename = _get_bzr_log_filename()
173
248
_rollover_trace_maybe(_bzr_log_filename)
176
#tf = codecs.open(trace_fname, 'at', 'utf8', buffering=LINE_BUFFERED)
177
tf = open(_bzr_log_filename, 'at', LINE_BUFFERED)
179
# tf.tell() on windows always return 0 until some writing done
182
tf.write("this is a debug log for diagnosing/reporting problems in bzr\n")
183
tf.write("you can delete or truncate this file, or include sections in\n")
184
tf.write("bug reports to bazaar@lists.canonical.com\n\n")
185
_file_handler = logging.StreamHandler(tf)
186
fmt = r'[%(process)5d] %(asctime)s.%(msecs)03d %(levelname)s: %(message)s'
187
datefmt = r'%a %H:%M:%S'
188
_file_handler.setFormatter(logging.Formatter(fmt, datefmt))
189
_file_handler.setLevel(logging.DEBUG)
190
logging.getLogger('').addHandler(_file_handler)
192
warning("failed to open trace file: %s" % (e))
250
bzr_log_file = _open_or_create_log_file(_bzr_log_filename)
251
bzr_log_file.write('\n')
252
if bzr_log_file.tell() <= 2:
253
bzr_log_file.write("this is a debug log for diagnosing/reporting problems in bzr\n")
254
bzr_log_file.write("you can delete or truncate this file, or include sections in\n")
255
bzr_log_file.write("bug reports to https://bugs.launchpad.net/bzr/+filebug\n\n")
259
except EnvironmentError, e:
260
# If we are failing to open the log, then most likely logging has not
261
# been set up yet. So we just write to stderr rather than using
262
# 'warning()'. If we using warning(), users get the unhelpful 'no
263
# handlers registered for "bzr"' when something goes wrong on the
264
# server. (bug #503886)
265
sys.stderr.write("failed to open trace file: %s\n" % (e,))
266
# TODO: What should happen if we fail to open the trace file? Maybe the
267
# objects should be pointed at /dev/null or the equivalent? Currently
268
# returns None which will cause failures later.
272
def enable_default_logging():
273
"""Configure default logging: messages to stderr and debug to .bzr.log
275
This should only be called once per process.
277
Non-command-line programs embedding bzrlib do not need to call this. They
278
can instead either pass a file to _push_log_file, or act directly on
279
logging.getLogger("bzr").
281
Output can be redirected away by calling _push_log_file.
283
:return: A memento from push_log_file for restoring the log state.
285
start_time = osutils.format_local_date(_bzr_log_start_time,
287
bzr_log_file = _open_bzr_log()
288
if bzr_log_file is not None:
289
bzr_log_file.write(start_time.encode('utf-8') + '\n')
290
memento = push_log_file(bzr_log_file,
291
r'[%(process)5d] %(asctime)s.%(msecs)03d %(levelname)s: %(message)s',
292
r'%Y-%m-%d %H:%M:%S')
293
# after hooking output into bzr_log, we also need to attach a stderr
294
# handler, writing only at level info and with encoding
295
stderr_handler = EncodedStreamHandler(sys.stderr,
296
osutils.get_terminal_encoding(), 'replace', level=logging.INFO)
297
logging.getLogger('bzr').addHandler(stderr_handler)
301
def push_log_file(to_file, log_format=None, date_format=None):
302
"""Intercept log and trace messages and send them to a file.
304
:param to_file: A file-like object to which messages will be sent.
306
:returns: A memento that should be passed to _pop_log_file to restore the
307
previously active logging.
311
new_handler = EncodedStreamHandler(to_file, "utf-8", level=logging.DEBUG)
312
if log_format is None:
313
log_format = '%(levelname)8s %(message)s'
314
new_handler.setFormatter(logging.Formatter(log_format, date_format))
315
# save and remove any existing log handlers
316
bzr_logger = logging.getLogger('bzr')
317
old_handlers = bzr_logger.handlers[:]
318
del bzr_logger.handlers[:]
319
# set that as the default logger
320
bzr_logger.addHandler(new_handler)
321
bzr_logger.setLevel(logging.DEBUG)
322
# TODO: check if any changes are needed to the root logger
324
# TODO: also probably need to save and restore the level on bzr_logger.
325
# but maybe we can avoid setting the logger level altogether, and just set
326
# the level on the handler?
328
# save the old trace file
329
old_trace_file = _trace_file
330
# send traces to the new one
331
_trace_file = to_file
332
result = new_handler, _trace_file
333
return ('log_memento', old_handlers, new_handler, old_trace_file, to_file)
336
def pop_log_file((magic, old_handlers, new_handler, old_trace_file, new_trace_file)):
337
"""Undo changes to logging/tracing done by _push_log_file.
339
This flushes, but does not close the trace file (so that anything that was
342
Takes the memento returned from _push_log_file."""
344
_trace_file = old_trace_file
345
bzr_logger = logging.getLogger('bzr')
346
bzr_logger.removeHandler(new_handler)
347
# must be closed, otherwise logging will try to close it at exit, and the
348
# file will likely already be closed underneath.
350
bzr_logger.handlers = old_handlers
351
if new_trace_file is not None:
352
new_trace_file.flush()
195
355
def log_exception_quietly():
196
356
"""Log the last exception to the trace file only.
198
Used for exceptions that occur internally and that may be
199
interesting to developers but not to users. For example,
358
Used for exceptions that occur internally and that may be
359
interesting to developers but not to users. For example,
200
360
errors loading plugins.
203
362
mutter(traceback.format_exc())
206
def enable_default_logging():
207
"""Configure default logging to stderr and .bzr.log"""
208
# FIXME: if this is run twice, things get confused
209
global _stderr_handler, _file_handler, _trace_file, _bzr_log_file
210
_stderr_handler = logging.StreamHandler()
211
logging.getLogger('').addHandler(_stderr_handler)
212
_stderr_handler.setLevel(logging.INFO)
213
if not _file_handler:
215
_trace_file = _bzr_log_file
217
_file_handler.setLevel(logging.DEBUG)
218
_bzr_logger.setLevel(logging.DEBUG)
365
def set_verbosity_level(level):
366
"""Set the verbosity level.
368
:param level: -ve for quiet, 0 for normal, +ve for verbose
370
global _verbosity_level
371
_verbosity_level = level
372
_update_logging_level(level < 0)
373
ui.ui_factory.be_quiet(level < 0)
376
def get_verbosity_level():
377
"""Get the verbosity level.
379
See set_verbosity_level() for values.
381
return _verbosity_level
221
384
def be_quiet(quiet=True):
222
global _stderr_handler, _stderr_quiet
224
_stderr_quiet = quiet
226
_stderr_handler.setLevel(logging.WARNING)
228
_stderr_handler.setLevel(logging.INFO)
386
set_verbosity_level(-1)
388
set_verbosity_level(0)
391
def _update_logging_level(quiet=True):
392
"""Hide INFO messages if quiet."""
394
_bzr_logger.setLevel(logging.WARNING)
396
_bzr_logger.setLevel(logging.INFO)
236
def disable_default_logging():
237
"""Turn off default log handlers.
239
This is intended to be used by the test framework, which doesn't
240
want leakage from the code-under-test into the main logs.
243
l = logging.getLogger('')
244
l.removeHandler(_stderr_handler)
246
l.removeHandler(_file_handler)
250
def enable_test_log(to_file):
251
"""Redirect logging to a temporary file for a test
253
returns an opaque reference that should be passed to disable_test_log
254
after the test completes.
256
disable_default_logging()
259
hdlr = logging.StreamHandler(to_file)
260
hdlr.setLevel(logging.DEBUG)
261
hdlr.setFormatter(logging.Formatter('%(levelname)8s %(message)s'))
262
_bzr_logger.addHandler(hdlr)
263
_bzr_logger.setLevel(logging.DEBUG)
264
result = hdlr, _trace_file, _trace_depth
265
_trace_file = to_file
270
def disable_test_log((test_log_hdlr, old_trace_file, old_trace_depth)):
271
_bzr_logger.removeHandler(test_log_hdlr)
272
test_log_hdlr.close()
275
_trace_file = old_trace_file
276
_trace_depth = old_trace_depth
278
enable_default_logging()
400
"""Is the verbosity level negative?"""
401
return _verbosity_level < 0
405
"""Is the verbosity level positive?"""
406
return _verbosity_level > 0
409
def debug_memory(message='', short=True):
410
"""Write out a memory dump."""
411
if sys.platform == 'win32':
412
from bzrlib import win32utils
413
win32utils.debug_memory_win32api(message=message, short=short)
415
_debug_memory_proc(message=message, short=short)
418
_short_fields = ('VmPeak', 'VmSize', 'VmRSS')
420
def _debug_memory_proc(message='', short=True):
422
status_file = file('/proc/%s/status' % os.getpid(), 'rb')
426
status = status_file.read()
431
for line in status.splitlines():
435
for field in _short_fields:
436
if line.startswith(field):
440
def _dump_memory_usage(err_file):
443
fd, name = tempfile.mkstemp(prefix="bzr_memdump", suffix=".json")
444
dump_file = os.fdopen(fd, 'w')
445
from meliae import scanner
446
scanner.dump_gc_objects(dump_file)
447
err_file.write("Memory dumped to %s\n" % name)
449
err_file.write("Dumping memory requires meliae module.\n")
450
log_exception_quietly()
452
err_file.write("Exception while dumping memory.\n")
453
log_exception_quietly()
455
if dump_file is not None:
461
def _qualified_exception_name(eclass, unqualified_bzrlib_errors=False):
462
"""Give name of error class including module for non-builtin exceptions
464
If `unqualified_bzrlib_errors` is True, errors specific to bzrlib will
465
also omit the module prefix.
467
class_name = eclass.__name__
468
module_name = eclass.__module__
469
if module_name in ("exceptions", "__main__") or (
470
unqualified_bzrlib_errors and module_name == "bzrlib.errors"):
472
return "%s.%s" % (module_name, class_name)
281
475
def report_exception(exc_info, err_file):
282
exc_type, exc_object, exc_tb = exc_info
476
"""Report an exception to err_file (typically stderr) and to .bzr.log.
478
This will show either a full traceback or a short message as appropriate.
480
:return: The appropriate exit code for this error.
283
482
# Log the full traceback to ~/.bzr.log
284
483
log_exception_quietly()
285
if (isinstance(exc_object, IOError)
286
and getattr(exc_object, 'errno', None) == errno.EPIPE):
287
print >>err_file, "bzr: broken pipe"
288
elif isinstance(exc_object, KeyboardInterrupt):
289
print >>err_file, "bzr: interrupted"
484
if 'error' in debug.debug_flags:
485
print_exception(exc_info, err_file)
486
return errors.EXIT_ERROR
487
exc_type, exc_object, exc_tb = exc_info
488
if isinstance(exc_object, KeyboardInterrupt):
489
err_file.write("bzr: interrupted\n")
490
return errors.EXIT_ERROR
491
elif isinstance(exc_object, MemoryError):
492
err_file.write("bzr: out of memory\n")
493
if 'mem_dump' in debug.debug_flags:
494
_dump_memory_usage(err_file)
496
err_file.write("Use -Dmem_dump to dump memory to a file.\n")
497
return errors.EXIT_ERROR
498
elif isinstance(exc_object, ImportError) \
499
and str(exc_object).startswith("No module named "):
500
report_user_error(exc_info, err_file,
501
'You may need to install this Python library separately.')
502
return errors.EXIT_ERROR
290
503
elif not getattr(exc_object, 'internal_error', True):
291
504
report_user_error(exc_info, err_file)
292
elif isinstance(exc_object, (OSError, IOError)):
505
return errors.EXIT_ERROR
506
elif osutils.is_environment_error(exc_object):
507
if getattr(exc_object, 'errno', None) == errno.EPIPE:
508
err_file.write("bzr: broken pipe\n")
509
return errors.EXIT_ERROR
293
510
# Might be nice to catch all of these and show them as something more
294
511
# specific, but there are too many cases at the moment.
295
512
report_user_error(exc_info, err_file)
513
return errors.EXIT_ERROR
297
515
report_bug(exc_info, err_file)
516
return errors.EXIT_INTERNAL_ERROR
519
def print_exception(exc_info, err_file):
520
exc_type, exc_object, exc_tb = exc_info
521
err_file.write("bzr: ERROR: %s.%s: %s\n" % (
522
exc_type.__module__, exc_type.__name__, exc_object))
524
traceback.print_exception(exc_type, exc_object, exc_tb, file=err_file)
300
527
# TODO: Should these be specially encoding the output?
301
def report_user_error(exc_info, err_file):
528
def report_user_error(exc_info, err_file, advice=None):
302
529
"""Report to err_file an error that's not an internal error.
304
531
These don't get a traceback unless -Derror was given.
533
:param exc_info: 3-tuple from sys.exc_info()
534
:param advice: Extra advice to the user to be printed following the
306
if 'error' in debug.debug_flags:
307
report_bug(exc_info, err_file)
309
print >>err_file, "bzr: ERROR:", str(exc_info[1])
537
err_file.write("bzr: ERROR: %s\n" % (exc_info[1],))
539
err_file.write("%s\n" % (advice,))
312
542
def report_bug(exc_info, err_file):
313
543
"""Report an exception that probably indicates a bug in bzr"""
315
exc_type, exc_object, exc_tb = exc_info
316
print >>err_file, "bzr: ERROR: %s.%s: %s" % (
317
exc_type.__module__, exc_type.__name__, exc_object)
319
traceback.print_exception(exc_type, exc_object, exc_tb, file=err_file)
321
print >>err_file, 'bzr %s on python %s (%s)' % \
323
'.'.join(map(str, sys.version_info)),
325
print >>err_file, 'arguments: %r' % sys.argv
327
print >>err_file, "** please send this report to bazaar@lists.ubuntu.com"
544
from bzrlib.crash import report_bug
545
report_bug(exc_info, err_file)
548
def _flush_stdout_stderr():
549
# called from the bzrlib library finalizer returned by bzrlib.initialize()
553
except ValueError, e:
554
# On Windows, I get ValueError calling stdout.flush() on a closed
559
if e.errno in [errno.EINVAL, errno.EPIPE]:
566
# called from the bzrlib library finalizer returned by bzrlib.initialize()
572
class EncodedStreamHandler(logging.Handler):
573
"""Robustly write logging events to a stream using the specified encoding
575
Messages are expected to be formatted to unicode, but UTF-8 byte strings
576
are also accepted. An error during formatting or a str message in another
577
encoding will be quitely noted as an error in the Bazaar log file.
579
The stream is not closed so sys.stdout or sys.stderr may be passed.
582
def __init__(self, stream, encoding=None, errors='strict', level=0):
583
logging.Handler.__init__(self, level)
586
encoding = getattr(stream, "encoding", "ascii")
587
self.encoding = encoding
591
flush = getattr(self.stream, "flush", None)
592
if flush is not None:
595
def emit(self, record):
597
line = self.format(record)
598
if not isinstance(line, unicode):
599
line = line.decode("utf-8")
600
self.stream.write(line.encode(self.encoding, self.errors) + "\n")
602
log_exception_quietly()
603
# Try saving the details that would have been logged in some form
604
msg = args = "<Unformattable>"
606
msg = repr(record.msg).encode("ascii")
607
args = repr(record.args).encode("ascii")
610
# Using mutter() bypasses the logging module and writes directly
611
# to the file so there's no danger of getting into a loop here.
612
mutter("Logging record unformattable: %s %% %s", msg, args)
615
class Config(object):
616
"""Configuration of message tracing in bzrlib.
618
This implements the context manager protocol and should manage any global
619
variables still used. The default config used is DefaultConfig, but
620
embedded uses of bzrlib may wish to use a custom manager.
624
return self # This is bound to the 'as' clause in a with statement.
626
def __exit__(self, exc_type, exc_val, exc_tb):
627
return False # propogate exceptions.
630
class DefaultConfig(Config):
631
"""A default configuration for tracing of messages in bzrlib.
633
This implements the context manager protocol.
637
self._original_filename = _bzr_log_filename
638
self._original_state = enable_default_logging()
639
return self # This is bound to the 'as' clause in a with statement.
641
def __exit__(self, exc_type, exc_val, exc_tb):
642
pop_log_file(self._original_state)
643
global _bzr_log_filename
644
_bzr_log_filename = self._original_filename
645
return False # propogate exceptions.