1

このサイトで質問するのは初めてですが、Stackoverflow を使用して、過去 2 年間に他のユーザーが抱えていた問題を調べました。残念ながら、私が現在抱えている問題に関連する投稿は見つかりませんでした。だからここに行きます...

ブート中にすぐに実行したい Python スクリプトがあります (つまり、Windows のログイン画面でも)。これを行うために、Python win32serviceutil フレームワークを使用して Windows サービスを作成し、サービスのインストール時にサービスを「自動」に設定します。それは非常に簡単に思え、Web に投稿された例を見て、すぐにサービスが機能するようになりました。私のコードは次のようになります。

import win32service
import win32serviceutil
import win32api
import win32con
import win32event
import win32evtlogutil
import os, sys, string, time
from STEL.ClientServer.StartClient import StartClient

class aservice(win32serviceutil.ServiceFramework):

   _svc_name_ = "STELClient"
   _svc_display_name_ = "My Service Long Fancy Name!"
   _svc_description_ = "THis is what my crazy little service does - aka a DESCRIPTION! WHoa!"

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

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

   def SvcDoRun(self):
      import servicemanager
      servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE,servicemanager.PYS_SERVICE_STARTED,(self._svc_name_, ''))

      self.timeout = 10000    
      # This is how long the service will wait to run / refresh itself (see script below)
      try:
          self.client.startClient()
      except:
          servicemanager.LogInfoMsg("STELClient - EXCEPTION !!")  #For Event Log

      while 1:
         # Wait for service stop signal, if I timeout, loop again
         rc = win32event.WaitForSingleObject(self.hWaitStop, self.timeout)
         # Check to see if self.hWaitStop happened
         if rc == win32event.WAIT_OBJECT_0:
            # Stop signal encountered
            self.client.stopClient()
            servicemanager.LogInfoMsg("STELClient - STOPPED!")  #For Event Log
            break
         else:
            execfile("C:\\STEL\\clientExample.py")
            pass


def ctrlHandler(ctrlType):
   return True

if __name__ == '__main__':
   win32api.SetConsoleCtrlHandler(ctrlHandler, True)
   win32serviceutil.HandleCommandLine(aservice)

関心のある主なコードは次のとおりです。

try:
      self.client.startClient()
  except:
      servicemanager.LogInfoMsg("STELClient - EXCEPTION !!")  #For Event Log

と:

else:
    execfile("C:\\STEL\\clientExample.py")
    pass

どちらのコード ブロックも同じサーバーに ping を送信します。唯一の違いは、スクリプトが呼び出されるたびに clientExample.py がサーバーに ping を実行することです。この場合、while ループのために 10 秒ごとに ping が実行されます。client.startClient() は独自のスレッドを生成し、5 秒ごとにサーバーに ping を送信します。

サービスをインストールしてからスクリプトを実行した後、client.startClient() スレッドが 3 ~ 4 分後に停止したように見えますが (私が計測した場合は約 3 分 40 秒)、clientExample.py は引き続き実行されます。 10秒ごと。

Python で次のコード行を実行すると、Python を停止/終了するまで、スレッドは無期限に実行されます。

from STEL.ClientServer.StartClient import StartClient
self.client = StartClient()
self.client.startClient()

Windows サービスでもこれと同じ動作を期待していましたが、そうではないようです。

誰にも手がかりはありますか?

4

1 に答える 1

1

そのため、WINPDB を使用してデバッガーをサービスに接続した後、問題を解決しました。

スレッドが停止していた理由は、キャッチされない例外 (dun dun DUNNNNN) があったためです。例外は、IOError: 9 (Bad file descriptor) をスローした print ステートメント (本当に???) によって引き起こされました。少しStackOverflow検索を行ったところ、これが見つかりました...

print ステートメントの作成中に IOError: (9, 'Bad file descriptor') エラーが発生するのはなぜですか?

printステートメントを削除して、問題が解決するかどうかを確認します...

生きて学ぶ...

于 2013-07-01T22:57:29.197 に答える