Skip to content

Commit b05ab2a

Browse files
Factor out support for running helper server apps in Darwin tests.
Switches MTROTAProviderTests to using the new factored-out bits. Other tests will be switched to those in a separate PR.
1 parent 7b2f729 commit b05ab2a

File tree

6 files changed

+271
-119
lines changed

6 files changed

+271
-119
lines changed

src/darwin/Framework/CHIPTests/MTROTAProviderTests.m

+13-118
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,10 @@
2020

2121
#import "MTRDeviceTestDelegate.h"
2222
#import "MTRErrorTestUtils.h"
23+
#import "MTRTestCase.h"
2324
#import "MTRTestKeys.h"
2425
#import "MTRTestResetCommissioneeHelper.h"
26+
#import "MTRTestServerAppRunner.h"
2527
#import "MTRTestStorage.h"
2628

2729
// system dependencies
@@ -73,35 +75,21 @@
7375

7476
static NSString * kUpdatedSoftwareVersionString_10 = @"10.0";
7577

76-
// kOtaRequestorBasePort gets the discriminator added to it to figure out the
77-
// port the ota-requestor app should be using. This ensures that apps with
78-
// distinct discriminators use distinct ports.
79-
static const uint16_t kOtaRequestorBasePort = 5542 - 1111;
80-
81-
@class MTROTARequestorAppRunner;
82-
83-
@interface MTROTAProviderTests : XCTestCase
84-
- (NSTask *)createTaskForPath:(NSString *)path;
78+
@interface MTROTAProviderTests : MTRTestCase
8579
- (NSString *)createImageFromRawImage:(NSString *)rawImage withVersion:(NSNumber *)version;
8680
- (MTRDevice *)commissionDeviceWithPayload:(NSString *)payloadString nodeID:(NSNumber *)nodeID;
87-
- (void)registerRunningRequestor:(MTROTARequestorAppRunner *)requestor;
8881
@end
8982

90-
static unsigned sAppRunnerIndex = 1;
91-
92-
@interface MTROTARequestorAppRunner : NSObject
83+
@interface MTROTARequestorAppRunner : MTRTestServerAppRunner
9384
@property (nonatomic, copy) NSString * downloadFilePath;
9485

9586
- (instancetype)initWithPayload:(NSString *)payload testcase:(MTROTAProviderTests *)testcase;
9687
- (MTRDevice *)commissionWithNodeID:(NSNumber *)nodeID;
9788
@end
9889

9990
@implementation MTROTARequestorAppRunner {
100-
unsigned _uniqueIndex;
101-
NSTask * _appTask;
10291
MTROTAProviderTests * _testcase;
10392
NSString * _payload;
104-
MTRDevice * commissionedDevice;
10593
}
10694

10795
- (MTRDevice *)commissionWithNodeID:(NSNumber *)nodeID
@@ -111,67 +99,24 @@ - (MTRDevice *)commissionWithNodeID:(NSNumber *)nodeID
11199

