warnings.py 15.5 KB
Newer Older
1 2
"""Python part of the warnings subsystem."""

Christian Heimes's avatar
Christian Heimes committed
3
import sys
4

5 6
__all__ = ["warn", "warn_explicit", "showwarning",
           "formatwarning", "filterwarnings", "simplefilter",
7
           "resetwarnings", "catch_warnings"]
8

9

Christian Heimes's avatar
Christian Heimes committed
10
def showwarning(message, category, filename, lineno, file=None, line=None):
11 12 13
    """Hook to write a warning to a file; replace if you like."""
    if file is None:
        file = sys.stderr
14
        if file is None:
15
            # sys.stderr is None when run with pythonw.exe - warnings get lost
16
            return
17
    try:
Christian Heimes's avatar
Christian Heimes committed
18
        file.write(formatwarning(message, category, filename, lineno, line))
19
    except OSError:
20
        pass # the file (probably stderr) is invalid - this warning gets lost.
21

Christian Heimes's avatar
Christian Heimes committed
22
def formatwarning(message, category, filename, lineno, line=None):
23
    """Function to format a warning the standard way."""
24
    s =  "%s:%s: %s: %s\n" % (filename, lineno, category.__name__, message)
25 26 27 28 29 30
    if line is None:
        try:
            import linecache
            line = linecache.getline(filename, lineno)
        except Exception:
            # When a warning is logged during Python shutdown, linecache
31
            # and the import machinery don't work anymore
32
            line = None
33
    if line:
Christian Heimes's avatar
Christian Heimes committed
34 35
        line = line.strip()
        s += "  %s\n" % line
36 37
    return s

38
def filterwarnings(action, message="", category=Warning, module="", lineno=0,
39
                   append=False):
40 41
    """Insert an entry into the list of warnings filters (at the front).

Georg Brandl's avatar
Georg Brandl committed
42 43 44 45 46 47 48 49
    'action' -- one of "error", "ignore", "always", "default", "module",
                or "once"
    'message' -- a regex that the warning message must match
    'category' -- a class that the warning must be a subclass of
    'module' -- a regex that the module name must match
    'lineno' -- an integer line number, 0 matches all warnings
    'append' -- if true, append to the list of filters
    """
50
    import re
51
    assert action in ("error", "ignore", "always", "default", "module",
52
                      "once"), "invalid action: %r" % (action,)
53
    assert isinstance(message, str), "message must be a string"
54
    assert isinstance(category, type), "category must be a class"
55
    assert issubclass(category, Warning), "category must be a Warning subclass"
56
    assert isinstance(module, str), "module must be a string"
57
    assert isinstance(lineno, int) and lineno >= 0, \
58
           "lineno must be an int >= 0"
59 60
    _add_filter(action, re.compile(message, re.I), category,
            re.compile(module), lineno, append=append)
61

62
def simplefilter(action, category=Warning, lineno=0, append=False):
63 64 65
    """Insert a simple entry into the list of warnings filters (at the front).

    A simple filter matches all modules and messages.
Georg Brandl's avatar
Georg Brandl committed
66 67 68 69 70
    'action' -- one of "error", "ignore", "always", "default", "module",
                or "once"
    'category' -- a class that the warning must be a subclass of
    'lineno' -- an integer line number, 0 matches all warnings
    'append' -- if true, append to the list of filters
71 72
    """
    assert action in ("error", "ignore", "always", "default", "module",
73
                      "once"), "invalid action: %r" % (action,)
74 75
    assert isinstance(lineno, int) and lineno >= 0, \
           "lineno must be an int >= 0"
76 77 78 79 80 81 82 83 84 85
    _add_filter(action, None, category, None, lineno, append=append)

def _add_filter(*item, append):
    # Remove possible duplicate filters, so new one will be placed
    # in correct place. If append=True and duplicate exists, do nothing.
    if not append:
        try:
            filters.remove(item)
        except ValueError:
            pass
86
        filters.insert(0, item)
87 88 89
    else:
        if item not in filters:
            filters.append(item)
90
    _filters_mutated()
91

92
def resetwarnings():
93
    """Clear the list of warning filters, so that no filters are active."""
94
    filters[:] = []
95
    _filters_mutated()
96 97 98 99 100 101 102 103 104 105

class _OptionError(Exception):
    """Exception used by option processing helpers."""
    pass

# Helper to process -W options passed via sys.warnoptions
def _processoptions(args):
    for arg in args:
        try:
            _setoption(arg)
106
        except _OptionError as msg:
107
            print("Invalid -W option ignored:", msg, file=sys.stderr)
108 109 110

# Helper for _processoptions()
def _setoption(arg):
111
    import re
112 113
    parts = arg.split(':')
    if len(parts) > 5:
114
        raise _OptionError("too many fields (max 5): %r" % (arg,))
115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
    while len(parts) < 5:
        parts.append('')
    action, message, category, module, lineno = [s.strip()
                                                 for s in parts]
    action = _getaction(action)
    message = re.escape(message)
    category = _getcategory(category)
    module = re.escape(module)
    if module:
        module = module + '$'
    if lineno:
        try:
            lineno = int(lineno)
            if lineno < 0:
                raise ValueError
        except (ValueError, OverflowError):
131
            raise _OptionError("invalid lineno %r" % (lineno,))
132 133 134
    else:
        lineno = 0
    filterwarnings(action, message, category, module, lineno)
135 136 137 138 139 140

# Helper for _setoption()
def _getaction(action):
    if not action:
        return "default"
    if action == "all": return "always" # Alias
141
    for a in ('default', 'always', 'ignore', 'module', 'once', 'error'):
142 143
        if a.startswith(action):
            return a
144
    raise _OptionError("invalid action: %r" % (action,))
145 146 147

# Helper for _setoption()
def _getcategory(category):
148
    import re
149 150 151 152 153
    if not category:
        return Warning
    if re.match("^[a-zA-Z0-9_]+$", category):
        try:
            cat = eval(category)
154
        except NameError:
155
            raise _OptionError("unknown warning category: %r" % (category,))
156 157 158 159
    else:
        i = category.rfind(".")
        module = category[:i]
        klass = category[i+1:]
160
        try:
161 162
            m = __import__(module, None, None, [klass])
        except ImportError:
163
            raise _OptionError("invalid module name: %r" % (module,))
164 165 166
        try:
            cat = getattr(m, klass)
        except AttributeError:
167
            raise _OptionError("unknown warning category: %r" % (category,))
168
    if not issubclass(cat, Warning):
169
        raise _OptionError("invalid warning category: %r" % (category,))
170 171
    return cat

Christian Heimes's avatar
Christian Heimes committed
172

173 174 175 176 177 178 179 180 181 182 183 184 185 186
def _is_internal_frame(frame):
    """Signal whether the frame is an internal CPython implementation detail."""
    filename = frame.f_code.co_filename
    return 'importlib' in filename and '_bootstrap' in filename


def _next_external_frame(frame):
    """Find the next frame that doesn't involve CPython internals."""
    frame = frame.f_back
    while frame is not None and _is_internal_frame(frame):
        frame = frame.f_back
    return frame


Christian Heimes's avatar
Christian Heimes committed
187 188 189 190 191 192 193 194 195
# Code typically replaced by _warnings
def warn(message, category=None, stacklevel=1):
    """Issue a warning, or maybe ignore it or raise an exception."""
    # Check if message is already a Warning object
    if isinstance(message, Warning):
        category = message.__class__
    # Check category argument
    if category is None:
        category = UserWarning
196 197 198
    if not (isinstance(category, type) and issubclass(category, Warning)):
        raise TypeError("category must be a Warning subclass, "
                        "not '{:s}'".format(type(category).__name__))
Christian Heimes's avatar
Christian Heimes committed
199 200
    # Get context information
    try:
201 202 203 204 205 206 207 208 209 210 211
        if stacklevel <= 1 or _is_internal_frame(sys._getframe(1)):
            # If frame is too small to care or if the warning originated in
            # internal code, then do not try to hide any frames.
            frame = sys._getframe(stacklevel)
        else:
            frame = sys._getframe(1)
            # Look for one frame less since the above line starts us off.
            for x in range(stacklevel-1):
                frame = _next_external_frame(frame)
                if frame is None:
                    raise ValueError
