Biniou is a local, event-driven job scheduler and automation framework. You can run commands that are triggered according to various conditions, and pipe them. The jobs can be started on a schedule, based on events generated by other jobs, and based on local file system changes.
The jobs can be defined based on the included templates or, for maximum flexibility, written in JavaScript. In that case you do any processing you want and dispatch the events in the format of your choice.
This utility is mostly designed as an offline tool running on your own computer, as an alternative to simpler "cron" tasks and less heavy than IFTTT or Huginn when you just want simple scripts running under various conditions locally.
- Install
biniou
globally, for example usingnpm install -g @laurent22/biniou
- You then need to let it run in the background, using
biniou start
- To create a job, create a folder in
~/.config/biniou/jobs
- the folder name will be the job ID. - Then add a
job.json
file in that folder - this will be used to describe the job
A simple job.json
file can be something like this:
{
"type": "shell",
"script": "/path/to/you/command.sh",
"trigger": "cron",
"triggerSpec": "0 * * * *"
}
This is essentially a simple cron command. In some context, it is easier than using the system crontab (especially on macOS) since the above command will run in the same environment as your user, which means paths, env variables, etc. will be defined.
{
"type": "js",
"trigger": "fileSystem",
"triggerSpec": {
"paths": ["/path/to/watch"],
"depth": 0,
"fileExtensions": ["txt"]
}
}
The above job will watch /path/to/watch
for any change. It will not do so recursively since depth
is set to 0
, and only the text file (.txt) will be watched.
Then the index.js
file can be used to do something when a file is changed:
exports = {
run: async (context) => {
const params = context.params;
const event = params.event;
const path = params.path;
if (event === 'add') {
console.info('A file has been added: ' + path);
// In this example, each file that is changed is copied to a backup folder
await biniou.execCommand(['cp', path, '/backup/']);
}
},
};
See the Chokidar documentation for the list of supported events. The most useful ones are probably add
, change
and unlink
.
A job can be setup to dispatch events. Events can be used by other jobs for further processing, thus creating a pipe.
To dispatch events, you need to create your script in JavaScript. For example, the following job will parse an RSS feed, and create an event for each RSS item:
job.json:
{
"type": "js",
"trigger": "cron",
"triggerSpec": "0,30 * * * *"
}
index.js:
exports = {
run: async () => {
const feed = await biniou.rssParser({
customFields: {
item: ['twitter-text'],
}
}).parseURL('https://joplinapp.org/rss');
const events = feed.items.map(item => {
return {
id: item.guid,
title: item.title,
content: item.content,
link: item.link,
twitterText: item['twitter-text'],
date: item.isoDate,
};
});
await biniou.dispatchEvents('joplin_post', events, { allowDuplicates: false });
},
};
Note in particular the biniou.*
function - several of these are available for various tasks to make it easier to define jobs.
Once a job has dispatched events, you will probably want to process these events in some way. For this you can create an event processing job.
Likewise you would create a job.json
file but this type the trigger
will be event
. You also set the triggerSpec
to the event types that your job can process. There can be any number of event types in that array.
job.json:
{
"type": "js",
"trigger": "event",
"triggerSpec": ["joplin_post"]
}
You then create an index.js
that's going to process the event. This file should contain a run
function which you use to receive and process the event. In the example below an email is going to be sent for each event:
index.js:
exports = {
run: async function(context) {
const mailerConfig = {
host: "smtp.example.com",
port: 465,
secure: true,
auth: {
user: '[email protected]',
pass: 'Moscow4',
},
tls: {
rejectUnauthorized: false
}
};
const content = JSON.parse(context.event.body);
await biniou.mailer(mailerConfig).sendMail({
from: '"Biniou" <[email protected]>',
to: '[email protected]',
subject: '[Biniou] ' + content.title,
text: context.event.body,
});
},
};
The job.json
file defines the basic job metadata:
Property | Type | Possible Values | Description |
---|---|---|---|
id |
string | Unique identifier for the job. This is the name of the folder that contains job.json | |
type |
JobType |
'js' , 'shell' , 'fileSystem' |
Type of the job. |
trigger |
JobTrigger |
'cron' , 'event' |
Defines when the job should be triggered. Can be based on a Cron schedule or an event. |
triggerSpec |
JobTriggerSpec |
string, string[] or FileSystemTriggerSpec | Specification of the trigger. It can be a single string or an array of strings (e.g., Cron expressions or event identifiers). If the job type is fileSystem , it should be a FileSystemTriggerSpec (see below). |
script |
string | The script to be executed. Defaults to index.js for js jobs. Otherwise you need to provide the path to the shell command. |
|
enabled |
boolean | true or false |
Indicates whether the job is enabled and should run. Defaults to true |
template |
string | Optional template name, if the job is based on a template. | |
depth |
number | If the job type is fileSystem this can be used to specify how deep the folders should be watched. undefined means all folders recursively. 0 means only the current folder, 1 is one level deep, etc. |
|
params |
any | Additional parameters that can be used for the job execution. |
Property | Type | Possible Values | Description |
---|---|---|---|
paths |
string[] | List of paths that must be watched - can be either files or directories | |
depth |
number | Optional. This can be used to specify how deep the folders should be watched. undefined means all folders recursively. 0 means only the current folder, 1 is one level deep, etc. |
|
fileExtensions |
string[] | Optional. File extensions that must be watched. Leave it empty to watch all files. |
Within a JavaScript job, you can access the biniou
global object that provide various utilities function. The full list of functions is described in JobSandbox.ts
Run yarn watch
to automatically build the application.
To run just one command, for testing, use:
yarn start run NAME_OF_JOB
NAME_OF_JOB
will be the name of the folder in ~/.config/biniou/jobs
MIT