112100
- (instancetype)initWithPayload:(NSString *)payload testcase:(MTROTAProviderTests *)testcase
113101
{
114-
if (!(self = [super init])) {
115-
return nil;
116-
}
117-
118-
_uniqueIndex = sAppRunnerIndex++;
119-
_testcase = testcase;
120-
_payload = payload;
121-
_downloadFilePath = [NSString stringWithFormat:@"/tmp/chip-ota-requestor-downloaded-image%u", _uniqueIndex];
122-
123-
NSError * error;
124-
__auto_type * parsedPayload = [MTRSetupPayload setupPayloadWithOnboardingPayload:payload error:&error];
125-
XCTAssertNotNil(parsedPayload);
126-
XCTAssertNil(error);
127-
128-
XCTAssertFalse(parsedPayload.hasShortDiscriminator);
129-
130-
__auto_type * discriminator = parsedPayload.discriminator;
131-
132-
_appTask = [testcase createTaskForPath:@"out/debug/ota-requestor-app/chip-ota-requestor-app"];
133-
134-
__auto_type * arguments = @[
135-
@"--interface-id",
136-
@"-1",
137-
@"--secured-device-port",
138-
[NSString stringWithFormat:@"%u", kOtaRequestorBasePort + discriminator.unsignedShortValue],
139-
@"--discriminator",
140-
[NSString stringWithFormat:@"%u", discriminator.unsignedShortValue],
141-
@"--KVS",
142-
[NSString stringWithFormat:@"/tmp/chip-ota-requestor-kvs%u", _uniqueIndex],
102+
__auto_type * downloadFilePath = [NSString stringWithFormat:@"/tmp/chip-ota-requestor-downloaded-image%u", [MTRTestServerAppRunner nextUniqueIndex]];
103+
__auto_type * extraArguments = @[
143104
@"--otaDownloadPath",
144-
_downloadFilePath,
105+
downloadFilePath,
145106
@"--autoApplyImage",
146107
];
147108

148-
[_appTask setArguments:arguments];
149-
150-
NSString * outFile = [NSString stringWithFormat:@"/tmp/darwin/framework-tests/ota-requestor-app-%u.log", _uniqueIndex];
151-
NSString * errorFile = [NSString stringWithFormat:@"/tmp/darwin/framework-tests/ota-requestor-app-err-%u.log", _uniqueIndex];
152-
153-
// Make sure the files exist.
154-
[[NSFileManager defaultManager] createFileAtPath:outFile contents:nil attributes:nil];
155-
[[NSFileManager defaultManager] createFileAtPath:errorFile contents:nil attributes:nil];
156-
157-
_appTask.standardOutput = [NSFileHandle fileHandleForWritingAtPath:outFile];
158-
_appTask.standardError = [NSFileHandle fileHandleForWritingAtPath:errorFile];
159-
160-
[_appTask launchAndReturnError:&error];
161-
XCTAssertNil(error);
162-
163-
NSLog(@"Started requestor with arguments %@ stdout=%@ and stderr=%@", arguments, outFile, errorFile);
109+
if (!(self = [super initWithAppName:@"ota-requestor" arguments:extraArguments payload:payload testcase:testcase])) {
110+
return nil;
111+
}
164112

165-
[_testcase registerRunningRequestor:self];
113+
_testcase = testcase;
114+
_payload = payload;
115+
_downloadFilePath = downloadFilePath;
166116

167117
return self;
168118
}
169119

170-
- (void)terminate
171-
{
172-
[_appTask terminate];
173-
}
174-
175120
@end
176121

177122
@interface MTROTAProviderTestControllerDelegate : NSObject <MTRDeviceControllerDelegate>
@@ -579,7 +524,6 @@ - (instancetype)initWithRawImagePath:(NSString *)rawImagePath
579524

580525
@implementation MTROTAProviderTests {
581526
NSMutableSet<NSNumber *> * _commissionedNodeIDs;
582-
NSMutableSet<MTROTARequestorAppRunner *> * _runningRequestors;
583527
}
584528

585529
+ (void)tearDown
@@ -605,7 +549,6 @@ - (void)setUp
605549
}
606550

607551
_commissionedNodeIDs = [[NSMutableSet alloc] init];
608-
_runningRequestors = [[NSMutableSet alloc] init];
609552

610553
XCTAssertNil(sOTAProviderDelegate.queryImageHandler);
611554
XCTAssertNil(sOTAProviderDelegate.applyUpdateRequestHandler);
@@ -637,12 +580,6 @@ - (void)tearDown
637580
ResetCommissionee(device, dispatch_get_main_queue(), self, kTimeoutInSeconds);
638581
}
639582

