Skip to content

Commit 27b7630

Browse files
committed
Changed logic to only report once, and also improve documentation in comments
1 parent e21e230 commit 27b7630

File tree

3 files changed

+35
-25
lines changed

3 files changed

+35
-25
lines changed

src/darwin/Framework/CHIP/MTRDevice.h

+4
Original file line numberDiff line numberDiff line change
@@ -405,6 +405,10 @@ MTR_EXTERN NSString * const MTRDataVersionKey MTR_NEWLY_AVAILABLE;
405405

406406
/**
407407
* Notifies delegate when the device attribute cache has been primed with initial configuration data of the device
408+
*
409+
* This is called when the MTRDevice object goes from not knowing the device to having cached the first attribute reports that include basic mandatory information, e.g. Descriptor clusters.
410+
*
411+
* The intention is that after this is called, the client should be able to call read for mandatory attributes and likely expect non-nil values.
408412
*/
409413
- (void)deviceCachePrimed:(MTRDevice *)device MTR_NEWLY_AVAILABLE;
410414

src/darwin/Framework/CHIP/MTRDevice.mm

+25-22
Original file line numberDiff line numberDiff line change
@@ -503,13 +503,15 @@ - (void)setDelegate:(id<MTRDeviceDelegate>)delegate queue:(dispatch_queue_t)queu
503503
_weakDelegate = [MTRWeakReference weakReferenceWithObject:delegate];
504504
_delegateQueue = queue;
505505

506+
// If Check if cache is already primed and client hasn't been informed yet, call the -deviceCachePrimed: callback
507+
if (!_delegateDeviceCachePrimedCalled && [self _isCachePrimedWithInitialConfigurationData]) {
508+
[self _callDelegateDeviceCachePrimed];
509+
}
510+
506511
if (setUpSubscription) {
507512
[self _setupSubscription];
508513
}
509514

510-
// Check if cache is already primed from storage
511-
[self _checkIfCacheIsPrimed];
512-
513515
os_unfair_lock_unlock(&self->_lock);
514516
}
515517

@@ -638,6 +640,11 @@ - (void)_handleSubscriptionEstablished
638640
// reset subscription attempt wait time when subscription succeeds
639641
_lastSubscriptionAttemptWait = 0;
640642

643+
// As subscription is established, check if the delegate needs to be informed
644+
if (!_delegateDeviceCachePrimedCalled) {
645+
[self _callDelegateDeviceCachePrimed];
646+
}
647+
641648
[self _changeState:MTRDeviceStateReachable];
642649

643650
os_unfair_lock_unlock(&self->_lock);
@@ -769,11 +776,6 @@ - (void)_handleReportEnd
769776
_receivingPrimingReport = NO;
770777
_estimatedStartTimeFromGeneralDiagnosticsUpTime = nil;
771778

772-
// First subscription report is priming report
773-
if (!_delegateDeviceCachePrimedCalled) {
774-
[self _callDelegateDeviceCachePrimed];
775-
}
776-
777779
// For unit testing only
778780
#ifdef DEBUG
779781
id delegate = _weakDelegate.strongObject;
@@ -1993,17 +1995,24 @@ - (NSArray *)_getAttributesToReportWithReportedValues:(NSArray<NSDictionary<NSSt
19931995

19941996
- (void)setAttributeValues:(NSArray<NSDictionary *> *)attributeValues reportChanges:(BOOL)reportChanges
19951997
{
1998+
os_unfair_lock_lock(&self->_lock);
1999+
19962000
if (reportChanges) {
1997-
[self _handleAttributeReport:attributeValues];
2001+
[self _reportAttributes:[self _getAttributesToReportWithReportedValues:attributeValues]];
19982002
} else {
1999-
os_unfair_lock_lock(&self->_lock);
20002003
for (NSDictionary * responseValue in attributeValues) {
20012004
MTRAttributePath * path = responseValue[MTRAttributePathKey];
20022005
NSDictionary * dataValue = responseValue[MTRDataKey];
20032006
_readCache[path] = dataValue;
20042007
}
2005-
os_unfair_lock_unlock(&self->_lock);
20062008
}
2009+
2010+
// If cache is set from storage and is primed with initial configuration data, then assume the client had beeen informed in the past, and mark that the callback has been called
2011+
if ([self _isCachePrimedWithInitialConfigurationData]) {
2012+
_delegateDeviceCachePrimedCalled = YES;
2013+
}
2014+
2015+
os_unfair_lock_unlock(&self->_lock);
20072016
}
20082017

20092018
// If value is non-nil, associate with expectedValueID
@@ -2181,25 +2190,19 @@ - (void)_removeExpectedValueForAttributePath:(MTRAttributePath *)attributePath e
21812190
}
21822191

