Services/Daemons

Any of the main programs can be run from the command line or, alternatively, installed as Services on machines running Windows or daemons on machines running Linux. Services can be installed for all users of a particular machine and, once installed, they will run automatically from when the machine boots until it is stopped and/or removed.

OpenMSIStream uses the Non-Sucking Service Manager (“NSSM”) to manage Windows Services. OpenMSIStream will try to download its own copy of the NSSM executable, but that functionality is rather unreliable, so users installing Windows Services are encouraged to download their own version of it directly from the website here, open the archive, and put the nssm.exe program in the directory in which they’re running InstallService (see below).

On Linux systems, OpenMSIStream installs and interacts with daemons using “systemd”. Most (but not all) Linux machines have systemd installed; OpenMSIStream will check for it and give you a command to install it (if you have such authority) if it’s not found.

Programs running as Services or daemons cannot be interacted with from the command line after they’ve been installed, but they will still produce output to log files. The OpenMSIStream interface to install and interact with persistently-running instances of programs is the same regardless of running of Windows or Linux.

Preparing the environment (note for Windows)

Issues with loading .dll files manifest differently when running OpenMSIStream code as Windows Services because Services run in %WinDir%/System32 and don’t read the same PATH locations as running interactively. Some workarounds are built into OpenMSIStream to mitigate these problems, but if you run into trouble with missing .dll files they can typically be resolved by copying those files directly into the %WinDir%/System32 directory.

Setup and installation

To install a Service or Daemon, type the following command in the openmsi environment in your Terminal or Anaconda Prompt (you will need to be in admin mode on Windows or have sudo privileges on Linux):

InstallService [program_name] [command_line_options] --service_name [name_for_service_or_daemon]

where:

  • [program_name] is the name of a main program,

  • [command_line_options] is a placeholder for that main program’s command line options, and

  • [name_for_service_or_daemon] is a unique name you’d like to use for the instance of the program (you’ll need to remember this to keep working with the Service or daemon).

Typing InstallService -h or InstallService [program_name] -h will give a helpful output detailing the required and optional command line arguments. These command line arguments will only need to be specified once, when the Service is installed.

Any optional command line arguments applicable to running a program are also applicable to installing the program as a Service or daemon. PLEASE NOTE that any arguments whose values are paths to directories or files must be given as absolute paths and not relative paths, because Services and daemons run from special locations independent of where you install them. This same principle holds for any parameter values in config files that are paths to files or directories.

While the script runs, you may be prompted to input values for any environment variables referenced in config files. If you input any new environment variable values you will need to close and reopen the Terminal or Admin mode Anaconda Prompt you were working in and rerun the InstallService command (the script will exit itself, and you’ll be prompted to do this).

If the script completes successfully on Windows, you should be able to see the Service listed in the Windows Service Manager window that pops up when you type mmc Services.msc. The Service or daemon will not start running until it is “start”-ed using the command in the next section. After being started, a Linux daemon will be listed if you type top in the terminal.

Starting and interacting with Services and daemons

After installing a Service/daemon, you can use the ManageService [name_for_service_or_daemon] [run_mode] command to perform several actions on the program, where [run_mode] is a text command telling the ServiceManager what to do:

  1. start the program running with [run_mode] given as “start”. You must do this after installing the Service or daemon to get it running.

  2. check the program status with [run_mode] given as “status” to make sure it’s running properly

  3. stop the program running with [run_mode] given as “stop”. If you temporarily stop the Service using this command, you can restart it afterward with ManageService [name_for_service_or_daemon] start.

  4. uninstall the Service or daemon completely with [run_mode] given as “remove”. You can also add the following flags when using this command:
    • --remove_env_vars to un-set any environment variable values

    • --remove_nssm to remove the NSSM executable that may have been downloaded to manage the Service on Windows

    • --remove_install_args to remove the installation arguments file (which otherwise allows reinstalling the Service or daemon with the same arguments as before)

  5. reinstall the Service or daemon using arguments from the most recent install run with [run_mode] given as “reinstall”. This is a short-hand way to call the last-used version of InstallService for a service of this same name.

  6. do more than one of the above at once with compound run modes. There are several compound run modes to do more than one action at once. They are :
    • “stop_and_remove” (calls “stop” and then “remove”),

    • “stop_and_reinstall” (calls “stop”, “remove”, and then “reinstall”), and

    • “stop_and_restart” (calls “stop”, “remove”, “reinstall”, and then “start”).

