Skip to content

Commit 5d2beb7

Browse files
TC-IDM-1.2 (#27024)
* TC-IDM-1.2 Adds automation for TC-IDM-1.2 Also Adds suppressResponse to CommandSender as well as a test-only function to test timedResponse flag with no corresponding TimedInvoke action + plumbing through the python layers * Restyled by isort * Updates from review comments * Couple formatting fixes * Cleanup. * Add a port to pase in python, fix filtering * Consolidate CommandSender functions * Timed invoke can be inferred --------- Co-authored-by: Restyled.io <commits@restyled.io>
1 parent 6e696b6 commit 5d2beb7

File tree

10 files changed

+451
-34
lines changed

10 files changed

+451
-34
lines changed

.github/workflows/tests.yaml

+1
Original file line numberDiff line numberDiff line change
@@ -448,6 +448,7 @@ jobs:
448448
scripts/run_in_python_env.sh out/venv './scripts/tests/run_python_test.py --app out/linux-x64-all-clusters-ipv6only-no-ble-no-wifi-tsan-clang-test/chip-all-clusters-app --factoryreset --app-args "--discriminator 1234 --KVS kvs1 --trace_decode 1" --script "src/python_testing/TC_CGEN_2_4.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021"'
449449
scripts/run_in_python_env.sh out/venv './scripts/tests/run_python_test.py --app out/linux-x64-all-clusters-ipv6only-no-ble-no-wifi-tsan-clang-test/chip-all-clusters-app --factoryreset --app-args "--discriminator 1234 --KVS kvs1 --trace_decode 1" --script "src/python_testing/TC_DA_1_2.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --PICS src/app/tests/suites/certification/ci-pics-values"'
450450
scripts/run_in_python_env.sh out/venv './scripts/tests/run_python_test.py --app out/linux-x64-all-clusters-ipv6only-no-ble-no-wifi-tsan-clang-test/chip-all-clusters-app --factoryreset --app-args "--discriminator 1234 --KVS kvs1 --trace_decode 1" --script "src/python_testing/TC_DA_1_5.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --PICS src/app/tests/suites/certification/ci-pics-values"'
451+
scripts/run_in_python_env.sh out/venv './scripts/tests/run_python_test.py --app out/linux-x64-all-clusters-ipv6only-no-ble-no-wifi-tsan-clang-test/chip-all-clusters-app --factoryreset --app-args "--discriminator 1234 --KVS kvs1 --trace_decode 1" --script "src/python_testing/TC_IDM_1_2.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021"'
451452
scripts/run_in_python_env.sh out/venv './scripts/tests/run_python_test.py --script "src/python_testing/TestMatterTestingSupport.py"'
452453
- name: Uploading core files
453454
uses: actions/upload-artifact@v3

config/python/CHIPProjectConfig.h

+2
Original file line numberDiff line numberDiff line change
@@ -59,4 +59,6 @@
5959
#define CHIP_DEVICE_CONFIG_ENABLE_COMMISSIONER_DISCOVERY 1
6060
#define CHIP_DEVICE_CONFIG_ENABLE_BOTH_COMMISSIONER_AND_COMMISSIONEE 1
6161

62+
#define CONFIG_BUILD_FOR_HOST_UNIT_TEST 1
63+
6264
#endif /* CHIPPROJECTCONFIG_H */

src/app/CommandSender.cpp

+28-13
Original file line numberDiff line numberDiff line change
@@ -32,9 +32,10 @@
3232
namespace chip {
3333
namespace app {
3434

35-
CommandSender::CommandSender(Callback * apCallback, Messaging::ExchangeManager * apExchangeMgr, bool aIsTimedRequest) :
36-
mExchangeCtx(*this), mpCallback(apCallback), mpExchangeMgr(apExchangeMgr), mSuppressResponse(false),
37-
mTimedRequest(aIsTimedRequest)
35+
CommandSender::CommandSender(Callback * apCallback, Messaging::ExchangeManager * apExchangeMgr, bool aIsTimedRequest,
36+
bool aSuppressResponse) :
37+
mExchangeCtx(*this),
38+
mpCallback(apCallback), mpExchangeMgr(apExchangeMgr), mSuppressResponse(aSuppressResponse), mTimedRequest(aIsTimedRequest)
3839
{}
3940

4041
CHIP_ERROR CommandSender::AllocateBuffer()
@@ -61,7 +62,7 @@ CHIP_ERROR CommandSender::AllocateBuffer()
6162
return CHIP_NO_ERROR;
6263
}
6364

64-
CHIP_ERROR CommandSender::SendCommandRequest(const SessionHandle & session, Optional<System::Clock::Timeout> timeout)
65+
CHIP_ERROR CommandSender::SendCommandRequestInternal(const SessionHandle & session, Optional<System::Clock::Timeout> timeout)
6566
{
6667
VerifyOrReturnError(mState == State::AddedCommand, CHIP_ERROR_INCORRECT_STATE);
6768

@@ -76,15 +77,6 @@ CHIP_ERROR CommandSender::SendCommandRequest(const SessionHandle & session, Opti
7677

7778
mExchangeCtx->SetResponseTimeout(timeout.ValueOr(session->ComputeRoundTripTimeout(app::kExpectedIMProcessingTime)));
7879

79-
if (mTimedRequest != mTimedInvokeTimeoutMs.HasValue())
80-
{
81-
ChipLogError(
82-
DataManagement,
83-
"Inconsistent timed request state in CommandSender: mTimedRequest (%d) != mTimedInvokeTimeoutMs.HasValue() (%d)",
84-
mTimedRequest, mTimedInvokeTimeoutMs.HasValue());
85-
return CHIP_ERROR_INCORRECT_STATE;
86-
}
87-
8880
if (mTimedInvokeTimeoutMs.HasValue())
8981
{
9082
ReturnErrorOnFailure(TimedRequest::Send(mExchangeCtx.Get(), mTimedInvokeTimeoutMs.Value()));
@@ -95,6 +87,29 @@ CHIP_ERROR CommandSender::SendCommandRequest(const SessionHandle & session, Opti
9587
return SendInvokeRequest();
9688
}
9789

90+
#if CONFIG_BUILD_FOR_HOST_UNIT_TEST
91+
CHIP_ERROR CommandSender::TestOnlyCommandSenderTimedRequestFlagWithNoTimedInvoke(const SessionHandle & session,
92+
Optional<System::Clock::Timeout> timeout)
93+
{
94+
VerifyOrReturnError(mTimedRequest, CHIP_ERROR_INCORRECT_STATE);
95+
return SendCommandRequestInternal(session, timeout);
96+
}
97+
#endif
98+
99+
CHIP_ERROR CommandSender::SendCommandRequest(const SessionHandle & session, Optional<System::Clock::Timeout> timeout)
100+
{
101+
102+
if (mTimedRequest != mTimedInvokeTimeoutMs.HasValue())
103+
{
104+
ChipLogError(
105+
DataManagement,
106+
"Inconsistent timed request state in CommandSender: mTimedRequest (%d) != mTimedInvokeTimeoutMs.HasValue() (%d)",
107+
mTimedRequest, mTimedInvokeTimeoutMs.HasValue());
108+
return CHIP_ERROR_INCORRECT_STATE;
109+
}
110+
return SendCommandRequestInternal(session, timeout);
111+
}
112+
98113
CHIP_ERROR CommandSender::SendGroupCommandRequest(const SessionHandle & session)
99114
{
100115
VerifyOrReturnError(mState == State::AddedCommand, CHIP_ERROR_INCORRECT_STATE);

src/app/CommandSender.h

+13-3
Original file line numberDiff line numberDiff line change
@@ -121,7 +121,8 @@ class CommandSender final : public Messaging::ExchangeDelegate
121121
* If used in a groups setting, callbacks do not need to be passed.
122122
* If callbacks are passed the only one that will be called in a group sesttings is the onDone
123123
*/
124-
CommandSender(Callback * apCallback, Messaging::ExchangeManager * apExchangeMgr, bool aIsTimedRequest = false);
124+
CommandSender(Callback * apCallback, Messaging::ExchangeManager * apExchangeMgr, bool aIsTimedRequest = false,
125+
bool aSuppressResponse = false);
125126
CHIP_ERROR PrepareCommand(const CommandPathParams & aCommandPathParams, bool aStartDataStruct = true);
126127
CHIP_ERROR FinishCommand(bool aEndDataStruct = true);
127128
TLV::TLVWriter * GetCommandDataIBTLVWriter();
@@ -164,11 +165,18 @@ class CommandSender final : public Messaging::ExchangeDelegate
164165
*/
165166
template <typename CommandDataT>
166167
CHIP_ERROR AddRequestDataNoTimedCheck(const CommandPathParams & aCommandPath, const CommandDataT & aData,
167-
const Optional<uint16_t> & aTimedInvokeTimeoutMs, bool aSuppressResponse = false)
168+
const Optional<uint16_t> & aTimedInvokeTimeoutMs)
168169
{
169-
mSuppressResponse = aSuppressResponse;
170170
return AddRequestDataInternal(aCommandPath, aData, aTimedInvokeTimeoutMs);
171171
}
172+
173+
/**
174+
* Version of SendCommandRequest that sets the TimedRequest flag but does not send the TimedInvoke
175+
* action. For use in tests only.
176+
*/
177+
CHIP_ERROR TestOnlyCommandSenderTimedRequestFlagWithNoTimedInvoke(const SessionHandle & session,
178+
Optional<System::Clock::Timeout> timeout = NullOptional);
179+
172180
#endif // CONFIG_BUILD_FOR_HOST_UNIT_TEST
173181

174182
private:
@@ -265,6 +273,8 @@ class CommandSender final : public Messaging::ExchangeDelegate
265273

266274
CHIP_ERROR Finalize(System::PacketBufferHandle & commandPacket);
267275

276+
CHIP_ERROR SendCommandRequestInternal(const SessionHandle & session, Optional<System::Clock::Timeout> timeout);
277+
268278
Messaging::ExchangeHolder mExchangeCtx;
269279
Callback * mpCallback = nullptr;
270280
Messaging::ExchangeManager * mpExchangeMgr = nullptr;

src/app/tests/suites/commands/interaction_model/InteractionModel.h

+3-4
Original file line numberDiff line numberDiff line change
@@ -234,12 +234,11 @@ class InteractionModelCommands
234234

235235
chip::app::CommandPathParams commandPath = { endpointId, clusterId, commandId,
236236
(chip::app::CommandPathFlags::kEndpointIdValid) };
237-
auto commandSender = std::make_unique<chip::app::CommandSender>(mCallback, device->GetExchangeManager(),
238-
mTimedInteractionTimeoutMs.HasValue());
237+
auto commandSender = std::make_unique<chip::app::CommandSender>(
238+
mCallback, device->GetExchangeManager(), mTimedInteractionTimeoutMs.HasValue(), mSuppressResponse.ValueOr(false));
239239
VerifyOrReturnError(commandSender != nullptr, CHIP_ERROR_NO_MEMORY);
240240

241-
ReturnErrorOnFailure(commandSender->AddRequestDataNoTimedCheck(commandPath, value, mTimedInteractionTimeoutMs,
242-
mSuppressResponse.ValueOr(false)));
241+
ReturnErrorOnFailure(commandSender->AddRequestDataNoTimedCheck(commandPath, value, mTimedInteractionTimeoutMs));
243242
ReturnErrorOnFailure(commandSender->SendCommandRequest(device->GetSecureSession().Value()));
244243
mCommandSender.push_back(std::move(commandSender));
245244

src/controller/python/ChipDeviceController-ScriptBinding.cpp

+6-2
Original file line numberDiff line numberDiff line change
@@ -139,7 +139,7 @@ PyChipError pychip_DeviceController_SetThreadOperationalDataset(const char * thr
139139
PyChipError pychip_DeviceController_SetWiFiCredentials(const char * ssid, const char * credentials);
140140
PyChipError pychip_DeviceController_CloseSession(chip::Controller::DeviceCommissioner * devCtrl, chip::NodeId nodeid);
141141
PyChipError pychip_DeviceController_EstablishPASESessionIP(chip::Controller::DeviceCommissioner * devCtrl, const char * peerAddrStr,
142-
uint32_t setupPINCode, chip::NodeId nodeid);
142+
uint32_t setupPINCode, chip::NodeId nodeid, uint16_t port);
143143
PyChipError pychip_DeviceController_EstablishPASESessionBLE(chip::Controller::DeviceCommissioner * devCtrl, uint32_t setupPINCode,
144144
uint16_t discriminator, chip::NodeId nodeid);
145145
PyChipError pychip_DeviceController_Commission(chip::Controller::DeviceCommissioner * devCtrl, chip::NodeId nodeid);
@@ -512,13 +512,17 @@ PyChipError pychip_DeviceController_CloseSession(chip::Controller::DeviceCommiss
512512
}
513513

514514
PyChipError pychip_DeviceController_EstablishPASESessionIP(chip::Controller::DeviceCommissioner * devCtrl, const char * peerAddrStr,
515-
uint32_t setupPINCode, chip::NodeId nodeid)
515+
uint32_t setupPINCode, chip::NodeId nodeid, uint16_t port)
516516
{
517517
chip::Inet::IPAddress peerAddr;
518518
chip::Transport::PeerAddress addr;
519519
RendezvousParameters params = chip::RendezvousParameters().SetSetupPINCode(setupPINCode);
520520
VerifyOrReturnError(chip::Inet::IPAddress::FromString(peerAddrStr, peerAddr), ToPyChipError(CHIP_ERROR_INVALID_ARGUMENT));
521521
addr.SetTransportType(chip::Transport::Type::kUdp).SetIPAddress(peerAddr);
522+
if (port != 0)
523+
{
524+
addr.SetPort(port);
525+
}
522526
params.SetPeerAddress(addr).SetDiscriminator(0);
523527
sPairingDelegate.SetExpectingPairingComplete(true);
524528
return ToPyChipError(devCtrl->EstablishPASEConnection(nodeid, params));

src/controller/python/chip/ChipDeviceCtrl.py

+26-5
Original file line numberDiff line numberDiff line change
@@ -473,13 +473,13 @@ def EstablishPASESessionBLE(self, setupPinCode: int, discriminator: int, nodeid:
473473
self.devCtrl, setupPinCode, discriminator, nodeid)
474474
)
475475

476-
def EstablishPASESessionIP(self, ipaddr: str, setupPinCode: int, nodeid: int):
476+
def EstablishPASESessionIP(self, ipaddr: str, setupPinCode: int, nodeid: int, port: int = 0):
477477
self.CheckIsActive()
478478

479479
self.state = DCState.RENDEZVOUS_ONGOING
480480
return self._ChipStack.CallAsync(
481481
lambda: self._dmLib.pychip_DeviceController_EstablishPASESessionIP(
482-
self.devCtrl, ipaddr.encode("utf-8"), setupPinCode, nodeid)
482+
self.devCtrl, ipaddr.encode("utf-8"), setupPinCode, nodeid, port)
483483
)
484484

485485
def GetTestCommissionerUsed(self):
@@ -779,9 +779,30 @@ def ComputeRoundTripTimeout(self, nodeid, upperLayerProcessingTimeoutMs: int = 0
779779
device.deviceProxy, upperLayerProcessingTimeoutMs))
780780
return res
781781

782+
async def TestOnlySendCommandTimedRequestFlagWithNoTimedInvoke(self, nodeid: int, endpoint: int,
783+
payload: ClusterObjects.ClusterCommand, responseType=None):
784+
'''
785+
786+
Please see SendCommand for description.
787+
'''
788+
self.CheckIsActive()
789+
790+
eventLoop = asyncio.get_running_loop()
791+
future = eventLoop.create_future()
792+
793+
device = self.GetConnectedDeviceSync(nodeid, timeoutMs=None)
794+
ClusterCommand.TestOnlySendCommandTimedRequestFlagWithNoTimedInvoke(
795+
future, eventLoop, responseType, device.deviceProxy, ClusterCommand.CommandPath(
796+
EndpointId=endpoint,
797+
ClusterId=payload.cluster_id,
798+
CommandId=payload.command_id,
799+
), payload).raise_on_error()
800+
return await future
801+
782802
async def SendCommand(self, nodeid: int, endpoint: int, payload: ClusterObjects.ClusterCommand, responseType=None,
783803
timedRequestTimeoutMs: typing.Union[None, int] = None,
784-
interactionTimeoutMs: typing.Union[None, int] = None, busyWaitMs: typing.Union[None, int] = None):
804+
interactionTimeoutMs: typing.Union[None, int] = None, busyWaitMs: typing.Union[None, int] = None,
805+
suppressResponse: typing.Union[None, bool] = None):
785806
'''
786807
Send a cluster-object encapsulated command to a node and get returned a future that can be awaited upon to receive
787808
the response. If a valid responseType is passed in, that will be used to deserialize the object. If not,
@@ -803,7 +824,7 @@ async def SendCommand(self, nodeid: int, endpoint: int, payload: ClusterObjects.
803824
ClusterId=payload.cluster_id,
804825
CommandId=payload.command_id,
805826
), payload, timedRequestTimeoutMs=timedRequestTimeoutMs,
806-
interactionTimeoutMs=interactionTimeoutMs, busyWaitMs=busyWaitMs).raise_on_error()
827+
interactionTimeoutMs=interactionTimeoutMs, busyWaitMs=busyWaitMs, suppressResponse=suppressResponse).raise_on_error()
807828
return await future
808829

809830
def SendGroupCommand(self, groupid: int, payload: ClusterObjects.ClusterCommand, busyWaitMs: typing.Union[None, int] = None):
@@ -1338,7 +1359,7 @@ def _InitLib(self):
13381359
self._dmLib.pychip_DeviceController_DiscoverCommissionableNodesCommissioningEnabled.restype = PyChipError
13391360

13401361
self._dmLib.pychip_DeviceController_EstablishPASESessionIP.argtypes = [
1341-
c_void_p, c_char_p, c_uint32, c_uint64]
1362+
c_void_p, c_char_p, c_uint32, c_uint64, c_uint16]
13421363
self._dmLib.pychip_DeviceController_EstablishPASESessionIP.restype = PyChipError
13431364

13441365
self._dmLib.pychip_DeviceController_EstablishPASESessionBLE.argtypes = [

src/controller/python/chip/clusters/Command.py

+28-3
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121
import logging
2222
import sys
2323
from asyncio.futures import Future
24-
from ctypes import CFUNCTYPE, c_char_p, c_size_t, c_uint8, c_uint16, c_uint32, c_void_p, py_object
24+
from ctypes import CFUNCTYPE, c_bool, c_char_p, c_size_t, c_uint8, c_uint16, c_uint32, c_void_p, py_object
2525
from dataclasses import dataclass
2626
from typing import Type, Union
2727

@@ -144,8 +144,30 @@ def _OnCommandSenderDoneCallback(closure):
144144
ctypes.pythonapi.Py_DecRef(ctypes.py_object(closure))
145145

146146

147+
def TestOnlySendCommandTimedRequestFlagWithNoTimedInvoke(future: Future, eventLoop, responseType, device, commandPath, payload):
148+
''' ONLY TO BE USED FOR TEST: Sends the payload with a TimedRequest flag but no TimedInvoke transaction
149+
'''
150+
if (responseType is not None) and (not issubclass(responseType, ClusterCommand)):
151+
raise ValueError("responseType must be a ClusterCommand or None")
152+
153+
handle = chip.native.GetLibraryHandle()
154+
transaction = AsyncCommandTransaction(future, eventLoop, responseType)
155+
156+
payloadTLV = payload.ToTLV()
157+
ctypes.pythonapi.Py_IncRef(ctypes.py_object(transaction))
158+
return builtins.chipStack.Call(
159+
lambda: handle.pychip_CommandSender_TestOnlySendCommandTimedRequestNoTimedInvoke(
160+
ctypes.py_object(transaction), device,
161+
commandPath.EndpointId, commandPath.ClusterId, commandPath.CommandId, payloadTLV, len(payloadTLV),
162+
ctypes.c_uint16(0), # interactionTimeoutMs
163+
ctypes.c_uint16(0), # busyWaitMs
164+
ctypes.c_bool(False) # suppressResponse
165+
))
166+
167+
147168
def SendCommand(future: Future, eventLoop, responseType: Type, device, commandPath: CommandPath, payload: ClusterCommand,
148-
timedRequestTimeoutMs: Union[None, int] = None, interactionTimeoutMs: Union[None, int] = None, busyWaitMs: Union[None, int] = None) -> PyChipError:
169+
timedRequestTimeoutMs: Union[None, int] = None, interactionTimeoutMs: Union[None, int] = None, busyWaitMs: Union[None, int] = None,
170+
suppressResponse: Union[None, bool] = None) -> PyChipError:
149171
''' Send a cluster-object encapsulated command to a device and does the following:
150172
- On receipt of a successful data response, returns the cluster-object equivalent through the provided future.
151173
- None (on a successful response containing no data)
@@ -175,6 +197,7 @@ def SendCommand(future: Future, eventLoop, responseType: Type, device, commandPa
175197
commandPath.ClusterId, commandPath.CommandId, payloadTLV, len(payloadTLV),
176198
ctypes.c_uint16(0 if interactionTimeoutMs is None else interactionTimeoutMs),
177199
ctypes.c_uint16(0 if busyWaitMs is None else busyWaitMs),
200+
ctypes.c_bool(False if suppressResponse is None else suppressResponse)
178201
))
179202

180203

@@ -203,7 +226,9 @@ def Init():
203226
setter = chip.native.NativeLibraryHandleMethodArguments(handle)
204227

205228
setter.Set('pychip_CommandSender_SendCommand',
206-
PyChipError, [py_object, c_void_p, c_uint16, c_uint32, c_uint32, c_char_p, c_size_t, c_uint16])
229+
PyChipError, [py_object, c_void_p, c_uint16, c_uint32, c_uint32, c_char_p, c_size_t, c_uint16, c_bool])
230+
setter.Set('pychip_CommandSender_TestOnlySendCommandTimedRequestNoTimedInvoke',
231+
PyChipError, [py_object, c_void_p, c_uint32, c_uint32, c_char_p, c_size_t, c_uint16, c_bool])
207232
setter.Set('pychip_CommandSender_SendGroupCommand',
208233
PyChipError, [c_uint16, c_void_p, c_uint32, c_uint32, c_char_p, c_size_t, c_uint16])
209234
setter.Set('pychip_CommandSender_InitCallbacks', None, [

0 commit comments

Comments
 (0)