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

Add ItemStateUpdatedEvent and enable group channel-links #3141

Merged
merged 2 commits into from
Mar 31, 2023

Conversation

J-N-K
Copy link
Member

@J-N-K J-N-K commented Nov 4, 2022

There has been some discussion in the past about the ItemStateEvent which in some cases may not carry the unit of a value even if the item has a unit (#2956).

This PR adds an ItemStateUpdatedEvent which is fired by the item itself (similar to the ItemStateChangedEvent. Since the item knows about its dimension and unit, the new event can carry this information even if the unit is derived from a unit provider or the state description. This also clearly separates the events when the item reports something and when the item is supposed to do something:

  • ItemStateEvent -> the item shall update to the contained state
  • ItemCommandEvent-> the item receives this command
  • ItemStateUpdatedEvent -> the item updated the state
  • ItemStateChangedEvent -> the item changed the state

Since this is rather big change in functionality I would suggest to consider this for OH4 because time is running out for extensive testing (even if unit/integration test coverage is quite good).

As a side-effect group items now also report their state which makes it possible to send aggregated states (AVG/SUM/MIN/MAX/...) to the outside world by using the FOLLOW profile.

@J-N-K J-N-K added this to the 4.0 milestone Nov 4, 2022
@J-N-K J-N-K added enhancement An enhancement or new feature of the Core API breaking labels Nov 4, 2022
@J-N-K J-N-K linked an issue Nov 4, 2022 that may be closed by this pull request
@openhab-bot
Copy link
Collaborator

This pull request has been mentioned on openHAB Community. There might be relevant details there:

https://community.openhab.org/t/follow-profile-on-group/113445/19

@spacemanspiff2007
Copy link
Contributor

spacemanspiff2007 commented Nov 4, 2022

Awesome! I really think the differentiation between report and todo makes things more clear for the user.
Could you also add an example how this would look as an SSE event?

@J-N-K
There was also some talk about adding a timestamp when the item state was set.
Have you already had some thoughts about that?

@J-N-K
Copy link
Member Author

J-N-K commented Nov 5, 2022

@spacemanspiff2007 Looks like the ItemStateEvent, it uses the same payload. It has a different topic (/stateupdated, in line with /statechanged) and always contains the new state of the item.

Regarding the timestamp: IMO it's still better to set the timestamp on the state, as that allows handling historic and future states with the same logic we have now. I wouldn't veto an addition to the events, but that should be a change for all events, not just some, so it's out of scope here.

@J-N-K J-N-K mentioned this pull request Nov 5, 2022
@openhab-bot
Copy link
Collaborator

This pull request has been mentioned on openHAB Community. There might be relevant details there:

https://community.openhab.org/t/habapp-items-some-times-none/140425/11

@J-N-K J-N-K marked this pull request as ready for review December 18, 2022 16:38
@J-N-K J-N-K requested a review from a team as a code owner December 18, 2022 16:38
@wborn wborn removed this from the 4.0 milestone Dec 19, 2022
@openhab-bot
Copy link
Collaborator

This pull request has been mentioned on openHAB Community. There might be relevant details there:

https://community.openhab.org/t/uom-default-units-and-consequences/142360/118

J-N-K added 2 commits March 23, 2023 19:38
Signed-off-by: Jan N. Klug <[email protected]>
@J-N-K J-N-K force-pushed the feature-updatedevent branch from 58e1ff0 to 3d79e0a Compare March 23, 2023 19:14
@J-N-K
Copy link
Member Author

J-N-K commented Mar 23, 2023

@openhab/core-maintainers Can we agree that this is useful and merge it? It makes the implementation of a solution for #3282 much easier because we have an event that carries the correct unit for state updates (which is needed e.g. for persistence).

@kaikreuzer
Copy link
Member

Thanks @J-N-K. Yes, I think it makes sense and the code looks all good to me.
Just one small question to not miss anything: Why did you label it as "API breaking"? Afaics, it merely adds the new event type(s), but there shouldn't really be an impact on the way the existing events are sent. Could you please briefly explain?

@J-N-K
Copy link
Member Author

J-N-K commented Mar 27, 2023

I didn't find a better label. The sending of events is not affected (there are only more events), but there is a change in behavior of the CommunicationManager which now listens for ItemStateUpdatedEvent instead of ItemStateEvent. As a result it no longer receives a DecimalType when this is part of the ItemStateEvent but the item has a dimension. Instead a QuantityType is received with the new state of the item (or the same if it doesn't change).

@spacemanspiff2007
Copy link
Contributor

Additionally it's somewhat API breaking for RestAPI consumers because

  • The new event should be used
  • The legacy code doing the event state to item state magic can be completely removed
    (e.g. ItemStateEvent(1) to Switch results in ON but to Number results in 1).

It's a huge simplification and makes dealing with the RestAPI events much more pleasant.

Copy link
Member

@kaikreuzer kaikreuzer left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you!

@kaikreuzer kaikreuzer merged commit 4182980 into openhab:main Mar 31, 2023
@kaikreuzer kaikreuzer added this to the 4.0 milestone Mar 31, 2023
@lolodomo
Copy link
Contributor

lolodomo commented Apr 1, 2023

I have difficulty to understand the impact of this change on the remote OH binding.
Is it ok to ignore this new event and only rely on old events?
Or should I ignore ItemStateEvent and consider rather ItemStateUpdatedEvent when and only when the remote server is OH 4 ?

https://github.com/openhab/openhab-addons/blob/main/bundles/org.openhab.binding.remoteopenhab/src/main/java/org/openhab/binding/remoteopenhab/internal/rest/RemoteopenhabRestClient.java#L416

@lolodomo
Copy link
Contributor

lolodomo commented Apr 1, 2023

Is it just an additional event that can be ignored or is the behavior with the previous events impacted?

@J-N-K
Copy link
Member Author

J-N-K commented Apr 2, 2023

There is a benefit in using it, but it does no harm ignoring it. All events are still fired like before.

One of the limitations of ItemStateEvent is that it does not contain information about the real state the item has. E.g. a ON event to a DimmerItem changes the internal state to 100% because internally the representation of the state is PercentType. The ItemStateChangedEvent however does report 100%. The ItemStateUpdatedEvent also contains the new state of the item in the same way it is stored internally. The same applies to DecimalTypes sent to UoM-Number items, Color items and roller shutters.

eventPublisher.post(ItemEventFactory.createStateChangedEvent(this.name, newState, oldState));
EventPublisher eventPublisher1 = this.eventPublisher;
if (eventPublisher1 != null) {
eventPublisher1.post(ItemEventFactory.createStateChangedEvent(this.name, newState, oldState));
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@J-N-K would you mind explaining about this change in particular? I am trying to understand why it was necessary. It seems to me that it does exactly the same thing. What have I missed?

I'm trying to understand the difference between:

        if (eventPublisher != null) {
            eventPublisher.post(ItemEventFactory.createStateChangedEvent(this.name, newState, oldState));
        }

vs:

        EventPublisher eventPublisher1 = this.eventPublisher;
        if (eventPublisher1 != null) {
            eventPublisher1.post(ItemEventFactory.createStateChangedEvent(this.name, newState, oldState));
        }

Why the need for a temporary variable? Is it for optimisation purpose, so that it's accessing a local variable instead of resolving a class-member variable twice?

Copy link
Member Author

@J-N-K J-N-K Apr 3, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The eventPublisher is @Nullable. There is a slight chance that it is set to null by another thread between the null-check and the usage. Therefore taking a copy ensures we don't get a NPE when using it: even if the field is set to null, we still have the non-null reference.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you! TIL!

florian-h05 added a commit to florian-h05/openhab-distro that referenced this pull request Apr 4, 2023
florian-h05 added a commit to florian-h05/openhab-distro that referenced this pull request Apr 4, 2023
@florian-h05
Copy link
Contributor

It seems that the events that trigger rules have changed with this PR, can this be caused here?
And if yes, how did they change? The JS library‘s event object is partly broken now and I need to fix it.

@J-N-K
Copy link
Member Author

J-N-K commented Apr 4, 2023

What exactly is broken? The ItemStateEvent was replaced by the ItemStateUpdatedEvent because that is what contains the new item's state, but I don't see why that would break something, they both implement .getItemState and inherit from ItemEvent?

@florian-h05
Copy link
Contributor

For file-based scripts you can create rules using rules.JSRule, the callback functions of these are provided the event data that triggered the rule run. The event is not passed 1:1 to the rule, because handling Java types in JS is confusing. Instead, based on the classname of the event, properties are added to a pure JS object.

I just need to adjust the classnames there.

The ItemStateEvent was replaced by the ItemStateUpdatedEvent

I‘ve already figured out that, thanks.

Did anything change for group events? On first look, it seems that GroupItemStateEvent was replaced by ItemUpdatedEvent.

@J-N-K
Copy link
Member Author

J-N-K commented Apr 4, 2023

GroupItemStateEvent never existed, even though it was in the JavaDoc. This PR introduced the updated event for groups (which was not reported before).

J-N-K pushed a commit to openhab/openhab-distro that referenced this pull request Apr 4, 2023
…atedEvent` (#1515)

* Add logger configuration for `ItemStateUpdatedEvent`
* Add logger config for `GroupStateUpdatedEvent`

Refs openhab/openhab-core#3141.

Signed-off-by: Florian Hotze <[email protected]>
@jimtng
Copy link
Contributor

jimtng commented Apr 4, 2023

I'm wondering why ItemStateUpdatedEvent extends ItemEvent and not ItemStateEvent. If it extended ItemStateEvent I think the jruby library would've been unaffected. But this is not a big problem and I've got a fix pending for it inside the jruby library.

The thing that puzzled me though is that previously (e.g. in openhab 3.4), I can send BusEvent.postUpdate("GroupItem", ON) and this will trigger an Item updated event for "GroupItem". You can test this using a MainUI rule in any language, then send the update event through karaf for example.

This stopped working in 4.0 recently. I don't know when exactly, but I know that it wasn't this PR that changed that behaviour. I looked around in the core code but even in 3.4 I can't actually find where this would have triggered, so the fact that it triggered in 3.4 is a mystery to me, but it did. We have a test to verify this behaviour since late last year and it has always passed until recently.

@lolodomo
Copy link
Contributor

lolodomo commented Apr 4, 2023

No impact on DSL rules?

@J-N-K
Copy link
Member Author

J-N-K commented Apr 4, 2023

I don't think so. They just pass the event payload and that is the same (except that you really get the item state instead of something that may or may not be the item state).

florian-h05 added a commit to florian-h05/openhab-js that referenced this pull request Apr 5, 2023
openhab/openhab-core#3141 changed the events
that are passed to the rule for **StateUpdateTriggers. This adds
support for the changed classnames whilst keeping backward
compatibility.

Signed-off-by: Florian Hotze <[email protected]>
florian-h05 added a commit to florian-h05/openhab-js that referenced this pull request Apr 5, 2023
openhab/openhab-core#3141 changed the events
that are passed to the rule for **StateUpdateTriggers. This adds
support for the changed classnames whilst keeping backward
compatibility.

Signed-off-by: Florian Hotze <[email protected]>
florian-h05 added a commit to openhab/openhab-js that referenced this pull request Apr 6, 2023
openhab/openhab-core#3141 changed the events that are passed to the rule for **StateUpdateTriggers. 
This adds support for the changed classnames whilst keeping backward compatibility.

Signed-off-by: Florian Hotze <[email protected]>
@openhab-bot
Copy link
Collaborator

This pull request has been mentioned on openHAB Community. There might be relevant details there:

https://community.openhab.org/t/openhab-4-0-milestone-discussion/145133/196

splatch pushed a commit to ConnectorIO/copybara-hab-core that referenced this pull request Jul 12, 2023
* Add (Group)ItemStateUpdatedEvent

Signed-off-by: Jan N. Klug <[email protected]>
GitOrigin-RevId: 4182980
@openhab-bot
Copy link
Collaborator

This pull request has been mentioned on openHAB Community. There might be relevant details there:

https://community.openhab.org/t/openhab-4-migration-faq/148144/18

@creativeprojects
Copy link

Is there any plan to deprecate the ItemStateEvent in the future?
I understand why the new one is better, but then I don't think the original event serves any purpose any more.

@J-N-K
Copy link
Member Author

J-N-K commented Aug 27, 2023

It still has an important purpose: it is issued to initiate an item state update. So the ItemStateEvent results in an update of the item, which is then communicated as ItemStateUpdatedEvent (and potentially an ItemStateChangedEvent).

@creativeprojects
Copy link

creativeprojects commented Aug 27, 2023

OK.
So if I understand well, it could mean that if a channel linked to the item state is down, we might only get ItemStateEvent but not a ItemStateUpdatedEvent in that case?

@spacemanspiff2007
Copy link
Contributor

No.
ItemStateEvent: Event that tells the item to update its state
ItemStateUpdatedEvent: Event that tells that an item has updated it's state

Do you see the subtle but important difference.

Let's take a DimmerItem:
ItemStateEvent: ON (Item should turn on)
ItemUpdatedEvent: 100% (Dimmer that is on has 100%)

@creativeprojects
Copy link

Right, I understand the difference, thank you for the explanation 👍🏻

I think we could update the documentation with this information: https://www.openhab.org/docs/developer/utils/events.html

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
API breaking enhancement An enhancement or new feature of the Core
Projects
None yet
Development

Successfully merging this pull request may close these issues.

RFC: Add an ItemStateUpdatedEvent