640-
for (MTROTARequestorAppRunner * runner in _runningRequestors) {
641-
[runner terminate];
642-
}
643-
// Break cycle.
644-
_runningRequestors = nil;
645-
646583
if (sController != nil) {
647584
[sController shutdown];
648585
XCTAssertFalse([sController isRunning]);
@@ -686,11 +623,6 @@ - (MTRDevice *)commissionDeviceWithPayload:(NSString *)payloadString nodeID:(NSN
686623
return [MTRDevice deviceWithNodeID:nodeID controller:sController];
687624
}
688625

689-
- (void)registerRunningRequestor:(MTROTARequestorAppRunner *)requestor
690-
{
691-
[_runningRequestors addObject:requestor];
692-
}
693-
694626
- (void)initStack
695627
{
696628
sStackInitRan = YES;
@@ -717,43 +649,6 @@ + (void)shutdownStack
717649
[[MTRDeviceControllerFactory sharedInstance] stopControllerFactory];
718650
}
719651

720-
/**
721-
* Given a path relative to the Matter root, create an absolute path to the file.
722-
*/
723-
- (NSString *)absolutePathFor:(NSString *)matterRootRelativePath
724-
{
725-
// Start with the absolute path to our file, then remove the suffix that
726-
// comes after the path to the Matter SDK root.
727-
NSString * pathToTest = [NSString stringWithUTF8String:__FILE__];
728-
NSMutableArray * pathComponents = [[NSMutableArray alloc] init];
729-
[pathComponents addObject:[pathToTest substringToIndex:(pathToTest.length - @"src/darwin/Framework/CHIPTests/MTROTAProviderTests.m".length)]];
730-
[pathComponents addObjectsFromArray:[matterRootRelativePath pathComponents]];
731-
return [NSString pathWithComponents:pathComponents];
732-
}
733-
734-
/**
735-
* Create a task given a path relative to the Matter root.
736-
*/
737-
- (NSTask *)createTaskForPath:(NSString *)path
738-
{
739-
NSTask * task = [[NSTask alloc] init];
740-
[task setLaunchPath:[self absolutePathFor:path]];
741-
return task;
742-
}
743-
744-
/**
745-
* Runs a task to completion and makes sure it succeeds.
746-
*/
747-
- (void)runTask:(NSTask *)task
748-
{
749-
NSError * launchError;
750-
[task launchAndReturnError:&launchError];
751-
XCTAssertNil(launchError);
752-
753-
[task waitUntilExit];
754-
XCTAssertEqual([task terminationStatus], 0);
755-
}
756-
757652
/**
758653
* Returns path to the raw image.
759654
*/

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

+31
Original file line numberDiff line numberDiff line change
@@ -16,13 +16,44 @@
1616

1717
#import <XCTest/XCTest.h>
1818

19+
#if TARGET_OS_IPHONE || TARGET_OS_SIMULATOR
20+
#define HAVE_NSTASK 0
21+
#else
22+
#define HAVE_NSTASK 1
23+
#endif
24+
1925
NS_ASSUME_NONNULL_BEGIN
2026

2127
@interface MTRTestCase : XCTestCase
2228
// It would be nice to do the leak-detection automatically, but running "leaks"
2329
// on every single sub-test is slow, and some of our tests seem to have leaks
2430
// outside Matter.framework. So have it be opt-in for now, and improve later.
2531
@property (nonatomic) BOOL detectLeaks;
32+
33+
#if HAVE_NSTASK
34+
/**
35+
* Create an NSTask for the given path. Path should be relative to the Matter
36+
* SDK root.
37+
*/
38+
- (NSTask *)createTaskForPath:(NSString *)path;
39+
40+
/**
41+
* Run a task to completion and make sure it succeeds.
42+
*/
43+
- (void)runTask:(NSTask *)task;
44+
45+
/**
46+
* Launch a task. The task will be automatically terminated when the testcase
47+
* tearDown happens.
48+
*/
49+
- (void)launchTask:(NSTask *)task;
50+
#endif // HAVE_NSTASK
51+
52+
/**
53+
* Get an absolute path from a path relative to the Matter SDK root.
54+
*/
55+
- (NSString *)absolutePathFor:(NSString *)matterRootRelativePath;
56+
2657
@end
2758

2859
NS_ASSUME_NONNULL_END

src/darwin/Framework/CHIPTests/TestHelpers/MTRTestCase.mm

+58-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,18 @@
1919

2020
#import "MTRTestCase.h"
2121

22-
@implementation MTRTestCase
22+
@implementation MTRTestCase {
23+
#if HAVE_NSTASK
24+
NSMutableSet<NSTask *> * _runningTasks;
25+
#endif // NSTask
26+
}
27+
28+
- (void)setUp
29+
{
30+
#if HAVE_NSTASK
31+
_runningTasks = [[NSMutableSet alloc] init];
32+
#endif // HAVE_NSTASK
33+
}
2334

2435
/**
2536
* Unfortunately, doing this in "+ (void)tearDown" (the global suite teardown)
@@ -36,7 +47,53 @@ - (void)tearDown
3647
}
3748
#endif
3849

50+
#if HAVE_NSTASK
51+
for (NSTask * task in _runningTasks) {
52+
[task terminate];
53+
}
54+
_runningTasks = nil;
55+
#endif // HAVE_NSTASK
56+
3957
[super tearDown];
4058
}
4159

60+
#if HAVE_NSTASK
61+
- (NSTask *)createTaskForPath:(NSString *)path
62+
{
63+
NSTask * task = [[NSTask alloc] init];
64+
[task setLaunchPath:[self absolutePathFor:path]];
65+
return task;
66+
}
67+
68+
- (void)runTask:(NSTask *)task
69+
{
70+
NSError * launchError;
71+
[task launchAndReturnError:&launchError];
72+
XCTAssertNil(launchError);
73+
74+
[task waitUntilExit];
75+
XCTAssertEqual([task terminationStatus], 0);
76+
}
77+
78+
- (void)launchTask:(NSTask *)task
79+
{
80+
NSError * launchError;
81+
[task launchAndReturnError:&launchError];
82+
XCTAssertNil(launchError);
83+
84+
[_runningTasks addObject:task];
85+
}
86+
#endif // HAVE_NSTASK
87+
88+
- (NSString *)absolutePathFor:(NSString *)matterRootRelativePath
89+
{
90+
// Start with the absolute path to our file, then remove the suffix that
91+
// comes after the path to the Matter SDK root.
92+
NSString * pathToTest = [NSString stringWithUTF8String:__FILE__];
93+
NSMutableArray * pathComponents = [[NSMutableArray alloc] init];
94+
[pathComponents addObject:[pathToTest substringToIndex:(pathToTest.length - @"src/darwin/Framework/CHIPTests/TestHelpers/MTRTestCase.mm".length)]];
95+
[pathComponents addObjectsFromArray:[matterRootRelativePath pathComponents]];
96+
return [NSString pathWithComponents:pathComponents];
97+
}
98+
4299
@end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
/**
2+
* Copyright (c) 2024 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+
19+
@class MTRTestCase;
20+
21+
NS_ASSUME_NONNULL_BEGIN
22+
23+
/**
24+
* A representation of a server application instance.
25+
*
26+
* Server applications are assumed to be compiled into out/debug/${APPNAME}-app,
27+
* with the binary being out/debug/${APPNAME}-app/chip-${APPNAME}-app.
28+
*/
29+
@interface MTRTestServerAppRunner : NSObject
30+
31+
/**
32+
* Initialize the app runner with the given app name, arguments, setup payload, and testcase
33+
* instance.
34+
*
35+
* The payload will be used to determine the discriminator and passcode
36+
* arguments the app should use, in addition to the provided arguments.
37+
* Discriminators should be at least 1111 (see documentation for port below).
38+
*
39+
* The --KVS argument for the app will be automatically set to something of the
40+
* format "/tmp/chip-${APPNAME}-kvs${UNIQUE_ID}".
41+
*
42+
* The port argument for the app will be determined automatically, by
43+
* subtracting 1111 from the discriminator and adding 5542 (so as not to collide
44+
* with any existing Matter things running on 5540/5541).
45+
*/
46+
- (instancetype)initWithAppName:(NSString *)name arguments:(NSArray<NSString *> *)arguments payload:(NSString *)payload testcase:(MTRTestCase *)testcase;
47+
48+
/**
49+
* Get the unique index that will be used for the next initialization. This
50+
* allows including that index in the arguments provided.
51+
*
52+
* TODO: Should we scan the provided arguments for %u and replace with the index
53+
* instead?
54+
*/
55+
+ (unsigned)nextUniqueIndex;
56+
57+
@end
58+
59+
NS_ASSUME_NONNULL_END

0 commit comments

Comments
 (0)