0005785: print statements in Check.py causing a crash in Windows (pythonw.exe) after outputting 4096 characters. Setup logging in gramps.py before anything else; for Windows divert stderr/stdout to a log file; use logging instead of print for errors or warnings.

svn: r20013
This commit is contained in:
Tim G L Lyons 2012-07-16 22:54:43 +00:00
parent 91586b708c
commit b377893a73
2 changed files with 75 additions and 49 deletions

View File

@ -34,6 +34,7 @@ import gettext
import sys import sys
import os import os
import locale import locale
import logging
#------------------------------------------------------------------------- #-------------------------------------------------------------------------
# #
@ -58,8 +59,9 @@ else:
if lang and lang[:2] == 'en': if lang and lang[:2] == 'en':
pass # No need to display warning, we're in English pass # No need to display warning, we're in English
else: else:
print 'Locale dir does not exist at ' + os.path.join(const.PREFIXDIR, "share/locale") logging.warning('Locale dir does not exist at ' +
print 'Running ./configure --prefix=YourPrefixDir might fix the problem' os.path.join(const.PREFIXDIR, "share/locale"))
logging.warning('Running ./configure --prefix=YourPrefixDir might fix the problem')
LOCALEDIR = None LOCALEDIR = None
LOCALEDOMAIN = 'gramps' LOCALEDOMAIN = 'gramps'
@ -77,7 +79,7 @@ else:
try: try:
lang = locale.getdefaultlocale()[0] + '.UTF-8' lang = locale.getdefaultlocale()[0] + '.UTF-8'
except TypeError: except TypeError:
print 'Unable to determine your Locale, using English' logging.warning('Unable to determine your Locale, using English')
lang = 'en.UTF-8' lang = 'en.UTF-8'
os.environ["LANG"] = lang os.environ["LANG"] = lang
@ -100,7 +102,7 @@ def setup_gettext():
try: try:
locale.bindtextdomain(LOCALEDOMAIN, LOCALEDIR) locale.bindtextdomain(LOCALEDOMAIN, LOCALEDIR)
except ValueError: except ValueError:
print 'Failed to bind text domain, gtk.Builder() has no translation' logging.warning('Failed to bind text domain, gtk.Builder() has no translation')
#following installs _ as a python function, we avoid this as TransUtils is #following installs _ as a python function, we avoid this as TransUtils is
#used sometimes: #used sometimes:
@ -217,8 +219,8 @@ def setup_windows_gettext():
return return
# No complete/working translation found # No complete/working translation found
print "Translation might not be complete/not working for",\ logging.warning("Translation might not be complete/not working for",\
locale.getlocale()[0] locale.getlocale()[0])
def get_localedomain(): def get_localedomain():

View File

