HaskellでWindowsサービスアプリケーションを書くのに苦労してきました。
バックグラウンド
サービスアプリケーションは、Windowsサービスコントロールマネージャーによって実行されます。起動すると、サービスのメイン関数として使用されるコールバックが提供されるStartServiceCtrlDispatcherへのブロッキング呼び出しを行います。
サービスのメイン関数は、start、stop、continueなどの着信コマンドを処理するための2番目のコールバックを登録することになっています。これは、RegisterServiceCtrlHandlerを呼び出すことによって行われます。
問題
サービスの主な機能を登録するプログラムを書くことができます。その後、プログラムをWindowsサービスとしてインストールし、サービス管理コンソールから起動できます。サービスを開始し、実行中であることを報告してから、着信要求を待つことができます。
問題は、サービスハンドラー関数を呼び出せないことです。サービスステータスをクエリすると、実行中であることがわかりますが、送信するとすぐに、「停止」コマンドウィンドウに次のメッセージが表示されます。
Windows could not stop the Test service on Local Computer.
Error 1061: The service cannot accept control messages at this time.
MSDNのドキュメントによると、StartServiceCtrlDispatcher関数は、すべてのサービスが停止したことを報告するまでブロックします。サービスのメイン関数が呼び出された後、ディスパッチャースレッドは、サービスコントロールマネージャーがコマンドを送信するまで待機することになっています。コマンドが送信されると、そのスレッドからハンドラー関数が呼び出されます。
詳細
以下は、私がやろうとしていることの非常に単純化されたバージョンですが、ハンドラー関数が呼び出されないという問題を示しています。
まず、いくつかの名前とインポート:
module Main where
import Control.Applicative
import Foreign
import System.Win32
wIN32_OWN_PROCESS :: DWORD
wIN32_OWN_PROCESS = 0x00000010
sTART_PENDING, rUNNING :: DWORD
sTART_PENDING = 0x00000002
rUNNING = 0x00000004
aCCEPT_STOP, aCCEPT_NONE :: DWORD
aCCEPT_STOP = 0x00000001
aCCEPT_NONE = 0x00000000
nO_ERROR :: DWORD
nO_ERROR = 0x00000000
type HANDLER_FUNCTION = DWORD -> IO ()
type MAIN_FUNCTION = DWORD -> Ptr LPTSTR -> IO ()
データマーシャリング用のStorableインスタンスを使用して、いくつかの特別なデータ型を定義する必要があります。
data TABLE_ENTRY = TABLE_ENTRY LPTSTR (FunPtr MAIN_FUNCTION)
instance Storable TABLE_ENTRY where
sizeOf _ = 8
alignment _ = 4
peek ptr = TABLE_ENTRY <$> peek (castPtr ptr) <*> peek (castPtr ptr `plusPtr` 4)
poke ptr (TABLE_ENTRY name proc) = do
poke (castPtr ptr) name
poke (castPtr ptr `plusPtr` 4) proc
data STATUS = STATUS DWORD DWORD DWORD DWORD DWORD DWORD DWORD
instance Storable STATUS where
sizeOf _ = 28
alignment _ = 4
peek ptr = STATUS
<$> peek (castPtr ptr)
<*> peek (castPtr ptr `plusPtr` 4)
<*> peek (castPtr ptr `plusPtr` 8)
<*> peek (castPtr ptr `plusPtr` 12)
<*> peek (castPtr ptr `plusPtr` 16)
<*> peek (castPtr ptr `plusPtr` 20)
<*> peek (castPtr ptr `plusPtr` 24)
poke ptr (STATUS a b c d e f g) = do
poke (castPtr ptr) a
poke (castPtr ptr `plusPtr` 4) b
poke (castPtr ptr `plusPtr` 8) c
poke (castPtr ptr `plusPtr` 12) d
poke (castPtr ptr `plusPtr` 16) e
poke (castPtr ptr `plusPtr` 20) f
poke (castPtr ptr `plusPtr` 24) g
外国からの輸入は3回だけです。Win32に提供する2つのコールバックには「ラッパー」インポートがあります。
foreign import stdcall "wrapper"
smfToFunPtr :: MAIN_FUNCTION -> IO (FunPtr MAIN_FUNCTION)
foreign import stdcall "wrapper"
handlerToFunPtr :: HANDLER_FUNCTION -> IO (FunPtr HANDLER_FUNCTION)
foreign import stdcall "windows.h RegisterServiceCtrlHandlerW"
c_RegisterServiceCtrlHandler
:: LPCTSTR -> FunPtr HANDLER_FUNCTION -> IO HANDLE
foreign import stdcall "windows.h SetServiceStatus"
c_SetServiceStatus :: HANDLE -> Ptr STATUS -> IO BOOL
foreign import stdcall "windows.h StartServiceCtrlDispatcherW"
c_StartServiceCtrlDispatcher :: Ptr TABLE_ENTRY -> IO BOOL
メインプログラム
最後に、主なサービスアプリケーションは次のとおりです。
main :: IO ()
main =
withTString "Test" $ \name ->
smfToFunPtr svcMain >>= \fpMain ->
withArray [TABLE_ENTRY name fpMain, TABLE_ENTRY nullPtr nullFunPtr] $ \ste ->
c_StartServiceCtrlDispatcher ste >> return ()
svcMain :: MAIN_FUNCTION
svcMain argc argv = do
appendFile "c:\\log.txt" "svcMain: svcMain here!\n"
args <- peekArray (fromIntegral argc) argv
fpHandler <- handlerToFunPtr svcHandler
h <- c_RegisterServiceCtrlHandler (head args) fpHandler
_ <- setServiceStatus h running
appendFile "c:\\log.txt" "svcMain: exiting\n"
svcHandler :: DWORD -> IO ()
svcHandler _ = appendFile "c:\\log.txt" "svcCtrlHandler: received.\n"
setServiceStatus :: HANDLE -> STATUS -> IO BOOL
setServiceStatus h status = with status $ c_SetServiceStatus h
running :: STATUS
running = STATUS wIN32_OWN_PROCESS rUNNING aCCEPT_STOP nO_ERROR 0 0 3000
出力
以前、を使用してサービスをインストールしましたsc create Test binPath= c:\Main.exe
。
プログラムのコンパイルからの出力は次のとおりです。
C:\path>ghc -threaded --make Main.hs
[1 of 1] Compiling Main ( Main.hs, Main.o )
Linking Main.exe ...
C:\path>
次に、ServiceControlMonitorからサービスを開始します。SetServiceStatusへの呼び出しが受け入れられたことの証拠は次のとおりです。
C:\Path>sc query Test
SERVICE_NAME: Test
TYPE : 10 WIN32_OWN_PROCESS
STATE : 4 RUNNING
(STOPPABLE, NOT_PAUSABLE, IGNORES_SHUTDOWN)
WIN32_EXIT_CODE : 0 (0x0)
SERVICE_EXIT_CODE : 0 (0x0)
CHECKPOINT : 0x0
WAIT_HINT : 0x0
C:\Path>
の内容は次のとおりです。これは、私の最初のコールバックが呼び出されたlog.txt
ことを証明しています。svcMain
svcMain: svcMain here!
svcMain: exiting
サービスコントロールマネージャーを使用して停止コマンドを送信するとすぐに、エラーメッセージが表示されます。私のハンドラー関数はログファイルに行を追加するはずでしたが、これは起こりません。その後、サービスは停止状態で表示されます。
C:\Path>sc query Test
SERVICE_NAME: Test
TYPE : 10 WIN32_OWN_PROCESS
STATE : 1 STOPPED
WIN32_EXIT_CODE : 0 (0x0)
SERVICE_EXIT_CODE : 0 (0x0)
CHECKPOINT : 0x0
WAIT_HINT : 0x0
C:\Path>
質問
ハンドラー関数を呼び出させるために何を試みることができるかについて誰かがアイデアを持っていますか?
20130306を更新
この問題はWindows764ビットで発生しますが、WindowsXPでは発生しません。他のバージョンのWindowsはまだテストされていません。コンパイルされた実行可能ファイルを複数のマシンにコピーして同じ手順を実行すると、異なる結果が得られます。