npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

progenic

v3.2.2

Published

Multi-workers daemon module

Downloads

14

Readme

Progenic

Multi-workers daemon module with advanced options.

NPM Downloads NPM Version GitHub Tag GitHub License Dependencies Status

(For a simpler and lightweight cousin of Progenic see Probiotic)

Table of Contents

Getting Started

Install the module with: npm install progenic
Then use it in your code to start a service with as many workers as needed:

const progenic = require('progenic');

progenic.run({
    name: 'myServiceName',
    main: 'path/to/myServiceScript.js',
    workers: 4,
    devMode: false,
    logsBasePath: '/mnt/logs-volume',
    checkPingsEnabled: true,
	disableRichLogger: false
});

Documentation

Progenic requires NodeJS 4.5.0+ and can be istalled via NPM:

npm install progenic
# or:
npm install --save progenic
// Import progenic module
const progenic = require('progenic');

// Define an options object
const options = {
    name: '...',         	 // MANDATORY
    main: '...',         	 // MANDATORY
    workers: 'auto',  		 // OPTIONAL, defaults to 'auto'
    devMode: false,   		 // OPTIONAL, defaults to false
    logsBasePath: '', 		 // OPTIONAL, defaults to '/var/log' if devMode === false, otherwise defaults to './'
    checkPingsEnabled: true, // OPTIONAL, defaults to true
	disableRichLogger: false // OPTIONAL, defaults to false
};

// Use the method 'run()' to start the service with the given options
progenic.run(options);

The progenic module starts a given service as a daemon, by spawning a configurable number of children that will act as workers. The father process is the one balancing the work among them.
Progenic leverages the cluster module under the hood.

Main options documentation

The mandatory name parameter is the name the service will be started with. It affects the process' PID file name (under /var/run) as well as the log files names.

The mandatory main parameter points to the JS file that is actually the service code.
This code will be spawned exactly workers-times in different processes children of the service containing the progenic code.

The optional workers parameter tells progenic how many workers ave to be spawn. When omitted progenic will assume the value 'auto', spawning exactly require('os').cpus().length - 1 workers (i.e. the number of CPUs of the system minus 1).

The optional devMode parameter tells progenic whether to start in development mode or in production one.
When devMode = true the following happens:

  • The main process doesn't daemonize
  • The PID file is written to the same folder the server is started from
  • Log files are written to the process' working directory as well
  • process.env.NODE_ENV is set to 'development' (instead of 'production')

Log files

By default your service will have log files created under /var/log, simply by running progenic like this:

progenic.run({
    name: 'myServiceName',
    main: 'path/to/myServiceScript.js',
    workers: 1
});
// => $ ls -lah /var/log/myServiceName*
//      -rw-r--r--   1 root    root             262 30 mar 02.26 myServiceName_1.log
//      -rw-r--r--   1 root    root             381 30 mar 02.26 myServiceName_master.log

We got two files created: one for the worker and one for the master process. First one will get the actual logs of your service while the latter logs startup info and all the actions that progenic takes in order to ensure that your service stays up, as well as the results of a periodic check on the workers to verify whether they are alive and running.

Each child worker gets its own log file.

The path where log files are created can be customized by passing in the logsBasePath option, like this:

progenic.run({
    name: 'myServiceName',
    main: 'path/to/myServiceScript.js',
    workers: 2,
    logsBasePath: '/mnt/logs-volume'
});
// => $ ls -lah /mnt/logs-volume/myServiceName*
//      -rw-r--r--   1 root    root             122 30 mar 02.41 myServiceName_1.log
//      -rw-r--r--   1 root    root             122 30 mar 02.41 myServiceName_2.log
//      -rw-r--r--   1 root    root             240 30 mar 02.41 myServiceName_master.log

Please note that writing logs to certain paths (/var/log is one of them) requires root provileges.
This behavior may vary depending on the linux distro being used.

PID file

Finally progenic will write one PID file for your process, under /var/run:

$ cat /var/run/myServiceName.pid
  4320

The PID file will not be deleted when your service exits.

Check-pings

When spawning workers progenic periodically checks whether they are still alive (and active) or not. This is done by pinging each one of them and waiting for an answer to come within 30 seconds.
If it fails to answer in time the worker gets killed and respawned.
This feature is enabled by default and can be disabled by passing the option checkPingsEnabled set to false:

progenic.run({
    name: 'myServiceName',
    main: 'path/to/myServiceScript.js',
    // [...]
    checkPingsEnabled: false
});

Rich-logging

Progenic usually adds a timestamp to each line logged by the master, like this:

$ tailf /var/log/myServiceName_master.log
[Wed Aug 17 2016 09:05:45 GMT+0000 (UTC)] [PROGENIC] Worker "1" is alive and kicking...
[Wed Aug 17 2016 09:05:46 GMT+0000 (UTC)] [PROGENIC] Worker "2" is alive and kicking...
[Wed Aug 17 2016 09:05:47 GMT+0000 (UTC)] [PROGENIC] Worker "3" is alive and kicking...
[Wed Aug 17 2016 09:05:48 GMT+0000 (UTC)] [PROGENIC] Worker "4" is alive and kicking...

There are times when this may not be the desired behavior, like when some logging libraries used by your code already provide a timestamp. Thus logging may become a bit too redundant (here using the bunyan logging library):

$ tailf /var/log/hephaestus.APTE_master.log | bunyan
[2016-08-17T09:05:45.500Z]  INFO: hephaestus/30956 on vp-pre1.xorovo.it: [Wed Aug 17 2016 09:05:45 GMT+0000 (UTC)] [PROGENIC] Worker "1" is alive and kicking...

At times like this the disableRichLogger option should be used, which provides a clean logger without additional information added to it:

progenic.run({
    name: 'myServiceName',
    main: 'path/to/myServiceScript.js',
    // [...]
    disableRichLogger: true
});

License

Copyright (c) 2016 Nicola Orritos
Licensed under the MIT license.