Skip to content

Commit

Permalink
[Darwin] Fix flaky subscription pool test
Browse files Browse the repository at this point in the history
  • Loading branch information
jtung-apple committed Sep 16, 2024
1 parent 4f35706 commit 25773eb
Show file tree
Hide file tree
Showing 2 changed files with 21 additions and 17 deletions.
13 changes: 10 additions & 3 deletions src/darwin/Framework/CHIP/MTRDevice_Concrete.mm
Original file line number Diff line number Diff line change
Expand Up @@ -1167,8 +1167,7 @@ - (void)_scheduleSubscriptionPoolWork:(dispatch_block_t)workBlock inNanoseconds:
return;
}

// Wait the required amount of time, then put it in the subscription pool to wait additionally for a spot, if needed
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, inNanoseconds), self.queue, ^{
dispatch_block_t workBlockToQueue = ^{
// In the case where a resubscription triggering event happened and already established, running the work block should result in a no-op
MTRAsyncWorkItem * workItem = [[MTRAsyncWorkItem alloc] initWithQueue:self.queue];
[workItem setReadyHandler:^(id _Nonnull context, NSInteger retryCount, MTRAsyncWorkCompletionBlock _Nonnull completion) {
Expand Down Expand Up @@ -1199,7 +1198,15 @@ - (void)_scheduleSubscriptionPoolWork:(dispatch_block_t)workBlock inNanoseconds:
}];
[self->_deviceController.concurrentSubscriptionPool enqueueWorkItem:workItem description:description];
MTR_LOG("%@ - enqueued in the subscription pool", self);
});
};

if (inNanoseconds > 0) {
// Wait the required amount of time, then put it in the subscription pool to wait additionally for a spot
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, inNanoseconds), self.queue, workBlockToQueue);
} else {
// Put in subscription pool directly if there is no wait time
workBlockToQueue();
}
}

- (void)_handleResubscriptionNeededWithDelay:(NSNumber *)resubscriptionDelayMs
Expand Down
25 changes: 11 additions & 14 deletions src/darwin/Framework/CHIPTests/MTRPerControllerStorageTests.m
Original file line number Diff line number Diff line change
Expand Up @@ -2588,20 +2588,17 @@ - (void)doTestSubscriptionPoolWithSize:(NSInteger)subscriptionPoolSize deviceOnb

// Create the base device to attempt to read from the 5th device
__auto_type * baseDeviceReadExpectation = [self expectationWithDescription:@"BaseDevice read"];
// Dispatch async to get around XCTest, so that this runs after the above devices queue their subscriptions
dispatch_async(queue, ^{
__auto_type * baseDevice = [MTRBaseDevice deviceWithNodeID:@(105) controller:controller];
__auto_type * onOffCluster = [[MTRBaseClusterOnOff alloc] initWithDevice:baseDevice endpointID:@(1) queue:queue];
[onOffCluster readAttributeOnOffWithCompletion:^(NSNumber * value, NSError * _Nullable error) {
XCTAssertNil(error);
// We expect the device to be off.
XCTAssertEqualObjects(value, @(0));
[baseDeviceReadExpectation fulfill];
os_unfair_lock_lock(&counterLock);
baseDeviceReadCompleted = YES;
os_unfair_lock_unlock(&counterLock);
}];
});
__auto_type * baseDevice = [MTRBaseDevice deviceWithNodeID:@(105) controller:controller];
__auto_type * onOffCluster = [[MTRBaseClusterOnOff alloc] initWithDevice:baseDevice endpointID:@(1) queue:queue];
[onOffCluster readAttributeOnOffWithCompletion:^(NSNumber * value, NSError * _Nullable error) {
XCTAssertNil(error);
// We expect the device to be off.
XCTAssertEqualObjects(value, @(0));
[baseDeviceReadExpectation fulfill];
os_unfair_lock_lock(&counterLock);
baseDeviceReadCompleted = YES;
os_unfair_lock_unlock(&counterLock);
}];

// Make the wait time depend on pool size and device count (can expand number of devices in the future)
NSArray * expectationsToWait = [subscriptionExpectations.allValues arrayByAddingObject:baseDeviceReadExpectation];
Expand Down

0 comments on commit 25773eb

Please sign in to comment.