Skip to content

Commit 4235ce5

Browse files
Add TestEventTrigger to the ICDManager
1 parent cd0e262 commit 4235ce5

File tree

9 files changed

+185
-25
lines changed

9 files changed

+185
-25
lines changed

.github/workflows/tests.yaml

+1
Original file line numberDiff line numberDiff line change
@@ -506,6 +506,7 @@ jobs:
506506
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-to json:out/trace_data/app-{SCRIPT_BASE_NAME}.json" --script "src/python_testing/TC_FAN_3_4.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --trace-to json:out/trace_data/test-{SCRIPT_BASE_NAME}.json --trace-to perfetto:out/trace_data/test-{SCRIPT_BASE_NAME}.perfetto"'
507507
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-to json:out/trace_data/app-{SCRIPT_BASE_NAME}.json" --script "src/python_testing/TC_FAN_3_5.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --trace-to json:out/trace_data/test-{SCRIPT_BASE_NAME}.json --trace-to perfetto:out/trace_data/test-{SCRIPT_BASE_NAME}.perfetto"'
508508
scripts/run_in_python_env.sh out/venv './scripts/tests/run_python_test.py --app out/linux-x64-lit-icd-ipv6only-no-ble-no-wifi-tsan-clang-test/lit-icd-app --factoryreset --app-args "--discriminator 1234 --KVS kvs1 --trace-to json:out/trace_data/app-{SCRIPT_BASE_NAME}.json" --script "src/python_testing/TC_ICDM_2_1.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 --trace-to json:out/trace_data/test-{SCRIPT_BASE_NAME}.json --trace-to perfetto:out/trace_data/test-{SCRIPT_BASE_NAME}.perfetto"'
509+
scripts/run_in_python_env.sh out/venv './scripts/tests/run_python_test.py --app out/linux-x64-lit-icd-ipv6only-no-ble-no-wifi-tsan-clang-test/lit-icd-app --factoryreset --app-args "--discriminator 1234 --KVS kvs1 --trace-to json:out/trace_data/app-{SCRIPT_BASE_NAME}.json --enable-key 000102030405060708090a0b0c0d0e0f" --script "src/python_testing/TC_ICDManagementCluster.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --hex-arg enableKey:000102030405060708090a0b0c0d0e0f --trace-to json:out/trace_data/test-{SCRIPT_BASE_NAME}.json --trace-to perfetto:out/trace_data/test-{SCRIPT_BASE_NAME}.perfetto"'
509510
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-to json:out/trace_data/app-{SCRIPT_BASE_NAME}.json" --script "src/python_testing/TC_IDM_1_2.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --trace-to json:out/trace_data/test-{SCRIPT_BASE_NAME}.json --trace-to perfetto:out/trace_data/test-{SCRIPT_BASE_NAME}.perfetto"'
510511
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-to json:out/trace_data/app-{SCRIPT_BASE_NAME}.json --enable-key 000102030405060708090a0b0c0d0e0f" --script "src/python_testing/TC_IDM_1_4.py" --script-args "--hex-arg PIXIT.DGGEN.TEST_EVENT_TRIGGER_KEY:000102030405060708090a0b0c0d0e0f --storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --trace-to json:out/trace_data/test-{SCRIPT_BASE_NAME}.json --trace-to perfetto:out/trace_data/test-{SCRIPT_BASE_NAME}.perfetto"'
511512
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-to json:out/trace_data/app-{SCRIPT_BASE_NAME}.json" --script "src/python_testing/TC_PWRTL_2_1.py" --script-args "--storage-path admin_storage.json --commissioning-method on-network --discriminator 1234 --passcode 20202021 --trace-to json:out/trace_data/test-{SCRIPT_BASE_NAME}.json --trace-to perfetto:out/trace_data/test-{SCRIPT_BASE_NAME}.perfetto"'

src/app/BUILD.gn

+4
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,10 @@ source_set("constants") {
146146
]
147147
}
148148

149+
source_set("test-event-trigger") {
150+
sources = [ "TestEventTriggerDelegate.h" ]
151+
}
152+
149153
# interaction-model is a static-library because it currently requires global functions (app/util/...) that are stubbed in different test files that depend on the app static_library
150154
# which in tern depens on the interaction-model.
151155
# Using source_set prevents the unit test to build correctly.

src/app/icd/server/BUILD.gn

