2021-05-06 07:08:47 +00:00
# JavaScript Scripting
This add-on provides support for JavaScript (ECMAScript 2021+) that can be used as a scripting language within automation rules.
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
Also included is [openhab-js ](https://github.com/openhab/openhab-js/ ), a fairly high-level ES6 library to support automation in openHAB. It provides convenient access
to common openHAB functionality within rules including items, things, actions, logging and more.
2021-05-06 07:08:47 +00:00
2021-12-17 07:00:44 +00:00
- [Configuration ](#configuration )
- [UI Based Rules ](#ui-based-rules )
2022-01-30 17:28:58 +00:00
- [Adding Triggers ](#adding-triggers )
- [Adding Actions ](#adding-actions )
2021-12-17 07:00:44 +00:00
- [Scripting Basics ](#scripting-basics )
2022-01-30 17:28:58 +00:00
- [Require ](#require )
- [Console ](#console )
- [SetTimeout ](#settimeout )
- [SetInterval ](#setinterval )
- [ScriptLoaded ](#scriptloaded )
- [ScriptUnLoaded ](#scriptunloaded )
- [Paths ](#paths )
2021-12-17 07:00:44 +00:00
- [Standard Library ](#standard-library )
2022-01-30 17:28:58 +00:00
- [Items ](#items )
- [Actions ](#actions )
- [Cache ](#cache )
- [Log ](#log )
- [Time ](#time )
- [Utils ](#utils )
2021-12-17 07:00:44 +00:00
- [File Based Rules ](#file-based-rules )
- [JSRule ](#jsrule )
- [Rule Builder ](#rule-builder )
2021-05-06 07:08:47 +00:00
2021-12-17 07:00:44 +00:00
## Configuration
2021-05-06 07:08:47 +00:00
2021-12-17 07:00:44 +00:00
This add-on includes by default the [openhab-js ](https://github.com/openhab/openhab-js/ ) NPM library and exports it's namespaces onto the global namespace. This allows the use of `items` , `actions` , `cache` and other objects without the need to explicitly import using `require()` . This functionality can be disabled for users who prefer to manage their own imports via the add-on configuration options.
2022-04-09 14:56:13 +00:00
![openHAB Rule Configuration ](doc/settings.png )
2021-12-17 07:00:44 +00:00
## UI Based Rules
The quickest way to add rules is through the openHAB Web UI.
Advanced users, or users migrating scripts from existing systems may want to use [File Based Rules ](#file-based-rules ) for managing rules using files in the user configuration directory.
### Adding Triggers
Using the openHAB UI, first create a new rule and set a trigger condition
2022-04-09 14:56:13 +00:00
![openHAB Rule Configuration ](doc/rule-config.png )
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
### Adding Actions
Select "Add Action" and then select "ECMAScript 262 Edition 11".
Its important this is "Edition 11" or higher, earlier versions will not work.
This will bring up a empty script editor where you can enter your javascript.
2021-12-17 07:00:44 +00:00
2022-04-09 14:56:13 +00:00
![openHAB Rule Engines ](doc/rule-engines.png )
2021-12-17 07:00:44 +00:00
You can now write rules using standard ES6 Javascript along with the included openHAB [standard library ](#standard-library ).
2022-04-09 14:56:13 +00:00
![openHAB Rule Script ](doc/rule-script.png )
2021-12-17 07:00:44 +00:00
For example, turning a light on:
```javascript
items.getItem("KitchenLight").sendCommand("ON");
console.log("Kitchen Light State", items.getItem("KitchenLight").state);
```
Sending a notification
```javascript
actions.NotificationAction.sendNotification("romeo@montague.org", "Balcony door is open");
2021-05-06 07:08:47 +00:00
```
2021-12-17 07:00:44 +00:00
Querying the status of a thing
```javascript
const thingStatusInfo = actions.Things.getThingStatusInfo("zwave:serial_zstick:512");
console.log("Thing status",thingStatusInfo.getStatus());
```
See [openhab-js ](https://openhab.github.io/openhab-js ) for a complete list of functionality
## Scripting Basics
The openHAB JSScripting runtime attempts to provide a familiar environment to Javascript developers.
### Require
2022-01-30 17:28:58 +00:00
Scripts may include standard NPM based libraries by using CommonJS require.
The library search path will look in `automation/js/node_modules` in the user configuration directory.
2021-12-17 07:00:44 +00:00
### Console
The JSScripting binding supports the standard `console` object for logging.
2022-01-30 17:28:58 +00:00
Script debug logging is enabled by default at the `INFO` level, but can be configured using the [console logging ]({{base}}/administration/logging.html ) commands.
2021-05-06 07:08:47 +00:00
```text
2021-12-17 07:00:44 +00:00
log:set DEBUG org.openhab.automation.script
2021-05-06 07:08:47 +00:00
```
2021-12-17 07:00:44 +00:00
Supported logging functions include:
- `console.log(obj1 [, obj2, ..., objN])`
- `console.info(obj1 [, obj2, ..., objN])`
- `console.warn(obj1 [, obj2, ..., objN])`
- `console.error(obj1 [, obj2, ..., objN])`
- `console.debug(obj1 [, obj2, ..., objN])`
- `console.trace(obj1 [, obj2, ..., objN])`
2021-05-06 07:08:47 +00:00
2022-01-30 17:28:58 +00:00
where `obj1 ... objN` is a list of JavaScript objects to output.
The string representations of each of these objects are appended together in the order listed and output.
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
see https://developer.mozilla.org/en-US/docs/Web/API/console for more information about console logging.
2021-05-06 07:08:47 +00:00
2022-01-30 17:28:58 +00:00
### SetTimeout
2021-05-06 07:08:47 +00:00
2021-12-17 07:00:44 +00:00
The global setTimeout() method sets a timer which executes a function or specified piece of code once the timer expires.
2021-11-29 08:44:26 +00:00
```javascript
2021-12-17 07:00:44 +00:00
var timeoutID = setTimeout(function[, delay, arg1, arg2, ...]);
var timeoutID = setTimeout(function[, delay]);
2021-05-06 07:08:47 +00:00
```
2021-12-17 07:00:44 +00:00
The global `clearTimeout()` method cancels a timeout previously established by calling `setTimeout()` .
2021-05-06 07:08:47 +00:00
2021-12-17 07:00:44 +00:00
see https://developer.mozilla.org/en-US/docs/Web/API/setTimeout for more information about setTimeout.
2022-01-30 17:28:58 +00:00
### SetInterval
2021-12-17 07:00:44 +00:00
The setInterval() method repeatedly calls a function or executes a code snippet, with a fixed time delay between each call.
```javascript
var intervalID = setInterval(func, [delay, arg1, arg2, ...]);
var intervalID = setInterval(function[, delay]);
```
The global `clearInterval()` method cancels a timed, repeating action which was previously established by a call to `setInterval()` .
2022-01-30 17:28:58 +00:00
NOTE: Timers will not be canceled if a script is deleted or modified, it is up to the user to manage timers. See using the [cache ](#cache ) namespace as well as [ScriptLoaded ](#scriptloaded ) and [ScriptUnLoaded ](#scriptunloaded ) for a convenient way of managing persisted objects, such as timers between reloads or deletions of scripts.
2021-12-17 07:00:44 +00:00
see https://developer.mozilla.org/en-US/docs/Web/API/setInterval for more information about setInterval.
### ScriptLoaded
For file based scripts, this function will be called if found when the script is loaded.
```javascript
scriptLoaded = function () {
console.log("script loaded");
loadedDate = Date.now();
}
```
### ScriptUnLoaded
For file based scripts, this function will be called if found when the script is unloaded.
```javascript
scriptUnloaded = function () {
console.log("script unloaded");
//clean up rouge timers
clearInterval(timer);
}
2021-11-29 08:44:26 +00:00
```
2021-12-17 07:00:44 +00:00
### Paths
For [file based rules ](#file-based-rules ), scripts will be loaded from `automation/js` in the user configuration directory.
NPM libraries will be loaded from `automation/js/node_modules` in the user configuration directory.
## Standard Library
Full documentation for the openHAB JavaScript library can be found at [openhab-js ](https://openhab.github.io/openhab-js )
### Items
2022-01-30 17:28:58 +00:00
The items namespace allows interactions with openHAB items.
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : items ](https://openhab.github.io/openhab-js/items.html ) for full API documentation
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
* items : < code > object</ code >
* .getItem(name, nullIfMissing) ⇒ < code > Item</ code >
2022-01-30 17:28:58 +00:00
* .getItems() ⇒ < code > Array.< Item> </ code >
2021-12-17 07:00:44 +00:00
* .getItemsByTag(...tagNames) ⇒ < code > Array.< Item> </ code >
* .createItem(itemName, [itemType], [category], [groups], [label], [tags], [giBaseType], [groupFunction], [itemMetadata])
* .addItem(itemName, [itemType], [category], [groups], [label], [tags], [giBaseType], [groupFunction])
* .removeItem(itemOrItemName) ⇒ < code > Boolean</ code >
* .replaceItem(itemName, [itemType], [category], [groups], [label], [tags], [giBaseType], [groupFunction])
* .safeItemName(s) ⇒ < code > String</ code >
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
```javascript
const item = items.getItem("KitchenLight");
console.log("Kitchen Light State", item.state);
```
Calling `getItem(...)` returns an `Item` object with the following properties:
* Item : < code > object</ code >
* .type ⇒ < code > String</ code >
* .name ⇒ < code > String</ code >
* .label ⇒ < code > String</ code >
* .history ⇒ < code > ItemHistory</ code >
* .state ⇒ < code > String</ code >
* .rawState ⇒ < code > HostState</ code >
* .members ⇒ < code > Array.< Item> </ code >
* .descendents ⇒ < code > Array.< Item> </ code >
* .isUninitialized ⇒ < code > Boolean</ code >
2022-01-30 17:28:58 +00:00
* .groupNames ⇒ < code > Array.< String> </ code >
2021-12-17 07:00:44 +00:00
* .tags ⇒ < code > Array.< String> </ code >
* .getMetadataValue(namespace) ⇒ < code > String</ code >
* .updateMetadataValue(namespace, value) ⇒ < code > String</ code >
* .upsertMetadataValue(namespace, value) ⇒ < code > Boolean</ code >
* .updateMetadataValues(namespaceToValues)
* .sendCommand(value)
* .sendCommandIfDifferent(value) ⇒ < code > Boolean</ code >
* .postUpdate(value)
* .addGroups(...groupNamesOrItems)
* .removeGroups(...groupNamesOrItems)
* .addTags(...tagNames)
* .removeTags(...tagNames)
2021-11-29 08:44:26 +00:00
```javascript
2021-12-17 07:00:44 +00:00
const item = items.getItem("KitchenLight");
//send a ON command
item.sendCommand("ON");
//Post an update
item.postUpdate("OFF");
//Get state
console.log("KitchenLight state", item.state)
2021-11-29 08:44:26 +00:00
```
2021-12-17 07:00:44 +00:00
calling `item.history...` returns a ItemHistory object with the following functions:
Note `serviceId` is optional, if omitted, the default persistance service will be used.
* ItemHistory : < code > object</ code >
* .averageSince(timestamp, serviceId) ⇒ < code > Number</ code >
2022-01-30 17:28:58 +00:00
* .changedSince(timestamp, serviceId) ⇒ < code > Number</ code >
2021-12-17 07:00:44 +00:00
* .deltaSince(timestamp, serviceId) ⇒ < code > Number</ code >
* .deviationSince(timestamp, serviceId) ⇒ < code > Number</ code >
* .evolutionRate(timestamp, serviceId) ⇒ < code > Number</ code >
* .historicState(timestamp, serviceId) ⇒ < code > state</ code >
* .lastUpdate(serviceId) ⇒ < code > Date</ code >
* .latestState(serviceId) ⇒ < code > state</ code >
* .maximumSince(timestamp,serviceId) ⇒ < code > state</ code >
* .minimumSince(timestamp,serviceId) ⇒ < code > state</ code >
* .persist(serviceId)
* .previousState(skipEqual,serviceId) ⇒ < code > state</ code >
* .sumSince(timestamp, serviceId) ⇒ < code > Number</ code >
* .updatedSince(timestamp, serviceId) ⇒ < code > Boolean</ code >
* .varianceSince(timestamp,serviceId) ⇒ < code > state</ code >
2021-11-29 08:44:26 +00:00
```javascript
2021-12-17 07:00:44 +00:00
var yesterday = new Date(new Date().getTime() - (24 * 60 * 60 * 1000));
var item = items.getItem("KitchenDimmer");
console.log("KitchenDimmer averageSince", item.history.averageSince(yesterday));
2021-11-29 08:44:26 +00:00
```
2021-12-17 07:00:44 +00:00
### Actions
2022-01-30 17:28:58 +00:00
The actions namespace allows interactions with openHAB actions.
The following are a list of standard actions.
2021-12-17 07:00:44 +00:00
Additional actions provided by user installed addons can be accessed using their common name on the actions name space
(example: `actions.Pushsafer.pushsafer(...)` )
2022-01-30 17:28:58 +00:00
See [openhab-js : actions ](https://openhab.github.io/openhab-js/actions.html ) for full API documentation and additional actions.
2021-12-17 07:00:44 +00:00
#### Audio Actions
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.Audio ](https://openhab.github.io/openhab-js/actions.html#.Audio ) for complete documentation
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
#### BusEvent
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.BusEvent ](https://openhab.github.io/openhab-js/actions.html#.BusEvent ) for complete documentation
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
#### Ephemeris Actions
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.Ephemeris ](https://openhab.github.io/openhab-js/actions.html#.Ephemeris ) for complete documentation
2021-12-17 07:00:44 +00:00
Ephemeris is a way to determine what type of day today or a number of days before or after today is. For example, a way to determine if today is a weekend, a bank holiday, someone’ s birthday, trash day, etc.
2022-01-30 17:28:58 +00:00
Additional information can be found on the [Ephemeris Actions Docs ](https://www.openhab.org/docs/configuration/actions.html#ephemeris ) as well as the [Ephemeris JavaDoc ](https://www.openhab.org/javadoc/latest/org/openhab/core/model/script/actions/ephemeris ).
2021-11-29 08:44:26 +00:00
```javascript
2021-12-17 07:00:44 +00:00
// Example
let weekend = actions.Ephemeris.isWeekend();
2021-11-29 08:44:26 +00:00
```
2021-12-17 07:00:44 +00:00
#### Exec Actions
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.Exec ](https://openhab.github.io/openhab-js/actions.html#.Exec ) for complete documentation
2021-11-29 08:44:26 +00:00
Execute a command line.
```javascript
// Execute command line.
2021-12-17 07:00:44 +00:00
actions.Exec.executeCommandLine('echo', 'Hello World!');
2021-11-29 08:44:26 +00:00
// Execute command line with timeout.
2021-12-17 07:00:44 +00:00
let Duration = Java.type('java.time.Duration');
actions.Exec.executeCommandLine(Duration.ofSeconds(20), 'echo', 'Hello World!');
2021-11-29 08:44:26 +00:00
// Get response from command line.
2021-12-17 07:00:44 +00:00
let response = actions.Exec.executeCommandLine('echo', 'Hello World!');
2021-11-29 08:44:26 +00:00
// Get response from command line with timeout.
2021-12-17 07:00:44 +00:00
response = actions.Exec.executeCommandLine(Duration.ofSeconds(20), 'echo', 'Hello World!');
2021-11-29 08:44:26 +00:00
```
2022-01-30 17:28:58 +00:00
#### HTTP Actions
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.HTTP ](https://openhab.github.io/openhab-js/actions.html#.HTTP ) for complete documentation
2021-11-29 08:44:26 +00:00
```javascript
// Example GET Request
2021-12-17 07:00:44 +00:00
var response = actions.HTTP.sendHttpGetRequest('< url > ');
2021-11-29 08:44:26 +00:00
```
Replace `<url>` with the request url.
2022-01-30 17:28:58 +00:00
#### ScriptExecution Actions
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.ScriptExecution ](https://openhab.github.io/openhab-js/actions.html#.ScriptExecution ) for complete documentation
2021-12-17 07:00:44 +00:00
2021-11-29 08:44:26 +00:00
```javascript
2021-12-17 07:00:44 +00:00
let now = time.ZonedDateTime.now();
2021-11-29 08:44:26 +00:00
// Function to run when the timer goes off.
function timerOver () {
logger.info('The timer is over.');
}
// Create the Timer.
2021-12-17 07:00:44 +00:00
this.myTimer = actions.ScriptExecution.createTimer(now.plusSeconds(10), timerOver);
2021-11-29 08:44:26 +00:00
// Cancel the timer.
this.myTimer.cancel();
// Check whether the timer is active. Returns true if the timer is active and will be executed as scheduled.
let active = this.myTimer.isActive();
// Reschedule the timer.
this.myTimer.reschedule(now.plusSeconds(5));
```
2022-01-30 17:28:58 +00:00
#### Semantics Actions
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.Semantics ](https://openhab.github.io/openhab-js/actions.html#.Semantics ) for complete documentation
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
#### Things Actions
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.Things ](https://openhab.github.io/openhab-js/actions.html#.Things ) for complete documentation
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
#### Voice Actions
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
See [openhab-js : actions.Voice ](https://openhab.github.io/openhab-js/actions.html#.Voice ) for complete documentation
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
#### Cloud Notification Actions
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
(optional action if openhab-cloud is installed)
2021-11-29 08:44:26 +00:00
Notification actions may be placed in Rules to send alerts to mobile devices registered with an [openHAB Cloud instance ](https://github.com/openhab/openhab-cloud ) such as [myopenHAB.org ](https://myopenhab.org/ ).
For available actions have a look at the [Cloud Notification Actions Docs ](https://www.openhab.org/docs/configuration/actions.html#cloud-notification-actions ).
```javascript
// Example
2021-12-17 07:00:44 +00:00
actions.NotificationAction.sendNotification('< email > ', '< message > '); // to a single myopenHAB user identified by e-mail
actions.NotificationAction.sendBroadcastNotification('< message > '); // to all myopenHAB users
2021-11-29 08:44:26 +00:00
```
Replace `<email>` with the e-mail address of the user.
Replace `<message>` with the notification text.
2021-12-17 07:00:44 +00:00
### Cache
The cache namespace provides a default cache that can be use to set and retrieve objects that will be persisted between reloads of scripts.
2022-01-30 17:28:58 +00:00
See [openhab-js : cache ](https://openhab.github.io/openhab-js/cache.html ) for full API documentation
2021-12-17 07:00:44 +00:00
* cache : < code > object</ code >
* .get(key, defaultSupplier) ⇒ < code > Object | null</ code >
* .put(key, value) ⇒ < code > Previous Object | null</ code >
* .remove(key) ⇒ < code > Previous Object | null</ code >
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
The `defaultSupplier` provided function will return a default value if a specified key is not already associated with a value
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
**Example** *(Get a previously set value with a default value (times = 0))*
2021-12-17 07:00:44 +00:00
```js
let counter = cache.get("counter", () => ({ "times": 0 }));
console.log("Count",counter.times++);
```
2022-01-30 17:28:58 +00:00
**Example** *(Get a previously set object)*
2021-12-17 07:00:44 +00:00
```js
let counter = cache.get("counter");
if(counter == null){
counter = {times: 0};
cache.put("counter", counter);
}
console.log("Count",counter.times++);
```
### Log
2022-01-30 17:28:58 +00:00
By default the JS Scripting binding supports console logging like `console.log()` and `console.debug()` to the openHAB default log.
Additionally scripts may create their own native openHAB logs using the log namespace.
2021-11-29 08:44:26 +00:00
```javascript
2021-12-17 07:00:44 +00:00
let logger = log('my_logger');
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
//prints "Hello World!"
logger.debug("Hello {}!", "world");
2021-05-06 07:08:47 +00:00
```
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
### Time
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
openHAB internally makes extensive use of the `java.time` package.
openHAB-JS exports the excellent [JS-Joda ](#https://js-joda.github.io/js-joda/ ) library via the `time` namespace, which is a native Javascript port of the same API standard used in Java for `java.time` .
Anywhere that a native Java `ZonedDateTime` or `Duration` is required, the runtime will automatically convert a JS-Joda `ZonedDateTime` or `Duration` to its Java counterpart.
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
Examples:
```javascript
var now = time.ZonedDateTime.now();
var yesterday = time.ZonedDateTime.now().minusHours(24);
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
var item = items.getItem("Kitchen");
console.log("averageSince", item.history.averageSince(yesterday));
```
```javascript
actions.Exec.executeCommandLine(time.Duration.ofSeconds(20), 'echo', 'Hello World!');
```
2022-01-03 12:52:05 +00:00
See [JS-Joda ](https://js-joda.github.io/js-joda/ ) for more examples and complete API usage.
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
### Utils
openHAB internally is a Java program.
openHAB-JS converts between Java and JavaScript data types and reverse.
See [openhab-js : utils ](https://openhab.github.io/openhab-js/utils.html ) for full API documentation
2021-12-17 07:00:44 +00:00
## File Based Rules
2022-01-30 17:28:58 +00:00
The JSScripting binding will load scripts from `automation/js` in the user configuration directory.
The system will automatically reload scripts when changes are detected to files.
Local variable state is not persisted among reloads, see using the [cache ](#cache ) for a connivent way to persist objects.
2021-12-17 07:00:44 +00:00
File based rules can be created in 2 different ways: using [JSRule ](#jsrule ) or the [Rule Builder ](#rule-builder ).
See [openhab-js : rules ](https://openhab.github.io/openhab-js/rules.html ) for full API documentation
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
### JSRule
JSRules provides a simple, declarative syntax for defining rules that will be executed based on a trigger condition
2021-11-29 08:44:26 +00:00
```javascript
2021-12-17 07:00:44 +00:00
const email = "juliet@capulet.org"
rules.JSRule({
name: "Balcony Lights ON at 5pm",
description: "Light will turn on when it's 5:00pm",
triggers: [triggers.GenericCronTrigger("0 0 17 * * ?")],
execute: data => {
items.getItem("BalconyLights").sendCommand("ON");
actions.NotificationAction.sendNotification(email, "Balcony lights are ON");
2022-01-30 17:28:58 +00:00
},
tags: ["Balcony", "Lights"],
id: "BalconyLightsOn"
2021-12-17 07:00:44 +00:00
});
```
2022-01-30 17:28:58 +00:00
Note: `description` , `tags` and `id` are optional.
2021-12-17 07:00:44 +00:00
Multiple triggers can be added, some example triggers include:
```javascript
triggers.ChannelEventTrigger('astro:sun:local:rise#event', 'START')
triggers.ItemStateChangeTrigger('my_item', 'OFF', 'ON')
triggers.ItemStateUpdateTrigger('my_item', 'OFF')
triggers.ItemCommandTrigger('my_item', 'OFF')
triggers.GroupStateChangeTrigger('my_group', 'OFF', 'ON')
triggers.GroupStateUpdateTrigger('my_group', 'OFF')
triggers.GroupCommandTrigger('my_group', 'OFF')
triggers.ThingStatusUpdateTrigger('some:thing:uuid','OFFLINE')
triggers.ThingStatusChangeTrigger('some:thing:uuid','ONLINE','OFFLINE')
triggers.SystemStartlevelTrigger(40) //Rules loaded
triggers.SystemStartlevelTrigger(50) //Rule engine started
triggers.SystemStartlevelTrigger(70) //User interfaces started
triggers.SystemStartlevelTrigger(80) //Things initialized
triggers.SystemStartlevelTrigger(100) //Startup Complete
triggers.GenericCronTrigger('0 30 16 * * ? *')
triggers.TimeOfDayTrigger('19:00')
2021-11-29 08:44:26 +00:00
```
2021-12-17 07:00:44 +00:00
See [openhab-js : triggers ](https://openhab.github.io/openhab-js/triggers.html ) in the API documentation for a full list of all triggers.
### Rule Builder
The Rule Builder provides a convenient API to write rules in a high-level, readable style using a builder pattern.
2021-11-29 08:44:26 +00:00
2021-12-17 07:00:44 +00:00
Rules are started by calling `rules.when()` and can chain together [triggers ](#rule-builder-triggers ),
[conditions ](#rule-builder-conditions ) and [operations ](#rule-builder-operations ) in the following pattern:
2021-11-29 08:44:26 +00:00
```javascript
2022-01-30 17:28:58 +00:00
rules.when().triggerType()...if().conditionType().then().operationType()...build(name, description, tags, id);
2021-12-17 07:00:44 +00:00
```
2021-11-29 08:44:26 +00:00
2022-01-30 17:28:58 +00:00
Rule are completed by calling `.build(name, description, tags, id)` , all parameters are optional and reasonable defaults will be used if omitted.
- `name` String rule name - defaults generated name
- `description` String Rule description - defaults generated description
- `tags` Array of string tag names - defaults empty array
- `id` String id - defaults random UUID
2021-12-17 07:00:44 +00:00
A simple example of this would look like:
```javascript
rules.when().item("F1_Light").changed().then().send("changed").toItem("F2_Light").build("My Rule", "My First Rule");
```
Operations and conditions can also optionally take functions:
```javascript
rules.when().item("F1_light").changed().then(event => {
console.log(event);
}).build("Test Rule", "My Test Rule");
```
see [Examples ](#rule-builder-examples ) for further patterns
#### Rule Builder Triggers
* `when()`
* `or()`
* `.channel(channelName)` Specifies a channel event as a source for the rule to fire.
* `.triggered(event)` Trigger on a specific event name
* `.cron(cronExpression)` Specifies a cron schedule for the rule to fire.
* `.item(itemName)` Specifies an item as the source of changes to trigger a rule.
* `.for(duration)`
* `.from(state)`
* `.to(state)`
* `.fromOff()`
* `.toOn()`
* `.receivedCommand()`
* `.receivedUpdate()`
* `.memberOf(groupName)`
* `.for(duration)`
* `.from(state)`
* `.to(state)`
* `.fromOff()`
* `.toOn()`
* `.receivedCommand()`
* `.receivedUpdate()`
* `.system()`
* `.ruleEngineStarted()`
* `.rulesLoaded()`
* `.startupComplete()`
* `.thingsInitialized()`
* `.userInterfacesStarted()`
* `.startLevel(level)`
* `.thing(thingName)`
* `changed()`
* `updated()`
* `from(state)`
* `to(state)`
Additionally all the above triggers have the following functions:
* `.if()` or `.if(fn)` -> a [rule condition ](#rule-builder-conditions )
* `.then()` or `.then(fn)` -> a [rule operation ](#rule-builder-operations )
* `.or()` -> a [rule trigger ](#rule-builder-triggers ) (chain additional triggers)
#### Rule Builder Conditions
* `if(optionalFunction)`
2022-01-30 17:28:58 +00:00
* `.stateOfItem(itemName)`
* `is(state)`
* `in(state...)`
2021-12-17 07:00:44 +00:00
#### Rule Builder Operations
* `then(optionalFunction)`
2022-01-30 17:28:58 +00:00
* `.build(name, description, tags, id)`
2021-12-17 07:00:44 +00:00
* `.copyAndSendState()`
* `.copyState()`
* `.inGroup(groupName)`
* `.postIt()`
* `.postUpdate(state)`
* `.send(command)`
* `.sendIt()`
* `.sendOff()`
* `.sendOn()`
* `.sendToggle()`
#### Rule Builder Examples
```javascript
//Basic rule, when the BedroomLight1 is changed, run a custom function
rules.when().item('BedroomLight1').changed().then(e => {
console.log("BedroomLight1 state", e.newState)
2022-01-30 17:28:58 +00:00
}.build();
2021-12-17 07:00:44 +00:00
//turn on the kitchen light at SUNSET
rules.when().timeOfDay("SUNSET").then().sendOn().toItem("KitchenLight").build("Sunset Rule","turn on the kitchen light
at SUNSET");
2022-01-30 17:28:58 +00:00
//turn off the kitchen light at 9PM and tag rule
2021-12-17 07:00:44 +00:00
rules.when().cron("0 0 21 * * ?").then().sendOff().toItem("KitchenLight").build("9PM Rule", "turn off the kitchen light
2022-01-30 17:28:58 +00:00
at 9PM", ["Tag1", "Tag2"]);
2021-12-17 07:00:44 +00:00
2022-01-30 17:28:58 +00:00
//set the colour of the hall light to pink at 9PM, tag rule and use a custom ID
2021-12-17 07:00:44 +00:00
rules.when().cron("0 0 21 * * ?").then().send("300,100,100").toItem("HallLight").build("Pink Rule", "set the colour of
2022-01-30 17:28:58 +00:00
the hall light to pink at 9PM", ["Tag1", "Tag2"], "MyCustomID");
2021-12-17 07:00:44 +00:00
//when the switch S1 status changes to ON, then turn on the HallLight
rules.when().item('S1').changed().toOn().then(sendOn().toItem('HallLight')).build("S1 Rule");
//when the HallLight colour changes pink, if the function fn returns true, then toggle the state of the OutsideLight
rules.when().item('HallLight').changed().to("300,100,100").if(fn).then().sendToggle().toItem('OutsideLight').build();
//and some rules which can be toggled by the items created in the 'gRules' Group:
//when the HallLight receives a command, send the same command to the KitchenLight
rules.when().item('HallLight').receivedCommand().then().sendIt().toItem('KitchenLight').build("Hall Light", "");
//when the HallLight is updated to ON, make sure that BedroomLight1 is set to the same state as the BedroomLight2
rules.when().item('HallLight').receivedUpdate().then().copyState().fromItem('BedroomLight1').toItem('BedroomLight2').build();
```