Вы не можете получить надежные результаты для некоторых (если не большинства) функций. Некоторые примеры:
функции, выполняющие произвольный код (например, exec(')(rorrEeulaV esiar'[::-1])
вызывает ValueError
)
функции, написанные не на Python
функции, которые вызывают другие функции, которые могут передавать ошибки вызывающей стороне
функции, повторно вызывающие активные исключения в блоке except:
К сожалению, этот список неполный.
Например. os.makedirs
написан на Python, и вы можете увидеть его исходный код:
...
try:
mkdir(name, mode)
except OSError as e:
if not exist_ok or e.errno != errno.EEXIST or not path.isdir(name):
raise
Bare raise
повторно вызывает последнее активное исключение (OSError
или один из его подклассов). Вот иерархия классов для OSError
:
+-- OSError
| +-- BlockingIOError
| +-- ChildProcessError
| +-- ConnectionError
| | +-- BrokenPipeError
| | +-- ConnectionAbortedError
| | +-- ConnectionRefusedError
| | +-- ConnectionResetError
| +-- FileExistsError
| +-- FileNotFoundError
| +-- InterruptedError
| +-- IsADirectoryError
| +-- NotADirectoryError
| +-- PermissionError
| +-- ProcessLookupError
| +-- TimeoutError
Чтобы получить точные типы исключений, вам нужно изучить mkdir
, функции, которые он вызывает, функции, которые вызывают эти функции и т. д.
Таким образом, получить возможные исключения без запуска функции очень сложно, и вам действительно не следует этого делать.
Однако для простых случаев, таких как
raise Exception # without arguments
raise Exception('abc') # with arguments
сочетание функциональности ast
модуля и inspect.getclosurevars
(чтобы получить классы исключений, введенный в Python 3.3) может дать довольно точные результаты:
from inspect import getclosurevars, getsource
from collections import ChainMap
from textwrap import dedent
import ast, os
class MyException(Exception):
pass
def g():
raise Exception
class A():
def method():
raise OSError
def f(x):
int()
A.method()
os.makedirs()
g()
raise MyException
raise ValueError('argument')
def get_exceptions(func, ids=set()):
try:
vars = ChainMap(*getclosurevars(func)[:3])
source = dedent(getsource(func))
except TypeError:
return
class _visitor(ast.NodeTransformer):
def __init__(self):
self.nodes = []
self.other = []
def visit_Raise(self, n):
self.nodes.append(n.exc)
def visit_Expr(self, n):
if not isinstance(n.value, ast.Call):
return
c, ob = n.value.func, None
if isinstance(c, ast.Attribute):
parts = []
while getattr(c, 'value', None):
parts.append(c.attr)
c = c.value
if c.id in vars:
ob = vars[c.id]
for name in reversed(parts):
ob = getattr(ob, name)
elif isinstance(c, ast.Name):
if c.id in vars:
ob = vars[c.id]
if ob is not None and id(ob) not in ids:
self.other.append(ob)
ids.add(id(ob))
v = _visitor()
v.visit(ast.parse(source))
for n in v.nodes:
if isinstance(n, (ast.Call, ast.Name)):
name = n.id if isinstance(n, ast.Name) else n.func.id
if name in vars:
yield vars[name]
for o in v.other:
yield from get_exceptions(o)
for e in get_exceptions(f):
print(e)
отпечатки
<class '__main__.MyException'>
<class 'ValueError'>
<class 'OSError'>
<class 'Exception'>
Имейте в виду, что этот код работает только для функций, написанных на Python.
person
vaultah
schedule
14.09.2015
compiler
. Затем, как предложено в вашей ссылке, используйтеparser
. Или встроенныйcompile
илиast.parse
- person 301_Moved_Permanently   schedule 14.09.2015Name
,Raise
,CallFunc
,Const
,Getattr
из пакетаcompiler
. где бы мне их найти? - person hiro protagonist   schedule 14.09.2015Call
,Name
иRaise
в модуле_ast
. я посмотрю, что я могу сделать оттуда. - person hiro protagonist   schedule 14.09.2015raise user_provided_callable('Ouch!')
илиraise CONFIG['exceptions'].get('nitpick', ValueError)('Whoops!')
. Такого рода вещи могут быть даже хорошей идеей в некоторых ситуациях, хотя я изо всех сил пытаюсь думать о чем-то еще, потому что я мог! - person Kevin J. Chase   schedule 14.09.2015os.makedirs
может подниматьFileExistsError
- но насколько я вижу на самом деле проглатываетFileExistsError
здесь. - person J Richard Snape   schedule 14.09.2015FileExistsError
не относится к исключениямmakedirs
может расие. моя ошибка. я исправил это в вопросе. - person hiro protagonist   schedule 15.09.2015