Как запустить скрипт 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. https://technet.microsoft.com/en-us/library/cc990289 (v=ws.11).aspx
  2. создание службы с помощью sc.exe; как передать в контекст параметры

пошаговое объяснение, как заставить его работать:

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