@ -31,7 +31,6 @@
#------------------------------------------------------------------------- #-------------------------------------------------------------------------
import sys import sys
import os import os
import const
import signal import signal
import gettext import gettext
_ = gettext.gettext _ = gettext.gettext
@ -47,9 +46,68 @@ from subprocess import Popen, PIPE
# GRAMPS modules # GRAMPS modules
# #
#------------------------------------------------------------------------- #-------------------------------------------------------------------------
import const
import constfunc
#-------------------------------------------------------------------------
#
# Setup logging
#
# Ideally, this needs to be done before any Gramps modules are imported, so that
# any code that is executed as the modules are imported can log errors or
# warnings. Errors and warnings are particularly applicable to TransUtils and
# setting up internationalisation in this module. const and constfunc have to be
# imported before this code is executed because they are used in this code.
#-------------------------------------------------------------------------
"""Setup basic logging support."""
# Setup a formatter
form = logging.Formatter(fmt="%(relativeCreated)d: %(levelname)s: %(filename)s: line %(lineno)d: %(message)s")
# Create the log handlers
if constfunc.win():
# If running in GUI mode redirect stdout and stderr to log file
if hasattr(sys.stdout, "fileno") and sys.stdout.fileno() < 0:
logfile = os.path.join(const.HOME_DIR,
"Gramps%s%s.log") % (const.VERSION_TUPLE[0],
const.VERSION_TUPLE[1])
# We now carry out the first step in build_user_paths(), to make sure
# that the user home directory is available to store the log file. When
# build_user_paths() is called, the call is protected by a try...except
# block, and any failure will be logged. However, if the creation of the
# user directory fails here, there is no way to report the failure,
# because stdout/stderr are not available, and neither is the logfile.
if os.path.islink(const.HOME_DIR):
pass # ok
elif not os.path.isdir(const.HOME_DIR):
os.makedirs(const.HOME_DIR)
sys.stdout = sys.stderr = open(logfile, "w")
stderrh = logging.StreamHandler(sys.stderr)
stderrh.setFormatter(form)
stderrh.setLevel(logging.DEBUG)
# Setup the base level logger, this one gets
# everything.
l = logging.getLogger()
l.setLevel(logging.WARNING)
l.addHandler(stderrh)
# put a hook on to catch any completely unhandled exceptions.
def exc_hook(type, value, tb):
if type == KeyboardInterrupt:
# Ctrl-C is not a bug.
return
if type == IOError:
# strange Windows logging error on close
return
import traceback
LOG.error("Unhandled exception\n" +
"".join(traceback.format_exception(type, value, tb)))
sys.excepthook = exc_hook
from gen.mime import mime_type_is_defined from gen.mime import mime_type_is_defined
import TransUtils import TransUtils
import constfunc
#------------------------------------------------------------------------- #-------------------------------------------------------------------------
# #
# Load internationalization setup # Load internationalization setup
@ -64,9 +122,8 @@ gettext.bindtextdomain(TransUtils.LOCALEDOMAIN, TransUtils.LOCALEDIR)
try: try:
locale.setlocale(locale.LC_ALL,'') locale.setlocale(locale.LC_ALL,'')
except: except:
# FIXME: This should use LOG.warn, but logging has not been initialised yet logging.warning(_("WARNING: Setting locale failed. Please fix the "
print >> sys.stderr, _("WARNING: Setting locale failed. Please fix the " "LC_* and/or the LANG environment variables to prevent this error"))
"LC_* and/or the LANG environment variables to prevent this error")
try: try:
# It is probably not necessary to set the locale to 'C' # It is probably not necessary to set the locale to 'C'
# because the locale will just stay at whatever it was, # because the locale will just stay at whatever it was,
@ -75,7 +132,7 @@ except:
# functions are working # functions are working
locale.setlocale(locale.LC_ALL,'C') locale.setlocale(locale.LC_ALL,'C')
except: except:
print >> sys.stderr, _("ERROR: Setting the 'C' locale didn't work either") logging.warning(_("ERROR: Setting the 'C' locale didn't work either"))
# FIXME: This should propagate the exception, # FIXME: This should propagate the exception,
# if that doesn't break Gramps under Windows # if that doesn't break Gramps under Windows
# raise # raise
@ -94,7 +151,7 @@ elif operating_system == 'FreeBSD':
try: try:
gettext.bindtextdomain(TransUtils.LOCALEDOMAIN, TransUtils.LOCALEDIR) gettext.bindtextdomain(TransUtils.LOCALEDOMAIN, TransUtils.LOCALEDIR)
except locale.Error: except locale.Error:
print 'No translation in some gtk.Builder strings, ' logging.warning('No translation in some gtk.Builder strings, ')
elif operating_system == 'OpenBSD': elif operating_system == 'OpenBSD':
pass pass
else: # normal case else: # normal case
@ -102,7 +159,7 @@ else: # normal case
locale.bindtextdomain(TransUtils.LOCALEDOMAIN, TransUtils.LOCALEDIR) locale.bindtextdomain(TransUtils.LOCALEDOMAIN, TransUtils.LOCALEDIR)
#locale.textdomain(TransUtils.LOCALEDOMAIN) #locale.textdomain(TransUtils.LOCALEDOMAIN)
except locale.Error: except locale.Error:
print 'No translation in some gtk.Builder strings, ' logging.warning('No translation in some gtk.Builder strings, ')
#------------------------------------------------------------------------- #-------------------------------------------------------------------------
# #
@ -112,7 +169,7 @@ else: # normal case
MIN_PYTHON_VERSION = (2, 6, 0, '', 0) MIN_PYTHON_VERSION = (2, 6, 0, '', 0)
if not sys.version_info >= MIN_PYTHON_VERSION : if not sys.version_info >= MIN_PYTHON_VERSION :
print (_("Your Python version does not meet the " logging.warning(_("Your Python version does not meet the "
"requirements. At least python %d.%d.%d is needed to" "requirements. At least python %d.%d.%d is needed to"
" start Gramps.\n\n" " start Gramps.\n\n"
"Gramps will terminate now.") % ( "Gramps will terminate now.") % (
@ -148,37 +205,6 @@ except:
args = sys.argv args = sys.argv
def setup_logging():
"""Setup basic logging support."""
# Setup a formatter
form = logging.Formatter(fmt="%(relativeCreated)d: %(levelname)s: %(filename)s: line %(lineno)d: %(message)s")
# Create the log handlers
stderrh = logging.StreamHandler(sys.stderr)
stderrh.setFormatter(form)
stderrh.setLevel(logging.DEBUG)
# Setup the base level logger, this one gets
# everything.
l = logging.getLogger()
l.setLevel(logging.WARNING)
l.addHandler(stderrh)
# put a hook on to catch any completely unhandled exceptions.
def exc_hook(type, value, tb):
if type == KeyboardInterrupt:
# Ctrl-C is not a bug.
return
if type == IOError:
# strange Windows logging error on close
return
import traceback
LOG.error("Unhandled exception\n" +
"".join(traceback.format_exception(type, value, tb)))
sys.excepthook = exc_hook
def build_user_paths(): def build_user_paths():
""" check/make user-dirs on each Gramps session""" """ check/make user-dirs on each Gramps session"""
for path in const.USER_DIRLIST: for path in const.USER_DIRLIST:
@ -346,8 +372,6 @@ def show_settings():
def run(): def run():
error = [] error = []
setup_logging()
try: try:
build_user_paths() build_user_paths()
except OSError, msg: except OSError, msg:
@ -442,4 +466,4 @@ def run():
errors = run() errors = run()
if errors and isinstance(errors, list): if errors and isinstance(errors, list):
for error in errors: for error in errors:
print error[0], error[1] logging.warning(error[0] + error[1])