Функция тайм-аута, если она занимает слишком много времени, чтобы закончить
У меня есть сценарий оболочки, который просматривает текстовый файл, содержащий URL-адреса, которые я хочу посетить, и делаю снимки экрана.
Все это сделано и просто. Сценарий инициализирует класс, который при запуске создает снимок экрана каждого сайта в списке. Некоторые сайты загружаются очень и очень долго, а некоторые могут вообще не загружаться. Поэтому я хочу обернуть функцию screengrabber в скрипт тайм-аута, возвращая функцию False
если это не могло закончиться в течение 10 секунд.
Я согласен с самым простым решением, возможно, установив асинхронный таймер, который будет возвращать False через 10 секунд, независимо от того, что на самом деле происходит внутри функции?
2 ответа
Процесс тайм-аута операции описан в документации для сигнала.
Основная идея состоит в том, чтобы использовать обработчики сигналов, чтобы установить тревогу на некоторый интервал времени и вызвать исключение по истечении этого таймера.
Обратите внимание, что это будет работать только в UNIX.
Вот реализация, которая создает декоратор (сохраните следующий код как timeout.py
).
from functools import wraps
import errno
import os
import signal
class TimeoutError(Exception):
pass
def timeout(seconds=10, error_message=os.strerror(errno.ETIME)):
def decorator(func):
def _handle_timeout(signum, frame):
raise TimeoutError(error_message)
def wrapper(*args, **kwargs):
signal.signal(signal.SIGALRM, _handle_timeout)
signal.alarm(seconds)
try:
result = func(*args, **kwargs)
finally:
signal.alarm(0)
return result
return wraps(func)(wrapper)
return decorator
Это создает декоратор под названием @timeout
это может быть применено к любым долгосрочным функциям.
Итак, в коде приложения вы можете использовать декоратор так:
from timeout import timeout
# Timeout a long running function with the default expiry of 10 seconds.
@timeout
def long_running_function1():
...
# Timeout after 5 seconds
@timeout(5)
def long_running_function2():
...
# Timeout after 30 seconds, with the error "Connection timed out"
@timeout(30, os.strerror(errno.ETIMEDOUT))
def long_running_function3():
...
Я переписал ответ Дэвида, используя with
Скажите, это позволяет вам сделать это:
with timeout(seconds=3):
time.sleep(4)
Что вызовет ошибку TimeoutError.
Код все еще использует signal
и, следовательно, только UNIX:
import signal
class timeout:
def __init__(self, seconds=1, error_message='Timeout'):
self.seconds = seconds
self.error_message = error_message
def handle_timeout(self, signum, frame):
raise TimeoutError(self.error_message)
def __enter__(self):
signal.signal(signal.SIGALRM, self.handle_timeout)
signal.alarm(self.seconds)
def __exit__(self, type, value, traceback):
signal.alarm(0)