Differences between revisions 6 and 7
Revision 6 as of 2015-02-03 17:01:55
Size: 4676
Editor: KaiJaeger
Comment:
Revision 7 as of 2017-01-24 09:30:52
Size: 3224
Editor: KaiJaeger
Comment:
Deletions are marked like this. Additions are marked like this.
Line 9: Line 9:
{{{ServiceState}}} is a class that offers an interface between the Windows SCM (Service Control Manager) and a Dyalog APL application running as a service. {{{ServiceState}}} is a namespace script that offers an interface between the Windows SCM (Service Control Manager) and a Dyalog APL application running as a service.
Line 11: Line 11:
In you application workspace you need to have a function on `⎕LX` like this: In you application workspace you need to have a function on `⎕LX` similar to this:
Line 14: Line 14:
      Start
[1] #.WinFile.PolishCurrentDir
[2] rideFlag0
[3] timeout←3
[4]    #.myServiceState←⎕NEW #.ServiceState rideFlag timeout
[5] '#'⎕WS'Event' #.myServiceState.Event_QuitDQ 1
[6]
:If myServiceState.IsRunningAsService
[7] {}myServiceState.EstablishServiceCallback ⍬ ⍝ This does not work!
[8
] {#.TestService.Run ⍵}&⍬
[9] ⎕DQ'.'
[10] #.TestService.Off
[11] :Else
[12] #.TestService.Run ⍬
[13] :EndIf
      Start;ps
[1] ps←#.ServiceState.CreateParmSpace
[2] ps.ride←1
[3] #.ServiceState.Init ps
[4] :If #.ServiceState.IsRunningAsService
[5] {#.TestService.Run ⍵}&⍬
[6] ⎕DQ'.'
[7] #.TestService.Off
[8] :Else
[9] #.TestService.Run ⍬
[10] :EndIf
Line 30: Line 27:
Note that this example focuses just on getting the application running as a service. Note that this example focuses on getting the application running as a service. For the purpose of demonstrating how to overwrite defaults (no RIDE) it shows how to create a parameter space with default settings (line [1]) and then overwrite a setting (line [2]).

If you are happy with the defaults (which you can check by executing `#.ServiceState.CreateParmSpace.∆List`) then you can specify `⍬` as the right argument of `#.ServiceState.Init` in line [3].
Line 34: Line 33:
'''[1]''': This makes sure that if the workspace carries a path the work (or current) directory is set accordingly. '''[4]''': Checks whether the application is actually running as a Windows Service. This allows to execute the application in its own thread in case its running as a service but not otherwise - that makes debugging the application significantly easier.
Line 36: Line 35:
'''[2]''': Flags that indicates whether the service is expected to connect to Ride or not. '''[5]''': Here we start the real application. Note that this '''must''' run in its own thread because otherwise the application might well become unresponsive.
Line 38: Line 37:
'''[3]''': Timeout: this defines how long the `ServiceState` class will wait for the application to conform a requested state change. '''[6]''': We need to execute `⎕DQ '.'` in order to make sure that the interpreter processes all events.
Line 40: Line 39:
'''[4]''': This instantiates the `ServiceState` class.

'''[5]''': `#.myServiceState.Event_QuitDQ` is a variable carrying an (any) integer. The application is supposed to send the same integer to `#` (when it wants to stop) in order to quit the `⎕DQ` on line 9.

'''[6]''': Checks whether the application is actually running as a Windows Service.

'''[7]''': This establishes `#.ServiceState.EstablishServiceCallback` as a callback that will handle all messages send by the Windows SCM. Well, it should. At the time of writing an instance methods that is associated with an event cause a NONCE ERROR, so we must circumvent thus Dyalog bug for the time being with something like this:

{{{
     '.'⎕WS'Event' 'ServiceNotification' '#.OnServiceHandler'
}}}

The `#.OnServiceHandler` function in turn calls the instance method:

{{{
 OnServiceHandler←{
   #.myServiceState.OnServiceHandler ⍵
 }
}}}

That works.

'''[8]''': Here we start the real application. Note that this '''must''' run in its own thread because otherwise the application might well become unresponsive.

'''[9]''': We need to execute `⎕DQ '.'` in order to make sure that the interpreter processes all events.

'''[10]''': At this stage we need to execute `⎕OFF`; that's what `#.TestService.Off` is doing.
'''[7]''': At this stage we need to execute `⎕OFF`; that's what `#.TestService.Off` is doing.
Line 70: Line 43:
The application must make sure that it checks whether a state change was requested. This can be done by calling certain helpers as shown in this example: The application must make sure in its main loop that it checks for state changes, and act accordingly. This is all done by the Operator `#.ServiceState.CheckServiceMessages` which takes the name of a logging function as operand:
Line 73: Line 46:
      MainLoop dummmy;buffer;data;event;obj;rc;r;msg;cs
[1] Log 'Server got started'
[2] :While 1
[3] ⎕DL 1
[4] :If #.myServiceState.ShallServicePause
[5] {}#.myServiceState.ConfirmStateChange ⍬
[6] Log'The service is pausing'
[7] {}#.myServiceState.WaitForContinue ⍬
[9] {}#.myServiceState.ConfirmStateChange ⍬
[10] Log'The service is no longer pausing'
[11] :EndIf
[12] :If #.myServiceState.ShallServiceQuit
[13] {}#.myServiceState.ConfirmStateChange ⍬
[14] Log'The service is in the process of shutting down...'
[15] :Leave
[16] :EndIf
[17] :EndWhile
[18] ⎕NQ'#'#.ServiceState.Events.QuitDQ
  MainLoop dummmy;buffer;data;event;obj;rc;r;msg;cs
[1] :Repeat
[2] ⎕DL 1 ⍝ What is reasonable here depends on the application
[3] :If (Log #.ServiceState.CheckServiceMessages)#.ServiceState.IsRunningAsService
[4] Log'The service is in the process of shutting down...'
[5] :Leave
[6] :EndIf
[7] :Until 0
Line 95: Line 58:
'''[14]''': This makes sure that the `⎕DQ` in `Start[7]` quits.

Apart from `ShallServicePause` and `ShallServiceQuit` there is also a function `ShallServiceContinue` but this is called by this function:

{{{
      {r}←WaitForContinue dummy
[1] ⍝ Returns always ⍬.
[2] ⍝ Consider calling this in case ShallServicePause has returned a 1 and you don't _
[3] ⍝ have to do any particular gymnastics but just want to wait for a state change.
[4] ⍝ It just waits until the status changes, although not necessarily to "continue".
[5] r←⍬
[6] :While 0=ShallServiceContinue
[7] ⎕DL 1
[8] :EndWhile
}}}
Line '''[3]''' checks for any changes signalled from the Service Control Manager (SCM). If a "Pause" is detected, then the function loops until either a "Continue" or a "Stop" is detected. The explicit result is 1 in case a "Stop" is detected and 0 otherwise.

ServiceState: interface between SCM and APL

(Hide table-of-contents)

ServiceState is part of the CategoryAplTree project.

Overview

ServiceState is a namespace script that offers an interface between the Windows SCM (Service Control Manager) and a Dyalog APL application running as a service.

In you application workspace you need to have a function on ⎕LX similar to this:

      Start;ps
[1]      ps←#.ServiceState.CreateParmSpace
[2]      ps.ride←1
[3]      #.ServiceState.Init ps
[4]   :If #.ServiceState.IsRunningAsService
[5]      {#.TestService.Run ⍵}&⍬
[6]      ⎕DQ'.'
[7]     #.TestService.Off
[8]  :Else
[9]     #.TestService.Run ⍬
[10] :EndIf

Note that this example focuses on getting the application running as a service. For the purpose of demonstrating how to overwrite defaults (no RIDE) it shows how to create a parameter space with default settings (line [1]) and then overwrite a setting (line [2]).

If you are happy with the defaults (which you can check by executing #.ServiceState.CreateParmSpace.∆List) then you can specify as the right argument of #.ServiceState.Init in line [3].

Remarks:

[4]: Checks whether the application is actually running as a Windows Service. This allows to execute the application in its own thread in case its running as a service but not otherwise - that makes debugging the application significantly easier.

[5]: Here we start the real application. Note that this must run in its own thread because otherwise the application might well become unresponsive.

[6]: We need to execute ⎕DQ '.' in order to make sure that the interpreter processes all events.

[7]: At this stage we need to execute ⎕OFF; that's what #.TestService.Off is doing.

The application

The application must make sure in its main loop that it checks for state changes, and act accordingly. This is all done by the Operator #.ServiceState.CheckServiceMessages which takes the name of a logging function as operand:

  MainLoop dummmy;buffer;data;event;obj;rc;r;msg;cs
[1]   :Repeat
[2]       ⎕DL 1                     ⍝ What is reasonable here depends on the application
[3]       :If (Log #.ServiceState.CheckServiceMessages)#.ServiceState.IsRunningAsService
[4]           Log'The service is in the process of shutting down...'
[5]           :Leave
[6]       :EndIf
[7]   :Until 0

Remarks:

Line [3] checks for any changes signalled from the Service Control Manager (SCM). If a "Pause" is detected, then the function loops until either a "Continue" or a "Stop" is detected. The explicit result is 1 in case a "Stop" is detected and 0 otherwise.

Project Page

For bug reports, future enhancements and a full version history see ServiceState/ProjectPage

Version Information

Original author:

KaiJaeger

Responsible:

KaiJaeger

Email:

kai@aplteam.com


CategoryAplTree

ServiceState (last edited 2018-03-03 11:39:17 by KaiJaeger)