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

keep subscriptions up to date when accessories are replaced #172

Merged
merged 2 commits into from
Nov 23, 2022
Merged
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
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@
* @author Andy Lintner
*/
public interface Characteristic {
/** @return The UUID type for this characteristic. */
String getType();

/**
* Adds an attribute to the passed JsonObjectBuilder named "value" with the current value of the
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,12 @@ public BaseCharacteristic(
this.unsubscriber = unsubscriber;
}

@Override
/** {@inheritDoc} */
public String getType() {
return type;
}

@Override
/** {@inheritDoc} */
public final CompletableFuture<JsonObject> toJson(int iid) {
Expand Down
10 changes: 7 additions & 3 deletions src/main/java/io/github/hapjava/server/impl/HomekitRegistry.java
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

import io.github.hapjava.accessories.HomekitAccessory;
import io.github.hapjava.characteristics.Characteristic;
import io.github.hapjava.server.impl.connections.SubscriptionManager;
import io.github.hapjava.services.Service;
import io.github.hapjava.services.impl.AccessoryInformationService;
import java.util.ArrayList;
Expand All @@ -19,14 +20,16 @@ public class HomekitRegistry {
private static final Logger logger = LoggerFactory.getLogger(HomekitRegistry.class);

private final String label;
private final SubscriptionManager subscriptions;
private final Map<Integer, HomekitAccessory> accessories;
private final Map<HomekitAccessory, Map<Integer, Service>> services = new HashMap<>();
private final Map<HomekitAccessory, Map<Integer, Characteristic>> characteristics =
new HashMap<>();
private boolean isAllowUnauthenticatedRequests = false;

public HomekitRegistry(String label) {
public HomekitRegistry(String label, SubscriptionManager subscriptions) {
this.label = label;
this.subscriptions = subscriptions;
this.accessories = new ConcurrentHashMap<>();
reset();
}
Expand Down Expand Up @@ -61,6 +64,7 @@ public synchronized void reset() {
services.put(accessory, newServicesByInterfaceId);
characteristics.put(accessory, newCharacteristicsByInterfaceId);
}
subscriptions.resync(this);
}

public String getLabel() {
Expand All @@ -87,8 +91,8 @@ public void add(HomekitAccessory accessory) {
accessories.put(accessory.getId(), accessory);
}

public void remove(HomekitAccessory accessory) {
accessories.remove(accessory.getId());
public boolean remove(HomekitAccessory accessory) {
return accessories.remove(accessory.getId()) != null;
}

public boolean isAllowUnauthenticatedRequests() {
Expand Down
52 changes: 37 additions & 15 deletions src/main/java/io/github/hapjava/server/impl/HomekitRoot.java
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,8 @@ public class HomekitRoot {
private final SubscriptionManager subscriptions = new SubscriptionManager();
private boolean started = false;
private int configurationIndex = 1;
private int nestedBatches = 0;
private boolean madeChanges = false;

HomekitRoot(
String label, HomekitWebHandler webHandler, InetAddress host, HomekitAuthInfo authInfo)
Expand Down Expand Up @@ -65,7 +67,7 @@ public class HomekitRoot {
this.authInfo = authInfo;
this.label = label;
this.category = category;
this.registry = new HomekitRegistry(label);
this.registry = new HomekitRegistry(label, subscriptions);
}

HomekitRoot(
Expand All @@ -83,11 +85,27 @@ public class HomekitRoot {
this(
label, DEFAULT_ACCESSORY_CATEGORY, webHandler, authInfo, new JmdnsHomekitAdvertiser(jmdns));
}

/**
* Add an accessory to be handled and advertised by this root. Any existing HomeKit connections
* will be terminated to allow the clients to reconnect and see the updated accessory list. When
* using this for a bridge, the ID of the accessory must be greater than 1, as that ID is reserved
* for the Bridge itself.
* Begin a batch update of accessories.
*
* <p>After calling this, you can call addAccessory() and removeAccessory() multiple times without
* causing HAP-Java to re-publishing the metadata to HomeKit. You'll need to call
* completeUpdateBatch in order to publish all accumulated changes.
*/
public synchronized void batchUpdate() {
if (this.nestedBatches == 0) madeChanges = false;
++this.nestedBatches;
}

/** Publish accumulated accessory changes since batchUpdate() was called. */
public synchronized void completeUpdateBatch() {
if (--this.nestedBatches == 0 && madeChanges) registry.reset();
}

/**
* Add an accessory to be handled and advertised by this root. When using this for a bridge, the
* ID of the accessory must be greater than 1, as that ID is reserved for the Bridge itself.
*
* @param accessory to advertise and handle.
*/
Expand All @@ -110,25 +128,28 @@ void addAccessorySkipRangeCheck(HomekitAccessory accessory) {
if (logger.isTraceEnabled()) {
accessory.getName().thenAccept(name -> logger.trace("Added accessory {}", name));
}
if (started) {
madeChanges = true;
if (started && nestedBatches == 0) {
registry.reset();
}
}

/**
* Removes an accessory from being handled or advertised by this root. Any existing HomeKit
* connections will be terminated to allow the clients to reconnect and see the updated accessory
* list.
* Removes an accessory from being handled or advertised by this root.
*
* @param accessory accessory to cease advertising and handling
*/
public void removeAccessory(HomekitAccessory accessory) {
this.registry.remove(accessory);
if (logger.isTraceEnabled()) {
accessory.getName().thenAccept(name -> logger.trace("Removed accessory {}", name));
}
if (started) {
registry.reset();
if (this.registry.remove(accessory)) {
if (logger.isTraceEnabled()) {
accessory.getName().thenAccept(name -> logger.trace("Removed accessory {}", name));
}
madeChanges = true;
if (started && nestedBatches == 0) {
registry.reset();
}
} else {
accessory.getName().thenAccept(name -> logger.warn("Could not remove accessory {}", name));
}
}

Expand All @@ -140,6 +161,7 @@ public void removeAccessory(HomekitAccessory accessory) {
*/
public void start() {
started = true;
madeChanges = false;
registry.reset();
webHandler
.start(
Expand Down
Loading