Christian Heimes's avatar
Christian Heimes committed
212 213 214 215
    except ValueError:
        globals = sys.__dict__
        lineno = 1
    else:
216 217
        globals = frame.f_globals
        lineno = frame.f_lineno
Christian Heimes's avatar
Christian Heimes committed
218 219 220 221 222 223 224
    if '__name__' in globals:
        module = globals['__name__']
    else:
        module = "<string>"
    filename = globals.get('__file__')
    if filename:
        fnl = filename.lower()
225
        if fnl.endswith(".pyc"):
Christian Heimes's avatar
Christian Heimes committed
226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241
            filename = filename[:-1]
    else:
        if module == "__main__":
            try:
                filename = sys.argv[0]
            except AttributeError:
                # embedded interpreters don't have sys.argv, see bug #839151
                filename = '__main__'
        if not filename:
            filename = module
    registry = globals.setdefault("__warningregistry__", {})
    warn_explicit(message, category, filename, lineno, module, registry,
                  globals)

def warn_explicit(message, category, filename, lineno,
                  module=None, registry=None, module_globals=None):
242
    lineno = int(lineno)
Christian Heimes's avatar
Christian Heimes committed
243 244 245 246 247 248
    if module is None:
        module = filename or "<unknown>"
        if module[-3:].lower() == ".py":
            module = module[:-3] # XXX What about leading pathname?
    if registry is None:
        registry = {}
249 250 251
    if registry.get('version', 0) != _filters_version:
        registry.clear()
        registry['version'] = _filters_version
Christian Heimes's avatar
Christian Heimes committed
252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
    if isinstance(message, Warning):
        text = str(message)
        category = message.__class__
    else:
        text = message
        message = category(message)
    key = (text, category, lineno)
    # Quick test for common case
    if registry.get(key):
        return
    # Search the filters
    for item in filters:
        action, msg, cat, mod, ln = item
        if ((msg is None or msg.match(text)) and
            issubclass(category, cat) and
            (mod is None or mod.match(module)) and
            (ln == 0 or lineno == ln)):
            break
    else:
        action = defaultaction
    # Early exit actions
    if action == "ignore":
        registry[key] = 1
        return

    # Prime the linecache for formatting, in case the
    # "file" is actually in a zipfile or something.
279
    import linecache
Christian Heimes's avatar
Christian Heimes committed
280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305
    linecache.getlines(filename, module_globals)

    if action == "error":
        raise message
    # Other actions
    if action == "once":
        registry[key] = 1
        oncekey = (text, category)
        if onceregistry.get(oncekey):
            return
        onceregistry[oncekey] = 1
    elif action == "always":
        pass
    elif action == "module":
        registry[key] = 1
        altkey = (text, category, 0)
        if registry.get(altkey):
            return
        registry[altkey] = 1
    elif action == "default":
        registry[key] = 1
    else:
        # Unrecognized actions are errors
        raise RuntimeError(
              "Unrecognized action (%r) in warnings.filters:\n %s" %
              (action, item))
306
    if not callable(showwarning):
307 308
        raise TypeError("warnings.showwarning() must be set to a "
                        "function or method")
Christian Heimes's avatar
Christian Heimes committed
309 310 311 312
    # Print message and context
    showwarning(message, category, filename, lineno)


313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334
class WarningMessage(object):

    """Holds the result of a single showwarning() call."""

    _WARNING_DETAILS = ("message", "category", "filename", "lineno", "file",
                        "line")

    def __init__(self, message, category, filename, lineno, file=None,
                    line=None):
        local_values = locals()
        for attr in self._WARNING_DETAILS:
            setattr(self, attr, local_values[attr])
        self._category_name = category.__name__ if category else None

    def __str__(self):
        return ("{message : %r, category : %r, filename : %r, lineno : %s, "
                    "line : %r}" % (self.message, self._category_name,
                                    self.filename, self.lineno, self.line))


class catch_warnings(object):

