Skip to content

Commit e21e230

Browse files
committed
[Darwin] MTRDevice delegate should be notified when cache is primed with basic info
1 parent f67c1af commit e21e230

File tree

5 files changed

+99
-3
lines changed

5 files changed

+99
-3
lines changed

src/darwin/Framework/CHIP/MTRDevice.h

+5
Original file line numberDiff line numberDiff line change
@@ -403,6 +403,11 @@ MTR_EXTERN NSString * const MTRDataVersionKey MTR_NEWLY_AVAILABLE;
403403
*/
404404
- (void)deviceBecameActive:(MTRDevice *)device MTR_AVAILABLE(ios(16.4), macos(13.3), watchos(9.4), tvos(16.4));
405405

406+
/**
407+
* Notifies delegate when the device attribute cache has been primed with initial configuration data of the device
408+
*/
409+
- (void)deviceCachePrimed:(MTRDevice *)device MTR_NEWLY_AVAILABLE;
410+
406411
@end
407412

408413
@interface MTRDevice (Deprecated)

src/darwin/Framework/CHIP/MTRDevice.mm

+75
Original file line numberDiff line numberDiff line change
@@ -225,6 +225,7 @@ @implementation MTRDevice {
225225
#ifdef DEBUG
226226
NSUInteger _unitTestAttributesReportedSinceLastCheck;
227227
#endif
228+
BOOL _delegateDeviceCachePrimedCalled;
228229
}
229230

230231
- (instancetype)initWithNodeID:(NSNumber *)nodeID controller:(MTRDeviceController *)controller
@@ -506,6 +507,9 @@ - (void)setDelegate:(id<MTRDeviceDelegate>)delegate queue:(dispatch_queue_t)queu
506507
[self _setupSubscription];
507508
}
508509

510+
// Check if cache is already primed from storage
511+
[self _checkIfCacheIsPrimed];
512+
509513
os_unfair_lock_unlock(&self->_lock);
510514
}
511515

@@ -574,6 +578,29 @@ - (BOOL)_subscriptionAbleToReport
574578
return (delegate != nil) && (state == MTRDeviceStateReachable);
575579
}
576580

581+
- (BOOL)_callDelegateWithBlock:(void (^)(id<MTRDeviceDelegate>))block
582+
{
583+
os_unfair_lock_assert_owner(&self->_lock);
584+
id<MTRDeviceDelegate> delegate = _weakDelegate.strongObject;
585+
if (delegate) {
586+
dispatch_async(_delegateQueue, ^{
587+
block(delegate);
588+
});
589+
return YES;
590+
}
591+
return NO;
592+
}
593+
594+
- (void)_callDelegateDeviceCachePrimed
595+
{
596+
os_unfair_lock_assert_owner(&self->_lock);
597+
_delegateDeviceCachePrimedCalled = [self _callDelegateWithBlock:^(id<MTRDeviceDelegate> delegate) {
598+
if ([delegate respondsToSelector:@selector(deviceCachePrimed:)]) {
599+
[delegate deviceCachePrimed:self];
600+
}
601+
}];
602+
}
603+
577604
// assume lock is held
578605
- (void)_changeState:(MTRDeviceState)state
579606
{
@@ -741,6 +768,12 @@ - (void)_handleReportEnd
741768
_receivingReport = NO;
742769
_receivingPrimingReport = NO;
743770
_estimatedStartTimeFromGeneralDiagnosticsUpTime = nil;
771+
772+
// First subscription report is priming report
773+
if (!_delegateDeviceCachePrimedCalled) {
774+
[self _callDelegateDeviceCachePrimed];
775+
}
776+
744777
// For unit testing only
745778
#ifdef DEBUG
746779
id delegate = _weakDelegate.strongObject;
@@ -2147,6 +2180,48 @@ - (void)_removeExpectedValueForAttributePath:(MTRAttributePath *)attributePath e
21472180
}
21482181
}
21492182

