Basic question: What’s the most Pythonic/logical way to make my own, custom warning classes? What are the correct warning and exception classes that I should be subclassing?
Motivation: The requirements for the library I’m writing specify that if a MyContainer object c contains an item x and the caller of the library tries to place a “duplicate” of x — call it y — into c, a warning is issued to the caller and the return value of c.my_transformation_method(x, y) is placed into c to replace x. In other words, MyContainers will replace elements with their duplicates, but must warn the user when doing so.
Based on my reading, the most flexible way to warn the caller of a library about a nonfatal action is with the warnings standard module. It allows the caller to handle the warning as it sees fit, doing anything from ignoring warnings to treating them as errors. (Note that I’m using Python 3, but I don’t think that’s essential to the question here.)
Example: What I’ve done is defined the following warning subclass:
class DuplicateItemWarning(UserWarning, ValueError):
pass
Then the add() method of MyContainer calls warnings.warn('detected duplicate', DuplicateItemWarning) when it detects an attempt to insert a duplicate item.
Specific questions:
-
Should I be subclassing
UserWarningas above, or just sublcassingWarning? -
It seems semantically sensible to subclass
ValueError(which, in the above example, merely insertsValueErrorin the MRO betweenWarningandException) in case a caller wants to treat warnings as errors. Is there a drawback to this I’m not seeing? -
I could find no previous questions on StackOverflow about customizing warning classes. Is this because Python programmers don’t even like using the
warningsmodule?
After reading the PEP 230 about the warning framework and the warnings docs, I think I have the answer to your questions:
UserWarningand all others are warning categories, they don’t seem to have another role than just classification. This way you could filter them out in your production environment for example. So, basically, you could subclass fromWarningif the warning does not fall in any other category. If in the context,UserWarningorRuntimeWarningseem enough, just use them.Warnings are alreadyExceptions. So, technically, to “catch” them as errors, you just need to change the filter, no need to subclass from anyXXXError. Now, again it’s all about making sense. If the warnings are about the values passed, you could subclass fromValueError, especially if there would be many different custom warnings, you would expect the caller to “catch” all warnings regarding values all at once.The
warningsmodule is Guido van Rossum’s idea. (See PEP 230). If that ain’t Pythonic enough … 😀