Как найти код выхода или причину, когда callback atexit вызывается в Python?

Я хочу знать, правильно ли работает Python script или нет. Для этого я использую atexit, но проблема в том, что я не знаю, как дифференцироваться, если atexit был вызван с sys.exit(0) или ноль или исключение.

Рассуждение: если программа закончится правильно, она ничего не сделает, но если программа закончится исключением или вернет код ошибки (статус выхода), отличный от нуля, я хочу вызвать какое-то действие.

В случае, если вы зададите себе вопрос, почему я не использую try/, это потому, что я хочу добавить такое же поведение для дюжины сценариев, которые импортируют общий модуль. Вместо того, чтобы модифицировать их все, я хочу добавить хакер atexit() к импортируемому модулю и получить это поведение бесплатно во всех них.

Ответ 1

Вы можете решить эту проблему с помощью sys.excepthook и путем патча обезьян sys.exit():

import atexit
import sys

class ExitHooks(object):
    def __init__(self):
        self.exit_code = None
        self.exception = None

    def hook(self):
        self._orig_exit = sys.exit
        sys.exit = self.exit
        sys.excepthook = self.exc_handler

    def exit(self, code=0):
        self.exit_code = code
        self._orig_exit(code)

    def exc_handler(self, exc_type, exc, *args):
        self.exception = exc

hooks = ExitHooks()
hooks.hook()

def foo():
    if hooks.exit_code is not None:
        print("death by sys.exit(%d)" % hooks.exit_code)
    elif hooks.exception is not None:
        print("death by exception: %s" % hooks.exception)
    else:
        print("natural death")
atexit.register(foo)

# test
sys.exit(1)