Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make it possible to see thing channels and linked items #212

Open
wants to merge 4 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
32 changes: 32 additions & 0 deletions doc/EXAMPLES.md
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@
+ [Example 41 - Get Metadata and Tags](#example-41---get-metadata-and-tags)
+ [Example 42 - Persist future data](#example-42---persist-future-data)
+ [Example 43 - Creating a rule dynamically using JRuleBuilder](#example-43---creating-a-rule-dynamically-using-jrulebuilder)
+ [Example 44 - Setting items linked to a thing to UNDEF when thing goes offline](#example-44---setting-items-linked-to-a-thing-to-undef-when-thing-goes-offline)

### Example 1 - Invoke another item Switch from rule

Expand Down Expand Up @@ -1125,3 +1126,34 @@ public class DynamicRuleModule extends JRule {
}
}
```

## Example 44 - Setting items linked to a thing to UNDEF when thing goes offline

Use case: No longer be fooled by outdated item states when a thing goes offline

```java
package org.openhab.automation.jrule.rules.user;

import java.util.List;

import org.openhab.automation.jrule.items.JRuleItem;
import org.openhab.automation.jrule.rules.JRule;
import org.openhab.automation.jrule.rules.JRuleName;
import org.openhab.automation.jrule.rules.JRuleWhenThingTrigger;
import org.openhab.automation.jrule.rules.event.JRuleThingEvent;
import org.openhab.automation.jrule.things.JRuleAbstractThing;
import org.openhab.automation.jrule.things.JRuleChannel;
import org.openhab.automation.jrule.things.JRuleThingRegistry;
import org.openhab.automation.jrule.things.JRuleThingStatus;

public class ChannelsToUndefWhenTingOffline extends JRule {
@JRuleName("Device monitoring - Set linked items to UNDEF if thing goes offline")
@JRuleWhenThingTrigger(thing = "*", from = JRuleThingStatus.ONLINE)
public void setChannelsToUndefWhenThingOffline(JRuleThingEvent thingEvent) {
String thingUID = thingEvent.getThing();
JRuleAbstractThing thing = JRuleThingRegistry.get(thingUID, JRuleAbstractThing.class);
List<JRuleChannel> channels = thing.getChannels();
channels.forEach(channel -> thing.getLinkedItems(channel).forEach(JRuleItem::postUndefUpdate));
}
}
```
7 changes: 4 additions & 3 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
<version>4.2.1-SNAPSHOT</version>

<properties>
<testcontainers.version>1.20.4</testcontainers.version>
<bnd.importpackage>com.sun.org.apache.xml.internal.utils.*;resolution:=optional,\
com.sun.org.apache.xpath.internal.*;resolution:=optional,\
org.apache.log.*;resolution:=optional,\
Expand Down Expand Up @@ -77,19 +78,19 @@
<dependency>
<groupId>org.testcontainers</groupId>
<artifactId>junit-jupiter</artifactId>
<version>1.17.6</version>
<version>${testcontainers.version}</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.testcontainers</groupId>
<artifactId>toxiproxy</artifactId>
<version>1.17.6</version>
<version>${testcontainers.version}</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.testcontainers</groupId>
<artifactId>mockserver</artifactId>
<version>1.17.6</version>
<version>${testcontainers.version}</version>
<scope>test</scope>
</dependency>
<dependency>
Expand Down
4 changes: 2 additions & 2 deletions src/main/history/dependencies.xml
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<features xmlns="http://karaf.apache.org/xmlns/features/v1.6.0" name="org.openhab.automation.jrule-4.2.0">
<features xmlns="http://karaf.apache.org/xmlns/features/v1.6.0" name="org.openhab.automation.jrule-4.2.1-SNAPSHOT">
<feature version="0.0.0">
<feature>openhab-runtime-base</feature>
<feature>wrap</feature>
<bundle>mvn:javax.el/javax.el-api/2.2.4</bundle>
<bundle>mvn:org.freemarker/freemarker/2.3.32</bundle>
<bundle>mvn:org.openhab.addons.bundles/org.openhab.automation.jrule/4.2.0</bundle>
<bundle>mvn:org.openhab.addons.bundles/org.openhab.automation.jrule/4.2.1-SNAPSHOT</bundle>
<bundle>wrap:mvn:javax.servlet/jsp-api/2.0</bundle>
<bundle>wrap:mvn:javax.servlet/servlet-api/2.4</bundle>
<bundle>wrap:mvn:org.lastnpe.eea/eea-all/2.2.1</bundle>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,14 @@ public void postUpdate(String itemName, JRuleValue value) {
}
}

public void postUndef(String itemName) {
postUpdate(itemName, UnDefType.UNDEF);
}

public void postNull(String itemName) {
postUpdate(itemName, UnDefType.NULL);
}

public void postUpdate(String itemName, double value, String unit) {
QuantityType<?> type = new QuantityType<>(value + " " + unit);
postUpdate(itemName, type);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -158,6 +158,7 @@ public JRuleHandler(JRuleConfig config, ItemRegistry itemRegistry, ItemChannelLi
final JRuleThingHandler thingHandler = JRuleThingHandler.get();
thingHandler.setThingManager(thingManager);
thingHandler.setThingRegistry(thingRegistry);
thingHandler.setItemChannelLinkRegistry(itemChannelLinkRegistry);

final JRuleItemHandler itemHandler = JRuleItemHandler.get();
itemHandler.setItemRegistry(itemRegistry);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,16 @@
package org.openhab.automation.jrule.internal.handler;

import java.util.Collection;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;

import org.openhab.automation.jrule.items.JRuleItem;
import org.openhab.automation.jrule.items.JRuleItemRegistry;
import org.openhab.automation.jrule.things.JRuleChannel;
import org.openhab.automation.jrule.things.JRuleThingStatus;
import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingManager;
import org.openhab.core.thing.ThingRegistry;
import org.openhab.core.thing.ThingUID;
import org.openhab.core.thing.*;
import org.openhab.core.thing.link.ItemChannelLinkRegistry;

/**
* The {@link JRuleThingHandler} provides access to thing actions
Expand All @@ -36,6 +40,8 @@ private JRuleThingHandler() {

private ThingManager thingManager;

private ItemChannelLinkRegistry itemChannelLinkRegistry;

public void setThingManager(ThingManager thingManager) {
this.thingManager = thingManager;
}
Expand All @@ -44,6 +50,10 @@ public void setThingRegistry(ThingRegistry thingRegistry) {
this.thingRegistry = thingRegistry;
}

public void setItemChannelLinkRegistry(ItemChannelLinkRegistry itemChannelLinkRegistry) {
this.itemChannelLinkRegistry = itemChannelLinkRegistry;
}

public static JRuleThingHandler get() {
if (instance == null) {
synchronized (JRuleThingHandler.class) {
Expand Down Expand Up @@ -83,4 +93,24 @@ public JRuleThingStatus getStatus(String thingUID) {
return JRuleThingStatus.THING_UNKNOWN;
}
}

/**
* Get all channels of a thing
*
* @param thingUID the thing UID
* @return list of all channels
*/
public List<JRuleChannel> getChannels(String thingUID) {
Thing thing = thingRegistry.get(new ThingUID(thingUID));
if (thing != null) {
return thing.getChannels().stream().map(channel -> new JRuleChannel(channel.getUID().toString())).toList();
} else {
return List.of();
}
}

public Set<JRuleItem> getLinkedItems(JRuleChannel channel) {
return itemChannelLinkRegistry.getLinkedItemNames(new ChannelUID(channel.getChannelUID())).stream()
.map(itemName -> JRuleItemRegistry.get(itemName)).collect(Collectors.toSet());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -98,7 +98,11 @@ default void postUpdate(JRuleRefreshValue state) {
}

default void postNullUpdate() {
JRuleEventHandler.get().postUpdate(getName(), null);
JRuleEventHandler.get().postNull(getName());
}

default void postUndefUpdate() {
JRuleEventHandler.get().postUndef(getName());
}

default Optional<ZonedDateTime> lastUpdated() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,11 @@
*/
package org.openhab.automation.jrule.things;

import java.util.List;
import java.util.Set;

import org.openhab.automation.jrule.internal.handler.JRuleThingHandler;
import org.openhab.automation.jrule.items.JRuleItem;

/**
* The {@link JRuleAbstractThing} represents a thing that is either a bridge, a bridged (sub thing of a bridge) or a
Expand Down Expand Up @@ -49,4 +53,12 @@ public void restart() {
disable();
enable();
}

public List<JRuleChannel> getChannels() {
return JRuleThingHandler.get().getChannels(thingUID);
}

public Set<JRuleItem> getLinkedItems(JRuleChannel channel) {
return JRuleThingHandler.get().getLinkedItems(channel);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.automation.jrule.things;

/**
* The {@link JRuleChannel} represents a Thing channel, possibly linked to an Item.
*
* @author Arne Seime - Initial contribution
*/
public class JRuleChannel {
private String channelUID;

public JRuleChannel(String channelUID) {
this.channelUID = channelUID;
}

public String getChannelUID() {
return channelUID;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,27 @@ public void mqttThingChangedToOffline() {
verifyRuleWasExecuted(TestRules.NAME_MQTT_THING_CHANGED_TO_OFFLINE);
}

@Test
public void mqttThingChangedFromOnline() throws MqttException {

Awaitility.await().with().pollDelay(1, TimeUnit.SECONDS).timeout(20, TimeUnit.SECONDS)
.pollInterval(200, TimeUnit.MILLISECONDS).await("thing online")
.until(() -> getThingState("mqtt:topic:mqtt:fromonlinetest"), s -> s.equals("ONLINE"));
publishMqttMessage("fromonlinetest/state", "1");
Awaitility.await().with().pollDelay(100, TimeUnit.MILLISECONDS).timeout(5, TimeUnit.SECONDS)
.pollInterval(100, TimeUnit.MILLISECONDS).await("item updated")
.until(() -> getState(TestRules.ITEM_MQTT_TOPIC_FROM_ONLINE_TEST), s -> "1".equals(s));
mqttProxy.setConnectionCut(true);
Awaitility.await().with().pollDelay(1, TimeUnit.SECONDS).timeout(20, TimeUnit.SECONDS)
.pollInterval(200, TimeUnit.MILLISECONDS).await("thing online")
.until(() -> getThingState("mqtt:topic:mqtt:fromonlinetest"), s -> s.equals("OFFLINE"));
Awaitility.await().with().pollDelay(1, TimeUnit.SECONDS).timeout(30, TimeUnit.SECONDS)
.pollInterval(200, TimeUnit.MILLISECONDS).await("item undef")
.until(() -> getState(TestRules.ITEM_MQTT_TOPIC_FROM_ONLINE_TEST), s -> "UNDEF".equals(s));

verifyRuleWasExecuted(TestRules.NAME_MQTT_THING_CHANGED_FROM_ONLINE);
}

@Test
public void memberOfGroupReceivedCommand() throws IOException {
sendCommand(TestRules.ITEM_SWITCH_GROUP_MEMBER1, JRuleSwitchItem.ON);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -312,7 +312,7 @@ private Optional<Double> getDoubleState(String itemName) throws IOException, Par
return Optional.ofNullable(getState(itemName)).map(Double::parseDouble);
}

private String getState(String itemName) throws IOException, ParseException {
protected static String getState(String itemName) throws IOException, ParseException {
try (CloseableHttpClient client = HttpClientBuilder.create().build()) {
HttpGet request = new HttpGet(String.format("http://%s:%s/rest/items/" + itemName + "/state",
getOpenhabHost(), getOpenhabPort()));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,7 @@
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.time.temporal.ChronoUnit;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Map;
import java.util.Set;
import java.util.*;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.IntFunction;
import java.util.stream.Collectors;
Expand All @@ -36,6 +33,9 @@
import org.openhab.automation.jrule.rules.event.JRuleThingEvent;
import org.openhab.automation.jrule.rules.event.JRuleTimerEvent;
import org.openhab.automation.jrule.rules.value.*;
import org.openhab.automation.jrule.things.JRuleChannel;
import org.openhab.automation.jrule.things.JRuleSubThing;
import org.openhab.automation.jrule.things.JRuleThingRegistry;
import org.openhab.automation.jrule.things.JRuleThingStatus;

/**
Expand All @@ -57,6 +57,8 @@ public class TestRules extends JRule {
public static final String NAME_EXEC_COMMAND_LINE = "Exec Command Line";
public static final String NAME_MQTT_CHANNEL_TRIGGERED = "Mqtt Channel Triggered";
public static final String NAME_MQTT_THING_CHANGED_TO_OFFLINE = "Mqtt Thing Changed To Offline";
public static final String NAME_MQTT_THING_CHANGED_FROM_ONLINE = "Mqtt Thing Changed From Online";
public static final String ITEM_MQTT_TOPIC_FROM_ONLINE_TEST = "ItemToUndef";
public static final String NAME_MEMBER_OF_GROUP_RECEIVED_COMMAND = "Member Of Group Received Command";
public static final String NAME_MEMBER_OF_GROUP_RECEIVED_UPDATE = "Member Of Group Received Update";
public static final String NAME_MEMBER_OF_GROUP_CHANGED = "Member Of Group Changed";
Expand Down Expand Up @@ -184,6 +186,24 @@ public void mqttThingChangedToOffline(JRuleThingEvent event) {
logInfo("thing '{}' goes '{}'", event.getThing(), event.getStatus());
}

@JRuleName(NAME_MQTT_THING_CHANGED_FROM_ONLINE)
@JRuleWhenThingTrigger(thing = "mqtt:topic:mqtt:fromonlinetest", from = JRuleThingStatus.ONLINE)
public void mqttThingChangedFromOnline(JRuleThingEvent event) {
logInfo("Thing '{}' goes '{}'", event.getThing(), event.getStatus());
String thing = event.getThing();
JRuleSubThing mqttTopicThing = JRuleThingRegistry.get(thing, JRuleSubThing.class);
List<JRuleChannel> channels = mqttTopicThing.getChannels();
logInfo("Channels size: {}", channels.size());
channels.stream().forEach(channel -> {
Set<JRuleItem> linkedItems = mqttTopicThing.getLinkedItems(channel);
logInfo("Linked items size: {}", linkedItems.size());
linkedItems.stream().forEach(item -> {
logInfo("Linked item: {}, setting to UNDEF", item.getName());
item.postUndefUpdate();
});
});
}

@JRuleName(NAME_MEMBER_OF_GROUP_RECEIVED_COMMAND)
@JRuleWhenItemReceivedCommand(item = ITEM_SWITCH_GROUP, memberOf = JRuleMemberOf.All)
public synchronized void memberOfGroupReceivedCommand(JRuleItemEvent event) {
Expand Down
2 changes: 2 additions & 0 deletions src/test/resources/docker/conf/items/default.items
Original file line number Diff line number Diff line change
Expand Up @@ -87,3 +87,5 @@ Number Number_To_Persist_Future (InfluxDbPersist)


Dimmer Dimmer_With_Tags_And_Metadata ["Control", "Light"] { Speech="SetLightState" [ location="Livingroom" ] }

Number ItemToUndef {channel="mqtt:topic:mqtt:fromonlinetest:number"}
4 changes: 4 additions & 0 deletions src/test/resources/docker/conf/things/mqtt.things
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,8 @@ Bridge mqtt:broker:mqtt [ host="mqtt", port=8666, username="admin", password="ad
Type number : number [ stateTopic="number/state" ]
Type number : numberTrigger [ stateTopic="number/state", trigger="true" ]
}
Thing topic fromonlinetest {
Channels:
Type number : number [ stateTopic="fromonlinetest/state" ]
}
}
Loading