21832192
// 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
2193+
- (BOOL)_isCachePrimedWithInitialConfigurationData
21862194
{
21872195
os_unfair_lock_assert_owner(&self->_lock);
21882196

2189-
// Only send the callback once per lifetime of MTRDevice
2190-
if (_delegateDeviceCachePrimedCalled) {
2191-
return;
2192-
}
2193-
21942197
// Check if root node descriptor exists
21952198
NSDictionary * rootDescriptorPartsListDataValue = _readCache[[MTRAttributePath attributePathWithEndpointID:@(0) clusterID:@(MTRClusterIDTypeDescriptorID) attributeID:@(MTRAttributeIDTypeClusterDescriptorAttributePartsListID)]];
21962199
if (!rootDescriptorPartsListDataValue || ![MTRArrayValueType isEqualToString:rootDescriptorPartsListDataValue[MTRTypeKey]]) {
2197-
return;
2200+
return NO;
21982201
}
21992202
NSArray * partsList = rootDescriptorPartsListDataValue[MTRValueKey];
22002203
if (![partsList isKindOfClass:[NSArray class]] || !partsList.count) {
22012204
MTR_LOG_ERROR("%@ unexpected type %@ for parts list %@", self, [partsList class], partsList);
2202-
return;
2205+
return NO;
22032206
}
22042207

22052208
// Check if we have cached descriptor clusters for each listed endpoint
@@ -2215,11 +2218,11 @@ - (void)_checkIfCacheIsPrimed
22152218
}
22162219
NSDictionary * descriptorDeviceTypeListDataValue = _readCache[[MTRAttributePath attributePathWithEndpointID:endpoint clusterID:@(MTRClusterIDTypeDescriptorID) attributeID:@(MTRAttributeIDTypeClusterDescriptorAttributeDeviceTypeListID)]];
22172220
if (![MTRArrayValueType isEqualToString:descriptorDeviceTypeListDataValue[MTRTypeKey]] || !descriptorDeviceTypeListDataValue[MTRValueKey]) {
2218-
return;
2221+
return NO;
22192222
}
22202223
}
22212224

2222-
[self _callDelegateDeviceCachePrimed];
2225+
return YES;
22232226
}
22242227

22252228
- (MTRBaseDevice *)newBaseDevice

src/darwin/Framework/CHIPTests/MTRDeviceTests.m

+6-3
Original file line numberDiff line numberDiff line change
@@ -2878,18 +2878,21 @@ - (void)test031_MTRDeviceAttributeCacheLocalTestStorage
28782878
device = [MTRDevice deviceWithNodeID:@(kDeviceId) controller:sController];
28792879

28802880
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"];
28822881
delegate.onReportEnd = ^{
28832882
[resubGotReportsExpectation fulfill];
28842883
__strong __auto_type strongDelegate = weakDelegate;
28852884
strongDelegate.onReportEnd = nil;
28862885
};
2886+
__block BOOL onDeviceCachePrimedCalled = NO;
28872887
delegate.onDeviceCachePrimed = ^{
2888-
[gotDeviceCachePrimedAgain fulfill];
2888+
onDeviceCachePrimedCalled = YES;
28892889
};
28902890
[device setDelegate:delegate queue:queue];
28912891

2892-
[self waitForExpectations:@[ gotDeviceCachePrimedAgain, resubGotReportsExpectation ] timeout:60];
2892+
[self waitForExpectations:@[ resubGotReportsExpectation ] timeout:60];
2893+
2894+
// Make sure that the new callback is only ever called once, the first time subscription was primed
2895+
XCTAssertFalse(onDeviceCachePrimedCalled);
28932896

28942897
NSUInteger attributesReportedWithSecondSubscription = [device unitTestAttributesReportedSinceLastCheck];
28952898

0 commit comments

Comments
 (0)