TC-SWTCH-2.3: Implement (#34526)

* TC-SWTCH-2.3: Implement

* Update src/python_testing/matter_testing_support.py

* Restyled by autopep8

* Restyled by isort

---------

Co-authored-by: Restyled.io <commits@restyled.io>
diff --git a/src/python_testing/TC_SWTCH.py b/src/python_testing/TC_SWTCH.py
index 867897e..5fb968b 100644
--- a/src/python_testing/TC_SWTCH.py
+++ b/src/python_testing/TC_SWTCH.py
@@ -1,5 +1,5 @@
 #
-#    Copyright (c) 2023 Project CHIP Authors
+#    Copyright (c) 2024 Project CHIP Authors
 #    All rights reserved.
 #
 #    Licensed under the Apache License, Version 2.0 (the "License");
@@ -30,13 +30,15 @@
 import logging
 import queue
 import time
+from datetime import datetime, timedelta
 from typing import Any
 
 import chip.clusters as Clusters
+import test_plan_support
 from chip.clusters import ClusterObjects as ClusterObjects
 from chip.clusters.Attribute import EventReadResult, TypedAttributePath
 from matter_testing_support import (AttributeValue, ClusterAttributeChangeAccumulator, EventChangeCallback, MatterBaseTest,
-                                    async_test_body, default_matter_test_main)
+                                    TestStep, async_test_body, default_matter_test_main)
 from mobly import asserts
 
 logger = logging.getLogger(__name__)
@@ -93,6 +95,25 @@
                             "ButtonId": pressed_position, "LongPressDelayMillis": 5000, "LongPressDurationMillis": 5500}
             self._send_named_pipe_command(command_dict)
 
+    def _ask_for_keep_pressed(self, endpoint_id: int, pressed_position: int):
+        if not self._use_button_simulator():
+            self.wait_for_user_input(
+                prompt_msg=f"Press switch position {pressed_position} for a long time (around 5 seconds) on the DUT, then release it.")
+        else:
+            # Using the long press here with a long duration so we can check the intermediate value.
+            command_dict = {"Name": "SimulateActionSwitchLongPress", "EndpointId": endpoint_id,
+                            "ButtonId": pressed_position, "LongPressDelayMillis": 0, "LongPressDurationMillis": self.keep_pressed_delay}
+            self._send_named_pipe_command(command_dict)
+
+    def _ask_for_release(self):
+        # Since we used a long press for this, "ask for release" on the button simulator just means waiting out the delay
+        if not self._use_button_simulator():
+            self.wait_for_user_input(
+                prompt_msg="Release the button."
+            )
+        else:
+            time.sleep(self.keep_pressed_delay/1000)
+
     def _placeholder_for_step(self, step_id: str):
         # TODO: Global search an replace of `self._placeholder_for_step` with `self.step` when done.
         logging.info(f"Step {step_id}")
@@ -285,6 +306,96 @@
             self._await_sequence_of_events(event_queue=event_listener.event_queue, endpoint_id=endpoint_id,
                                            sequence=expected_events, timeout_sec=post_prompt_settle_delay_seconds)
 
+    def _received_event(self, event_listener: EventChangeCallback, target_event: ClusterObjects.ClusterEvent, timeout_s: int) -> bool:
+        """
+            Returns true if this event was received, false otherwise
+        """
+        remaining = timedelta(seconds=timeout_s)
+        end_time = datetime.now() + remaining
+        while (remaining.seconds > 0):
+            try:
+                event = event_listener.event_queue.get(timeout=remaining.seconds)
+            except queue.Empty:
+                return False
+
+            if event.Header.EventId == target_event.event_id:
+                return True
+            remaining = end_time - datetime.now()
+        return False
+
+    def pics_TC_SWTCH_2_3(self):
+        return ['SWTCH.S.F01']
+
+    def steps_TC_SWTCH_2_3(self):
+        return [TestStep(1, test_plan_support.commission_if_required(), "", is_commissioning=True),
+                TestStep(2, "Set up subscription to all events of Switch cluster on the endpoint"),
+                TestStep(3, "Operator does not operate switch on the DUT"),
+                TestStep(4, "TH reads the CurrentPosition attribute from the DUT", "Verify that the value is 0"),
+                TestStep(5, "Operator operates switch (keep it pressed)",
+                         "Verify that the TH receives InitialPress event with NewPosition set to 1 on the DUT"),
+                TestStep(6, "TH reads the CurrentPosition attribute from the DUT", "Verify that the value is 1"),
+                TestStep(7, "Operator releases switch on the DUT"),
+                TestStep("8a", "If the DUT implements the MSR feature, verify that the TH receives ShortRelease event with NewPosition set to 0 on the DUT", "Event received"),
+                TestStep(
+                    "8b", "If the DUT implements the AS feature, verify that the TH does not receive ShortRelease event on the DUT", "No event received"),
+                TestStep(9, "TH reads the CurrentPosition attribute from the DUT", "Verify that the value is 0"),
+                ]
+
+    @async_test_body
+    async def test_TC_SWTCH_2_3(self):
+        # Commissioning - already done
+        self.step(1)
+        cluster = Clusters.Switch
+        feature_map = await self.read_single_attribute_check_success(cluster, attribute=cluster.Attributes.FeatureMap)
+
+        has_msr_feature = (feature_map & cluster.Bitmaps.Feature.kMomentarySwitchRelease) != 0
+        has_as_feature = (feature_map & cluster.Bitmaps.Feature.kActionSwitch) != 0
+
+        endpoint_id = self.matter_test_config.endpoint
+
+        self.step(2)
+        event_listener = EventChangeCallback(cluster)
+        await event_listener.start(self.default_controller, self.dut_node_id, endpoint=endpoint_id)
+
+        self.step(3)
+        self._ask_for_switch_idle()
+
+        self.step(4)
+        button_val = await self.read_single_attribute_check_success(cluster=cluster, attribute=cluster.Attributes.CurrentPosition)
+        asserts.assert_equal(button_val, 0, "Button value is not 0")
+
+        self.step(5)
+        # We're using a long press here with a very long duration (in computer-land). This will let us check the intermediate values.
+        # This is 1s larger than the subscription ceiling
+        self.keep_pressed_delay = 6000
+        self.pressed_position = 1
+        self._ask_for_keep_pressed(endpoint_id, self.pressed_position)
+        event_listener.wait_for_event_report(cluster.Events.InitialPress)
+
+        self.step(6)
+        button_val = await self.read_single_attribute_check_success(cluster=cluster, attribute=cluster.Attributes.CurrentPosition)
+        asserts.assert_equal(button_val, self.pressed_position, f"Button value is not {self.pressed_position}")
+
+        self.step(7)
+        self._ask_for_release()
+
+        self.step("8a")
+        if has_msr_feature:
+            asserts.assert_true(self._received_event(event_listener, cluster.Events.ShortRelease, 10),
+                                "Did not receive short release")
+        else:
+            self.mark_current_step_skipped()
+
+        self.step("8b")
+        if has_as_feature:
+            asserts.assert_false(self._received_event(event_listener, cluster.Events.ShortRelease, 10), "Received short release")
+        else:
+            self.mark_current_step_skipped()
+
+        self.step(9)
+        button_val = await self.read_single_attribute_check_success(cluster=cluster, attribute=cluster.Attributes.CurrentPosition)
+        asserts.assert_equal(button_val, 0, "Button value is not 0")
+
 
 if __name__ == "__main__":
     default_matter_test_main()