335 336 337 338 339 340 341 342
    """A context manager that copies and restores the warnings filter upon
    exiting the context.

    The 'record' argument specifies whether warnings should be captured by a
    custom implementation of warnings.showwarning() and be appended to a list
    returned by the context manager. Otherwise None is returned by the context
    manager. The objects appended to the list are arguments whose attributes
    mirror the arguments to showwarning().
343

344 345 346
    The 'module' argument is to specify an alternative module to the module
    named 'warnings' and imported under that name. This argument is only useful
    when testing the warnings module itself.
347 348 349 350 351 352 353 354 355 356 357

    """

    def __init__(self, *, record=False, module=None):
        """Specify whether to record warnings and if an alternative module
        should be used other than sys.modules['warnings'].

        For compatibility with Python 3.0, please consider all arguments to be
        keyword-only.

        """
358
        self._record = record
359
        self._module = sys.modules['warnings'] if module is None else module
Benjamin Peterson's avatar
Benjamin Peterson committed
360 361 362 363 364 365 366 367 368 369
        self._entered = False

    def __repr__(self):
        args = []
        if self._record:
            args.append("record=True")
        if self._module is not sys.modules['warnings']:
            args.append("module=%r" % self._module)
        name = type(self).__name__
        return "%s(%s)" % (name, ", ".join(args))
370 371

    def __enter__(self):
Benjamin Peterson's avatar
Benjamin Peterson committed
372 373 374
        if self._entered:
            raise RuntimeError("Cannot enter %r twice" % self)
        self._entered = True
375 376
        self._filters = self._module.filters
        self._module.filters = self._filters[:]
377
        self._module._filters_mutated()
378
        self._showwarning = self._module.showwarning
379 380 381 382 383 384 385 386
        if self._record:
            log = []
            def showwarning(*args, **kwargs):
                log.append(WarningMessage(*args, **kwargs))
            self._module.showwarning = showwarning
            return log
        else:
            return None
387 388

    def __exit__(self, *exc_info):
Benjamin Peterson's avatar
Benjamin Peterson committed
389 390
        if not self._entered:
            raise RuntimeError("Cannot exit %r without entering first" % self)
391
        self._module.filters = self._filters
392
        self._module._filters_mutated()
393 394 395
        self._module.showwarning = self._showwarning


Christian Heimes's avatar
Christian Heimes committed
396 397 398 399 400 401 402 403 404 405
# filters contains a sequence of filter 5-tuples
# The components of the 5-tuple are:
# - an action: error, ignore, always, default, module, or once
# - a compiled regex that must match the warning message
# - a class representing the warning category
# - a compiled regex that must match the module that is being warned
# - a line number for the line being warning, or 0 to mean any line
# If either if the compiled regexs are None, match anything.
_warnings_defaults = False
try:
406
    from _warnings import (filters, _defaultaction, _onceregistry,
407
                           warn, warn_explicit, _filters_mutated)
408 409
    defaultaction = _defaultaction
    onceregistry = _onceregistry
Christian Heimes's avatar
Christian Heimes committed
410
    _warnings_defaults = True
411
except ImportError:
Christian Heimes's avatar
Christian Heimes committed
412 413 414 415
    filters = []
    defaultaction = "default"
    onceregistry = {}

416 417 418 419 420 421
    _filters_version = 1

    def _filters_mutated():
        global _filters_version
        _filters_version += 1

Christian Heimes's avatar
Christian Heimes committed
422

423
# Module initialization
424
_processoptions(sys.warnoptions)
Christian Heimes's avatar
Christian Heimes committed
425
if not _warnings_defaults:
426 427 428 429
    silence = [ImportWarning, PendingDeprecationWarning]
    silence.append(DeprecationWarning)
    for cls in silence:
        simplefilter("ignore", category=cls)
Christian Heimes's avatar
Christian Heimes committed
430 431 432 433 434 435 436 437
    bytes_warning = sys.flags.bytes_warning
    if bytes_warning > 1:
        bytes_action = "error"
    elif bytes_warning:
        bytes_action = "default"
    else:
        bytes_action = "ignore"
    simplefilter(bytes_action, category=BytesWarning, append=1)
438 439 440 441 442 443 444
    # resource usage warnings are enabled by default in pydebug mode
    if hasattr(sys, 'gettotalrefcount'):
        resource_action = "always"
    else:
        resource_action = "ignore"
    simplefilter(resource_action, category=ResourceWarning, append=1)

Christian Heimes's avatar
Christian Heimes committed
445
del _warnings_defaults