Differences between revisions 1 and 2
Revision 1 as of 2014-11-03 10:54:31
Size: 4348
Editor: KaiJaeger
Comment:
Revision 2 as of 2014-11-10 12:16:22
Size: 4660
Editor: KaiJaeger
Comment:
Deletions are marked like this. Additions are marked like this.
Line 9: Line 9:
{{{ServiceState}}} is a namespace that offers an interface between the Windows SCM (Service Control Manager) and a Dyalog APL application running as a service. {{{ServiceState}}} is a class that offers an interface between the Windows SCM (Service Control Manager) and a Dyalog APL application running as a service.
Line 16: Line 16:
[2] {}#.ServiceState.Init 1 3
[
3] '#'⎕WS'Event'#.ServiceState.Events.QuitDQ
[4
] :If #.ServiceState.IsRunningAsService
[5] {}#.ServiceState.EstablishServiceCallback ⍬
[6
] {#.TestService.Run ⍵}&⍬
[7] ⎕DQ'.'
[8] #.TestService.Off
[9] :Else
[10] #.TestService.Run ⍬
[11] :EndIf
[2] rideFlag←0
[
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
Line 34: Line 36:
'''[2]''': This initializes the `ServiceState` interface. There are two arguments: '''[2]''': Flags that indicates whether the service is expected to connect to Ride or not.
Line 36: Line 38:
 1. is a boolean that indicates whether the service shall connect to Ride (1) or not (0).
 2. is a positive integer specifying the number of seconds until `ServiceState` gives up waiting for the application to confirm a state change requested by the SCM (timeout).
'''[3]''': Timeout: this defines how long the `ServiceState` class will wait for the application to conform a requested state change.
Line 39: Line 40:
Note that the two values are considered as constants; they are therefore established as niladic functions. For example, after having executed line [2] of the aforementioned `Start` function you can say: '''[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:
Line 42: Line 49:
      #.ServiceState.States.ride
1
      #.ServiceState.States.timeout
3
     '.'⎕WS'Event' 'ServiceNotification' '#.OnServiceHandler'
Line 48: Line 52:
'''[3]''': `#.ServiceState.Events.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 7. The `#.OnServiceHandler` function in turn calls the instance method:
Line 50: Line 54:
'''[4]''': Checks whether the application is actually running as a Windows Service. {{{
 OnServiceHandler←{
   #.myServiceState.OnServiceHandler ⍵
 }
}}}
Line 52: Line 60:
'''[5]''': This establishes `#.ServiceState.EstablishServiceCallback` as a callback that will handle all messages send by the Windows SCM. That works.
Line 54: Line 62:
'''[6]''': Here we start the real application. Note that this '''must''' run in its own thread because otherwise the application might well become unresponsive. '''[8]''': Here we start the real application. Note that this '''must''' run in its own thread because otherwise the application might well become unresponsive.
Line 56: Line 64:
'''[7]''': We need to execute `⎕DQ '.'` in order to make sure that the interpreter processes all events. '''[9]''': We need to execute `⎕DQ '.'` in order to make sure that the interpreter processes all events.
Line 58: Line 66:
'''[8]''': At this stage we need to execute `⎕OFF`; that's what `#.TestService.Off` is doing. '''[10]''': At this stage we need to execute `⎕OFF`; that's what `#.TestService.Off` is doing.
Line 66: Line 74:
[1] Log INI.CONFIG.appshortname,' server started'
[2] :While 1
[3] ⎕DL 1
[4]    :If #.ServiceState.ShallServicePause
[5]          Log'The service is pausing'
[6]
{}#.ServiceState.WaitForContinue ⍬
[7] Log'The service is resuming'
[8]
:EndIf
[9] :If #.ServiceState.ShallServiceQuit
[10] Log'The service is in the process of shutting down...'
[11] :Leave
[12] :EndIf
[13] :EndWhile
[14] ⎕NQ'#'#.ServiceState.Events.QuitDQ
[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] {}#.my
ServiceState.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

ServiceState

(Hide table-of-contents)

ServiceState is part of the CategoryAplTree project.

Overview

ServiceState is a class 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 like this:

      Start
[1]   #.WinFile.PolishCurrentDir
[2]   rideFlag←0
[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

Note that this example focuses just on getting the application running as a service.

Remarks:

[1]: This makes sure that if the workspace carries a path the work (or current) directory is set accordingly.

[2]: Flags that indicates whether the service is expected to connect to Ride or not.

[3]: Timeout: this defines how long the ServiceState class will wait for the application to conform a requested state change.

[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.

The application

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:

      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

Remarks:

[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

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 CategoryAPLTreeProject

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