Как очистить ресурсы задач и процессы-зомбиPython

Программы на Python
Ответить
Anonymous
 Как очистить ресурсы задач и процессы-зомби

Сообщение Anonymous »

Я создаю приложение celery + django + selenium. Я запускаю браузеры на основе селена в отдельных процессах с помощью сельдерея. Версии:

Код: Выделить всё

celery==5.2.6
redis==3.4.1
selenium-wire==5.1.0
Django==4.0.4
djangorestframework==3.13.1
Я обнаружил, что через несколько часов приложение генерирует тысячи процессов-зомби. Также выяснилось, что проблема связана с контейнером докеров сельдерея, потому что после sudo /usr/local/bin/docker-compose -f /data/new_app/docker-compose.yml перезапустить сельдерей у меня 0 зомби процессы.
Мой код

Код: Выделить всё

from rest_framework.decorators import api_view

@api_view(['POST'])
def periodic_check_all_urls(request): # web-service endpoint
...
check_urls.delay(parsing_results_ids) # call celery task

Код задачи «Сельдерей»

Код: Выделить всё

from celery import shared_task

@shared_task()
def check_urls(parsing_result_ids: List[int]):
"""
Run Selenium-based parser
the parser exctracts data and saves in database
"""
try:
logger.info(f"{datetime.now()} Start check_urls")
parser = Parser() # open selenium browser
parsing_results = ParsingResult.objects.filter(pk__in=parsing_result_ids).exclude(status__in=["DONE", "FAILED"])
parser.check_parsing_result(parsing_results)
except Exception as e:
full_trace = traceback.format_exc()
finally:
if 'parser' in locals():
parser.stop()

Функция остановки браузера Selenium и деструктор

Код: Выделить всё

class Parser():
def __init__(self):
"""
Prepare parser
"""
if not USE_GUI:
self.display = Display(visible=0, size=(800, 600))
self.display.start()

""" Replaced with FireFox
self.driver = get_chromedriver(proxy_data)
"""
proxy_data = {
...
}
self.driver = get_firefox_driver(proxy_data=proxy_data)

def __del__(self):
self.stop()

def stop(self):
try:
self.driver.quit()
logger.info("Selenium driver closed")
except:
pass

try:
self.display.stop()
logger.info("Display stopped")
except:
pass
Также я пробовал несколько настроек, чтобы ограничить ресурсы задачи сельдерея и время работы (не помогло с процессами Zombie)
Мои настройки сельдерея в dgango settings.py

Код: Выделить всё

# celery setting (documents generation)
CELERY_BROKER_URL = os.environ.get("CELERY_BROKER", "redis://redis:6379/0")
CELERY_RESULT_BACKEND = os.environ.get("CELERY_BROKER", "redis://redis:6379/0")
CELERY_IMPORTS = ("core_app.celery",)
CELERY_TASK_TIME_LIMIT = 10 * 60
Мои настройки сельдерея в докерах

Код: Выделить всё

celery:
build: ./project
command: celery -A core_app worker  --loglevel=info --concurrency=15 --max-memory-per-child=1000000
volumes:
- ./project:/usr/src/app
- ./project/media:/project/media
- ./project/logs:/project/logs
env_file:
- .env
environment:
# environment variables declared in the environment section override env_file
- DJANGO_ALLOWED_HOSTS=localhost 127.0.0.1 [::1]
- CELERY_BROKER=redis://redis:6379/0
- CELERY_BACKEND=redis://redis:6379/0
depends_on:
- django
- redis
Я читаю Django/Celery. Как завершить задачу сельдерея? но это не помогло
Также прочитайте отзыв сельдерея, оставив зомби-процесс ffmpeg, но моя задача уже содержит try/Exception

Подробнее здесь: https://stackoverflow.com/questions/792 ... -processes
Ответить

Быстрый ответ

Изменение регистра текста: 
Смайлики
:) :( :oops: :roll: :wink: :muza: :clever: :sorry: :angel: :read: *x)
Ещё смайлики…
   
К этому ответу прикреплено по крайней мере одно вложение.

Если вы не хотите добавлять вложения, оставьте поля пустыми.

Максимально разрешённый размер вложения: 15 МБ.

Вернуться в «Python»