Я пытаюсь написать декоратор, который добавляет подробное ведение журнала к функции через декоратор (метод тоже был бы хорош, но я еще не пробовал это). Мотивация заключается в том, что исправление в однострочном вызове декоратора add_logs в поле в производстве намного проще (и безопаснее), чем добавление 100 строк отладки.Редактирование кода функции во время выполнения
Ex:
def hey(name):
print("Hi " + name)
t = 1 + 1
if t > 6:
t = t + 1
print("I was bigger")
else:
print("I was not.")
print("t = ", t)
return t
Я хотел бы сделать декоратора, который преобразует это в код, который делает это:
def hey(name):
print("line 1")
print("Hi " + name)
print("line 2")
t = 1 + 1
print("line 3")
if t > 6:
print("line 4")
t = t + 1
print("line 5")
print("I was bigger")
else:
print("line 6")
print("I was not.")
print("line 7")
print("t = ", t)
print("line 8")
return t
То, что я получил до сих пор:
import inspect, ast
import itertools
import imp
def log_maker():
line_num = 1
while True:
yield ast.parse('print("line {line_num}")'.format(line_num=line_num)).body[0]
line_num = line_num + 1
def add_logs(function):
def dummy_function(*args, **kwargs):
pass
lines = inspect.getsourcelines(function)
code = "".join(lines[0][1:])
ast_tree = ast.parse(code)
body = ast_tree.body[0].body
#I realize this doesn't do exactly what I want.
#(It doesn't add debug lines inside the if statement)
#Once I get it almost working, I will rewrite this
#to use something like node visitors
body = list(itertools.chain(*zip(log_maker(), body)))
ast_tree.body[0].body = body
fix_line_nums(ast_tree)
code = compile(ast_tree,"<string>", mode='exec')
dummy_function.__code__ = code
return dummy_function
def fix_line_nums(node):
if hasattr(node, "body"):
for index, child in enumerate(node.body):
if hasattr(child, "lineno"):
if index == 0:
if hasattr(node, "lineno"):
child.lineno = node.lineno + 1
else:
# Hopefully this only happens if the parent is a module...
child.lineno = 1
else:
child.lineno = node.body[index - 1].lineno + 1
fix_line_nums(child)
@add_logs
def hey(name):
print("Hi " + name)
t = 1 + 1
if t > 6:
t = t + 1
print("I was bigger")
else:
print("I was not.")
print("t = ", t)
return t
if __name__ == "__main__":
print(hey("mark"))
print(hey)
Произошла ошибка:
Traceback (most recent call last):
File "so.py", line 76, in <module>
print(hey("mark"))
TypeError: <module>() takes no arguments (1 given)
, что имеет смысл, поскольку компиляция создает модуль и, конечно, модули не являются вызывающими. Я пробовал сотню различных способов сделать эту работу на данный момент, но не могу найти рабочего решения. Любые рекомендации? Неужели я ошибаюсь?
(я не смог найти учебник для модуля Аст, который фактически изменяет код во время выполнения, как это указатель на такой учебник будет полезным.)
Примечания: В настоящее время я тестирование это на CPython 3.2, но будет оценено решение 2.6/3.3_and_up. В настоящее время поведение одинаково на 2.7 и 3.3.
Спасибо, это работает отлично. (И мои АСТ-манипуляции, похоже, работают правильно, как мы ;;.)) – marky1991
@ marky1991: Кстати, вы можете посмотреть ['sys.settrace'] (https://docs.python.org/2/ library/sys.html # sys.settrace) и/или модуль ['trace'] (https://docs.python.org/2/library/trace.html). Они могут позволить вам получить вывод трассировки, который вы хотите, менее громоздким способом. – BrenBarn