|
| 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 | +import logging |
| 19 | +import time |
| 20 | + |
| 21 | +import chip.clusters as Clusters |
| 22 | +from chip.exceptions import ChipStackError |
| 23 | +from matter_testing_support import MatterBaseTest, TestStep, async_test_body, default_matter_test_main |
| 24 | +from mobly import asserts |
| 25 | + |
| 26 | +kRootEndpointId = 0 |
| 27 | +kInvalidEndpointId = 0xFFFF |
| 28 | +kWiFiFeature = 1 |
| 29 | +kThreadFeature = 2 |
| 30 | +kExpiryLengthSeconds = 900 |
| 31 | +kChipErrorTimeout = 0x32 |
| 32 | +kMaxCommissioningCompleteRetryTimes = 5 |
| 33 | + |
| 34 | + |
| 35 | +class TC_CNET_4_24(MatterBaseTest): |
| 36 | + def steps_TC_CNET_4_24(self): |
| 37 | + return [TestStep("precondition", "TH is commissioned", is_commissioning=True), |
| 38 | + TestStep(1, 'TH sends ArmFailSafe command to the DUT with ExpiryLengthSeconds set to 900. Verify that DUT sends ArmFailSafeResponse command to the TH'), |
| 39 | + TestStep(2, 'TH reads Networks attribute on endpoint PIXIT.CNET.ENDPOINT_WIFI from the DUT. Verify that the Networks attribute list has an entry with the following fields: network_id matches the bytes in PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID, Connected is of type bool and is TRUE'), |
| 40 | + TestStep(3, 'TH finds the index of the Networks list entry with network_id for PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID and saves it as `Userwifi_netidx`'), |
| 41 | + TestStep(4, 'TH sends RemoveNetwork Command to the DUT PIXIT.CNET.ENDPOINT_WIFI endpoint with network_id field set to PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID. Verify that DUT sends NetworkConfigResponse to command with the following fields: NetworkingStatus is Success, NetworkIndex matches previously saved `Userwifi_netidx`'), |
| 42 | + TestStep(5, 'TH sends AddOrUpdateThreadNetwork command to the DUT PIXIT.CNET.ENDPOINT_THREAD endpoint with operational dataset field set to PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET. Verify that DUT sends the NetworkConfigResponse command to the TH with the following fields: NetworkingStatus is Success, DebugText is of type string with max length 512 or empty'), |
| 43 | + TestStep(6, 'TH reads Networks attribute from the DUT PIXIT.CNET.ENDPOINT_THREAD endpoint. Verify that the Networks attribute list has an entry with the following fields: network_id is the extended PAN ID of PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET, Connected is of type bool and is FALSE'), |
| 44 | + TestStep(7, 'TH sends ConnectNetwork command to the DUT PIXIT.CNET.ENDPOINT_THREAD endpoint with network_id field set to the extended PAN ID of PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET'), |
| 45 | + TestStep(8, 'TH sends the CommissioningComplete command to the DUT. Verify that DUT sends CommissioningCompleteResponse with the ErrorCode field set to OK'), |
| 46 | + TestStep(9, 'TH reads Networks attribute from the DUT PIXIT.CNET.ENDPOINT_THREAD endpoint. Verify that the Networks attribute list has an entry with the following values: network_id is the extended PAN ID of PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET, Connected is of type bool and is TRUE. TH saves the index of the Networks list entry as `Userth_netidx`'), |
| 47 | + TestStep(10, 'TH sends ArmFailSafe command to the DUT with ExpiryLengthSeconds set to 900| Verify that DUT sends ArmFailSafeResponse command to the TH'), |
| 48 | + TestStep(11, 'TH sends RemoveNetwork Command to the DUT PIXIT.CNET.ENDPOINT_THREAD endpoint with network_id field set to the extended PAN ID of PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET. Verify that DUT sends NetworkConfigResponse to command with the following response fields: NetworkingStatus is Success, NetworkIndex is `Userth_netidx`'), |
| 49 | + TestStep(12, 'TH sends AddOrUpdateWiFiNetwork command to the DUT PIXIT.CNET.ENDPOINT_WIFI endpoint with SSID field set to PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID, Credentials field set to PIXIT.CNET.WIFI_1ST_ACCESSPOINT_CREDENTIALS. Verify that DUT sends the NetworkConfigResponse command to the TH with the following response fields: NetworkingStatus is Success, DebugText is of type string with max length 512 or empty'), |
| 50 | + TestStep(13, 'TH sends ConnectNetwork command to the DUT PIXIT.CNET.ENDPOINT_WIFI endpoint with network_id field set to PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID'), |
| 51 | + TestStep(14, 'TH sends the CommissioningComplete command to the DUT. Verify that DUT sends CommissioningCompleteResponse with the ErrorCode field set to OK'), |
| 52 | + TestStep(15, 'TH reads Networks attribute from the DUT PIXIT.CNET.ENDPOINT_WIFI endpoint. Verify that the Networks attribute list has an entry with the following fields: network_id matches the bytes in PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID, Connected is of type bool and is TRUE')] |
| 53 | + |
| 54 | + def def_TC_CNET_4_24(self): |
| 55 | + return '[TC-CNET-1.4] verification for Secondary Network Interface [DUT-Server]' |
| 56 | + |
| 57 | + def pics_TC_CNET_4_24(self): |
| 58 | + return ["CNET.S.F00", "CNET.S.F01"] |
| 59 | + |
| 60 | + # Override default timeout. |
| 61 | + @property |
| 62 | + def default_timeout(self) -> int: |
| 63 | + return 200 |
| 64 | + |
| 65 | + async def SendConnectNetworkWithFailure( |
| 66 | + self, networkID: int, endpoint: int): |
| 67 | + try: |
| 68 | + cmd = Clusters.NetworkCommissioning.Commands.ConnectNetwork(networkID=networkID) |
| 69 | + await self.send_single_cmd(endpoint=endpoint, cmd=cmd) |
| 70 | + except ChipStackError as e: |
| 71 | + asserts.assert_equal(e.err, kChipErrorTimeout, "Unexpected error while trying to send ConnectNetwork command") |
| 72 | + logging.exception(e) |
| 73 | + |
| 74 | + async def SendCommissioningCompleteWithRetry(self): |
| 75 | + commissioning_complete_cmd = Clusters.GeneralCommissioning.Commands.CommissioningComplete() |
| 76 | + # After switching networks, it may resolve to an incorrect address. Implement a retry mechanism for failure recovery. |
| 77 | + for i in range(kMaxCommissioningCompleteRetryTimes): |
| 78 | + try: |
| 79 | + commissioning_complete_results = await self.send_single_cmd(cmd=commissioning_complete_cmd) |
| 80 | + asserts.assert_true(commissioning_complete_results.errorCode == Clusters.GeneralCommissioning.Enums.CommissioningErrorEnum.kOk, "CommissioningComplete command failed") |
| 81 | + break |
| 82 | + except ChipStackError as e: |
| 83 | + asserts.assert_equal(e.err, kChipErrorTimeout, "Unexpected error while trying to send CommissioningComplete") |
| 84 | + time.sleep(10) |
| 85 | + logging.exception(e) |
| 86 | + else: |
| 87 | + asserts.assert_true(False, "CommissioningComplete command failed") |
| 88 | + |
| 89 | + @async_test_body |
| 90 | + async def test_TC_CNET_4_24(self): |
| 91 | + self.step("precondition") |
| 92 | + asserts.assert_true('PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET' in self.matter_test_config.global_test_params, |
| 93 | + "PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET must be included on the command line in " |
| 94 | + "the --hex-arg flag as PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET:<Thread dataset in hex>") |
| 95 | + operational_dataset = self.matter_test_config.global_test_params['PIXIT.CNET.THREAD_1ST_OPERATIONALDATASET'] |
| 96 | + |
| 97 | + asserts.assert_true('PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID' in self.matter_test_config.global_test_params, |
| 98 | + "PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID must be included on the command line in " |
| 99 | + "the --hex-arg flag as PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID:<Wi-Fi ssid in hex>") |
| 100 | + wifi_ssid = self.matter_test_config.global_test_params['PIXIT.CNET.WIFI_1ST_ACCESSPOINT_SSID'] |
| 101 | + |
| 102 | + asserts.assert_true('PIXIT.CNET.WIFI_1ST_ACCESSPOINT_CREDENTIALS' in self.matter_test_config.global_test_params, |
| 103 | + "PIXIT.CNET.WIFI_1ST_ACCESSPOINT_CREDENTIALS must be included on the command line in " |
| 104 | + "the --hex-arg flag as PIXIT.CNET.WIFI_1ST_ACCESSPOINT_CREDENTIALS:<Wi-Fi credentials in hex>") |
| 105 | + wifi_credential = self.matter_test_config.global_test_params['PIXIT.CNET.WIFI_1ST_ACCESSPOINT_CREDENTIALS'] |
| 106 | + |
| 107 | + feature_map_response = await self.default_controller.ReadAttribute(self.dut_node_id, [(Clusters.NetworkCommissioning.Attributes.FeatureMap)], fabricFiltered=True) |
| 108 | + |
| 109 | + wifi_endpoint = kInvalidEndpointId |
| 110 | + thread_endpoint = kInvalidEndpointId |
| 111 | + for endpoint, value in feature_map_response.items(): |
| 112 | + feature_dict = value.get(Clusters.Objects.NetworkCommissioning, {}) |
| 113 | + feature_map = feature_dict.get(Clusters.Objects.NetworkCommissioning.Attributes.FeatureMap, None) |
| 114 | + if feature_map == kWiFiFeature: |
| 115 | + wifi_endpoint = endpoint |
| 116 | + elif feature_map == kThreadFeature: |
| 117 | + thread_endpoint = endpoint |
| 118 | + else: |
| 119 | + continue |
| 120 | + asserts.assert_true(wifi_endpoint != kInvalidEndpointId and thread_endpoint != kInvalidEndpointId, "Should support both Wi-Fi and Thread") |
| 121 | + |
| 122 | + self.step(1) |
| 123 | + arm_failsafe_cmd = Clusters.GeneralCommissioning.Commands.ArmFailSafe(expiryLengthSeconds=kExpiryLengthSeconds) |
| 124 | + arm_fail_safe_results = await self.send_single_cmd(cmd=arm_failsafe_cmd) |
| 125 | + asserts.assert_true(arm_fail_safe_results.errorCode == Clusters.GeneralCommissioning.Enums.CommissioningErrorEnum.kOk, "ArmFailSafe command failed") |
| 126 | + |
| 127 | + self.step(2) |
| 128 | + networks_response = await self.read_single_attribute_check_success(cluster=Clusters.NetworkCommissioning, |
| 129 | + attribute=Clusters.NetworkCommissioning.Attributes.Networks, endpoint=wifi_endpoint) |
| 130 | + self.step(3) |
| 131 | + index = 0 |
| 132 | + Userwifi_netidx = 0 |
| 133 | + verification = False |
| 134 | + for network_info in networks_response: |
| 135 | + if network_info.connected and network_info.networkID == wifi_ssid: |
| 136 | + Userwifi_netidx = index |
| 137 | + verification = True |
| 138 | + index += 1 |
| 139 | + asserts.assert_true(verification, "There is no correct entry in Networks attribute list") |
| 140 | + |
| 141 | + self.step(4) |
| 142 | + remove_network_cmd = Clusters.NetworkCommissioning.Commands.RemoveNetwork(networkID=wifi_ssid) |
| 143 | + remove_network_results = await self.send_single_cmd(endpoint=wifi_endpoint, cmd=remove_network_cmd) |
| 144 | + asserts.assert_true(remove_network_results.networkIndex == Userwifi_netidx and remove_network_results.networkingStatus == Clusters.NetworkCommissioning.Enums.NetworkCommissioningStatusEnum.kSuccess, "RemoveNetwork command failed") |
| 145 | + |
| 146 | + self.step(5) |
| 147 | + add_thread_network_cmd = Clusters.NetworkCommissioning.Commands.AddOrUpdateThreadNetwork(operationalDataset=operational_dataset) |
| 148 | + add_network_results = await self.send_single_cmd(endpoint=thread_endpoint, cmd=add_thread_network_cmd) |
| 149 | + asserts.assert_true(add_network_results.networkingStatus == Clusters.NetworkCommissioning.Enums.NetworkCommissioningStatusEnum.kSuccess, "AddOrUpdateNetwork command failed") |
| 150 | + |
| 151 | + self.step(6) |
| 152 | + networks_response = await self.read_single_attribute_check_success(cluster=Clusters.NetworkCommissioning, |
| 153 | + attribute=Clusters.NetworkCommissioning.Attributes.Networks, endpoint=thread_endpoint) |
| 154 | + |
| 155 | + thread_extend_panid = None |
| 156 | + for network in networks_response: |
| 157 | + thread_extend_panid = network.networkID |
| 158 | + asserts.assert_false(thread_extend_panid == None, "There is no networkID in Networks attribute") |
| 159 | + |
| 160 | + self.step(7) |
| 161 | + await self.SendConnectNetworkWithFailure(networkID=thread_extend_panid, endpoint=thread_endpoint) |
| 162 | + |
| 163 | + self.step(8) |
| 164 | + await self.SendCommissioningCompleteWithRetry() |
| 165 | + |
| 166 | + self.step(9) |
| 167 | + networks_response = await self.read_single_attribute_check_success(cluster=Clusters.NetworkCommissioning, |
| 168 | + attribute=Clusters.NetworkCommissioning.Attributes.Networks, endpoint=thread_endpoint) |
| 169 | + verification = False |
| 170 | + for network_info in networks_response: |
| 171 | + if network_info.connected and network_info.networkID == thread_extend_panid: |
| 172 | + verification = True |
| 173 | + asserts.assert_true(verification, "There is no correct entry in Networks attribute list") |
| 174 | + |
| 175 | + self.step(10) |
| 176 | + arm_fail_safe_results = await self.send_single_cmd(cmd=arm_failsafe_cmd) |
| 177 | + asserts.assert_true(arm_fail_safe_results.errorCode == Clusters.GeneralCommissioning.Enums.CommissioningErrorEnum.kOk, "ArmFailSafe command failed") |
| 178 | + |
| 179 | + self.step(11) |
| 180 | + remove_network_cmd = Clusters.NetworkCommissioning.Commands.RemoveNetwork(networkID=thread_extend_panid) |
| 181 | + remove_network_results = await self.send_single_cmd(endpoint=thread_endpoint, cmd=remove_network_cmd) |
| 182 | + asserts.assert_true(remove_network_results.networkingStatus == Clusters.NetworkCommissioning.Enums.NetworkCommissioningStatusEnum.kSuccess, "RemoveNetwork command failed") |
| 183 | + |
| 184 | + self.step(12) |
| 185 | + add_wifi_network_cmd = Clusters.NetworkCommissioning.Commands.AddOrUpdateWiFiNetwork(ssid=wifi_ssid, credentials=wifi_credential) |
| 186 | + add_network_results = await self.send_single_cmd(endpoint=wifi_endpoint, cmd=add_wifi_network_cmd) |
| 187 | + asserts.assert_true(add_network_results.networkingStatus == Clusters.NetworkCommissioning.Enums.NetworkCommissioningStatusEnum.kSuccess, "AddOrUpdateNetwork command failed") |
| 188 | + |
| 189 | + self.step(13) |
| 190 | + await self.SendConnectNetworkWithFailure(networkID=wifi_ssid, endpoint=wifi_endpoint) |
| 191 | + |
| 192 | + self.step(14) |
| 193 | + await self.SendCommissioningCompleteWithRetry() |
| 194 | + |
| 195 | + self.step(15) |
| 196 | + networks_response = await self.read_single_attribute_check_success(cluster=Clusters.NetworkCommissioning, |
| 197 | + attribute=Clusters.NetworkCommissioning.Attributes.Networks, endpoint=wifi_endpoint) |
| 198 | + verification = False |
| 199 | + for network_info in networks_response: |
| 200 | + if network_info.connected and network_info.networkID == wifi_ssid: |
| 201 | + verification = True |
| 202 | + asserts.assert_true(verification, "There is no correct entry in Networks attribute list") |
| 203 | + |
| 204 | +if __name__ == "__main__": |
| 205 | + default_matter_test_main() |
0 commit comments