2011-03-15 87 views
1

以下是我正在运行的Windows服务的框架。如果发生错误,则会记录错误并可以在事件查看器中查看。问题是脚本退出并且不会重新启动,即使我已将恢复设置为在第一次,第二次和随后的故障时重新启动服务。目前我的错误处理很少,因为我想查看事件查看器中可能出现的错误,以便我可以编写相应的代码来处理这些错误。如何使Windows服务保持运行

from win32api import CloseHandle, GetLastError, SetConsoleCtrlHandler 
import os 
import sys 
import time 
import pythoncom 
import win32serviceutil 
import win32service 
import win32event 
import servicemanager 
import socket 

class AppServerSvc (win32serviceutil.ServiceFramework): 
    _svc_name_ = "my_service_name" 
    _svc_display_name_ = "my service" 

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

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

    def SvcDoRun(self): 
     servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE, 
           servicemanager.PYS_SERVICE_STARTED, 
           (self._svc_name_,'')) 
     self.run = True 
     self.main() 

    def main(self): 
     while self.run == True 
      pass   

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

编辑:

我想尝试:除了围绕self.main但是结果还是一样。该服务在崩溃时不会重新启动...请任何人有任何想法?如果它不能在发生碰撞的情况下,重新启动服务是不是所有的有用......还不如运行它作为一个.pyc文件

编辑:

下面

是中可能出现的错误的例子我的脚本......我不相信这个错误消息是特别有用的,因为我试图实现的是让服务重新启动,但是这里没有一个是在没有重新启动的情况下崩溃我的服务的错误的示例:

The instance's SvcRun() method failed 
Traceback (most recent call last): 
    File "C:\Python27\lib\site-packages\win32\lib\win32serviceutil.py", line 806, in SvcRun 
    self.SvcDoRun() 
    File "C:\Some_Service.py", line 46, in SvcDoRun 
    self.main() 
    File "Some_Service.py", line 61, in main 
    ser = self.open_serial_port() 
    File "Some_Service.py", line 70, in open_serial_port 
    serial_connection.open() 
    File "C:\Python27\lib\site-packages\serial\serialwin32.py", line 56, in open 
    raise SerialException("could not open port %s: %s" % (self.portstr, ctypes.WinError())) 
SerialException: could not open port COM6: [Error 1225] The remote system refused the network connection. 
%2: %3 
+0

什么是导致它崩溃的错误?除了在main()中不睡觉,你的代码看起来很好用。 – AlG 2011-03-22 12:21:43

+0

@ qor72这只是使脚本成为服务的框架。我正在运行的脚本用于解析串行数据。问题是可能会发生一些错误,例如服务可能无法连接到我的数据库。我现在想要休息几秒钟,然后重新启动服务。我不知道为什么它应该死了,而不是重新启动,如果我有服务设置重新启动失败..我必须在某处丢失一段代码来处理? – Richard 2011-03-22 12:30:04

+0

在我编写的服务中,不是使用Python,错误检查将错误写入日志并继续,而不需要重新启动。这就是为什么我对你的信息很好奇;例如,服务是否无法写入错误日志,导致一个破坏服务的错误? – AlG 2011-03-22 12:58:08

回答

5

下面是一个以零分出错的服务。如果发生错误,则会发送事件并使用os._exit(-1)退出服务。该值必须是0以外的任何值,以便窗口知道服务没有很好地退出。

from win32api import CloseHandle, GetLastError, SetConsoleCtrlHandler 
import os 
import sys 
import time 

import win32serviceutil 
import win32service 
import win32event 
import servicemanager 

import traceback 



class AppServerSvc (win32serviceutil.ServiceFramework): 
    _svc_name_ = "test" 
    _svc_display_name_ = "test" 

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




    def SvcStop(self): 
     self.ReportServiceStatus(win32service.SERVICE_STOP_PENDING) 
     win32event.SetEvent(self.hWaitStop) 
     self.run = False 
    def SvcDoRun(self): 

     servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE, 
           servicemanager.PYS_SERVICE_STARTED, 
           (self._svc_name_,'')) 
     self.run = True 
     try: # try main 
      self.main() 
     except: 
      servicemanager.LogErrorMsg(traceback.format_exc()) # if error print it to event log 
      os._exit(-1)#return some value other than 0 to os so that service knows to restart 


    def main(self): 

     while self.run == True: 
      time.sleep(30)   
      t = 1/0 

if __name__ == '__main__': 
    win32serviceutil.HandleCommandLine(AppServerSvc) 
+0

如果有人知道这样做的更好方法,请发布以便获得赏金... – Richard 2011-03-22 13:45:25

+3

是否有理由使用os._exit(-1)而不是sys。退出(-1)? – 2013-09-08 05:44:18

0

我怀疑你的服务实际上并没有崩溃,而是在它有机会运行之前完成。您是否尝试在main函数中插入几秒钟的睡眠声明?

+0

直到服务get的停止命令为止,主函数是一个无限循环。该脚本实际上是崩溃的,因为导致崩溃的错误被记录下来。如果出现错误,我希望服务重新启动... – Richard 2011-03-18 13:44:26

+0

什么是您的错误消息?有可能当它崩溃时它仍然返回一个0的错误代码?也许你需要一些东西:'尝试:\ n main()\ nexcept:\ n返回-1' – theheadofabroom 2011-03-18 13:56:58

0

为什么不使用try-except?

class AppServerSvc (win32serviceutil.ServiceFramework): 
    _svc_name_ = "my_service_name" 
    _svc_display_name_ = "my service" 

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

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

    def SvcDoRun(self): 
     servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE, 
           servicemanager.PYS_SERVICE_STARTED, 
           (self._svc_name_,'')) 
     self.run = True 
     self.main() 

    def main(self): 
     while self.run == True 
      try : 
       <your code> 
      except : 
       time.sleep(3) 
       <may be an error log here> 



if __name__ == '__main__': 
    win32serviceutil.HandleCommandLine(AppServerSvc) 
+0

在我的答案中,我做到了。就在我称之为main的地方,而不是在函数本身。 – Richard 2011-03-24 13:33:32