forked from project-chip/connectedhomeip
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Test automation for FabricSync ICD BridgedDeviceBasicInfoCluster (pro…
…ject-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 <[email protected]> * comments addressed * more comments addressed * lint pass * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: C Freeman <[email protected]> * 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 <[email protected]> * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: Terence Hampson <[email protected]> * Update src/python_testing/TC_BRBINFO_4_1.py Co-authored-by: Terence Hampson <[email protected]> * comments addressed * Restyled by autopep8 --------- Co-authored-by: Terence Hampson <[email protected]> Co-authored-by: C Freeman <[email protected]> Co-authored-by: Restyled.io <[email protected]>
- Loading branch information
Showing
1 changed file
with
275 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,275 @@ | ||
# | ||
# Copyright (c) 2024 Project CHIP Authors | ||
# All rights reserved. | ||
# | ||
# Licensed under the Apache License, Version 2.0 (the "License"); | ||
# you may not use this file except in compliance with the License. | ||
# You may obtain a copy of the License at | ||
# | ||
# http://www.apache.org/licenses/LICENSE-2.0 | ||
# | ||
# Unless required by applicable law or agreed to in writing, software | ||
# distributed under the License is distributed on an "AS IS" BASIS, | ||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
# See the License for the specific language governing permissions and | ||
# limitations under the License. | ||
# | ||
|
||
# This test requires a TH_SERVER application. Please specify with --string-arg th_server_app_path:<path_to_app> | ||
# TH_SERVER must support following arguments: --secured-device-port --discriminator --passcode --KVS | ||
# E.g: python3 src/python_testing/TC_BRBINFO_4_1.py --commissioning-method on-network --qr-code MT:-24J042C00KA0648G00 \ | ||
# --string-arg th_server_app_path:out/linux-x64-lit-icd/lit-icd-app | ||
|
||
import logging | ||
import os | ||
import queue | ||
import signal | ||
import subprocess | ||
import time | ||
import uuid | ||
|
||
import chip.clusters as Clusters | ||
from chip import ChipDeviceCtrl | ||
from matter_testing_support import MatterBaseTest, SimpleEventCallback, TestStep, async_test_body, default_matter_test_main | ||
from mobly import asserts | ||
|
||
logger = logging.getLogger(__name__) | ||
_ROOT_ENDPOINT_ID = 0 | ||
|
||
|
||
class TC_BRBINFO_4_1(MatterBaseTest): | ||
|
||
# | ||
# Class Helper functions | ||
# | ||
|
||
async def _read_attribute_expect_success(self, endpoint, cluster, attribute, node_id): | ||
return await self.read_single_attribute_check_success(endpoint=endpoint, cluster=cluster, attribute=attribute, node_id=node_id) | ||
|
||
# Override default timeout to support a 60 min wait | ||
@property | ||
def default_timeout(self) -> int: | ||
return 63*60 | ||
|
||
def desc_TC_BRBINFO_4_1(self) -> str: | ||
"""Returns a description of this test""" | ||
return "[TC_BRBINFO_4_1] Verification of KeepActive Command [DUT-Server]" | ||
|
||
def steps_TC_BRBINFO_4_1(self) -> list[TestStep]: | ||
steps = [ | ||
TestStep("0", "Preconditions"), | ||
TestStep("1a", "TH reads from the ICD the A_IDLE_MODE_DURATION, A_ACTIVE_MODE_DURATION, and ACTIVE_MODE_THRESHOLD attributes"), | ||
TestStep("1b", "Simple KeepActive command w/ subscription. ActiveChanged event received by TH contains PromisedActiveDuration"), | ||
TestStep("2", "Sends 3x KeepActive commands w/ subscription. ActiveChanged event received ONCE and contains PromisedActiveDuration"), | ||
TestStep("3", "KeepActive not returned after 60 minutes of offline ICD"), | ||
] | ||
return steps | ||
|
||
def _ask_for_vendor_commissioniong_ux_operation(self, discriminator, setupPinCode, setupManualCode, setupQRCode): | ||
self.wait_for_user_input( | ||
prompt_msg=f"Using the DUT vendor's provided interface, commission the ICD device using the following parameters:\n" | ||
f"- discriminator: {discriminator}\n" | ||
f"- setupPinCode: {setupPinCode}\n" | ||
f"- setupQRCode: {setupQRCode}\n" | ||
f"- setupManualcode: {setupManualCode}\n" | ||
f"If using FabricSync Admin test app, you may type:\n" | ||
f">>> pairing onnetwork 111 {setupPinCode}") | ||
|
||
async def _send_keep_active_command(self, duration, endpoint_id) -> int: | ||
logging.info("Sending keep active command") | ||
keep_active = await self.default_controller.SendCommand(nodeid=self.dut_node_id, endpoint=endpoint_id, payload=Clusters.Objects.BridgedDeviceBasicInformation.Commands.KeepActive(stayActiveDuration=duration)) | ||
return keep_active | ||
|
||
async def _wait_for_active_changed_event(self, timeout) -> int: | ||
try: | ||
promised_active_duration = self.q.get(block=True, timeout=timeout) | ||
logging.info(f"PromisedActiveDuration: {promised_active_duration}") | ||
return promised_active_duration | ||
except queue.Empty: | ||
asserts.fail("Timeout on event ActiveChanged") | ||
|
||
async def _get_dynamic_endpoint(self) -> int: | ||
root_part_list = await self.read_single_attribute_check_success(cluster=Clusters.Descriptor, attribute=Clusters.Descriptor.Attributes.PartsList, endpoint=_ROOT_ENDPOINT_ID) | ||
set_of_endpoints_after_adding_device = set(root_part_list) | ||
|
||
asserts.assert_true(set_of_endpoints_after_adding_device.issuperset( | ||
self.set_of_dut_endpoints_before_adding_device), "Expected only new endpoints to be added") | ||
unique_endpoints_set = set_of_endpoints_after_adding_device - self.set_of_dut_endpoints_before_adding_device | ||
asserts.assert_equal(len(unique_endpoints_set), 1, "Expected only one new endpoint") | ||
newly_added_endpoint = list(unique_endpoints_set)[0] | ||
return newly_added_endpoint | ||
|
||
@async_test_body | ||
async def setup_class(self): | ||
# These steps are not explicitly, but they help identify the dynamically added endpoint | ||
# The second part of this process happens on _get_dynamic_endpoint() | ||
root_part_list = await self.read_single_attribute_check_success(cluster=Clusters.Descriptor, attribute=Clusters.Descriptor.Attributes.PartsList, endpoint=_ROOT_ENDPOINT_ID) | ||
self.set_of_dut_endpoints_before_adding_device = set(root_part_list) | ||
|
||
super().setup_class() | ||
app = self.user_params.get("th_server_app_path", None) | ||
if not app: | ||
asserts.fail('This test requires a TH_SERVER app. Specify app path with --string-arg th_server_app_path:<path_to_app>') | ||
|
||
self.kvs = f'kvs_{str(uuid.uuid4())}' | ||
self.port = 5543 | ||
discriminator = 3850 | ||
passcode = 20202021 | ||
app_args = f'--secured-device-port {self.port} --discriminator {discriminator} --passcode {passcode} --KVS {self.kvs} ' | ||
cmd = f'{app} {app_args}' | ||
|
||
logging.info("Starting ICD Server App") | ||
self.app_process = subprocess.Popen(cmd, bufsize=0, shell=True) | ||
logging.info("ICD started") | ||
time.sleep(3) | ||
|
||
logging.info("Commissioning of ICD to fabric one (TH)") | ||
self.icd_nodeid = 1111 | ||
|
||
await self.default_controller.CommissionOnNetwork(nodeId=self.icd_nodeid, setupPinCode=passcode, filterType=ChipDeviceCtrl.DiscoveryFilterType.LONG_DISCRIMINATOR, filter=discriminator) | ||
|
||
logging.info("Commissioning of ICD to fabric two (DUT)") | ||
params = await self.openCommissioningWindow(dev_ctrl=self.default_controller, node_id=self.icd_nodeid) | ||
|
||
self._ask_for_vendor_commissioniong_ux_operation(params.randomDiscriminator, params.commissioningParameters.setupPinCode, | ||
params.commissioningParameters.setupManualCode, params.commissioningParameters.setupQRCode) | ||
|
||
def teardown_class(self): | ||
logging.warning("Stopping app with SIGTERM") | ||
self.app_process.send_signal(signal.SIGTERM.value) | ||
self.app_process.wait() | ||
os.remove(self.kvs) | ||
super().teardown_class() | ||
|
||
# | ||
# BRBINFO 4.1 Test Body | ||
# | ||
|
||
@async_test_body | ||
async def test_TC_BRBINFO_4_1(self): | ||
self.is_ci = self.check_pics('PICS_SDK_CI_ONLY') | ||
icdm_cluster = Clusters.Objects.IcdManagement | ||
icdm_attributes = icdm_cluster.Attributes | ||
brb_info_cluster = Clusters.Objects.BridgedDeviceBasicInformation | ||
basic_info_cluster = Clusters.Objects.BasicInformation | ||
basic_info_attributes = basic_info_cluster.Attributes | ||
|
||
dynamic_endpoint_id = await self._get_dynamic_endpoint() | ||
logging.info(f"Dynamic endpoint is {dynamic_endpoint_id}") | ||
|
||
# Preconditions | ||
self.step("0") | ||
|
||
logging.info("Ensuring DUT is commissioned to TH") | ||
|
||
# Confirms commissioning of DUT on TH as it reads its fature map | ||
await self._read_attribute_expect_success( | ||
_ROOT_ENDPOINT_ID, | ||
basic_info_cluster, | ||
basic_info_attributes.FeatureMap, | ||
self.dut_node_id | ||
) | ||
|
||
logging.info("Ensuring ICD is commissioned to TH") | ||
|
||
# Confirms commissioning of ICD on TH as it reads its feature map | ||
await self._read_attribute_expect_success( | ||
_ROOT_ENDPOINT_ID, | ||
basic_info_cluster, | ||
basic_info_attributes.FeatureMap, | ||
self.icd_nodeid | ||
) | ||
|
||
self.step("1a") | ||
|
||
idle_mode_duration = await self._read_attribute_expect_success( | ||
_ROOT_ENDPOINT_ID, | ||
icdm_cluster, | ||
icdm_attributes.IdleModeDuration, | ||
self.icd_nodeid | ||
) | ||
logging.info(f"IdleModeDuration: {idle_mode_duration}") | ||
|
||
active_mode_duration = await self._read_attribute_expect_success( | ||
_ROOT_ENDPOINT_ID, | ||
icdm_cluster, | ||
icdm_attributes.ActiveModeDuration, | ||
self.icd_nodeid | ||
) | ||
logging.info(f"ActiveModeDuration: {active_mode_duration}") | ||
|
||
self.step("1b") | ||
|
||
# Subscription to ActiveChanged | ||
event = brb_info_cluster.Events.ActiveChanged | ||
self.q = queue.Queue() | ||
urgent = 1 | ||
cb = SimpleEventCallback("ActiveChanged", event.cluster_id, event.event_id, self.q) | ||
subscription = await self.default_controller.ReadEvent(nodeid=self.dut_node_id, events=[(dynamic_endpoint_id, event, urgent)], reportInterval=[1, 3]) | ||
subscription.SetEventUpdateCallback(callback=cb) | ||
|
||
stay_active_duration = 1000 | ||
logging.info(f"Sending KeepActiveCommand({stay_active_duration}ms)") | ||
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id) | ||
|
||
logging.info("Waiting for ActiveChanged from DUT...") | ||
promised_active_duration = await self._wait_for_active_changed_event((idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000) | ||
|
||
asserts.assert_greater_equal(promised_active_duration, stay_active_duration, "PromisedActiveDuration < StayActiveDuration") | ||
|
||
self.step("2") | ||
|
||
stay_active_duration = 1500 | ||
logging.info(f"Sending KeepActiveCommand({stay_active_duration}ms)") | ||
self._send_keep_active_command(stay_active_duration) | ||
|
||
logging.info("Waiting for ActiveChanged from DUT...") | ||
promised_active_duration = await self._wait_for_active_changed_event((idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000) | ||
|
||
# wait for active time duration | ||
time.sleep(max(stay_active_duration/1000, promised_active_duration)) | ||
# ICD now should be in idle mode | ||
|
||
# sends 3x keep active commands | ||
logging.info(f"Sending KeepActiveCommand({stay_active_duration})") | ||
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id) | ||
time.sleep(100) | ||
logging.info(f"Sending KeepActiveCommand({stay_active_duration})") | ||
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id) | ||
time.sleep(100) | ||
logging.info(f"Sending KeepActiveCommand({stay_active_duration})") | ||
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id) | ||
time.sleep(100) | ||
|
||
logging.info("Waiting for ActiveChanged from DUT...") | ||
promised_active_duration = await self._wait_for_active_changed_event((idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000) | ||
|
||
asserts.assert_equal(self.q.qSize(), 0, "More than one event received from DUT") | ||
|
||
self.step("3") | ||
|
||
stay_active_duration = 10000 | ||
logging.info(f"Sending KeepActiveCommand({stay_active_duration})") | ||
self._send_keep_active_command(stay_active_duration, dynamic_endpoint_id) | ||
|
||
# stops (halts) the ICD server process by sending a SIGTOP signal | ||
self.app_process.send_signal(signal.SIGSTOP.value) | ||
|
||
if not self.is_ci: | ||
logging.info("Waiting for 60 minutes") | ||
self._timeout | ||
time.sleep(60*60) | ||
|
||
# resumes (continues) the ICD server process by sending a SIGCONT signal | ||
self.app_process.send_signal(signal.SIGCONT.value) | ||
|
||
# wait for active changed event, expect no event will be sent | ||
event_timeout = (idle_mode_duration + max(active_mode_duration, stay_active_duration))/1000 | ||
try: | ||
promised_active_duration = self.q.get(block=True, timeout=event_timeout) | ||
finally: | ||
asserts.assert_true(queue.Empty(), "ActiveChanged event received when not expected") | ||
|
||
|
||
if __name__ == "__main__": | ||
default_matter_test_main() |