123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- #!/usr/bin/env python
- # -*- Mode: Python -*-
- # GObject-Introspection - a framework for introspecting GObject libraries
- # Copyright (C) 2010 Red Hat, Inc.
- # Copyright (C) 2010 Johan Dahlin
- #
- # This program is free software; you can redistribute it and/or
- # modify it under the terms of the GNU General Public License
- # as published by the Free Software Foundation; either version 2
- # of the License, or (at your option) any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program; if not, write to the Free Software
- # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- # 02110-1301, USA.
- #
- from __future__ import absolute_import
- from __future__ import division
- from __future__ import print_function
- from __future__ import unicode_literals
- import os
- import sys
- import operator
- from . import utils
- (WARNING,
- ERROR,
- FATAL) = range(3)
- class Position(object):
- """
- Represents a position in the source file which we
- want to inform about.
- """
- __slots__ = ('filename', 'line', 'column')
- def __init__(self, filename=None, line=None, column=None):
- self.filename = filename
- self.line = line
- self.column = column
- def _compare(self, other, op):
- return op((self.filename, self.line, self.column),
- (other.filename, other.line, other.column))
- def __lt__(self, other):
- return self._compare(other, operator.lt)
- def __gt__(self, other):
- return self._compare(other, operator.gt)
- def __ge__(self, other):
- return self._compare(other, operator.ge)
- def __le__(self, other):
- return self._compare(other, operator.le)
- def __eq__(self, other):
- return self._compare(other, operator.eq)
- def __ne__(self, other):
- return self._compare(other, operator.ne)
- def __hash__(self):
- return hash((self.filename, self.line, self.column))
- def __repr__(self):
- return '<Position %s:%d:%d>' % (os.path.basename(self.filename),
- self.line or -1,
- self.column or -1)
- def format(self, cwd):
- filename = os.path.realpath(self.filename)
- cwd = os.path.realpath(cwd)
- common_prefix = os.path.commonprefix((filename, cwd))
- if common_prefix:
- filename = os.path.relpath(filename, common_prefix)
- if self.column is not None:
- return '%s:%d:%d' % (filename, self.line, self.column)
- elif self.line is not None:
- return '%s:%d' % (filename, self.line, )
- else:
- return '%s:' % (filename, )
- class MessageLogger(object):
- _instance = None
- def __init__(self, namespace=None, output=None):
- if output is None:
- output = sys.stderr
- self._cwd = os.getcwd()
- self._output = output
- self._namespace = namespace
- self._enable_warnings = []
- self._warning_count = 0
- self._error_count = 0
- @classmethod
- def get(cls, *args, **kwargs):
- if cls._instance is None:
- cls._instance = cls(*args, **kwargs)
- return cls._instance
- def enable_warnings(self, log_types):
- self._enable_warnings = log_types
- def get_warning_count(self):
- return self._warning_count
- def get_error_count(self):
- return self._error_count
- def log(self, log_type, text, positions=None, prefix=None, marker_pos=None, marker_line=None):
- """
- Log a warning, using optional file positioning information.
- If the warning is related to a ast.Node type, see log_node().
- """
- utils.break_on_debug_flag('warning')
- self._warning_count += 1
- if log_type not in self._enable_warnings:
- return
- if type(positions) == set:
- positions = list(positions)
- if isinstance(positions, Position):
- positions = [positions]
- if not positions:
- positions = [Position('<unknown>')]
- for position in positions[:-1]:
- self._output.write("%s:\n" % (position.format(cwd=self._cwd), ))
- last_position = positions[-1].format(cwd=self._cwd)
- if log_type == WARNING:
- error_type = "Warning"
- elif log_type == ERROR:
- error_type = "Error"
- self._error_count += 1
- elif log_type == FATAL:
- error_type = "Fatal"
- if marker_pos is not None and marker_line is not None:
- text = '%s\n%s\n%s' % (text, marker_line, ' ' * marker_pos + '^')
- if prefix:
- if self._namespace:
- text = ('%s: %s: %s: %s: %s\n' % (last_position, error_type,
- self._namespace.name, prefix, text))
- else:
- text = ('%s: %s: %s: %s\n' % (last_position, error_type,
- prefix, text))
- else:
- if self._namespace:
- text = ('%s: %s: %s: %s\n' % (last_position, error_type,
- self._namespace.name, text))
- else:
- text = ('%s: %s: %s\n' % (last_position, error_type, text))
- self._output.write(text)
- if log_type == FATAL:
- utils.break_on_debug_flag('fatal')
- raise SystemExit(text)
- def log_node(self, log_type, node, text, context=None, positions=None):
- """
- Log a warning, using information about file positions from
- the given node. The optional context argument, if given, should be
- another ast.Node type which will also be displayed. If no file position
- information is available from the node, the position data from the
- context will be used.
- """
- if positions:
- pass
- elif getattr(node, 'file_positions', None):
- positions = node.file_positions
- elif context and context.file_positions:
- positions = context.file_positions
- else:
- positions = set()
- if context:
- text = "%s: %s" % (getattr(context, 'symbol', context.name), text)
- elif not positions and hasattr(node, 'name'):
- text = "(%s)%s: %s" % (node.__class__.__name__, node.name, text)
- self.log(log_type, text, positions)
- def log_symbol(self, log_type, symbol, text):
- """Log a warning in the context of the given symbol."""
- self.log(log_type, text, symbol.position,
- prefix="symbol='%s'" % (symbol.ident, ))
- def log_node(log_type, node, text, context=None, positions=None):
- ml = MessageLogger.get()
- ml.log_node(log_type, node, text, context=context, positions=positions)
- def warn(text, positions=None, prefix=None, marker_pos=None, marker_line=None):
- ml = MessageLogger.get()
- ml.log(WARNING, text, positions, prefix, marker_pos, marker_line)
- def warn_node(node, text, context=None, positions=None):
- log_node(WARNING, node, text, context=context, positions=positions)
- def error_node(node, text, context=None, positions=None):
- log_node(ERROR, node, text, context=context, positions=positions)
- def warn_symbol(symbol, text):
- ml = MessageLogger.get()
- ml.log_symbol(WARNING, symbol, text)
- def error(text, positions=None, prefix=None, marker_pos=None, marker_line=None):
- ml = MessageLogger.get()
- ml.log(ERROR, text, positions, prefix, marker_pos, marker_line)
- def fatal(text, positions=None, prefix=None, marker_pos=None, marker_line=None):
- ml = MessageLogger.get()
- ml.log(FATAL, text, positions, prefix, marker_pos, marker_line)
|