Debugging problems

If something goes wrong while the program is running as a Service/daemon, a file called [service_name]_ERROR_LOG.txt should be created in the directory that you ran the installation command from. That file should contain a traceback for the error that killed the Service. If that file doesn’t exist, then there was likely an issue with installing the Service in the first place, and not with the OpenMSIStream Python code it was trying to run.

Output in the “working directory”

Working with Services will create a few files in a “working directory”. The default working directory is the working_dir subdirectory of the OpenMSIStream repo, but you can change its location by setting the OPENMSISTREAM_SERVICES_WORKING_DIR environment variable on your system. A logfile called “Services.log” will contain some lines related to installing or working with any Services or daemons. Python files in that directory will correspond to the executables that are installed as Services, and so checking these files will give details on the setup for their corresponding Services/daemons. The directory may also contain text files listing environment variables values or installation arguments for services that you install. None of these created files will be tracked in the repo if they’re in the default location (they’re in the .gitignore).

Running other Python code as Services/Daemons

In addition to the main programs provided by OpenMSIStream, the infrastructure used to work with Services/daemons can be applied to arbitrary Python code as well. To install your own code as a Service or daemon, you can run the same commands as above, except the InstallService command’s [program_name] should be a specification string instead of the name of a main program. That specification string can have two different formats, depending on the code you want to install.

Installing any class that extends Runnable

First, if you’ve written your own new class that extends Runnable, you can give the name of the class and the path to it as the specification string, like:

InstallService [class_name]=[path.to.class.file] [command_line_options] --service_name [name_for_service_or_daemon]

where [class_name] is the name of the class, and [path.to.class.file] is a Python path to the file containing it. You can find an example custom Runnable class in the OpenMSIStream repository, called runnable_example.py in the openmsistream/services/examples directory. The openmsistream.utilities.Runnable.run_from_command_line() function in that class can be run as a Service by installing it with the command:

InstallService RunnableExample=openmsistream.services.examples.runnable_example [absolute_path_to_output_dir] --service_name RunnableExampleServiceTest

and then starting/stopping/removing it with:

ManageService RunnableExampleServiceTest start
ManageService RunnableExampleServiceTest stop_and_remove

Running this successfully will create a file called runnable_example_service_test.txt in the directory at [absolute_path_to_output_dir]; timestamped lines will be written to the file when the Python code is run. On Windows, the Service will be restarted each time it completes and so there will be several lines in the file and more will be added until the Service is stopped.

Installing a general Python function

If instead of a custom Runnable class you’ve written just a Python script containing a function you’d like to run, the specification string format is slightly different; the InstallService command would look like:

InstallService [path.to.script.file]:[func_name] [command_line_options] --service_name [name_for_service_or_daemon]

where [path.to.script.file] is a Python path to the file containing the function to run, and [func_name] is the name of the function in the file. The function you write should accept one argument: the command_line_options as a list.

You can find an example custom Service/daemon script in the OpenMSIStream repository, called script_example.py in the openmsistream/services/examples directory. In this case the main function in the script can be run as a Service by installing it with the command:

InstallService openmsistream.services.examples.script_example:main [absolute_path_to_output_dir] --service_name ScriptExampleServiceTest

and then starting/stopping/removing it with:

ManageService ScriptExampleServiceTest start
ManageService ScriptExampleServiceTest stop_and_remove

Running this successfully will create a file called script_example_service_test.txt in the directory at [absolute_path_to_output_dir]; timestamped lines will be written to the file when the Python code is run. On Windows, the Service will be restarted each time it completes and so there will be several lines in the file and more will be added until the Service is stopped.