2011-03-15 99 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