Skip to content

Commit ad792f1

Browse files
rochaferraztehampsoncecillerestyled-commits
authored andcommitted
Test automation for FabricSync ICD BridgedDeviceBasicInfoCluster (project-chip#34628)
* WIP Bridged ICD, commissioning to both fabrics * wip testing sending KeepActive * wip most steps implemented * using SIGSTOP and SIGCONT to control ICD server pausing * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: Terence Hampson <thampson@google.com> * comments addressed * more comments addressed * lint pass * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: C Freeman <cecille@google.com> * comments addressed, incl TH_SERVER configurable * added setupQRCode and setupManualCode as options for DUT commissioning * Restyled by autopep8 * Restyled by isort * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: Terence Hampson <thampson@google.com> * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: Terence Hampson <thampson@google.com> * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: Terence Hampson <thampson@google.com> * comments addressed * Restyled by autopep8 --------- Co-authored-by: Terence Hampson <thampson@google.com> Co-authored-by: C Freeman <cecille@google.com> Co-authored-by: Restyled.io <commits@restyled.io>
1 parent dc312d7 commit ad792f1

File tree

1 file changed

+275
-0
lines changed

1 file changed

+275
-0
lines changed

src/python_testing/TC_BRBINFO_4_1.py

+275
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,275 @@
1+
#
2+
# Copyright (c) 2024 Project CHIP Authors
3+
# All rights reserved.
4+
#
5+
# Licensed under the Apache License, Version 2.0 (the "License");
6+
# you may not use this file except in compliance with the License.
7+
# You may obtain a copy of the License at
8+
#
9+
# http://www.apache.org/licenses/LICENSE-2.0
10+
#
11+
# Unless required by applicable law or agreed to in writing, software
12+
# distributed under the License is distributed on an "AS IS" BASIS,
13+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
# See the License for the specific language governing permissions and
15+
# limitations under the License.
16+
#
17+
18+
# This test requires a TH_SERVER application. Please specify with --string-arg th_server_app_path:<path_to_app>
19+
# TH_SERVER must support following arguments: --secured-device-port --discriminator --passcode --KVS
20+
# E.g: python3 src/python_testing/TC_BRBINFO_4_1.py --commissioning-method on-network --qr-code MT:-24J042C00KA0648G00 \
21+
# --string-arg th_server_app_path:out/linux-x64-lit-icd/lit-icd-app
22+
23+
import logging
24+
import os
25+
import queue
26+
import signal
27+
import subprocess
28+
import time
29+
import uuid
30+
31+
import chip.clusters as Clusters
32+
from chip import ChipDeviceCtrl
33+
from matter_testing_support import MatterBaseTest, SimpleEventCallback, TestStep, async_test_body, default_matter_test_main
34+
from mobly import asserts
35+
36+
logger = logging.getLogger(__name__)
37+
_ROOT_ENDPOINT_ID = 0
38+
39+
40+
class TC_BRBINFO_4_1(MatterBaseTest):
41+
42+
#
43+
# Class Helper functions
44+
#
45+
46+
async def _read_attribute_expect_success(self, endpoint, cluster, attribute, node_id):
47+
return await self.read_single_attribute_check_success(endpoint=endpoint, cluster=cluster, attribute=attribute, node_id=node_id)
48+
49+
# Override default timeout to support a 60 min wait
50+
@property
51+
def default_timeout(self) -> int:
52+
return 63*60
53+
54+
def desc_TC_BRBINFO_4_1(self) -> str:
55+
"""Returns a description of this test"""
56+
return "[TC_BRBINFO_4_1] Verification of KeepActive Command [DUT-Server]"
57+
58+
def steps_TC_BRBINFO_4_1(self) -> list[TestStep]:
59+
steps = [
60+
TestStep("0", "Preconditions"),
61+
TestStep("1a", "TH reads from the ICD the A_IDLE_MODE_DURATION, A_ACTIVE_MODE_DURATION, and ACTIVE_MODE_THRESHOLD attributes"),
62+
TestStep("1b", "Simple KeepActive command w/ subscription. ActiveChanged event received by TH contains PromisedActiveDuration"),
63+
TestStep("2", "Sends 3x KeepActive commands w/ subscription. ActiveChanged event received ONCE and contains PromisedActiveDuration"),
64+
TestStep("3", "KeepActive not returned after 60 minutes of offline ICD"),
65+
]
66+
return steps
67+
68+
def _ask_for_vendor_commissioniong_ux_operation(self, discriminator, setupPinCode, setupManualCode, setupQRCode):
69+
self.wait_for_user_input(
70+
prompt_msg=f"Using the DUT vendor's provided interface, commission the ICD device using the following parameters:\n"
71+
f"- discriminator: {discriminator}\n"
72+
f"- setupPinCode: {setupPinCode}\n"
73+
f"- setupQRCode: {setupQRCode}\n"
74+
f"- setupManualcode: {setupManualCode}\n"
75+
f"If using FabricSync Admin test app, you may type:\n"
76+
f">>> pairing onnetwork 111 {setupPinCode}")
77+
78+
async def _send_keep_active_command(self, duration, endpoint_id) -> int:
79+
logging.info("Sending keep active command")
80+
keep_active = await self.default_controller.SendCommand(nodeid=self.dut_node_id, endpoint=endpoint_id, payload=Clusters.Objects.BridgedDeviceBasicInformation.Commands.KeepActive(stayActiveDuration=duration))
81+
return keep_active
82+
83+
async def _wait_for_active_changed_event(self, timeout) -> int:
84+
try:
85+
promised_active_duration = self.q.get(block=True, timeout=timeout)
86+
logging.info(f"PromisedActiveDuration: {promised_active_duration}")
87+
return promised_active_duration
88+
except queue.Empty:
89+
asserts.fail("Timeout on event ActiveChanged")
90+
91+
async def _get_dynamic_endpoint(self) -> int:
92+
root_part_list = await self.read_single_attribute_check_success(cluster=Clusters.Descriptor, attribute=Clusters.Descriptor.Attributes.PartsList, endpoint=_ROOT_ENDPOINT_ID)
93+
set_of_endpoints_after_adding_device = set(root_part_list)
94+
95+
asserts.assert_true(set_of_endpoints_after_adding_device.issuperset(
96+
self.set_of_dut_endpoints_before_adding_device), "Expected only new endpoints to be added")
97+
unique_endpoints_set = set_of_endpoints_after_adding_device - self.set_of_dut_endpoints_before_adding_device
98+
asserts.assert_equal(len(unique_endpoints_set), 1, "Expected only one new endpoint")
99+
newly_added_endpoint = list(unique_endpoints_set)[0]
100+
return newly_added_endpoint
101+
102+
@async_test_body
103+
async def setup_class(self):
104+
# These steps are not explicitly, but they help identify the dynamically added endpoint
105+
# The second part of this process happens on _get_dynamic_endpoint()
106+
root_part_list = await self.read_single_attribute_check_success(cluster=Clusters.Descriptor, attribute=Clusters.Descriptor.Attributes.PartsList, endpoint=_ROOT_ENDPOINT_ID)
107+
self.set_of_dut_endpoints_before_adding_device = set(root_part_list)
108+
109+
super().setup_class()
110+
app = self.user_params.get("th_server_app_path", None)
111+
if not app:
112+
asserts.fail('This test requires a TH_SERVER app. Specify app path with --string-arg th_server_app_path:<path_to_app>')
113+
114+
self.kvs = f'kvs_{str(uuid.uuid4())}'
115+
self.port = 5543
116+
discriminator = 3850
117+
passcode = 20202021
118+
app_args = f'--secured-device-port {self.port} --discriminator {discriminator} --passcode {passcode} --KVS {self.kvs} '
119+
cmd = f'{app} {app_args}'
120+
121+
logging.info("Starting ICD Server App")
122+
self.app_process = subprocess.Popen(cmd, bufsize=0, shell=True)
123+
logging.info("ICD started")
124+
time.sleep(3)
125+
126+
logging.info("Commissioning of ICD to fabric one (TH)")
127+
self.icd_nodeid = 1111
128+
129+
await self.default_controller.CommissionOnNetwork(nodeId=self.icd_nodeid, setupPinCode=passcode, filterType=ChipDeviceCtrl.DiscoveryFilterType.LONG_DISCRIMINATOR, filter=discriminator)
130+
131+
logging.info("Commissioning of ICD to fabric two (DUT)")
132+
params = await self.openCommissioningWindow(dev_ctrl=self.default_controller, node_id=self.icd_nodeid)
133+
134+
self._ask_for_vendor_commissioniong_ux_operation(params.randomDiscriminator, params.commissioningParameters.setupPinCode,
135+
params.commissioningParameters.setupManualCode, params.commissioningParameters.setupQRCode)
136+
137+
def teardown_class(self):
138+
logging.warning("Stopping app with SIGTERM")
139+
self.app_process.send_signal(signal.SIGTERM.value)
140+
self.app_process.wait()
141+
os.remove(self.kvs)
142+
super().teardown_class()
143+
144+
#
145+
# BRBINFO 4.1 Test Body
146+
#
147+
148+
@async_test_body
149+
async def test_TC_BRBINFO_4_1(self):
150+
self.is_ci = self.check_pics('PICS_SDK_CI_ONLY')
151+
icdm_cluster = Clusters.Objects.IcdManagement
152+
icdm_attributes = icdm_cluster.Attributes
153+
brb_info_cluster = Clusters.Objects.BridgedDeviceBasicInformation
154+
basic_info_cluster = Clusters.Objects.BasicInformation
155+
basic_info_attributes = basic_info_cluster.Attributes
156+
157+
dynamic_endpoint_id = await self._get_dynamic_endpoint()
158+
logging.info(f"Dynamic endpoint is {dynamic_endpoint_id}")
159+
160+
# Preconditions
161+
self.step("0")
162+
163+
logging.info("Ensuring DUT is commissioned to TH")
164+
165+
# Confirms commissioning of DUT on TH as it reads its fature map
166+
await self._read_attribute_expect_success(
167+
_ROOT_ENDPOINT_ID,
168+
basic_info_cluster,
169+
basic_info_attributes.FeatureMap,
170+
self.dut_node_id
171+
)
172+
173+
logging.info("Ensuring ICD is commissioned to TH")
174+
175+
# Confirms commissioning of ICD on TH as it reads its feature map
176+
await self._read_attribute_expect_success(
177+
_ROOT_ENDPOINT_ID,
178+
basic_info_cluster,
179+
basic_info_attributes.FeatureMap,
180+
self.icd_nodeid
181+
)
182+
183+
self.step("1a")
184+
185+
idle_mode_duration = await self._read_attribute_expect_success(
186+
_ROOT_ENDPOINT_ID,
187+
icdm_cluster,
188+
icdm_attributes.IdleModeDuration,
189+
self.icd_nodeid
190+
)
191+
logging.info(f"IdleModeDuration: {idle_mode_duration}")
192+
193+
active_mode_duration = await self._read_attribute_expect_success(
194+
_ROOT_ENDPOINT_ID,
195+
icdm_cluster,
196+
icdm_attributes.ActiveModeDuration,
197+
self.icd_nodeid
198+
)
199+
logging.info(f"ActiveModeDuration: {active_mode_duration}")
200+
201+
self.step("1b")
202+
203+
# Subscription to ActiveChanged
204+
event = brb_info_cluster.Events.ActiveChanged
205+
self.q = queue.Queue()
206+
urgent = 1
207+
cb = SimpleEventCallback("ActiveChanged", event.cluster_id, event.event_id, self.q)
208+
subscription = await self.default_controller.ReadEvent(nodeid=self.dut_node_id, events=[(dynamic_endpoint_id, event, urgent)], reportInterval=[1, 3])
209+
subscription.SetEventUpdateCallback(callback=cb)
210+
211+
stay_active_duration = 1000
212+
logging.info(f"Sending KeepActiveCommand({stay_active_duration}ms)")
213+
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id)
214+
215+
logging.info("Waiting for ActiveChanged from DUT...")
216+
promised_active_duration = await self._wait_for_active_changed_event((idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000)
217+
218+
asserts.assert_greater_equal(promised_active_duration, stay_active_duration, "PromisedActiveDuration < StayActiveDuration")
219+
220+
self.step("2")
221+
222+
stay_active_duration = 1500
223+
logging.info(f"Sending KeepActiveCommand({stay_active_duration}ms)")
224+
self._send_keep_active_command(stay_active_duration)
225+
226+
logging.info("Waiting for ActiveChanged from DUT...")
227+
promised_active_duration = await self._wait_for_active_changed_event((idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000)
228+
229+
# wait for active time duration
230+
time.sleep(max(stay_active_duration/1000, promised_active_duration))
231+
# ICD now should be in idle mode
232+
233+
# sends 3x keep active commands
234+
logging.info(f"Sending KeepActiveCommand({stay_active_duration})")
235+
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id)
236+
time.sleep(100)
237+
logging.info(f"Sending KeepActiveCommand({stay_active_duration})")
238+
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id)
239+
time.sleep(100)
240+
logging.info(f"Sending KeepActiveCommand({stay_active_duration})")
241+
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id)
242+
time.sleep(100)
243+
244+
logging.info("Waiting for ActiveChanged from DUT...")
245+
promised_active_duration = await self._wait_for_active_changed_event((idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000)
246+
247+
asserts.assert_equal(self.q.qSize(), 0, "More than one event received from DUT")
248+
249+
self.step("3")
250+
251+
stay_active_duration = 10000
252+
logging.info(f"Sending KeepActiveCommand({stay_active_duration})")
253+
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id)
254+
255+
# stops (halts) the ICD server process by sending a SIGTOP signal
256+
self.app_process.send_signal(signal.SIGSTOP.value)
257+
258+
if not self.is_ci:
259+
logging.info("Waiting for 60 minutes")
260+
self._timeout
261+
time.sleep(60*60)
262+
263+
# resumes (continues) the ICD server process by sending a SIGCONT signal
264+
self.app_process.send_signal(signal.SIGCONT.value)
265+
266+
# wait for active changed event, expect no event will be sent
267+
event_timeout = (idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000
268+
try:
269+
promised_active_duration = self.q.get(block=True, timeout=event_timeout)
270+
finally:
271+
asserts.assert_true(queue.Empty(), "ActiveChanged event received when not expected")
272+
273+
274+
if __name__ == "__main__":
275+
default_matter_test_main()

0 commit comments

Comments
 (0)