Как запустить скрипт Python в качестве службы в Windows?
я набрасываю архитектуру для набора программ, которые совместно используют различные взаимосвязанные объекты, хранящиеся в базе данных. Я хочу, чтобы одна из программ действовала как служба, которая обеспечивает интерфейс более высокого уровня для операций над этими объектами, а другие программы-для доступа к объектам через эту службу.
в настоящее время я стремлюсь к Python и платформе Django в качестве технологий для реализации этой службы. Я почти уверен, что понимаю, как демонизировать питона программы в 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-sucking Service Manager, что сделало его очень простым и разумным, чтобы иметь дело с Windows Services. Я подумал, так как я мог бы передать варианты установленная служба, я мог бы также выбрать свой исполняемый файл Python и передать свой скрипт в качестве опции.
Я еще не пробовал это решение, но я сделаю это прямо сейчас и обновлю этот пост по ходу процесса. Я также заинтересован в использовании virtualenvs на Windows, так что я мог бы придумать учебник рано или поздно и ссылку на него здесь.
есть несколько альтернатив для установки в качестве службы практически любой исполняемый файл Windows.
метод 1: Используйте instsrv и srvany из rktools.exe
для Windows Home Server или Windows Server 2003 (работает и с WinXP),Windows Server 2003 Resource Kit Tools поставляется с утилитами, которые могут быть использованы в тандеме для этого, называется instsrv.exe и srvany.exe. См. эту статью базы знаний Майкрософт 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 create TestService binpath= "C:\Python36\Python.exe c:\PythonFiles\AppServerSvc.py" DisplayName=" TestService " start= auto
первый аргумент binpath - это путь python.exe
второй аргумент binpath и путь к вашему файлу python что мы уже создали
Не пропустите, что вы должны поставить один пробел после "=" знак.
тогда, если все в порядке, вы должны увидеть
[SC] CreateService Успех
теперь ваша служба python установлена как служба windows. Вы можете увидеть его в Диспетчере служб и реестре в разделе:
HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\TestService
3 - Ок теперь. Вы можете запустить свою службу в Service manager.
вы можете выполнить каждый файл python, который предоставляет этот скелет службы.
самый простой способ-это использовать: NSSM-менеджер услуг без сосания:
1-сделать загрузку наhttps://nssm.cc/download
2-Установите программу python в качестве службы: win prompt as admin
c:>nssm.exe install WinService
3-на консоли NSSMs:
путь: C:\Python27\Python27.exe
запуск каталог: C:\Python27
Аргументы: c:\WinService.py
4-Проверьте созданные сервисы на сервисах.msc
Я начал хостинг как сервис с pywin32.
все было хорошо, но я столкнулся с проблемой, что служба не смогла запустить в течение 30 секунд (тайм-аут по умолчанию для Windows) при запуске системы. Это было критично для меня, потому что запуск Windows происходил одновременно на нескольких виртуальных машинах, размещенных на одной физической машине, и загрузка ввода-вывода была огромной. Сообщения об ошибках:
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