Вопрос-Ответ

Timeout function if it takes too long to finish [duplicate]

Функция тайм-аута, если для завершения требуется слишком много времени

У меня есть скрипт оболочки, который перебирает текстовый файл, содержащий URL: ы, который я хочу посетить и сделать скриншоты.

Все это делается просто. Скрипт инициализирует класс, который при запуске создает скриншот каждого сайта в списке. Некоторым сайтам требуется очень, очень много времени для загрузки, а некоторые могут вообще не загружаться. Итак, я хочу обернуть screengrabber-функцию в скрипт тайм-аута, заставляя функцию возвращаться False если она не может завершиться в течение 10 секунд.

Я доволен самым простым возможным решением, возможно, установкой асинхронного таймера, который будет возвращать False через 10 секунд независимо от того, что на самом деле происходит внутри функции?

Переведено автоматически
Ответ 1

Процесс тайм-аута операций описан в документации для signal.

Основная идея заключается в использовании обработчиков сигналов для установки будильника на некоторый интервал времени и создания исключения по истечении этого таймера.

Обратите внимание, что это будет работать только в UNIX.

Вот реализация, которая создает декоратор (сохраните следующий код как timeout.py).

import errno
import os
import signal
import functools

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)

@functools.wraps(func)
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 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():
...
Ответ 2

Я переписал ответ Дэвида, используя оператор 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)
python