Запустите php скрипт как процесс демона - PullRequest
144 голосов
/ 10 января 2010

Мне нужно запустить скрипт php как процесс демона (ждать инструкций и делать что-то). Работа cron не сделает это для меня, потому что действия должны быть предприняты, как только прибудет инструкция. Я знаю, что PHP не самый лучший вариант для процессов-демонов из-за проблем с управлением памятью, но по разным причинам мне приходится использовать PHP в этом случае. Я наткнулся на инструмент libslack под названием Daemon (http://libslack.org/daemon), который, кажется, помогает мне управлять процессами демона, но за последние 5 лет не было обновлений, поэтому мне интересно, знаете ли вы какие-нибудь другие альтернативы, подходящие для мой случай. Любая информация будет по достоинству оценена.

Ответы [ 14 ]

159 голосов
/ 10 января 2010

Вы можете запустить свой php-скрипт из командной строки (т.е. bash), используя

nohup php myscript.php &

& помещает ваш процесс в фоновый режим.

Edit:
Да, есть некоторые недостатки, но их невозможно контролировать? Это просто неправильно.
Простой kill processid остановит это. И это все еще лучшее и простое решение.

155 голосов
/ 16 мая 2013

Другой вариант - использовать Upstart . Первоначально он был разработан для Ubuntu (и поставляется с ним по умолчанию), но предназначен для всех дистрибутивов Linux.

Этот подход аналогичен Supervisord и daemontools в том, что он автоматически запускает демон при загрузке системы и восстанавливается при завершении сценария.

Как настроить:

Создайте новый файл сценария на /etc/init/myphpworker.conf. Вот пример:

# Info
description "My PHP Worker"
author      "Jonathan"

# Events
start on startup
stop on shutdown

# Automatically respawn
respawn
respawn limit 20 5

# Run the script!
# Note, in this example, if your PHP script returns
# the string "ERROR", the daemon will stop itself.
script
    [ $(exec /usr/bin/php -f /path/to/your/script.php) = 'ERROR' ] && ( stop; exit 1; )
end script

Запуск и остановка вашего демона:

sudo service myphpworker start
sudo service myphpworker stop

Проверьте, работает ли ваш демон:

sudo service myphpworker status

Спасибо

Огромное спасибо Кевину ван Зонневельду , откуда я выучил эту технику.

50 голосов
/ 07 июня 2017

С новым systemd вы можете создать сервис.

Вы должны создать файл или символическую ссылку в /etc/systemd/system/, например. myphpdaemon.service и размещайте контент, подобный этому, myphpdaemon будет названием службы:

[Unit]
Description=My PHP Daemon Service
#May your script needs MySQL or other services to run, eg. MySQL Memcached
Requires=mysqld.service memcached.service 
After=mysqld.service memcached.service

[Service]
User=root
Type=simple
TimeoutSec=0
PIDFile=/var/run/myphpdaemon.pid
ExecStart=/usr/bin/php -f /srv/www/myphpdaemon.php arg1 arg2> /dev/null 2>/dev/null
#ExecStop=/bin/kill -HUP $MAINPID #It's the default you can change whats happens on stop command
#ExecReload=/bin/kill -HUP $MAINPID
KillMode=process

Restart=on-failure
RestartSec=42s

StandardOutput=null #If you don't want to make toms of logs you can set it null if you sent a file or some other options it will send all php output to this one.
StandardError=/var/log/myphpdaemon.log
[Install]
WantedBy=default.target

Вы сможете запускать, получать статус, перезапускать и останавливать службы, используя команду

systemctl <start|status|restart|stop|enable> myphpdaemon

PHP-скрипт должен иметь своего рода «цикл» для продолжения работы.

<?php
gc_enable();//
while (!connection_aborted() || PHP_SAPI == "cli") {

  //Code Logic

  //sleep and usleep could be useful
    if (PHP_SAPI == "cli") {
        if (rand(5, 100) % 5 == 0) {
            gc_collect_cycles(); //Forces collection of any existing garbage cycles
        }
    }
}

Рабочий пример:

[Unit]
Description=PHP APP Sync Service
Requires=mysqld.service memcached.service
After=mysqld.service memcached.service

[Service]
User=root
Type=simple
TimeoutSec=0
PIDFile=/var/run/php_app_sync.pid
ExecStart=/bin/sh -c '/usr/bin/php -f /var/www/app/private/server/cron/app_sync.php  2>&1 > /var/log/app_sync.log'
KillMode=mixed

Restart=on-failure
RestartSec=42s

[Install]
WantedBy=default.target

Если ваша подпрограмма PHP должна выполняться один раз в цикле (например, дайджест), вы можете использовать сценарий оболочки или bash для вызова в служебный файл systemd вместо PHP напрямую, например:

#!/usr/bin/env bash
script_path="/app/services/"

while [ : ]
do
#    clear
    php -f "$script_path"${1}".php" fixedparameter ${2}  > /dev/null 2>/dev/null
    sleep 1
done

Если вы выбрали эту опцию, вы должны изменить KillMode на mixed, чтобы процессы, bash (main) и PHP (child) были убиты.

ExecStart=/app/phpservice/runner.sh phpfile parameter  > /dev/null 2>/dev/null
KillMode=process

This method also is effective if you're facing a memory leak.

Примечание. Каждый раз, когда вы меняете свой «myphpdaemon.service», вы должны запустите `systemctl daemon-reload ', но не беспокойтесь, если это не так, это будет оповещается, когда это необходимо.

47 голосов
/ 10 января 2010

Если вы можете - возьмите копию Расширенное программирование в среде UNIX . Вся глава 13 посвящена программированию демонов. Примеры написаны на C, но все необходимые функции имеют обертки в PHP (в основном расширения pcntl и posix ).

В двух словах - написание демона (это возможно только на ОС * nix - Windows использует службы) выглядит так:

  1. Позвоните umask(0), чтобы предотвратить проблемы с разрешениями.
  2. fork() и есть родительский выход.
  3. Звоните setsid().
  4. Настройка обработки сигнала SIGHUP (обычно это игнорируется или используется для подачи сигнала демону для перезагрузки его конфигурации) и SIGTERM (чтобы сообщить процессу о корректном выходе из процесса).
  5. fork() снова и родительский выход.
  6. Изменить текущий рабочий каталог с помощью chdir().
  7. fclose() stdin, stdout и stderr и не пишите им. Правильный способ - перенаправить их либо в /dev/null, либо в файл, но я не смог найти способ сделать это в PHP. Когда вы запускаете демон, возможно, перенаправить его с помощью оболочки (вам придется самому выяснить, как это сделать, я не знаю :).
  8. Делай свою работу!

Кроме того, поскольку вы используете PHP, будьте осторожны с циклическими ссылками, поскольку сборщик мусора PHP, до PHP 5.3, не имел возможности собирать эти ссылки, и процесс будет иметь утечку памяти до тех пор, пока не произойдет сбой.

24 голосов
/ 19 января 2010

Я запускаю большое количество PHP-демонов.

Я согласен с вами, что PHP не лучший (или даже хороший) язык для этого, но демоны делятся кодом с веб-компонентами, поэтому в целом это хорошее решение для нас.

Для этого мы используем daemontools. Это умный, чистый и надежный. На самом деле мы используем его для запуска всех наших демонов.

Вы можете проверить это на http://cr.yp.to/daemontools.html.

РЕДАКТИРОВАТЬ: быстрый список функций.

  • Автоматически запускает демон при перезагрузке
  • Автоматический перезапуск dameon при неудаче
  • Ведение журнала для вас, включая опрокидывание и сокращение
  • Интерфейс управления: 'svc' и 'svstat'
  • UNIX дружественный (возможно, не для всех плюс)
14 голосов
/ 19 января 2010

Вы можете

  1. Используйте nohup, как предложил Хенрик.
  2. Используйте screen и запускайте свою PHP-программу как обычный процесс внутри нее. Это дает вам больше контроля, чем использование nohup.
  3. Используйте демон-демон, подобный http://supervisord.org/ (он написан на Python, но может демонизировать любую программу командной строки и дает вам удаленный контроль для управления им).
  4. Напишите свою собственную оболочку для демона, как предложил Эмиль, но это лишнее ИМО.

Я бы порекомендовал самый простой метод (на мой взгляд, экран), а затем, если вам нужны дополнительные функции или функции, перейдите к более сложным методам.

11 голосов
/ 19 января 2010

Существует несколько способов решения этой проблемы.

Я не знаю специфики, но, возможно, есть другой способ запустить процесс PHP. Например, если вам нужен код для запуска на основе событий в базе данных SQL, вы можете настроить триггер для выполнения вашего скрипта. Это действительно легко сделать в PostgreSQL: http://www.postgresql.org/docs/current/static/external-pl.html.

Честно говоря, я думаю, что вам лучше всего создать процесс Damon с использованием nohup. nohup позволяет продолжить выполнение команды даже после выхода пользователя из системы:

nohup php myscript.php &

Однако существует очень серьезная проблема. Как вы сказали, менеджер памяти PHP является полной фигней, он был построен с учетом того, что сценарий выполняется только в течение нескольких секунд, а затем существует. Ваш скрипт PHP начнет использовать гигабайты памяти только через несколько дней. Вы ДОЛЖНЫ ТАКЖЕ создать скрипт cron, который запускается каждые 12 или, может быть, 24 часа, который убивает и повторно запускает ваш php-скрипт следующим образом:

killall -3 php
nohup php myscript.php &

Но что, если сценарий был в середине работы? Ну, kill -3 - это прерывание, это то же самое, что делать Ctrl + C на CLI. Ваш php-скрипт может перехватить это прерывание и корректно завершить работу с помощью библиотеки PHP pcntl: http://php.oregonstate.edu/manual/en/function.pcntl-signal.php

Вот пример:

function clean_up() {
  GLOBAL $lock;
  mysql_close();
  fclose($lock)
  exit();
}
pcntl_signal(SIGINT, 'clean_up');

Идея, стоящая за $ lock, заключается в том, что скрипт PHP может открывать файл с помощью fopen ("file", "w") ;. Только один процесс может иметь блокировку записи в файл, поэтому, используя это, вы можете убедиться, что запущена только одна копия вашего PHP-скрипта.

Удачи!

10 голосов
/ 15 января 2010

Кевин ван Зонневельд написал очень хорошую подробную статью об этом , в своем примере он использует пакет System_Daemon PEAR (дата последнего выпуска 2009-09-02 ).

6 голосов
/ 16 февраля 2013

Выезд https://github.com/shaneharter/PHP-Daemon

Это объектно-ориентированная библиотека демонов. Он имеет встроенную поддержку таких вещей, как ведение журнала и устранение ошибок, а также поддержку создания фоновых рабочих.

3 голосов
/ 12 июня 2016

У меня недавно была потребность в кроссплатформенном решении (Windows, Mac и Linux) для проблемы запуска PHP-скриптов в качестве демонов. Я решил проблему, написав собственное решение на C ++ и создав двоичные файлы:

https://github.com/cubiclesoft/service-manager/

Полная поддержка Linux (через sysvinit), а также служб Windows NT и запуска Mac OSX.

Если вам просто нужен Linux, то пара других решений, представленных здесь, работают достаточно хорошо и, в зависимости от вкуса. В наши дни также есть Upstart и systemd, у которых есть запасные варианты для сценариев sysvinit. Но половина смысла использования PHP в том, что он кроссплатформенный по своей природе, поэтому код, написанный на языке, имеет довольно хорошие шансы работать везде, как есть. Недостатки начинают проявляться, когда в картину входят некоторые внешние нативные аспекты уровня ОС, такие как системные службы, но эта проблема возникает с большинством языков сценариев.

Попытка поймать сигналы, как это предложил кто-то из пользователей PHP, не очень хорошая идея. Внимательно прочитайте документацию по pcntl_signal(), и вы быстро поймете, что PHP обрабатывает сигналы, используя некоторые довольно неприятные методы (в частности, «тики»), которые сжимают кучу циклов для чего-то, что процессы редко замечают (то есть сигналы). Обработка сигналов в PHP также едва доступна на платформах POSIX, и поддержка отличается в зависимости от версии PHP. Изначально это звучит как приличное решение, но оно не очень полезно.

С течением времени PHP также стал лучше справляться с проблемами утечки памяти. Вы все еще должны быть осторожны (синтаксический анализатор DOM XML имеет тенденцию к утечке), но я редко вижу сбежавшие процессы в наши дни, а средство отслеживания ошибок PHP довольно тихо по сравнению с прошлыми днями.

...