Как запустить скрипт Python в качестве службы в Windows?
я набрасываю архитектуру для набора программ, которые совместно используют различные взаимосвязанные объекты, хранящиеся в базе данных. Я хочу, чтобы одна из программ действовала как служба, которая предоставляет интерфейс более высокого уровня для операций с этими объектами, а другие программы для доступа к объектам через эту службу.
в настоящее время я нацелен на Python и Django framework в качестве технологий для реализации этой службы. Я уверен, что я понимаю, как демонизировать Python программы в Linux. Однако это необязательный элемент спецификации, который система должна поддерживать Windows. У меня мало опыта программирования Windows и никакого опыта работы со службами Windows.
можно ли запускать программы на Python как служба Windows (у меня. е. запустить его автоматически без входа пользователя в систему)? мне не обязательно будет реализовывать эту часть, но мне нужна грубая идея, как это будет сделано, чтобы решить, следует ли проектировать вдоль этих русло.
Edit: Спасибо за все ответы до сих пор, они довольно полные. Я хотел бы знать еще одну вещь: как Windows знает о моей службе? Могу ли я управлять им с помощью собственных утилит Windows? что эквивалентно помещению скрипта start/stop в /etc/init.Ди?
8 ответов
Да, вы можете. Я делаю это с помощью библиотек pythoncom, которые входят в комплект ActivePython или может быть установлен с pywin32 (Python для расширений Windows).
это основной скелет для простой службы:
import win32serviceutil
import win32service
import win32event
import servicemanager
import socket
class AppServerSvc (win32serviceutil.ServiceFramework):
_svc_name_ = "TestService"
_svc_display_name_ = "Test Service"
def __init__(self,args):
win32serviceutil.ServiceFramework.__init__(self,args)
self.hWaitStop = win32event.CreateEvent(None,0,0,None)
socket.setdefaulttimeout(60)
def SvcStop(self):
self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING)
win32event.SetEvent(self.hWaitStop)
def SvcDoRun(self):
servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE,
servicemanager.PYS_SERVICE_STARTED,
(self._svc_name_,''))
self.main()
def main(self):
pass
if __name__ == '__main__':
win32serviceutil.HandleCommandLine(AppServerSvc)
ваш код будет идти в main()
метод-обычно с каким-то бесконечным циклом, который может быть прерван проверкой флага, который вы устанавливаете в SvcStop
метод
хотя я поддержал выбранный ответ пару недель назад, тем временем я боролся намного больше с этой темой. Похоже, что специальная установка Python и использование специальных модулей для запуска скрипта в качестве службы-это просто неправильный способ. Как насчет переносимости и тому подобного?
я наткнулся на замечательный Non-сосание Service Manager, что сделало его очень простым и здравомыслящим, чтобы иметь дело со службами Windows. Я подумал, что, поскольку я могу передать варианты установленная служба, я мог бы также выбрать свой исполняемый файл Python и передать свой скрипт в качестве опции.
Я еще не пробовал это решение, но я сделаю это прямо сейчас и обновлю этот пост по ходу процесса. Мне также интересно использовать virtualenvs в Windows, поэтому я могу рано или поздно придумать учебник и ссылку на него здесь.
существует несколько альтернатив для установки в качестве службы практически любого исполняемого файла Windows.
метод 1: Используйте instsrv и srvany из rktools.exe
для Windows Home Server или Windows Server 2003 (также работает с WinXP),Средства Набора Ресурсов Windows Server 2003 поставляется с утилитами, которые могут быть использованы в тандеме для этого, называется instsrv.exe и srvany.exe. См. эту статью Microsoft KB KB137890 для получения подробной информации о том, как использовать эти utils.
для Windows Home Server Есть отличная удобная оболочка для этих утилит с именем aptly"Любой Установщик Службы".
Метод 2: Используйте ServiceInstaller для Windows NT
есть еще одна альтернатива с помощью ServiceInstaller для Windows NT (скачать-может здесь) С доступны инструкции python. Вопреки имя, он работает как с Windows 2000, так и с Windows XP. Ниже приведены некоторые инструкции по установке скрипта python в качестве службы.
установка скрипта Python
запустите ServiceInstaller для создания нового услуга. (В данном примере это предполагается, что python установлен на c:\python25)
Service Name : PythonTest Display Name : PythonTest Startup : Manual (or whatever you like) Dependencies : (Leave blank or fill to fit your needs) Executable : c:\python25\python.exe Arguments : c:\path_to_your_python_script\test.py Working Directory : c:\path_to_your_python_script
после установки, откройте контроль Апплет служб панели, выберите и запустить PythonTest услуга.
после моего первоначального ответа я заметил, что на SO уже размещены тесно связанные вопросы и ответы. См. также:
могу ли я запустить скрипт Python в качестве службы (в Windows)? Как?
как сделать Windows в курсе службы, которую я написал на Python?
самый простой способ достичь этого-использовать собственную команду sc.exe:
sc create PythonApp binPath= "C:\Python34\Python.exe --C:\tmp\pythonscript.py"
пошаговое объяснение, как заставить его работать:
1-сначала создайте файл python в соответствии с базовым скелетом, упомянутым выше. И сохраните его на пути, например: "c:\PythonFiles\AppServerSvc.py"
import win32serviceutil
import win32service
import win32event
import servicemanager
import socket
class AppServerSvc (win32serviceutil.ServiceFramework):
_svc_name_ = "TestService"
_svc_display_name_ = "Test Service"
def __init__(self,args):
win32serviceutil.ServiceFramework.__init__(self,args)
self.hWaitStop = win32event.CreateEvent(None,0,0,None)
socket.setdefaulttimeout(60)
def SvcStop(self):
self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING)
win32event.SetEvent(self.hWaitStop)
def SvcDoRun(self):
servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE,
servicemanager.PYS_SERVICE_STARTED,
(self._svc_name_,''))
self.main()
def main(self):
# Your business logic or call to any class should be here
# this time it creates a text.txt and writes Test Service in a daily manner
f = open('C:\test.txt', 'a')
rc = None
while rc != win32event.WAIT_OBJECT_0:
f.write('Test Service \n')
f.flush()
# block for 24*60*60 seconds and wait for a stop event
# it is used for a one-day loop
rc = win32event.WaitForSingleObject(self.hWaitStop, 24 * 60 * 60 * 1000)
f.write('shut down \n')
f.close()
if __name__ == '__main__':
win32serviceutil.HandleCommandLine(AppServerSvc)
2 - На этом шаге мы должны зарегистрировать наши услуги.
запустить командную строку администратор и типа как:
sc создать testservice binpath= "C:\Python36\Python - ... exe c:\PythonFiles\AppServerSvc.py" DisplayName=" TestService " start= auto
первый аргумент binpath - это путь python.exe
второй аргумент binpath is путь к вашему файлу python что мы уже создали
Не пропустите, что вы должны поставить один пробел после "=" знак.
тогда, если все в порядке, вы должны увидеть
[SC] CreateService Успех
теперь ваша служба python установлена как служба windows. Вы можете увидеть его в Service Manager и registry в разделе:
раздел HKEY_LOCAL_MACHINE\система\CurrentControlSet на\сервисы\службы testservice
3 - Ок теперь. Вы можете запустить службу в service manager.
вы можете выполнить каждый файл python, который предоставляет этот скелет службы.
самый простой способ-это использовать: NSSM-не всасывающий Диспетчер служб:
1-сделать загрузку на https://nssm.cc/download
2-Установите программу python в качестве службы: win prompt as admin
c:>nssm.exe установить WinService
3-на консоли NSSMs:
путь: C:\Python27\Python27.exe
запуск каталог: C:\Python27
Аргументы: c:\WinService.py
4-Проверьте созданные службы в службах.msc
Я начал хостинг как сервис с pywin32.
все было хорошо, но я столкнулся с проблемой, что служба не смогла запустить в течение 30 секунд (тайм-аут по умолчанию для Windows) при запуске системы. Это было критично для меня, потому что запуск Windows происходил одновременно на нескольких виртуальных машинах, размещенных на одной физической машине, и загрузка IO была огромной. Сообщения об ошибках:
Error 1053: The service did not respond to the start or control request in a timely fashion.
Error 7009: Timeout (30000 milliseconds) waiting for the <ServiceName> service to connect.
Я много боролся с pywin, но в конечном итоге с использованием NSSM, как было предложено в ответ. Мигрировать туда было очень легко.
pysc: диспетчер управления службами на Python
пример скрипта для запуска в качестве службы взято из pythonhosted.org:
from xmlrpc.server import SimpleXMLRPCServer from pysc import event_stop class TestServer: def echo(self, msg): return msg if __name__ == '__main__': server = SimpleXMLRPCServer(('127.0.0.1', 9001)) @event_stop def stop(): server.server_close() server.register_instance(TestServer()) server.serve_forever()
создать и запустить службу
import os import sys from xmlrpc.client import ServerProxy import pysc if __name__ == '__main__': service_name = 'test_xmlrpc_server' script_path = os.path.join( os.path.dirname(__file__), 'xmlrpc_server.py' ) pysc.create( service_name=service_name, cmd=[sys.executable, script_path] ) pysc.start(service_name) client = ServerProxy('http://127.0.0.1:9001') print(client.echo('test scm'))
остановить и удалить службу
import pysc service_name = 'test_xmlrpc_server' pysc.stop(service_name) pysc.delete(service_name)
pip install pysc