# JRuby Scripting This add-on provides [JRuby](https://www.jruby.org/) scripting language for automation rules. Also included is [openhab-scripting](https://openhab.github.io/openhab-jruby/), a fairly high-level Ruby gem to support automation in openHAB. It provides native Ruby access to common openHAB functionality within rules including items, things, actions, logging and more. If you're new to Ruby, you may want to check out [Ruby Basics](https://openhab.github.io/openhab-jruby/main/file.ruby-basics.html). - [Why Ruby?](#why-ruby) - [Installation](#installation) - [Configuration](#configuration) - [Usage](#usage) - [UI Based Scripts](#ui-based-scripts) - [File Based Scripts](#file-based-scripts) - [Event Object](#event-object) - [Library Details](#library-details) - [Items](#items) - [Groups](#groups) - [Commands](#commands) - [Updates](#updates) - [State](#state) - [Metadata](#metadata) - [Persistence](#persistence) - [Semantic Model](#semantic-model) - [Linked Things](#linked-things) - [Item Builder](#item-builder) - [Things](#things) - [Actions](#actions) - [Logging](#logging) - [Timers](#timers) - [Cache](#cache) - [Time](#time) - [Ephemeris](#ephemeris) - [Rules, Scripts, and Scenes](#rules-scripts-and-scenes) - [Gems](#gems) - [Shared Code](#shared-code) - [Transformations](#transformations) - [Profile](#profile) - [Sitemaps](#sitemaps) - [File Based Rules](#file-based-rules) - [Basic Rule Structure](#basic-rule-structure) - [Rule Triggers](#rule-triggers) - [Item or Thing Changed](#item-or-thing-changed) - [Item Updated](#item-updated) - [Item Received a Command](#item-received-a-command) - [Member-of-Group Trigger](#member-of-group-trigger) - [Script is Loaded](#script-is-loaded) - [openHAB System Started](#openhab-system-started) - [Cron Trigger](#cron-trigger) - [Other Triggers](#other-triggers) - [Combining Multiple Triggers](#combining-multiple-triggers) - [Combining Multiple Conditions](#combining-multiple-conditions) - [Rule Conditions](#rule-conditions) - [Rule Executions](#rule-executions) - [Run Execution Block](#run-execution-block) - [Triggered Execution Block](#triggered-execution-block) - [Delay Execution Block](#delay-execution-block) - [Terse Rules](#terse-rules) - [Early Exit From a Rule](#early-exit-from-a-rule) - [Dynamic Generation of Rules](#dynamic-generation-of-rules) - [Hooks](#hooks) - [Calling Java From JRuby](#calling-java-from-jruby) - [Full Documentation](#full-documentation) Additional [example rules are available](https://openhab.github.io/openhab-jruby/main/file.examples.html), as well as examples of [conversions from DSL and Python rules](https://openhab.github.io/openhab-jruby/main/file.conversions.html). ## Why Ruby? - Ruby is designed for programmers' productivity with the idea that programming should be fun for programmers. - Ruby emphasizes the necessity for software to be understood by humans first and computers second. - Ruby makes writing automation enjoyable without having to fight with compilers and interpreters. - Rich ecosystem of tools, including things like Rubocop to help developers write clean code and RSpec to test the libraries. - Ruby is really good at letting one express intent and create a DSL to make that expression easier. ### Design points - Create an intuitive method of defining rules and automation - Rule language should "flow" in a way that you can read the rules out loud - Abstract away complexities of openHAB - Enable all the power of Ruby and openHAB - Create a Frictionless experience for building automation - The common, yet tricky tasks are abstracted and made easy, e.g. creating a timer that automatically reschedules itself. - Tested - Designed and tested using [Test-Driven Development](https://en.wikipedia.org/wiki/Test-driven_development) with [RSpec](https://rspec.info/) - Extensible. - Anyone should be able to customize and add/remove core language features - Easy access to the Ruby ecosystem in rules through Ruby gems. ## Installation ### Prerequisites 1. openHAB 3.4+ 1. The JRuby Scripting Language Addon ### From the User Interface 1. Go to `Settings -> Add-ons -> Automation` and install the jrubyscripting automation addon following the [openHAB instructions](https://www.openhab.org/docs/configuration/addons.html). In openHAB 4.0+ the defaults are set so the next step can be skipped. 1. Go to `Settings -> Other Services -> JRuby Scripting`: - **Ruby Gems**: `openhab-scripting=~>5.0` - **Require Scripts**: `openhab/dsl` (not required, but recommended) ### Using Files 1. Edit `/services/addons.cfg` and ensure that `jrubyscripting` is included in an uncommented `automation=` list of automations to install. In openHAB 4.0+ the defaults are set so the next step can be skipped. 1. Configure JRuby openHAB services Create a file called `jruby.cfg` in `/services/` with the following content: ```ini org.openhab.automation.jrubyscripting:gems=openhab-scripting=~>5.0 org.openhab.automation.jrubyscripting:require=openhab/dsl ``` ## Configuration After installing this add-on, you will find configuration options in the openHAB portal under _Settings -> Other Services -> JRuby Scripting_. Alternatively, JRuby configuration parameters may be set by creating a `jruby.cfg` file in `conf/services/`. By default this add-on includes the [openhab-scripting](https://github.com/openhab/openhab-jruby) Ruby gem and automatically `require`s it. This allows the use of [items](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#items-class_method), [rules](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#rules-class_method), [shared_cache](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#shared_cache-class_method) and other objects in your scripts. This functionality can be disabled for users who prefer to manage their own gems and `require`s via the add-on configuration options. Simply change the `gems` and `require` configuration settings. | Parameter | Description | | --------------------- | -------------------------------------------------------------------------------------------------------- | | `gem_home` | The path to store Ruby Gems.

Default: `$OPENHAB_CONF/automation/ruby/.gem/RUBY_ENGINE_VERSION` | | `gems` | A list of gems to install.

Default: `openhab-scripting=~>5.0` | | `check_update` | Check for updated version of `gems` on start up or settings change.

Default: `true` | | `require` | List of scripts to be required automatically.

Default: `openhab/dsl` | | `rubylib` | Search path for user libraries.

Default: `$OPENHAB_CONF/automation/ruby/lib` | | `dependency_tracking` | Enable dependency tracking.

Default: `true` | | `local_context` | See notes below.

Default: `singlethread` | | `local_variables` | See notes below.

