Skip to content

Commit 11e2592

Browse files
Add a Matter.framework API for invoking multiple commands. (project-chip#37398)
* Add a Matter.framework API for invoking multiple commands. The API allows grouping the commands, so that later groups don't get invoked if anything in an earlier group fails. The changes to _invokeCommandWithEndpointID in MTRDevice_XPC.mm are fixing a pre-existing bug where it was not dispatching its completions to the right queue. * Add the new selectors/types to the XPC allow-list. * Add NSError to completion for invokeCommands:...
1 parent 13f68b9 commit 11e2592

11 files changed

+745
-31
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
/**
2+
* Copyright (c) 2025 Project CHIP Authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
#import <Foundation/Foundation.h>
18+
#import <Matter/MTRBaseDevice.h> // For MTRCommandPath
19+
20+
NS_ASSUME_NONNULL_BEGIN
21+
22+
/**
23+
* An object representing a single command to be invoked and the expected
24+
* result of invoking it.
25+
*/
26+
MTR_AVAILABLE(ios(18.4), macos(15.4), watchos(11.4), tvos(18.4))
27+
@interface MTRCommandWithExpectedResult : NSObject <NSCopying, NSSecureCoding>
28+
29+
/**
30+
* The path of the command being invoked.
31+
*/
32+
@property (nonatomic, retain) MTRCommandPath * path;
33+
34+
/**
35+
* The command fields to pass for the command invoke. nil if this command does
36+
* not have any fields. If not nil, this should be a data-value dictionary of
37+
* MTRStructureValueType.
38+
*/
39+
@property (nonatomic, retain, nullable) NSDictionary<NSString *, id> * commandFields;
40+
41+
/**
42+
* The expected result of invoking the command.
43+
*
44+
* If this is nil, that indicates that the invoke is considered successful if it
45+
* does not result in an error status response.
46+
*
47+
* If this is is not nil, then invoke is considered successful if
48+
* it results in a data response and for each entry in the provided
49+
* expectedResult the field whose field ID matches the key of the entry has a
50+
* value that equals the value of the entry. Values of entries are data-value
51+
* dictionaries.
52+
*/
53+
@property (nonatomic, copy, nullable) NSDictionary<NSNumber *, NSDictionary<NSString *, id> *> * expectedResult;
54+
55+
- (instancetype)initWithPath:(MTRCommandPath *)path
56+
commandFields:(nullable NSDictionary<NSString *, id> *)commandFields
57+
expectedResult:(nullable NSDictionary<NSNumber *, NSDictionary<NSString *, id> *> *)expectedResult;
58+
59+
@end
60+
61+
NS_ASSUME_NONNULL_END
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,119 @@
1+
/**
2+
* Copyright (c) 2025 Project CHIP Authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
#import "MTRDeviceDataValidation.h"
18+
#import "MTRLogging_Internal.h"
19+
#import <Matter/Matter.h>
20+
21+
@implementation MTRCommandWithExpectedResult
22+
- (instancetype)initWithPath:(MTRCommandPath *)path
23+
commandFields:(nullable NSDictionary<NSString *, id> *)commandFields
24+
expectedResult:(nullable NSDictionary<NSNumber *, NSDictionary<NSString *, id> *> *)expectedResult
25+
{
26+
if (self = [super init]) {
27+
self.path = path;
28+
self.commandFields = commandFields;
29+
self.expectedResult = expectedResult;
30+
}
31+
32+
return self;
33+
}
34+
35+
- (id)copyWithZone:(NSZone *)zone
36+
{
37+
return [[MTRCommandWithExpectedResult alloc] initWithPath:self.path commandFields:self.commandFields expectedResult:self.expectedResult];
38+
}
39+
40+
- (NSString *)description
41+
{
42+
return [NSString stringWithFormat:@"<%@: %p, path: %@, fields: %@, expectedResult: %@", NSStringFromClass(self.class), self, self.path, self.commandFields, self.expectedResult];
43+
}
44+
45+
#pragma mark - MTRCommandWithExpectedResult NSSecureCoding implementation
46+
47+
static NSString * const sPathKey = @"pathKey";
48+
static NSString * const sFieldsKey = @"fieldsKey";
49+
static NSString * const sExpectedResultKey = @"expectedResultKey";
50+
51+
+ (BOOL)supportsSecureCoding
52+
{
53+
return YES;
54+
}
55+
56+
- (nullable instancetype)initWithCoder:(NSCoder *)decoder
57+
{
58+
self = [super init];
59+
if (self == nil) {
60+
return nil;
61+
}
62+
63+
_path = [decoder decodeObjectOfClass:MTRCommandPath.class forKey:sPathKey];
64+
if (!_path || ![_path isKindOfClass:MTRCommandPath.class]) {
65+
MTR_LOG_ERROR("MTRCommandWithExpectedResult decoded %@ for endpoint, not MTRCommandPath.", _path);
66+
return nil;
67+
}
68+
69+
_commandFields = [decoder decodeObjectOfClass:NSDictionary.class forKey:sFieldsKey];
70+
if (_commandFields) {
71+
if (![_commandFields isKindOfClass:NSDictionary.class]) {
72+
MTR_LOG_ERROR("MTRCommandWithExpectedResult decoded %@ for commandFields, not NSDictionary.", _commandFields);
73+
return nil;
74+
}
75+
76+
if (!MTRDataValueDictionaryIsWellFormed(_commandFields) || ![MTRStructureValueType isEqual:_commandFields[MTRTypeKey]]) {
77+
MTR_LOG_ERROR("MTRCommandWithExpectedResult decoded %@ for commandFields, not a structure-typed data-value dictionary.", _commandFields);
78+
return nil;
79+
}
80+
}
81+
82+
_expectedResult = [decoder decodeObjectOfClass:NSDictionary.class forKey:sExpectedResultKey];
83+
if (_expectedResult) {
84+
if (![_expectedResult isKindOfClass:NSDictionary.class]) {
85+
MTR_LOG_ERROR("MTRCommandWithExpectedResult decoded %@ for expectedResult, not NSDictionary.", _expectedResult);
86+
return nil;
87+
}
88+
89+
for (id key in _expectedResult) {
90+
if (![key isKindOfClass:NSNumber.class]) {
91+
MTR_LOG_ERROR("MTRCommandWithExpectedResult decoded key %@ in expectedResult", key);
92+
return nil;
93+
}
94+
95+
if (![_expectedResult[key] isKindOfClass:NSDictionary.class] || !MTRDataValueDictionaryIsWellFormed(_expectedResult[key])) {
96+
MTR_LOG_ERROR("MTRCommandWithExpectedResult decoded value %@ for key %@ in expectedResult", _expectedResult[key], key);
97+
return nil;
98+
}
99+
}
100+
}
101+
102+
return self;
103+
}
104+
105+
- (void)encodeWithCoder:(NSCoder *)coder
106+
{
107+
// In theory path is not nullable, but we don't really enforce that in init.
108+
if (self.path) {
109+
[coder encodeObject:self.path forKey:sPathKey];
110+
}
111+
if (self.commandFields) {
112+
[coder encodeObject:self.commandFields forKey:sFieldsKey];
113+
}
114+
if (self.expectedResult) {
115+
[coder encodeObject:self.expectedResult forKey:sExpectedResultKey];
116+
}
117+
}
118+
119+
@end

src/darwin/Framework/CHIP/MTRDevice.h

+27-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/**
22
*
3-
* Copyright (c) 2022-2023 Project CHIP Authors
3+
* Copyright (c) 2022-2025 Project CHIP Authors
44
*
55
* Licensed under the Apache License, Version 2.0 (the "License");
66
* you may not use this file except in compliance with the License.
@@ -19,6 +19,7 @@
1919
#import <Matter/MTRAttributeValueWaiter.h>
2020
#import <Matter/MTRBaseClusters.h>
2121
#import <Matter/MTRBaseDevice.h>
22+
#import <Matter/MTRCommandWithExpectedResult.h>
2223
#import <Matter/MTRDefines.h>
2324

2425
NS_ASSUME_NONNULL_BEGIN
@@ -294,6 +295,31 @@ MTR_AVAILABLE(ios(16.1), macos(13.0), watchos(9.1), tvos(16.1))
294295
completion:(MTRDeviceResponseHandler)completion
295296
MTR_AVAILABLE(ios(16.4), macos(13.3), watchos(9.4), tvos(16.4));
296297

298+
/**
299+
* Invoke one or more groups of commands.
300+
*
301+
* For any given group, if any command in any preceding group failed, the group
302+
* will be skipped. If all commands in all preceding groups succeeded, the
303+
* commands within the group will be invoked, with no ordering guarantees within
304+
* that group.
305+
*
306+
* Results from all commands that were invoked will be passed to the provided
307+
* completion as an array of response-value dictionaries. Each of these will
308+
* have the command path of the command (see MTRCommandPathKey) and one of three
309+
* things:
310+
*
311+
* 1) No other fields, indicating that the command invoke returned a succcess
312+
* status.
313+
* 2) A field for MTRErrorKey, indicating that the invoke returned a failure
314+
* status (which is the value of the field).
315+
* 3) A field for MTRDataKey, indicating that the invoke returned a data
316+
* response. In this case the data-value representing the response will be
317+
* the value of this field.
318+
*/
319+
- (void)invokeCommands:(NSArray<NSArray<MTRCommandWithExpectedResult *> *> *)commands
320+
queue:(dispatch_queue_t)queue
321+
completion:(MTRDeviceResponseHandler)completion MTR_AVAILABLE(ios(18.4), macos(15.4), watchos(11.4), tvos(18.4));
322+
297323
/**
298324
* Open a commissioning window on the device.
299325
*

src/darwin/Framework/CHIP/MTRDevice.mm

+10
Original file line numberDiff line numberDiff line change
@@ -523,6 +523,16 @@ - (void)_invokeKnownCommandWithEndpointID:(NSNumber *)endpointID
523523
completion:responseHandler];
524524
}
525525

526+
- (void)invokeCommands:(NSArray<NSArray<MTRCommandWithExpectedResult *> *> *)commands
527+
queue:(dispatch_queue_t)queue
528+
completion:(MTRDeviceResponseHandler)completion
529+
{
530+
MTR_ABSTRACT_METHOD();
531+
dispatch_async(queue, ^{
532+
completion(nil, [MTRError errorForCHIPErrorCode:CHIP_ERROR_INCORRECT_STATE]);
533+
});
534+
}
535+
526536
- (void)openCommissioningWindowWithSetupPasscode:(NSNumber *)setupPasscode
527537
discriminator:(NSNumber *)discriminator
528538
duration:(NSNumber *)duration

src/darwin/Framework/CHIP/MTRDeviceController_XPC.mm

+16-1
Original file line numberDiff line numberDiff line change
@@ -137,14 +137,29 @@ - (NSXPCInterface *)_interfaceForServerProtocol
137137
NSMutableSet * allowedClasses = [MTRDeviceController_XPC _allowedClasses];
138138
[allowedClasses addObjectsFromArray:@[
139139
[MTRCommandPath class],
140-
[MTRAttributePath class],
141140
]];
142141

143142
[interface setClasses:allowedClasses
144143
forSelector:@selector(deviceController:nodeID:invokeCommandWithEndpointID:clusterID:commandID:commandFields:expectedValues:expectedValueInterval:timedInvokeTimeout:serverSideProcessingTimeout:completion:)
145144
argumentIndex:0
146145
ofReply:YES];
147146

147+
// invokeCommands has the same reply types as invokeCommandWithEndpointID.
148+
[interface setClasses:allowedClasses
149+
forSelector:@selector(deviceController:nodeID:invokeCommands:completion:)
150+
argumentIndex:0
151+
ofReply:YES];
152+
153+
// invokeCommands gets handed MTRCommandWithExpectedResult (which includes
154+
// MTRCommandPath, which is already in allowedClasses).
155+
[allowedClasses addObjectsFromArray:@[
156+
[MTRCommandWithExpectedResult class],
157+
]];
158+
[interface setClasses:allowedClasses
159+
forSelector:@selector(deviceController:nodeID:invokeCommands:completion:)
160+
argumentIndex:2
161+
ofReply:NO];
162+
148163
// readAttributePaths: gets handed an array of MTRAttributeRequestPath.
149164
allowedClasses = [MTRDeviceController_XPC _allowedClasses];
150165
[allowedClasses addObjectsFromArray:@[

0 commit comments

Comments
 (0)