2183+
// This method checks if there is a need to inform delegate that the attribute cache has been "primed"
2184+
// - The delegate callback is only called once
2185+
- (void)_checkIfCacheIsPrimed
2186+
{
2187+
os_unfair_lock_assert_owner(&self->_lock);
2188+
2189+
// Only send the callback once per lifetime of MTRDevice
2190+
if (_delegateDeviceCachePrimedCalled) {
2191+
return;
2192+
}
2193+
2194+
// Check if root node descriptor exists
2195+
NSDictionary * rootDescriptorPartsListDataValue = _readCache[[MTRAttributePath attributePathWithEndpointID:@(0) clusterID:@(MTRClusterIDTypeDescriptorID) attributeID:@(MTRAttributeIDTypeClusterDescriptorAttributePartsListID)]];
2196+
if (!rootDescriptorPartsListDataValue || ![MTRArrayValueType isEqualToString:rootDescriptorPartsListDataValue[MTRTypeKey]]) {
2197+
return;
2198+
}
2199+
NSArray * partsList = rootDescriptorPartsListDataValue[MTRValueKey];
2200+
if (![partsList isKindOfClass:[NSArray class]] || !partsList.count) {
2201+
MTR_LOG_ERROR("%@ unexpected type %@ for parts list %@", self, [partsList class], partsList);
2202+
return;
2203+
}
2204+
2205+
// Check if we have cached descriptor clusters for each listed endpoint
2206+
for (NSDictionary * endpointDataValue in partsList) {
2207+
if (![MTRUnsignedIntegerValueType isEqual:endpointDataValue[MTRTypeKey]]) {
2208+
MTR_LOG_ERROR("%@ unexpected type for parts list item %@", self, endpointDataValue);
2209+
continue;
2210+
}
2211+
NSNumber * endpoint = endpointDataValue[MTRValueKey];
2212+
if (![endpoint isKindOfClass:[NSNumber class]]) {
2213+
MTR_LOG_ERROR("%@ unexpected type for parts list item %@", self, endpointDataValue);
2214+
continue;
2215+
}
2216+
NSDictionary * descriptorDeviceTypeListDataValue = _readCache[[MTRAttributePath attributePathWithEndpointID:endpoint clusterID:@(MTRClusterIDTypeDescriptorID) attributeID:@(MTRAttributeIDTypeClusterDescriptorAttributeDeviceTypeListID)]];
2217+
if (![MTRArrayValueType isEqualToString:descriptorDeviceTypeListDataValue[MTRTypeKey]] || !descriptorDeviceTypeListDataValue[MTRValueKey]) {
2218+
return;
2219+
}
2220+
}
2221+
2222+
[self _callDelegateDeviceCachePrimed];
2223+
}
2224+
21502225
- (MTRBaseDevice *)newBaseDevice
21512226
{
21522227
return [MTRBaseDevice deviceWithNodeID:self.nodeID controller:self.deviceController];

src/darwin/Framework/CHIPTests/MTRDeviceTests.m

+11-3
Original file line numberDiff line numberDiff line change
@@ -2843,7 +2843,7 @@ - (void)test031_MTRDeviceAttributeCacheLocalTestStorage
28432843
{
28442844
dispatch_queue_t queue = dispatch_get_main_queue();
28452845

2846-
// First start with clean slate and
2846+
// First start with clean slate by removing the MTRDevice and clearing the persisted cache
28472847
__auto_type * device = [MTRDevice deviceWithNodeID:@(kDeviceId) controller:sController];
28482848
[sController removeDevice:device];
28492849
[sController.controllerDataStore clearAllStoredAttributes];
@@ -2853,16 +2853,20 @@ - (void)test031_MTRDeviceAttributeCacheLocalTestStorage
28532853
// Now recreate device and get subscription primed
28542854
device = [MTRDevice deviceWithNodeID:@(kDeviceId) controller:sController];
28552855
XCTestExpectation * gotReportsExpectation = [self expectationWithDescription:@"Attribute and Event reports have been received"];
2856+
XCTestExpectation * gotDeviceCachePrimed = [self expectationWithDescription:@"Device cache primed for the first time"];
28562857
__auto_type * delegate = [[MTRDeviceTestDelegate alloc] init];
28572858
__weak __auto_type weakDelegate = delegate;
28582859
delegate.onReportEnd = ^{
28592860
[gotReportsExpectation fulfill];
28602861
__strong __auto_type strongDelegate = weakDelegate;
28612862
strongDelegate.onReportEnd = nil;
28622863
};
2864+
delegate.onDeviceCachePrimed = ^{
2865+
[gotDeviceCachePrimed fulfill];
2866+
};
28632867
[device setDelegate:delegate queue:queue];
28642868

2865-
[self waitForExpectations:@[ gotReportsExpectation ] timeout:60];
2869+
[self waitForExpectations:@[ gotReportsExpectation, gotDeviceCachePrimed ] timeout:60];
28662870

28672871
NSUInteger attributesReportedWithFirstSubscription = [device unitTestAttributesReportedSinceLastCheck];
28682872

@@ -2874,14 +2878,18 @@ - (void)test031_MTRDeviceAttributeCacheLocalTestStorage
28742878
device = [MTRDevice deviceWithNodeID:@(kDeviceId) controller:sController];
28752879

28762880
XCTestExpectation * resubGotReportsExpectation = [self expectationWithDescription:@"Attribute and Event reports have been received for resubscription"];
2881+
XCTestExpectation * gotDeviceCachePrimedAgain = [self expectationWithDescription:@"Device cache primed upon load from persistence"];
28772882
delegate.onReportEnd = ^{
28782883
[resubGotReportsExpectation fulfill];
28792884
__strong __auto_type strongDelegate = weakDelegate;
28802885
strongDelegate.onReportEnd = nil;
28812886
};
2887+
delegate.onDeviceCachePrimed = ^{
2888+
[gotDeviceCachePrimedAgain fulfill];
2889+
};
28822890
[device setDelegate:delegate queue:queue];
28832891

2884-
[self waitForExpectations:@[ resubGotReportsExpectation ] timeout:60];
2892+
[self waitForExpectations:@[ gotDeviceCachePrimedAgain, resubGotReportsExpectation ] timeout:60];
28852893

28862894
NSUInteger attributesReportedWithSecondSubscription = [device unitTestAttributesReportedSinceLastCheck];
28872895

src/darwin/Framework/CHIPTests/TestHelpers/MTRDeviceTestDelegate.h

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ typedef void (^MTRDeviceTestDelegateDataHandler)(NSArray<NSDictionary<NSString *
2727
@property (nonatomic, nullable) MTRDeviceTestDelegateDataHandler onAttributeDataReceived;
2828
@property (nonatomic, nullable) MTRDeviceTestDelegateDataHandler onEventDataReceived;
2929
@property (nonatomic, nullable) dispatch_block_t onReportEnd;
30+
@property (nonatomic, nullable) dispatch_block_t onDeviceCachePrimed;
3031
@end
3132

3233
NS_ASSUME_NONNULL_END

src/darwin/Framework/CHIPTests/TestHelpers/MTRDeviceTestDelegate.m

+7
Original file line numberDiff line numberDiff line change
@@ -53,4 +53,11 @@ - (NSNumber *)unitTestMaxIntervalOverrideForSubscription:(MTRDevice *)device
5353
return @(2); // seconds
5454
}
5555

56+
- (void)deviceCachePrimed:(MTRDevice *)device
57+
{
58+
if (self.onDeviceCachePrimed != nil) {
59+
self.onDeviceCachePrimed();
60+
}
61+
}
62+
5663
@end

0 commit comments

Comments
 (0)