Default: `transient` | When using file-based configuration, these parameters must be prefixed with `org.openhab.automation.jrubyscripting:`, for example: ```ini org.openhab.automation.jrubyscripting:gems=openhab-scripting=~>5.0 org.openhab.automation.jrubyscripting:require=openhab/dsl ``` ### gem_home Path to where Ruby Gems will be installed to and loaded from. The directory will be created if necessary. You can use `RUBY_ENGINE_VERSION`, `RUBY_ENGINE` and/or `RUBY_VERSION` replacements in this value to automatically point to a new directory when the addon is updated with a new version of JRuby. ### gems A comma separated list of [Ruby Gems](https://rubygems.org/) to install. The default installs the version of the helper for this version of openHAB. When overriding the default, be sure to still include the `openhab-scripting` gem in the list of gems to install. Each gem can have version specifiers which uses [pessimistic versioning](https://thoughtbot.com/blog/rubys-pessimistic-operator). Multiple version specifiers can be added by separating them with a semicolon. Examples: | gem setting | Description | | ------------------------------------------------ | -------------------------------------------------------------------------------------------------------- | | `openhab-scripting` | install the latest version of `openhab-scripting` gem | | `openhab-scripting=~>5.0.0` | install the latest version 5.0.x but not 5.1.x | | `openhab-scripting=~>5.0` | install the latest version 5.x but not 6.x | | `openhab-scripting=~>5.0, faraday=~>2.7;>=2.7.4` | install `openhab-scripting` gem version 5.x and `faraday` gem version 2.7.4 or higher, but less than 3.0 | | `gem1= >= 2.2.1; <= 2.2.5` | install `gem1` gem version 2.2.1 or above, but less than or equal to version 2.2.5 | ### check_update Check RubyGems for updates to the above gems when openHAB starts or JRuby settings are changed. Otherwise it will try to fulfil the requirements with locally installed gems, and you can manage them yourself with an external Ruby by setting the same GEM_HOME. ### require A comma separated list of script names to be required by the JRuby Scripting Engine at the beginning of user scripts. The default is to require the helper library. ### rubylib Search path for user libraries. Separate each path with a colon (semicolon in Windows). ### dependency_tracking Dependency tracking allows your scripts to automatically reload when one of its dependencies is updated. You may want to disable dependency tracking if you plan on editing or updating a shared library, but don't want all your scripts to reload until you can test it. ### local_context The local context holds Ruby runtime, name-value pairs for sharing variables between Java and Ruby. Valid values are: `singleton`, `threadsafe`, `singlethread`, or `concurrent`. See [this](https://github.com/jruby/jruby/wiki/RedBridge#context-instance-type) for options and details. ### local_variables Defines how variables are shared between Ruby and Java. Valid values are: `transient`, `persistent`, or `global`. See the [JRuby documentation](https://github.com/jruby/jruby/wiki/RedBridge#local-variable-behavior-options) for options and details. ## Usage ### UI Based Scripts 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 Scripts](#file-based-scripts) 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. ![openHAB Rule Configuration](doc/rule-config.png) #### Adding Actions Select "Add Action" and then select "Run Script" with "Ruby". This will bring up an empty script editor where you can enter your JavaScript. ![openHAB Rule Engines](doc/rule-engines.png) You can now write rules using standard Ruby along with the included openHAB [library](#library-details). ![openHAB Rule Script](doc/rule-script.png) For example, turning a light on: ```ruby KitchenLight.on logger.info("Kitchen Light State: #{KitchenLight.state}") ``` Sending a notification: ```ruby notify("romeo@montague.org", "Balcony door is open") ``` Querying the status of a thing: ```ruby logger.info("Thing status: #{things["zwave:serial_zstick:512"].status}")" ``` Theoretically you could even use a system start trigger with a UI rule, and then use the [syntax](#file-based-rules) mostly developed for file based rules to create multiple rules. ### File Based Scripts The JRuby Scripting addon will load scripts from `automation/ruby` 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 convenient way to persist objects. See [File Based Rules](#file-based-rules) for examples of creating rules within your scripts. ### Event Object When you use "Item event" as trigger (i.e. "[item] received a command", "[item] was updated", "[item] changed"), there is additional context available for the action in a variable called `event`. This tables gives an overview of the `event` object for most common trigger types. For full details, explore [OpenHAB::Core::Events](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Events.html). | Property Name | Type | Trigger Types | Description | Rules DSL Equivalent | | ------------- | -------------------------------------------------------------------------------------------- | -------------------------------------- | ---------------------------------------------------- | ---------------------- | | `state` | [State](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/State.html) or `nil` | `[item] changed`, `[item] was updated` | State that triggered event | `triggeringItem.state` | | `was` | [State](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/State.html) or `nil` | `[item] changed` | Previous state of Item or Group that triggered event | `previousState` | | `command` | [Command](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/Command.html) | `[item] received a command` | Command that triggered event | `receivedCommand` | | `item` | [Item](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Item.html) | all | Item that triggered event | `triggeringItem` | ```ruby logger.info(event.state == ON) ``` ```ruby event.item ``` Get the Triggering Item's Name: ```ruby event.item.name ``` Get the Triggering Item's Label: ```ruby event.item.label ``` Get the Triggering Item's State: ```ruby event.state # this version retrieves the item's state when the event was generated ``` or ```ruby event.item.state # this version will re-query the item for its state ``` ```ruby if event.item.state == ON # do something end # or (preferable) if event.item.on? # do something end ``` Get the Triggering Item's Previous State: ```ruby event.was ``` ```ruby if event.was.on? # do something end ``` Compare Triggering Item's State Against Previous State: ```ruby event.state > event.was ``` Get the Received Command: ```ruby event.command ``` ```ruby if event.command.on? # do something end ``` ## Library Details The openHAB JRuby Scripting runtime attempts to provide a familiar environment to Ruby developers. ### Items The [items](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#items-class_method) object allows interactions with openHAB items. However, most items can be referred to directly by name: ```ruby My_Item gWindowBlinds ``` Items can be retrieved dynamically: ```ruby the_item = items['My_Item'] # This returns an Item object, not just its state # For all intents and purposes, the_item variable is the same as My_Item in the previous example ``` Get the Item's Name as a String: ```ruby My_Item.name ``` Get the Item's Label: ```ruby My_Item.label ``` Get a Related Item: ```ruby my_light_item = items[My_Switch.name.sub('_Switch', '_Light')] ``` #### Groups Groups are regular items, but can also contain other items. ```ruby # direct members gTest.members # direct members and all their descendents gTest.all_members ``` Group members work like a [Ruby array](https://docs.ruby-lang.org/en/master/Array.html) so you can use `&` for intersection, `|` for union, and `-` for difference. ```ruby curtains_in_family_room = gFamilyRoom.members & gCurtains.members ``` You can iterate over group members with Ruby's ubiquitous `#each` method: ```ruby gTest.members.each do |item| # process item end # Iterate over all members, including members of members gTest.all_members.each do |item| # process item end ``` Group members are also [Enumerable](https://docs.ruby-lang.org/en/master/Enumerable.html), so can use any of tthose included methods: ```ruby members_that_are_on = gTest.members.select(&:on?) # exclude state members_that_are_not_on = gTest.members.reject(&:on?) # Filter with code: high_temperatures = gTemperatures.members.select(&:state?).select { |item| item.state > 30 | '°C' } ``` See also [Accessing elements in a Ruby array](https://docs.ruby-lang.org/en/master/Array.html#class-Array-label-Accessing+Elements). Get a sorted list of Group members matching a condition: ```ruby sorted_items_by_battery_level = gBattery.members .select(&:state?) # only include non NULL / UNDEF members .select { |item| item.state < 20 } # select only those with low battery .sort_by(&:state) ``` Get a list of values mapped from the members of a group: ```ruby battery_levels = gBattery.select(&:state?) # only include non NULL / UNDEF members .sort_by(&:state) .map { |item| "#{item.label}: #{item.state}" } # Use item state default formatting ``` Perform arithmetic on values from members of a group: ```ruby weekly_rainfall = gRainWeeklyForecast.members.sum(&:state) ``` #### Commands These three variants do the same thing: ```ruby My_Item.on My_Item.command ON My_Item << ON ``` Note: all possible commands are supported on the corresponding item types, e.g. `on`, `off`, `up`, `down`, `play`, `pause`, `stop`, etc. For more details, see the individual item classes under [OpenHAB::Core::Items](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items.html). ##### Sending Commands to an Item Only When Its State is Different ```ruby My_Item.ensure.on My_Item.ensure.command ON My_Item.ensure << ON # ensure causes the command to return nil if the item is already in the same state logger.info("Turning off the light") if My_Item.ensure.off ``` ##### Timed Commands A [Timed Command](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Items/TimedCommand.html) is similar to the openHAB Item's [expire parameter](https://www.openhab.org/docs/configuration/items.html#parameter-expire) but it offers more flexibility. It removes the need to manually create a timer. The command is sent to the item, then after the duration has elapsed, reverted. It also handles automatically canceling the timer if the item changes states before it reverts. ```ruby My_Switch.on for: 5.minutes ``` #### Updates Post an update to an item: ```ruby My_Switch.update ON ``` #### State The Item's state is accessible through [Item#state](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/GenericItem.html#state-instance_method): ```ruby if My_Item.state == ON # do something end # This syntax is equivalent and preferred: if My_Item.on? # do something end if Indoor_Temperature.state > 20 | '°C' || Indoor_Temperature.state > Outdoor_Temperature.state # do something end ``` Note: Boolean helper methods are available depending on the item / state type. For example `up?`, `down?`, `closed?`, `open?`, etc. Check if an Item's state is [NULL](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/UnDefType.html#NULL-constant) of [UNDEF](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/UnDefType.html#UNDEF-constant): ```ruby if My_Item.state? logger.info 'My_Item is not NULL nor UNDEF' end ``` ##### Comparing Item's State ```ruby String_Item.state == 'test string' Number_Item.state > 5.3 items['Number_Item'].state == 10 # Compare Quantity Types Temperature_Item.state > 24 | '°C' Indoor_Temperature.state > Outdoor_Temperature.state Indoor_Temperature.state > Outdoor_Temperature.state + 5 | '°C' Indoor_Temperature.state - Outdoor_Temperature.state > 5 | '°C' ``` See [unit block](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#unit-class_method) ##### Range checking Types that are comparable, such as [StringType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/StringType.html), [DateTimeType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/DateTimeType.html), [DecimalType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/DecimalType.html), [PercentType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/PercentType.html), include Ruby's [Comparable](https://docs.ruby-lang.org/en/master/Comparable.html) module which provides the handy [between?](https://docs.ruby-lang.org/en/master/Comparable.html#method-i-between-3F) method. ```ruby String_Item.update("Freddy") String_Item.state.between?("E", "G") # => true Number_Item.update(10) if Number_Item.state.between?(5, 20) logger.info "Number_Item falls within the expected range" end Temperature_Item.update(16 | "°C") Temperature_Item.state.between?(20 | "°C", 24 | "°C") # => false ``` Alternatively, a Ruby [range](https://docs.ruby-lang.org/en/master/Range.html) can be used. This can be handy for excluding the end of the range with the `...` operator. ```ruby if (5..10).cover?(Number_Item.state) logger.info "Number_Item is in the expected range" end ((20|"°C")..(24|"°C")).cover?(Temperature_Item.state) ``` ##### Loose Type Comparisons Some openHAB item types can accept different command types. For example, a [DimmerItem](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/DimmerItem.html) can accept a command with an [OnOffType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/OnOffType.html), [IncreaseDecreaseType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/IncreaseDecreaseType.html) or a [PercentType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/PercentType.html). However, ultimately an item only stores its state in its native type, e.g. a [DimmerItems](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/DimmerItem.html)'s native type is [PercentType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/PercentType.html). In some contexts, you don't care about the precise value of a particular state, and just want to know if it fits the general definition of [ON](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/OnOffType.html#ON-constant), etc. You can either explicitly convert to the general type, _or_ all of the state predicate methods available on [Item](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Item.html), [ItemStateEvent](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Events/ItemStateEvent.html), [ItemStateChangedEvent](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Events/ItemStateChangedEvent.html), [ItemCommandEvent](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Events/ItemCommandEvent.html), as well as specific types such as [PercentType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/PercentType.html) and [HSBType](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Types/HSBType.html), will do the conversion internally. ```ruby DimmerItem1.update(10) sleep 1 DimmerItem1.state == 10 # => true DimmerItem1.state == ON # => false DimmerItem1.state.as(OnOffType) == ON # => true DimmerItem1.state.on? # => true DimmerItem1.on? # => true DimmerItem1.off? # => false rule 'command' do received_command DimmerItem1 run do |event| if event.on? # This will be executed even when the command is a positive PercentType # instead of an actual ON command logger.info("DimmerItem1 is being turned on") end end end DimmerItem1 << 100 # => This will trigger the logger.info above ``` #### Metadata Metadata is accessed through [Item#metadata](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Item.html#metadata-instance_method). ```ruby metadata = My_Item.metadata['namespace'].value ``` #### Persistence [Persistence](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Persistence.html) methods are available directly on [Items](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Item.html). ```ruby logger.info("KitchenDimmer average_since #{KitchenDimmer.average_since(1.day.ago)}") daily_max = My_Item.maximum_since(24.hours.ago) ``` #### Semantic Model Many [helper methods](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Semantics.html) are available to make it easy to navigate the semantic model to get related items. ```ruby LivingRoom_Motion.location # Location of the motion sensor .equipments(Semantics::Lightbulb) # Get all Lightbulb Equipments in the location .members # Get all the member items of the equipments .points(Semantics::Switch) # Select only items that are Switch Points .on # Send an ON command to the items ``` #### Linked Things If an [Item](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Item.html) is linked to a [Thing](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Things/Thing.html), you can easily retrieve it. ```ruby linked_thing = My_Item.thing thing_uid = My_Item.thing.uid ``` An item can be linked to multiple things: ```ruby My_Item.things.each do |thing| logger.info("Thing: #{thing.uid}") end ``` #### Item Builder New items can be created via [items.build](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Items/Registry.html#build-instance_method). Note that by default items are not persisted to storage, and will be removed when the script unloads. ```ruby items.build do switch_item MySwitch, "My Switch" switch_item NotAutoupdating, autoupdate: false, channel: "mqtt:topic:1#light" group_item MyGroup do contact_item ItemInGroup, channel: "binding:thing#channel" end # passing `thing` to a group item will automatically use it as the base # for item channels group_item Equipment, tags: Semantics::HVAC, thing: "binding:thing" string_item Mode, tags: Semantics::Control, channel: "mode" end # dimension Temperature inferred number_item OutdoorTemp, format: "%.1f %unit%", unit: "°F" # unit lx, dimension Illuminance, format "%s %unit%" inferred number_item OutdoorBrightness, state: 10_000 | "lx" end ``` ### Things The [things](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#things-class_method) object allows interactions with openHAB things. Get Thing Status: ```ruby things['lgwebos:WebOSTV:main-tv'].status ``` Check if Thing is Online: ```ruby things['lgwebos:WebOSTV:main-tv'].online? ``` or ```ruby things['lgwebos:WebOSTV:main-tv'].status == ThingStatus::ONLINE ``` Enable/Disable a Thing: ```ruby thing = things['lgwebos:WebOSTV:main-tv'] thing.disable logger.info "TV enabled: #{thing.enabled?}" thing.enable logger.info "TV enabled: #{thing.enabled?}" ``` Get Thing's configurations: ```ruby server = things["smtp:mail:local"].configuration["hostname"] logger.info "Configured SMTP Server: #{server}" frontporch_cam_ip = things["ipcamera:dahua:frontporch"].configuration["ipAddress"] logger.info "Front Porch Camera's IP Address: #{frontporch_cam_ip}" ``` Get Thing's property: ```ruby model_id = things["fronius:meter:mybridge:mymeter"].properties["modelId"] logger.info "Fronius Smart Meter model: #{model_id}" ``` ### Actions [openHAB built-in actions](https://www.openhab.org/docs/configuration/actions.html) are available as children of the [Actions](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Actions.html) module. Action classes are also imported into the top-level namespace. Thing actions can be called directly on the [Thing](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Things/Thing.html). Publish an MQTT Message: ```ruby things['mqtt:broker:mybroker'].publish_mqtt('topic/name', 'payload') ``` Send an Email: ```ruby things['mail:smtp:mymailthing'].send_mail('me@example.com', 'Subject', 'message body') ``` Play Sound Through the Default Audio Sink: ```ruby Audio.play_sound 'sound_file.mp3' ``` Execute a Command: ```ruby Exec.execute_command_line('/path/to/program') ``` ### Logging The JRuby Scripting addon has a global `logger` object for logging. To log a message on `INFO` log level: ```ruby logger.info("The current time is #{Time.now}") ``` The default logger name for UI rules is `org.openhab.automation.jrubyscripting.script`. For file-based rules, it's based on the rule's ID, such as `org.openhab.automation.jrubyscripting.rule.myrule.rb:15`. To use a custom logger name: ```ruby logger = OpenHAB::Log.logger("org.openhab.custom") ``` Please be aware that messages might not appear in the logs if the logger name does not start with `org.openhab`. This behaviour is due to [log4j2](https://logging.apache.org/log4j/2.x/) requiring definition for each logger prefix. The [logger](https://openhab.github.io/openhab-jruby/main/OpenHAB/Logger.html) is similar to a standard [Ruby Logger](https://docs.ruby-lang.org/en/master/Logger.html). Supported logging functions include: - `logger.log(severity, obj)` - `logger.info(obj)` - `logger.warn(obj)` - `logger.error(obj)` - `logger.debug(obj)` - `logger.trace(obj)` `obj` is any Ruby (or Java) object. `#to_s` (or `toString()` if it's a Java object) is called on `obj`, and the result is output to the openHAB log. Additionally, all of these methods can take a [Ruby block](https://ruby-doc.com/docs/ProgrammingRuby/html/tut_containers.html#S2) instead, which will only be called if logging is enabled at the given level, and the result of the block will be treated as the log message. ### Timers ```ruby sleep 1.5 # sleep for 1.5 seconds ``` See Ruby docs on [sleep](https://docs.ruby-lang.org/en/master/Kernel.html#method-i-sleep). `sleep` should be avoided if possible. A [delay](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#delay-instance_method) can be inserted in between two execution blocks to achieve the same result. This delay is implemented with a timer. This is available only on file-based rules. ```ruby rule "delay something" do on_load run { logger.info "This will run immediately" } delay 10.seconds run { logger.info "This will run 10 seconds after" } end ``` Alternatively a timer can be used in either a file-based rule or in a UI based rule using [after](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#after-class_method). After takes a [Duration](#durations), e.g. `10.minutes` instead of using [ZonedDateTime](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/ZonedDateTime.html). ```ruby rule "simple timer" do changed Watering_System, to: ON run do after(5.minutes) { Watering_System.off } end end ``` When a script is unloaded, all created timers are automatically cancelled. #### Accessing Variables You can access all variables of the current context in the created timers. Note: Variables can be mutated (changed) after the timer has been created. Be aware that this can lead to unintentional side effects, e.g. when you change the variable after timer creation, which can make debugging quite difficult! ```ruby my_var = "Hello world!"; # Schedule a timer that expires in ten seconds after(10.seconds) do logger.info("Timer expired with my_var = '#{my_var}'") end my_var = "Hello mutation!" # When the timer runs, it will log "Hello mutation!" instead of "Hello world!" ``` #### Reschedule a Timer A timer can be rescheduled inside the timer body ```ruby after(3.minutes) do |timer| My_Light.on timer.reschedule # This will reschedule it for the same initial duration, i.e. 3 minutes in this case end ``` Or it can be rescheduled from outside the timer ```ruby my_timer = after(3.minutes) do My_Light.on end my_timer.reschedule # Use the same initial duration ``` It can be rescheduled to a different duration ```ruby after(3.minutes) do |timer| My_Light.on timer.reschedule(1.minute) end ``` It can also be canceled: ```ruby rule 'cancel timer' do changed Light_Item, to: OFF run { my_timer&.cancel } end ``` #### Manage Multiple Timers Multiple timers can be managed in the traditional way by storing the timer objects in a Hash: ```ruby @timers ||= {} if @timers[event.item] @timers[event.item].reschedule else @timers[event.item] = after 3.minutes do # Use the triggering item as the timer ID event.item.off @timers.delete(event.item) end end ``` However, a built in mechanism is available to help manage multiple timers, and is done in a thread-safe manner. This is done using timer IDs. The following rule automatically finds and reschedules the timer matching the same ID, which corresponds to each group member. ```ruby after 3.minutes, id: event.item do # Use the triggering item as the timer ID event.item.off end ``` Furthermore, you can manipulate the managed timers using the built-in [timers](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/TimerManager.html) object. ```ruby # timers is a special object to access the timers created with an id rule "cancel all timers" do received_command Cancel_All_Timers, to: ON # Send a command to this item to cancel all timers run do gOutdoorLights.members.each do |item_as_timer_id| timers.cancel(item_as_timer_id) end end end rule "reschedule all timers" do received_command Reschedule_All_Timers, to: ON # Send a command to this item to restart all timers run do gOutdoorLights.members.each do |item_as_timer_id| timers.reschedule(item_as_timer_id) end end end ``` ### Cache The [shared_cache](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#shared_cache-class_method) object provides a cache that can be used to set and retrieve objects that will be persisted between reloads of scripts, and available between different rules. It acts similarly to a regular Ruby Hash. Just be wary of Ruby-only data types (such as Symbols) that won't be accessible between different scripts. Get a previously set object with a default value: ```ruby shared_cache.compute_if_absent(:counter) { 0 } # Initialize with 0 if it didn't exist logger.info("Count: #{shared_cache[:counter] += 1}") ``` Get a previously set object, or assign it (this version is subject to race conditions with other scripts): ```ruby shared_cache[:counter] ||= 0 logger.info("Count: #{shared_cache[:counter] += 1}") ``` Get a previously set object with a default value, without assigning it (this version has an even longer amount of time between fetching the value and assigning it): ```ruby count = shared_cache.fetch(:counter) { 0 } shared_cache[:counter] = count + 1 ``` ### Time Several options are available for time related code, including but not limited to: - Java [LocalDate](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/LocalDate.html) - represents a date with no time - Java [LocalTime](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/LocalTime.html) - represents a time with no date - Java [Month](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Month.html) - Java [MonthDay](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/MonthDay.html) - represents a date with no time or year - Java [ZonedDateTime](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/ZonedDateTime.html) - represents a specific instance with a date and time - Java [Duration](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Duration.html) - Java [Period](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Period.html) - Ruby [Date](https://docs.ruby-lang.org/en/master/Date.html) - represents a date with no time - Ruby [Time](https://docs.ruby-lang.org/en/master/Time.html) - represents a specific instant with a date and time - Ruby [DateTime](https://docs.ruby-lang.org/en/master/DateTime.html) - represents a specific instant with a date and time #### Durations Ruby [integers](https://docs.ruby-lang.org/en/master/Integer.html) and [floats](https://docs.ruby-lang.org/en/master/Float.html) are extended with several methods to support durations. These methods create a new [Duration](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Duration.html) or [Period](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Period.html) object that is used by the [every](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#every-instance_method) trigger, [delay](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#delay-instance_method) block, the for option of [changed](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#changed-instance_method) triggers, and [timers](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Timer.html). ```ruby rule "run every 30 seconds" do every 30.seconds run { logger.info("Hello") } end ``` ```ruby rule "Warn about open door" do changed FrontDoor, to: OPEN, for: 10.minutes run { |event| logger.info("#{event.item.name} has been open for 10 minutes") } end ``` ```ruby rule "Timer example" do on_load run do after(3.hours) { logger.info("3 hours have passed") } end end ``` #### Time Comparisons, Conversions, and Arithmetic Comparisons, conversions and arithmetic are automatic between Java and Ruby types. Note that anytime you do a comparison between a type with more specific data, and a type missing specific data, the comparison is done as if the more specific data is at the beginning of its period. I.e. comparing a time to a month, the month will be treated as 00:00:00 on the first day of the month. When comparing with a type that's missing more generic data, it will be filled in from the other object. I.e. comparing a time to a month, the month will be assumed to be in the same year as the time. ```ruby # Get current date/time now = ZonedDateTime.now one_hour_from_now = ZonedDateTime.now + 60.minutes one_hour_from_now = 1.hour.from_now # or use the simpler helper method that also returns a ZonedDateTime # Or use Ruby time ruby_now = Time.now # Compare them if one_hour_from_now > now logger.info "As it should be" end # Comparing Ruby Time and ZonedDateTime works just fine if one_hour_from_now > ruby_now logger.info "It works too" end if Time.now > LocalTime.parse('7am') logger.info 'Wake up!' end if MonthDay.now == MonthDay.parse('02-14') logger.info "Happy Valentine's Day!" end # Ranges can cross midnight if Time.now.between?('10pm'..'5am') logger.info 'Sleep time' end # Explicit conversions ZonedDateTime.now.to_time Time.now.to_zoned_date_time # You can parse string as time wake_up_time = LocalTime.parse("6:00 am") # Compare now against LocalTime if ZonedDateTime.now >= wake_up_time Wake_Up_Alarm.on end # Even compare against Ruby Time if Time.now >= wake_up_time Wake_Up_Alarm.on end # Get today's start of the day (midnight) start_of_day = ZonedDateTime.now.with(LocalTime::MIDNIGHT) # or start_of_day = LocalTime::MIDNIGHT.to_zoned_date_time # Comparing ZonedDateTime against LocalTime with `<` max = Solar_Power.maximum_since(24.hours.ago) if max.timestamp < LocalTime::NOON logger.info "Max solar power #{max} happened before noon, at: #{max.timestamp}" end # Comparing Time against ZonedDateTime with `>` sunset = things["astro:sun:home"].get_event_time("SUN_SET", nil, nil) if Time.now > sunset logger.info "it is after sunset" end # Subtracting Duration from Time and comparing Time against ZonedDateTime Motion_Sensor.last_update < Time.now - 10.minutes # Alternatively: Motion_Sensor.last_update < 10.minutes.ago # Finding The Duration Between Two Times elapsed_time = Time.now - Motion_Sensor.last_update # Alternatively: elapsed_time = ZonedDateTime.now - Motion_Sensor.last_update # Using `-` operator with ZonedDateTime # Comparing two ZonedDateTime using `<` Motion_Sensor.last_update < Light_Item.last_update - 10.minutes # is the same as: Motion_Sensor.last_update.before?(Light_Item.last_update.minus_minutes(10)) # Getting Epoch Second Time.now.to_i ZonedDateTime.now.to_i ZonedDateTime.now.to_epoch_second # Convert Epoch second to time Time.at(1669684403) # Convert Epoch second to ZonedDateTime Time.at(1669684403).to_zoned_date_time # or java.time.Instant.of_epoch_second(1669684403).at_zone(ZoneId.system_default) ``` #### Ranges Ranges of date time objects work as expected. Make sure to use `#cover?` instead of `#include?` to do a simple comparison, instead of generating an array and searching it linearly. Ranges of non-absolute, "circular" types ([LocalTime](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/LocalTime.html), [Month](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Month.html), [MonthDay](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/MonthDay.html)) are smart enough to automatically handle boundary issues. Coarse types (like [LocalDate](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/LocalDate.html), [Month](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Month.html), [MonthDay](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/MonthDay.html)) will also work correctly when checking against a more specific type. To easily parse strings into date-time ranges, use the [OpenHAB::DSL.between](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#between-class_method) helper. [Duration](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Duration.html), [ZonedDateTime](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/ZonedDateTime.html), [LocalTime](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/LocalTime.html), [LocalDate](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/LocalDate.html), [MonthDay](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/MonthDay.html), [Month](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Java/Month.html), [Time](https://openhab.github.io/openhab-jruby/main/Time.html), [Date](https://openhab.github.io/openhab-jruby/main/Date.html), and [DateTime](https://openhab.github.io/openhab-jruby/main/DateTime.html) classes include [between?](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Between.html#between%3F-instance_method) method that accepts a range of string or any of the date/time objects. ```ruby between("10:00".."14:00").cover?(Time.now) between("11pm".."1am").cover?(Time.now) # Or use the alternative syntax: Time.now.between?("10:00".."14:00") Time.now.between?("11pm".."1am") case Time.now when between("6:00"..."12:00") logger.info("Morning Time") when between('12:00'..'15:00') logger.info("Afternoon") else logger.info("Not in time range") end # Compare against Month Time.now.between?(Month::NOVEMBER..Month::DECEMBER) Date.today.between?(Month::NOVEMBER..Month::DECEMBER) ZonedDateTime.now.between?(Month::NOVEMBER..Month::DECEMBER) # Compare against MonthDay Time.now.between?("05-01".."12-01") # Compare against time of day Time.now.between?("5am".."11pm") ``` ### Ephemeris [Helper methods](https://openhab.github.io/openhab-jruby/main/OpenHAB/CoreExt/Ephemeris.html) to easily access [openHAB's Ephemeris action](https://www.openhab.org/docs/configuration/actions.html#ephemeris) are provided on all date-like objects: ```ruby Time.now.holiday? # => false MonthDay.parse("12-25").holiday # => :christmas 1.day.from_now.next_holiday # => :thanksgiving notify("It's #{Ephemeris.holiday_name(Date.today)}!") if Date.today.holiday? Date.today.weekend? # => true Date.today.in_dayset?(:school) # => false ``` ### Rules, Scripts, and Scenes [Rules](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Rules/Rule.html), Scenes and Scripts can be accessed using the [rules](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Rules/Registry.html) object. For example, to execute/trigger a rule: ```ruby rules[rule_uid].trigger ``` Scenes are rules with a `Scene` tag, and Scripts are rules with a `Script` tag. They can be found using their uid just like normal rules, i.e. `rules[uid]`. For convenience, a list of all Scenes are available through the enumerable [rules.scenes](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Rules/Registry.html#scenes-instance_method), and a list of all Scripts through [rules.scripts](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Rules/Registry.html#scripts-instance_method). Example: All scenes tagged `sunrise` will be triggered at sunrise, and all scenes tagged `sunset` will be triggered at sunset. Note: these use the [Terse Rule](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/Terse.html) syntax. ```ruby channel("astro:sun:home:rise#event") { rules.scenes.tagged("sunrise").each(&:trigger) } channel("astro:sun:home:set#event") { rules.scenes.tagged("sunset").each(&:trigger) } ``` Or it can be written as one rule with the help of [trigger attachments](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#Triggers-group). ```ruby rule "Activate scenes at sunset/sunrise" do channel "astro:sun:home:rise#event", attach: "sunrise" channel "astro:sun:home:set#event", attach: "sunset" run { |event| rules.scenes.tagged(event.attachment).each(&:trigger) } end ``` Get the UID of a Rule ```ruby rule_obj = rule 'my rule name' do received_command My_Item run do # rule code here end end rule_uid = rule_obj.uid ``` A rule's UID can also be specified at rule creation ```ruby rule "my rule name", id: "my_unique_rule_uid" do # ... end # or rule "my rule name" do uid "my_unique_rule_uid" # ... end ``` Get the UID of a Rule by Name ```ruby rule_uid = rules.find { |rule| rule.name == 'This is the name of my rule' }.uid ``` Enable or Disable a Rule by UID ```ruby rules[rule_uid].enable rules[rule_uid].disable ``` #### Passing Values to Rules A rule/script may be given additional context/data by the caller. This additional data is available within the rule by referring to the names of the context variable. This is applicable to both UI rules and file-based rules. Within the script/rule body (either UI or file rule) ```ruby script id: "check_temp" do if CPU_Temperature.state > maxTemperature logger.warn "The CPU is overheating!" end end ``` The above script can be executed, passing it the `maxTemperature` argument from any supported scripting language, e.g.: ```ruby rules["check_temp"].trigger(maxTemperature: 80 | "°C") ``` ### Gems [Bundler](https://bundler.io/) is integrated, enabling any [Ruby gem](https://rubygems.org/) compatible with JRuby to be used within rules. This permits easy access to the vast ecosystem of libraries within the Ruby community. Gems are available using the [inline bundler syntax](https://bundler.io/guides/bundler_in_a_single_file_ruby_script.html). The require statement can be omitted. ```ruby gemfile do source 'https://rubygems.org' gem 'json', require: false gem 'nap', '1.1.0', require: 'rest' end logger.info("The nap gem is at version #{REST::VERSION}") ``` ### Shared Code If you would like to easily share code among multiple scripts, you can place it in `/automation/ruby/lib`. You can then simply `require` the file from your rules files. Files located in `$RUBYLIB` won't be automatically loaded individually by openHAB, only when you `require` them. `automation/ruby/myrule.rb` OR a UI Rule's script: ```ruby require "my_lib" logger.info(my_lib_version) ``` `automation/ruby/lib/my_lib.rb` ```ruby def my_lib_version "1.0" end ``` ### Transformations #### Using openHAB Transformations Existing openHAB transformations can also be used by calling the [transform](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#transform-class_method) method. This enables the use of any transformations from the `/transform` folder or managed through the UI, such as MAP, JSONPATH, Jinja Transformation, etc. ```ruby # Convert OPEN/CLOSED to Online/Offline using availability.map # OPEN=Online # CLOSED=OFFLINE logger.info transform(:map, "availability.map", LivingRoom_Switch_Availability.state) ``` #### Writing Custom Transformations in Ruby This add-on also provides the necessary infrastructure to use Ruby for writing [transformations](https://www.openhab.org/docs/configuration/transformations.html). The main value to be transformed is given to the script in a variable called `input`. Note that the values are passed to the transformation as Strings even for numeric items and data types. **Note**: In openHAB 3.4, due to an [issue](https://github.com/jruby/jruby/issues/5876) in the current version of JRuby, you will need to begin your script with `input ||= nil` (and `a ||= nil` etc. for additional query variables) so that JRuby will recognize the variables as variables--rather than method calls--when it's parsing the script. Otherwise you will get errors like `(NameError) undefined local variable or method 'input' for main:Object`. This is not necessary in openHAB 4.0+. #### File Based Transformations Once the addon is installed, you can create a Ruby file in the `$OPENHAB_CONF/transform` directory, with the extension `.rb`. When referencing the file, you need to specify the `RB` transform: `RB(mytransform.rb):%s`. You can also specify additional variables to be set in the script using a URI-like query syntax: `RB(mytransform.rb?a=1&b=c):%s` in order to share a single script with slightly different parameters for different items. ##### Example: Display the wind direction in degrees and cardinal direction `weather.items` ```Xtend Number:Angle Exterior_WindDirection "Wind Direction [RB(compass.rb):%s]" ``` `compass.rb` ```ruby DIRECTIONS = %w[N NE E SE S SW W NW N].freeze if input.nil? || input == "NULL" || input == "UNDEF" "-" else cardinal = DIRECTIONS[(input.to_f / 45).round] "#{cardinal} (#{input.to_f.round}°)" end ``` Given a state of `82 °`, this will produce a formatted state of `E (82°)`. ##### Example: Display the number of lights that are on/off within a group ```Xtend Group gIndoorLights "Indoor Lights [RB(group_count.rb?group=gIndoorLights):%s]" Group gOutdoorLights "Outdoor Lights [RB(group_count.rb?group=gOutdoorLights):%s]" ``` `group_count.rb` ```ruby items[group].all_members.then { |all| "#{all.select(&:on?).size}/#{all.size}" } ``` When 3 lights out of 10 lights are on, this will produce a formatted state of `3/10` #### Inline Transformations Inline transformations are supported too. For example, to display the temperature in both °C and °F: ```Xtend Number:Temperature Outside_Temperature "Outside Temperature [RB(| input.to_f.|('°C').then { |t| %(#{t.format('%d °C')} / #{t.to_unit('°F').format('%d °F')}) } ):%s]" ``` When the item contains `0 °C`, this will produce a formatted state of `0 °C / 32 °F`. ### Profile You can create an openHAB profile in JRuby that can be applied to item channel links. For more details, see [#profile](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL.html#profile-class_method). ### Sitemaps Sitemaps can be created via [sitemaps.build](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/Sitemaps/Provider.html#build-instance_method). ```ruby sitemaps.build do sitemap "default", "My Residence" do frame label: "Control" do text label: "Climate", icon: "if:mdi:home-thermometer-outline" do frame label: "Main Floor" do text item: MainFloor_AmbTemp switch item: MainFloorThermostat_TargetMode, label: "Mode", mappings: %w[off auto cool heat] setpoint item: MainFloorThermostat_SetPoint, label: "Set Point", visibility: "MainFloorThermostat_TargetMode!=off" end frame label: "Basement" do text item: Basement_AmbTemp switch item: BasementThermostat_TargetMode, label: "Mode", mappings: { OFF: "off", COOL: "cool", HEAT: "heat" } setpoint item: BasementThermostat_SetPoint, label: "Set Point", visibility: "BasementThermostat_TargetMode!=off" end end end end end ``` ## File Based Rules ### Basic Rule Structure See [OpenHAB::DSL::Rules::Builder](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/Builder.html) for full details. ```ruby rule "name" do end ``` Jump to: [Rule Triggers](#rule-triggers), [Rule Executions](#rule-executions), [Rule Conditions](#rule-conditions) ### Rule Triggers #### Item or Thing Changed ```ruby rule "Log (or notify) when the secret door is open" do changed SecretDoor, to: OPEN run { |event| logger.info("#{event.item} is opened") } end ``` ```ruby rule "Log when Fronius Inverter goes offline" do changed things["fronius:bridge:mybridge"], from: :online, to: :offline run { |event| logger.info("Thing #{event.uid} went #{event.status}!") } end ``` See [#changed](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#changed-instance_method) ##### Detecting Change Duration Only execute a rule when an item state changed and stayed the same for a period of time. This method can only be done using a file-based rule. ```ruby rule "Garage Door Alert" do changed GarageDoor, to: OPEN, for: 20.minutes run { Voice.say "The garage door has been open for 20 minutes!" } end ``` #### Item Updated ```ruby rule "Calculate" do updated Camera_Event_Data run do |event| logger.info "Camera event: #{event.state}" end end ``` See [#updated](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#updated-instance_method) #### Item Received a Command ```ruby rule "Received a command" do received_command DoorBell, to: ON run do |event| notify "Someone pressed the door bell" play_sound "doorbell.mp3" end end ``` See [#received_command](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#received_command-instance_method) #### Member-of-Group Trigger Add `.members` to the GroupItem in order to trigger on its members. ```ruby rule "Trigger by Member of" do changed gGroupName.members run do |event| logger.info "Triggered item: #{event.item.name}" end end ``` #### Script is Loaded ```ruby rule "initialize things" do on_load # This triggers whenever the script (re)loads run { logger.info "Here we go!" } end ``` See [#on_load](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#on_load-instance_method) #### openHAB System Started ```ruby rule "System startup rule" do on_start at_level: 80 run { logger.info "I'm glad to be alive!" } end ``` See [#on_start](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#on_start-instance_method) #### Cron Trigger Traditional cron trigger: ```ruby rule "cron rule" do cron "0 0,15 15-19 L * ?"" run { logger.info "Cron run" } end ``` Or an easier syntax: ```ruby rule "cron rule" do cron second: 0, minute: "0,15", hour: "15-19", dom: "L" run { logger.info "Cron run" } end ``` See [#cron](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#cron-instance_method) ##### `every` Trigger ```ruby rule "run every day" do every :day, at: "2:35pm" run { Amazon_Echo_TTS << "It's time to pick up the kids!" } end ``` ```ruby rule "run every 5 mins" do every 5.minutes run { logger.info "openHAB is awesome" } end ``` ```ruby rule "Anniversary Reminder" do every "10-15" # This takes a MM-DD syntax to trigger on 15th of October at midnight run do things["mail:smtp:mymailthing"].send_mail("me@example.com", "Anniversary Reminder!", "Today is your anniversary!") end end ``` See [#every](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#every-instance_method) #### Other Triggers There are more triggers supported by this library. See the [full list of supported triggers](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#Triggers-group). #### Combining Multiple Triggers ```ruby rule "multiple triggers" do changed Switch1, to: ON changed Switch2, to: ON run { |event| logger.info "Switch: #{event.item.name} changed to: #{event.state}" } end ``` When the trigger conditions are the same, the triggers can be combined. ```ruby rule "multiple triggers" do changed Switch1, Switch2, to: ON run { |event| logger.info "Switch: #{event.item.name} changed to: #{event.state}" } end ``` #### Combining Multiple Conditions ```ruby rule "multiple conditions" do changed Button_Action, to: ["single", "double"] run { |event| logger.info "Action: #{event.state}" } end ``` ### Rule Conditions ```ruby rule "motion sensor" do updated Motion_Sensor, to: ON only_if { Sensor_Enable.on? } # Run rule only if Sensor_Enable item is ON not_if { Sun_Elevation.positive? } # and not while the sun is up run { LightItem.on } end ``` Restrict Rule Executions to Certain Time of Day: ```ruby rule "doorbell" do updated DoorBell_Button, to: "single" between "6am".."8:30pm" run { play_sound "doorbell_chime.mp3" } end ``` See [Rule Guards](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#Guards-group) ### Rule Executions Execution blocks are executed when a rule is triggered and all the rule conditions are met. Multiple execution blocks can be specified. This can be useful especially when using a delay execution block inbetween two run or triggered blocks. #### Run Execution Block A run execution block is the most commonly used execution block. It provides the full [event object](#event-object) to the block. ```ruby rule "Rule with a run block" do received_command SwitchItem1 run do |event| logger.info "#{event.item} received this command: #{event.command}" end end ``` #### Triggered Execution Block A triggered execution block passes the `TriggeringItem` object directly to the block. It is handy when combined with Ruby's pretzel-colon operator to act directly on the object. ```ruby rule "Limit the duration of TV watching" do changed gTVPower.members, to: ON, for: 2.hours triggered(&:off) end ``` #### Delay Execution Block A delay exection block is useful for adding a delay inbetween rule executions or even at the beginning of the trigger event without having to manually create a timer. Unlike `sleep`, a delay block does not block the current executing thread. It actually sets a timer for you behind the scenes. ```ruby rule "Check for offline things 15 minutes after openHAB had started" do on_start delay 15.minutes run do offline_things = things.select(&:offline?).map(&:uid).join(", ") notify("Things that are still offline: #{offline_things}") end end ``` See [Execution Blocks](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/BuilderDSL.html#Execution-Blocks-group) ### Terse Rules A rule with a trigger and an execution block can be created with just one line. ```ruby received_command(My_Switch, to: ON) { My_Light.on } ``` See [Terse Rules](https://openhab.github.io/openhab-jruby/main/OpenHAB/DSL/Rules/Terse.html) for full details. ### Early Exit From a Rule You can use `next` within a file-based rule, because it's in a block: ```ruby rule "doorbell" do updated DoorBell_Button, to: "single" run do next unless Time.now.between?("6am".."8:30pm") play_sound "doorbell_chime.mp3" end end ``` Use `return` within a UI rule: ```ruby return unless Time.now.between?("6am".."8:30pm") play_sound "doorbell_chime.mp3" ``` ### Dynamic Generation of Rules The rule definition itself is just Ruby code, which means you can use code to generate your rules. Take care when doing this as the the items/groups are processed when the rules file is processed, meaning that new items/groups will not automatically generate new rules. ```ruby rule "Log whenever a Virtual Switch Changes" do items.grep(SwitchItem) .select { |item| item.label&.include?("Virtual") } .each do |item| changed item end run { |event| logger.info "#{event.item.name} changed from #{event.was} to #{event.state}" } end ``` This rule is effectively the same: ```ruby virtual_switches = items.grep(SwitchItem) .select { |item| item.label&.include?("Virtual") } rule "Log whenever a Virtual Switch Changes 2" do changed(*virtual_switches) run { |event| logger.info "#{event.item.name} changed from #{event.was} to #{event.state} 2" } end ``` This will accomplish the same thing, but create a new rule for each virtual switch: ```ruby virtual_switches = items.grep(SwitchItem) .select { |item| item.label&.include?("Virtual") } virtual_switches.each do |switch| rule "Log whenever a #{switch.label} Changes" do changed switch run { |event| logger.info "#{event.item.name} changed from #{event.was} to #{event.state} 2" } end end ``` ### Hooks File based scripts can also register [hooks](https://openhab.github.io/openhab-jruby/main/OpenHAB/Core/ScriptHandling.html) that will be called when the script has completed loading (`script_loaded`) and when it gets unloaded (`script_unloaded`). ```ruby x = 1 script_loaded do logger.info("script loaded!") logger.info(x) # this will log 2, since it won't execute until the entire script loads. end x = 2 script_unloaded do logger.info("script unloaded") end ``` ## Calling Java From JRuby JRuby can [access almost any Java object](https://github.com/jruby/jruby/wiki/CallingJavaFromJRuby) that's available in the current JVM. This is how the library is implemented internally. ```ruby # you can `java_import` classes and interfaces, which will become Ruby constants java_import java.time.format.DateTimeFormatter formatter = DateTimeFormatter.of_pattern("yyyy MM dd") # or you can just reference them directly to avoid polluting the global namespace formatter = java.time.format.DateTimeFormatter.of_pattern("yyyy MM dd") ``` ## Full Documentation Visit for the full documentation of the **openHAB JRuby Helper Library**.