无法启动用Python编写的Windows服务(win32serviceutil)

20 投票
8 回答
32760 浏览
提问于 2025-04-17 10:42

我正在尝试启动一个简单的服务示例:

someservice.py:

import win32serviceutil 
import win32service 
import win32event

class SmallestPythonService(win32serviceutil.ServiceFramework):
    _svc_name_ = "SmallestPythonService"
    _svc_display_name_ = "display service"

    def __init__(self, args):
        win32serviceutil.ServiceFramework.__init__(self, args)
        self.hWaitStop = win32event.CreateEvent(None, 0, 0, None)

    def SvcStop(self):
        self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING)
        win32event.SetEvent(self.hWaitStop)

    def SvcDoRun(self):
        win32event.WaitForSingleObject(self.hWaitStop, win32event.INFINITE)

if __name__=='__main__':
    win32serviceutil.HandleCommandLine(SmallestPythonService)

当我运行

python someservice.py install

一切正常,服务出现在Windows服务列表中,但是

python someservice.py start

却出现了“错误1053:服务没有及时响应启动或控制请求”的问题,但实际上并没有任何延迟。

我在网上查找了解决方案,发现这个问题发生在 pythonservice.exe 找不到 python27.dll 的时候。确实是找不到,所以我把 C:\Python27 加入了 PATH 环境变量。现在 pythonservice.exe 可以正常运行了,但错误1053依然存在。

我在Windows 7 Ultimate上运行的是Python 2.7.2和pywin32 216,并且有管理员权限。

8 个回答

4

我觉得你的问题可以通过修改方法 SvcDoRun 来解决。

把它从

   def  SvcDoRun(self):
        win32event.WaitForSingleObject(self.hWaitStop, win32event.INFINITE)

改成

   def  SvcDoRun(self):
        self.ReportServiceStatus(win32service.SERVICE_RUNNING)
        win32event.WaitForSingleObject(self.hWaitStop, win32event.INFINITE)
6

我也遇到过这个问题,后来通过在我的 "__main__" 执行块中添加以下内容解决了它:

if len(sys.argv) == 1:
    servicemanager.Initialize()
    servicemanager.PrepareToHostSingle(RouterService)
    servicemanager.StartServiceCtrlDispatcher()
else:
    win32serviceutil.HandleCommandLine(RouterService)

(别忘了在文件顶部导入 servicemanager)。

我觉得问题出在 Windows 服务管理器默认情况下运行可执行文件时没有传递任何参数,这样一来,应用程序就需要明确告诉它要启动服务,似乎 SvcDoRun 并不会自动被调用。

正如其他人提到的,如果你是从命令行运行这个程序,你确实需要一个路径映射。在我的应用程序中,我使用 cx_freeze 冻结了服务,并用这个可执行文件来安装服务,这样所有的依赖项都包含在内了。

16

另外,感谢你提到可能是DLL的问题,这让我找到了正确的解决办法。

你需要做的是把Python27添加到系统路径(SYSTEM PATH),而不是用户路径(USER PATH)。因为默认情况下,Python服务会以“本地系统”(LocalSystem)身份安装,所以当它尝试启动时,会使用系统路径变量。这就是为什么你可以在命令提示符下运行它,而你的用户路径是正确的。

希望这对你有帮助!

撰写回答