+1
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,7 @@ source_set("manager") {
8181
":notifier",
8282
":observer",
8383
"${chip_root}/src/app:subscription-info-provider",
84+
"${chip_root}/src/app:test-event-trigger",
8485
"${chip_root}/src/credentials:credentials",
8586
"${chip_root}/src/messaging",
8687
]

src/app/icd/server/ICDManager.cpp

+30-2
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,8 @@ static_assert(UINT8_MAX >= CHIP_CONFIG_MAX_EXCHANGE_CONTEXTS,
4242
"ICDManager::mOpenExchangeContextCount cannot hold count for the max exchange count");
4343

4444
void ICDManager::Init(PersistentStorageDelegate * storage, FabricTable * fabricTable, Crypto::SymmetricKeystore * symmetricKeystore,
45-
Messaging::ExchangeManager * exchangeManager, SubscriptionsInfoProvider * subInfoProvider)
45+
Messaging::ExchangeManager * exchangeManager, SubscriptionsInfoProvider * subInfoProvider,
46+
TestEventTriggerDelegate * testEventTriggerDelegate)
4647
{
4748
#if CHIP_CONFIG_ENABLE_ICD_CIP
4849
VerifyOrDie(storage != nullptr);
@@ -52,6 +53,9 @@ void ICDManager::Init(PersistentStorageDelegate * storage, FabricTable * fabricT
5253
VerifyOrDie(subInfoProvider != nullptr);
5354
#endif // CHIP_CONFIG_ENABLE_ICD_CIP
5455

56+
VerifyOrDie(testEventTriggerDelegate);
57+
testEventTriggerDelegate->AddHandler(this);
58+
5559
#if CHIP_CONFIG_ENABLE_ICD_LIT
5660
// LIT ICD Verification Checks
5761
if (SupportsFeature(Feature::kLongIdleTimeSupport))
@@ -87,8 +91,11 @@ void ICDManager::Init(PersistentStorageDelegate * storage, FabricTable * fabricT
8791
UpdateOperationState(OperationalState::IdleMode);
8892
}
8993

90-
void ICDManager::Shutdown()
94+
void ICDManager::Shutdown(TestEventTriggerDelegate * testEventTriggerDelegate)
9195
{
96+
VerifyOrDie(testEventTriggerDelegate);
97+
testEventTriggerDelegate->RemoveHandler(this);
98+
9299
ICDNotifier::GetInstance().Unsubscribe(this);
93100

94101
// cancel any running timer of the icd
@@ -643,6 +650,27 @@ void ICDManager::ExtendActiveMode(Milliseconds16 extendDuration)
643650
}
644651
}
645652

653+
CHIP_ERROR ICDManager::HandleEventTrigger(uint64_t eventTrigger)
654+
{
655+
ICDTestEventTriggerEvent trigger = static_cast<ICDTestEventTriggerEvent>(eventTrigger);
656+
CHIP_ERROR err = CHIP_NO_ERROR;
657+
658+
switch (trigger)
659+
{
660+
case ICDTestEventTriggerEvent::kAddActiveModeReq:
661+
SetKeepActiveModeRequirements(KeepActiveFlag::kTestEventTriggerActiveMode, true);
662+
break;
663+
case ICDTestEventTriggerEvent::kRemoveActiveModeReq:
664+
SetKeepActiveModeRequirements(KeepActiveFlag::kTestEventTriggerActiveMode, false);
665+
break;
666+
default:
667+
err = CHIP_ERROR_INVALID_ARGUMENT;
668+
break;
669+
}
670+
671+
return err;
672+
}
673+
646674
ICDManager::ObserverPointer * ICDManager::RegisterObserver(ICDStateObserver * observer)
647675
{
648676
return mStateObserverPool.CreateObject(observer);

src/app/icd/server/ICDManager.h

+28-12
Original file line numberDiff line numberDiff line change
@@ -16,16 +16,11 @@
1616
*/
1717
#pragma once
1818

19-
#include <app-common/zap-generated/cluster-enums.h>
20-
2119
#include <app/icd/server/ICDServerConfig.h>
2220

23-
#if CHIP_CONFIG_ENABLE_ICD_CIP
24-
#include <app/icd/server/ICDCheckInSender.h> // nogncheck
25-
#include <app/icd/server/ICDMonitoringTable.h> // nogncheck
26-
#endif // CHIP_CONFIG_ENABLE_ICD_CIP
27-
21+
#include <app-common/zap-generated/cluster-enums.h>
2822
#include <app/SubscriptionsInfoProvider.h>
23+
#include <app/TestEventTriggerDelegate.h>
2924
#include <app/icd/server/ICDConfigurationData.h>
3025
#include <app/icd/server/ICDNotifier.h>
3126
#include <app/icd/server/ICDStateObserver.h>
@@ -38,6 +33,11 @@
3833
#include <platform/internal/CHIPDeviceLayerInternal.h>
3934
#include <system/SystemClock.h>
4035

36+
#if CHIP_CONFIG_ENABLE_ICD_CIP
37+
#include <app/icd/server/ICDCheckInSender.h> // nogncheck
38+
#include <app/icd/server/ICDMonitoringTable.h> // nogncheck
39+
#endif // CHIP_CONFIG_ENABLE_ICD_CIP
40+
4141
namespace chip {
4242
namespace Crypto {
4343
using SymmetricKeystore = SessionKeystore;
@@ -54,7 +54,7 @@ class TestICDManager;
5454
/**
5555
* @brief ICD Manager is responsible of processing the events and triggering the correct action for an ICD
5656
*/
57-
class ICDManager : public ICDListener
57+
class ICDManager : public ICDListener, public TestEventTriggerHandler
5858
{
5959
public:
6060
// This structure is used for the creation an ObjectPool of ICDStateObserver pointers
@@ -93,8 +93,9 @@ class ICDManager : public ICDListener
9393

9494
ICDManager() {}
9595
void Init(PersistentStorageDelegate * storage, FabricTable * fabricTable, Crypto::SymmetricKeystore * symmetricKeyStore,
96-
Messaging::ExchangeManager * exchangeManager, SubscriptionsInfoProvider * manager);
97-
void Shutdown();
96+
Messaging::ExchangeManager * exchangeManager, SubscriptionsInfoProvider * manager,
97+
TestEventTriggerDelegate * testEventTriggerDelegate);
98+
void Shutdown(TestEventTriggerDelegate * testEventTriggerDelegate);
9899
void UpdateICDMode();
99100
void UpdateOperationState(OperationalState state);
100101
void SetKeepActiveModeRequirements(KeepActiveFlags flag, bool state);
@@ -129,6 +130,15 @@ class ICDManager : public ICDListener
129130
*/
130131
uint32_t StayActiveRequest(uint32_t stayActiveDuration);
131132

133+
/**
134+
* @brief TestEventTriggerHandler for the ICD feature set
135+
*
136+
* @param eventTrigger Event trigger to handle.
137+
* @return CHIP_ERROR CHIP_NO_ERROR - No erros during the processing
138+
* CHIP_ERROR_INVALID_ARGUMENT - eventTrigger isn't a valid value
139+
*/
140+
CHIP_ERROR HandleEventTrigger(uint64_t eventTrigger) override;
141+
132142
#if CHIP_CONFIG_ENABLE_ICD_CIP
133143
void SendCheckInMsgs();
134144

@@ -165,14 +175,14 @@ class ICDManager : public ICDListener
165175
void OnSubscriptionReport() override;
166176

167177
protected:
178+
friend class TestICDManager;
179+
168180
/**
169181
* @brief Hepler function that extends the Active Mode duration as well as the Active Mode Jitter timer for the transition to
170182
* iddle mode.
171183
*/
172184
void ExtendActiveMode(System::Clock::Milliseconds16 extendDuration);
173185

174-
friend class TestICDManager;
175-
176186
static void OnIdleModeDone(System::Layer * aLayer, void * appState);
177187
static void OnActiveModeDone(System::Layer * aLayer, void * appState);
178188

@@ -191,6 +201,12 @@ class ICDManager : public ICDListener
191201
uint8_t mOpenExchangeContextCount = 0;
192202

193203
private:
204+
enum class ICDTestEventTriggerEvent : uint64_t
205+
{
206+
kAddActiveModeReq = 0x0046'0000'00000001,
207+
kRemoveActiveModeReq = 0x0046'0000'00000002,
208+
};
209+
194210
#if CHIP_CONFIG_ENABLE_ICD_CIP
195211
bool ShouldCheckInMsgsBeSentAtActiveModeFunction(FabricIndex aFabricIndex, NodeId subjectID);
196212

src/app/icd/server/ICDNotifier.h

+6-5
Original file line numberDiff line numberDiff line change
@@ -39,11 +39,12 @@ class ICDListener
3939
public:
4040
enum class KeepActiveFlagsValues : uint8_t
4141
{
42-
kCommissioningWindowOpen = 0x01,
43-
kFailSafeArmed = 0x02,
44-
kExchangeContextOpen = 0x04,
45-
kCheckInInProgress = 0x08,
46-
kInvalidFlag = 0x10, // Move up when adding more flags
42+
kCommissioningWindowOpen = 0x01,
43+
kFailSafeArmed = 0x02,
44+
kExchangeContextOpen = 0x04,
45+
kCheckInInProgress = 0x08,
46+
kTestEventTriggerActiveMode = 0x10,
47+
kInvalidFlag = 0x20, // Move up when adding more flags
4748
};
4849

4950
enum class ICDManagementEvents : uint8_t

src/app/server/Server.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -347,7 +347,7 @@ CHIP_ERROR Server::Init(const ServerInitParams & initParams)
347347
mICDManager.RegisterObserver(&app::DnssdServer::Instance());
348348

349349
mICDManager.Init(mDeviceStorage, &GetFabricTable(), mSessionKeystore, &mExchangeMgr,
350-
chip::app::InteractionModelEngine::GetInstance());
350+
chip::app::InteractionModelEngine::GetInstance(), mTestEventTriggerDelegate);
351351
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER
352352

353353
// This code is necessary to restart listening to existing groups after a reboot
@@ -592,7 +592,7 @@ void Server::Shutdown()
592592
Access::ResetAccessControlToDefault();
593593
Credentials::SetGroupDataProvider(nullptr);
594594
#if CHIP_CONFIG_ENABLE_ICD_SERVER
595-
mICDManager.Shutdown();
595+
mICDManager.Shutdown(mTestEventTriggerDelegate);
596596
#endif // CHIP_CONFIG_ENABLE_ICD_SERVER
597597
mAttributePersister.Shutdown();
598598
// TODO(16969): Remove chip::Platform::MemoryInit() call from Server class, it belongs to outer code

src/app/tests/TestICDManager.cpp

+47-4
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
*/
1818
#include <app/EventManagement.h>
1919
#include <app/SubscriptionsInfoProvider.h>
20+
#include <app/TestEventTriggerDelegate.h>
2021
#include <app/icd/server/ICDConfigurationData.h>
2122
#include <app/icd/server/ICDManager.h>
2223
#include <app/icd/server/ICDNotifier.h>
@@ -53,6 +54,8 @@ constexpr NodeId kClientNodeId12 = 0x100002;
5354
constexpr NodeId kClientNodeId21 = 0x200001;
5455
constexpr NodeId kClientNodeId22 = 0x200002;
5556

57+
const uint8_t kTestKey[16] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 };
58+
5659
constexpr uint8_t kKeyBuffer1a[] = {
5760
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f
5861
};
@@ -91,6 +94,22 @@ class TestSubscriptionsInfoProvider : public SubscriptionsInfoProvider
9194
bool mHasPersistedSubscription = false;
9295
};
9396

97+
class TestEventDelegate : public TestEventTriggerDelegate
98+
{
99+
public:
100+
TestEventDelegate() { mEnableKey = ByteSpan{ kTestKey }; };
101+
102+
explicit TestEventDelegate(const ByteSpan & enableKey) : mEnableKey(enableKey) {}
103+
104+
bool DoesEnableKeyMatch(const ByteSpan & enableKey) const override
105+
{
106+
return !mEnableKey.empty() && mEnableKey.data_equal(enableKey);
107+
}
108+
109+
private:
110+
ByteSpan mEnableKey;
111+
};
112+
94113
class TestContext : public chip::Test::AppContext
95114
{
96115
public:
@@ -115,16 +134,18 @@ class TestContext : public chip::Test::AppContext
115134
// Performs setup for each individual test in the test suite
116135
CHIP_ERROR SetUp() override
117136
{
137+
118138
ReturnErrorOnFailure(chip::Test::AppContext::SetUp());
119-
mICDManager.Init(&testStorage, &GetFabricTable(), &mKeystore, &GetExchangeManager(), &mSubInfoProvider);
139+
mICDManager.Init(&testStorage, &GetFabricTable(), &mKeystore, &GetExchangeManager(), &mSubInfoProvider,
140+
&mTestEventDelagate);
120141
mICDManager.RegisterObserver(&mICDStateObserver);
121142
return CHIP_NO_ERROR;
122143
}
123144

124145
// Performs teardown for each individual test in the test suite
125146
void TearDown() override
126147
{
127-
mICDManager.Shutdown();
148+
mICDManager.Shutdown(&mTestEventDelagate);
128149
chip::Test::AppContext::TearDown();
129150
}
130151

@@ -134,6 +155,7 @@ class TestContext : public chip::Test::AppContext
134155
TestSubscriptionsInfoProvider mSubInfoProvider;
135156
TestPersistentStorageDelegate testStorage;
136157
TestICDStateObserver mICDStateObserver;
158+
TestEventDelegate mTestEventDelagate;
137159

138160
private:
139161
System::Clock::ClockBase * mRealClock;
@@ -514,9 +536,9 @@ class TestICDManager
514536
uint32_t counter = ICDConfigurationData::GetInstance().GetICDCounter().GetValue();
515537

516538
// Shut down and reinit ICDManager to increment counter
517-
ctx->mICDManager.Shutdown();
539+
ctx->mICDManager.Shutdown(&(ctx->mTestEventDelagate));
518540
ctx->mICDManager.Init(&(ctx->testStorage), &(ctx->GetFabricTable()), &(ctx->mKeystore), &(ctx->GetExchangeManager()),
519-
&(ctx->mSubInfoProvider));
541+
&(ctx->mSubInfoProvider), &(ctx->mTestEventDelagate));
520542
ctx->mICDManager.RegisterObserver(&(ctx->mICDStateObserver));
521543

522544
NL_TEST_ASSERT_EQUALS(aSuite, counter + ICDConfigurationData::kICDCounterPersistenceIncrement,
@@ -692,6 +714,26 @@ class TestICDManager
692714
NL_TEST_ASSERT(aSuite, ctx->mICDManager.ShouldCheckInMsgsBeSentAtActiveModeFunction(kTestFabricIndex1, kClientNodeId11));
693715
}
694716
#endif // CHIP_CONFIG_PERSIST_SUBSCRIPTIONS
717+
718+
static void TestHandleTestEventTriggerActiveModeReq(nlTestSuite * aSuite, void * aContext)
719+
{
720+
TestContext * ctx = static_cast<TestContext *>(aContext);
721+
722+
// Verify That ICDManager starts in Idle
723+
NL_TEST_ASSERT(aSuite, ctx->mICDManager.mOperationalState == ICDManager::OperationalState::IdleMode);
724+
725+
// Add ActiveMode req for the Test event trigger event
726+
ctx->mICDManager.HandleEventTrigger(static_cast<uint64_t>(ICDManager::ICDTestEventTriggerEvent::kAddActiveModeReq));
727+
NL_TEST_ASSERT(aSuite, ctx->mICDManager.mOperationalState == ICDManager::OperationalState::ActiveMode);
728+
729+
// Advance clock by the ActiveModeDuration and check that the device is still in ActiveMode
730+
AdvanceClockAndRunEventLoop(ctx, ICDConfigurationData::GetInstance().GetActiveModeDuration() + 1_ms32);
731+
NL_TEST_ASSERT(aSuite, ctx->mICDManager.mOperationalState == ICDManager::OperationalState::ActiveMode);
732+
733+
// Remove req and device should go to IdleMode
734+
ctx->mICDManager.HandleEventTrigger(static_cast<uint64_t>(ICDManager::ICDTestEventTriggerEvent::kRemoveActiveModeReq));
735+
NL_TEST_ASSERT(aSuite, ctx->mICDManager.mOperationalState == ICDManager::OperationalState::IdleMode);
736+
}
695737
};
696738

697739
} // namespace app
@@ -710,6 +752,7 @@ static const nlTest sTests[] = {
710752
NL_TEST_DEF("TestICDCounter", TestICDManager::TestICDCounter),
711753
NL_TEST_DEF("TestICDStayActive", TestICDManager::TestICDMStayActive),
712754
NL_TEST_DEF("TestShouldCheckInMsgsBeSentAtActiveModeFunction", TestICDManager::TestShouldCheckInMsgsBeSentAtActiveModeFunction),
755+
NL_TEST_DEF("TestHandleTestEventTriggerActiveModeReq", TestICDManager::TestHandleTestEventTriggerActiveModeReq),
713756
NL_TEST_SENTINEL(),
714757
};
715758

0 commit comments

Comments
 (0)