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

batch notifications when possible #66

Merged
merged 1 commit into from
Mar 9, 2019

Conversation

ccutrer
Copy link
Collaborator

@ccutrer ccutrer commented Feb 12, 2019

I've been running this for several days on my home installation, and it's definitely improved responsiveness when asking Siri to do something that involves multiple accessories.

often, HomeKit will will change multiple characteristics at once
(on and set brightness of a dimmable bulb, for example) or change
multiple items at once ("hey siri, close all the shades in the house").
instead of sending a separate event message for every characteristic
of every item that changed, batch them all up, and send as a single
message
@andylintner
Copy link
Collaborator

I like the idea here. I'm going to leave this pending until I have some time to dig into the implementation - should get to it in the next few days.

@ccutrer
Copy link
Collaborator Author

ccutrer commented Feb 12, 2019

👍 it's definitely a bit more complex than the other PRs. One thing that could probably use another pass is the thread safety. Like I said, I'm still a Java greenie, so I wasn't sure here. I used a ConcurrentMap for the pendingNotifications because that's what the other fields are. But if my understanding of synchronized it correct, all accesses to member fields are already protected by that, so we don't need to use that thread safe data structure. It made me wonder if this class started without using synchronized, and then ran in to problems, so additional thread safety measures were added without simplifying other code (see also the double-check on the set when adding a subscription).

@yfre
Copy link
Contributor

yfre commented Feb 22, 2019

as first: batch is very useful and actually required by HAP spec.

but im wondering whether we should reply with "EVENT" to a subscription immediately (as batch or not).

currently we have following sequence of messages (recorded from home app on mac os):

Receive GET /characteristics?id=1958124085.9
Send HTTP/1.1 200 OK. {"characteristics":[{"value":false,"aid":1958124085,"iid":9}]}

Receive PUT /characteristics {"characteristics":[{"aid":1958124085,"iid":9,"ev":true}]}
Send EVENT/1.0 200 OK{"characteristics":[{"aid":1958124085,"iid":9,"value":false}]}
Send HTTP/1.1 204 No Content

so,
it as first for value via characteristics and then subscriber to the same characteristic and we replay firt with EVENT and then with No Content.

according to HAP specification, we should first send "no Content" and
"When the value changes, the accessory sends .. EVENT". i.e. only in case of changes.

@ccutrer
Copy link
Collaborator Author

ccutrer commented Feb 22, 2019

This is an interesting point. Technically there's actually a race condition with how the controller gets the current value, then registers for notifications (the value could change between the initial GET, and the notification registration). While the probability is small, it goes up as the number of devices exposed on your bridge goes up, both from more devices, and for it taking longer to process these requests and responses. I read the spec when it says "When the value changes, the accessory sends .. EVENT" as emphasizing not to send EVENTs on a timer or something, but only on actual changes. Not necessarily that you shouldn't send an initial EVENT with the current state of the item.

I do agree that the first EVENT for a subscription shouldn't be sent until after we respond 204 to the request to register it.

@timcharper timcharper merged commit 661e6e0 into hap-java:master Mar 9, 2019
@ccutrer ccutrer deleted the aggregate_notifications branch October 18, 2022 15:53
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants