python-peps/pep-0409.txt

188 lines
5.5 KiB
Plaintext
Raw Normal View History

PEP: 409
Title: Suppressing exception context
Version: $Revision$
Last-Modified: $Date$
Author: Ethan Furman <ethan@stoneleaf.us>
Status: Accepted
Type: Standards Track
Content-Type: text/x-rst
Created: 26-Jan-2012
2012-02-03 03:34:26 -05:00
Post-History: 30-Aug-2002, 01-Feb-2012, 03-Feb-2012
Resolution: http://mail.python.org/pipermail/python-dev/2012-February/116136.html
Abstract
========
One of the open issues from PEP 3134 is suppressing context: currently
2012-01-29 03:41:13 -05:00
there is no way to do it. This PEP proposes one.
2012-02-03 03:34:26 -05:00
Rationale
=========
There are two basic ways to generate exceptions:
1) Python does it (buggy code, missing resources, ending loops, etc.)
2) manually (with a raise statement)
2012-01-29 03:41:13 -05:00
When writing libraries, or even just custom classes, it can become
necessary to raise exceptions; moreover it can be useful, even
necessary, to change from one exception to another. To take an example
from my dbf module::
2012-02-03 03:34:26 -05:00
try:
value = int(value)
except Exception:
raise DbfError(...)
2012-02-03 03:34:26 -05:00
Whatever the original exception was (``ValueError``, ``TypeError``, or
2012-01-29 03:41:13 -05:00
something else) is irrelevant. The exception from this point on is a
2012-02-03 03:34:26 -05:00
``DbfError``, and the original exception is of no value. However, if
this exception is printed, we would currently see both.
Alternatives
============
Several possibilities have been put forth:
2012-02-03 03:34:26 -05:00
* ``raise as NewException()``
2012-02-03 03:34:26 -05:00
Reuses the ``as`` keyword; can be confusing since we are not really
reraising the originating exception
2012-02-03 03:34:26 -05:00
* ``raise NewException() from None``
2012-02-03 03:34:26 -05:00
Follows existing syntax of explicitly declaring the originating
exception
2012-02-03 03:34:26 -05:00
* ``exc = NewException(); exc.__context__ = None; raise exc``
2012-01-29 03:41:13 -05:00
Very verbose way of the previous method
2012-02-03 03:34:26 -05:00
* ``raise NewException.no_context(...)``
2012-01-29 03:41:13 -05:00
Make context suppression a class method.
All of the above options will require changes to the core.
Proposal
========
2012-01-29 03:41:13 -05:00
I proprose going with the second option::
raise NewException from None
It has the advantage of using the existing pattern of explicitly setting
the cause::
2012-01-29 03:41:13 -05:00
raise KeyError() from NameError()
but because the cause is ``None`` the previous context is not displayed
by the default exception printing routines.
2012-02-03 03:34:26 -05:00
Implementation Discussion
=========================
Currently, ``None`` is the default for both ``__context__`` and ``__cause__``.
In order to support ``raise ... from None`` (which would set ``__cause__`` to
``None``) we need a different default value for ``__cause__``. Several ideas
were put forth on how to implement this at the language level:
2012-02-03 03:34:26 -05:00
* Overwrite the previous exception information (side-stepping the issue and
leaving ``__cause__`` at ``None``).
2012-02-03 03:34:26 -05:00
Rejected as this can seriously hinder debugging due to
`poor error messages`_.
2012-02-03 03:34:26 -05:00
* Use one of the boolean values in ``__cause__``: ``False`` would be the
default value, and would be replaced when ``from ...`` was used with the
explicity chained exception or ``None``.
2012-02-03 03:34:26 -05:00
Rejected as this encourages the use of two different objects types for
``__cause__`` with one of them (boolean) not allowed to have the full range
of possible values (``True`` would never be used).
2012-02-03 03:34:26 -05:00
* Create a special exception class, ``__NoException__``.
Rejected as possibly confusing, possibly being mistakenly raised by users,
and not being a truly unique value as ``None``, ``True``, and ``False`` are.
2012-02-03 03:34:26 -05:00
* Use ``Ellipsis`` as the default value (the ``...`` singleton).
Accepted. There are no other possible values; it cannot be raised as it is
not an exception; it has the connotation of 'fill in the rest...' as in
``__cause__`` is not set, look in ``__context__`` for it.
2012-02-03 03:34:26 -05:00
Language Details
================
To support ``from None``, ``__context__`` will stay as it is, but
``__cause__`` will start out as ``Ellipsis`` and will change to ``None``
when the ``raise ... from None`` method is used.
2012-02-03 03:34:26 -05:00
============================================ ================== =======================================
form __context__ __cause__
============================================ ================== =======================================
raise ``None`` ``Ellipsis``
reraise previous exception ``Ellipsis``
reraise from ``None`` | ``ChainedException`` previous exception ``None`` | explicitly chained exception
============================================ ================== =======================================
The default exception printing routine will then:
* If ``__cause__`` is ``Ellipsis`` the ``__context__`` (if any) will be
2012-02-03 03:34:26 -05:00
printed.
* If ``__cause__`` is ``None`` the ``__context__`` will not be printed.
* if ``__cause__`` is anything else, ``__cause__`` will be printed.
In both of the latter cases the exception chain will stop being followed.
Because the default value for ``__cause__`` is now ``Ellipsis``::
raise Exception from Ellipsis
is allowed and will (re)set ``__cause__`` to ``Ellipsis``.
2012-02-03 03:34:26 -05:00
Patches
=======
There is a patch for CPython implementing this attached to `Issue 6210`_.
2012-01-27 10:47:48 -05:00
References
==========
2012-02-03 03:34:26 -05:00
Discussion and refinements in this `thread on python-dev`_.
.. _poor error messages:
http://bugs.python.org/msg152294
.. _issue 6210:
2012-01-27 10:47:48 -05:00
http://bugs.python.org/issue6210
2012-02-03 03:34:26 -05:00
.. _Thread on python-dev:
http://mail.python.org/pipermail/python-dev/2012-January/115838.html
2012-01-29 03:41:13 -05:00
Copyright
=========
This document has been placed in the public domain.
..
2012-01-29 03:41:13 -05:00
Local Variables:
mode: indented-text
indent-tabs-mode: nil
sentence-end-double-space: t
fill-column: 70
coding: utf-8
End: