Skip to content

Commit de342bd

Browse files
andy31415andreilitvinrestyled-commitsbzbarsky-apple
authored
Remove a few more calls of IME to ember-compatibility. (project-chip#35749)
* Remove a few more calls of IME to ember-compatibility. This works to move more logic into DataModel::Provider calls to decouple interactionmodel engine from ember calls. * Remove auto-include logic * Some work to make tests actually pass. Starts adding CHI-based enumeration of commands * Restyle * Add unit tests and fixes for codgen data provider actually honoring the generated/accepted command callbacks * Add support for using CommandHandlerInterface for accepted/generated commands. Note that iteration is still O(N^2) which is not ideal, however at least the use of this is infrequent and list of commands is somewhat short. * Fix includes * Comment update * Comment update * Make TestServerCommandDispatch pass * Unit tests pass * Remove unused include * Remove unused include * Restyled by clang-format * Remove debug code * Update src/app/InteractionModelEngine.cpp Co-authored-by: Boris Zbarsky <bzbarsky@apple.com> * Make TestRead actually pass * Restyle * Make TestAclAttribute pass * make test reporting engine pass * Undo hardcoding of attribute info and make test read interaction use a custom mock model * Remove do not submit code * Test passes now * Restyle * Config reset not used * Better fix for TestCommandInteraction * Restyle * Fix up data model for test commands as well * Self review: fix up includes * Self review: fix up includes * Self review: fix up includes * Restyle * DELETE IntnteractionModelTest.py and chip im-responder/im-initiator * Method rename * Revert "DELETE IntnteractionModelTest.py and chip im-responder/im-initiator" This reverts commit 4a6dd20. * Disable InteractionModelTest without deletion --------- Co-authored-by: Andrei Litvin <andreilitvin@google.com> Co-authored-by: Restyled.io <commits@restyled.io> Co-authored-by: Boris Zbarsky <bzbarsky@apple.com>
1 parent f54e3c8 commit de342bd

14 files changed

+405
-40
lines changed

scripts/tests/cirque_tests.sh

+5-1
Original file line numberDiff line numberDiff line change
@@ -37,13 +37,17 @@ OT_SIMULATION_CACHE="$CIRQUE_CACHE_PATH/ot-simulation-cmake.tgz"
3737
OT_SIMULATION_CACHE_STAMP_FILE="$CIRQUE_CACHE_PATH/ot-simulation.commit"
3838

3939
# Append test name here to add more tests for run_all_tests
40+
#
41+
# NOTE:
42+
# "InteractionModelTest" is currently disabled due to it overriding
43+
# internal data model methods (for example it says "CommandExists" for
44+
# paths where endpoint/cluster do not)
4045
CIRQUE_TESTS=(
4146
"EchoTest"
4247
"EchoOverTcpTest"
4348
"FailsafeTest"
4449
"MobileDeviceTest"
4550
"CommissioningTest"
46-
"InteractionModelTest"
4751
"IcdWaitForActiveTest"
4852
"SplitCommissioningTest"
4953
"CommissioningFailureTest"

src/app/InteractionModelEngine.cpp

+55-8
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@
3636
#include <app/util/af-types.h>
3737
#include <app/util/ember-compatibility-functions.h>
3838
#include <app/util/endpoint-config-api.h>
39+
#include <lib/core/DataModelTypes.h>
3940
#include <lib/core/Global.h>
4041
#include <lib/core/TLVUtilities.h>
4142
#include <lib/support/CHIPFaultInjection.h>
@@ -515,7 +516,8 @@ CHIP_ERROR InteractionModelEngine::ParseAttributePaths(const Access::SubjectDesc
515516
{
516517
ConcreteAttributePath concretePath(paramsList.mValue.mEndpointId, paramsList.mValue.mClusterId,
517518
paramsList.mValue.mAttributeId);
518-
if (ConcreteAttributePathExists(concretePath))
519+
520+
if (IsExistentAttributePath(concretePath))
519521
{
520522
Access::RequestPath requestPath{ .cluster = concretePath.mClusterId,
521523
.endpoint = concretePath.mEndpointId,
@@ -1584,16 +1586,19 @@ CHIP_ERROR InteractionModelEngine::PushFrontAttributePathList(SingleLinkedListNo
15841586
return err;
15851587
}
15861588

1587-
bool InteractionModelEngine::IsExistingAttributePath(const ConcreteAttributePath & path)
1589+
bool InteractionModelEngine::IsExistentAttributePath(const ConcreteAttributePath & path)
15881590
{
15891591
#if CHIP_CONFIG_USE_DATA_MODEL_INTERFACE
15901592
#if CHIP_CONFIG_USE_EMBER_DATA_MODEL
1591-
// Ensure that Provider interface and ember are IDENTICAL in attribute location (i.e. "check" mode)
1592-
VerifyOrDie(GetDataModelProvider()
1593-
->GetAttributeInfo(ConcreteAttributePath(path.mEndpointId, path.mClusterId, path.mAttributeId))
1594-
.has_value() == emberAfContainsAttribute(path.mEndpointId, path.mClusterId, path.mAttributeId)
15951593

1596-
);
1594+
bool providerResult = GetDataModelProvider()
1595+
->GetAttributeInfo(ConcreteAttributePath(path.mEndpointId, path.mClusterId, path.mAttributeId))
1596+
.has_value();
1597+
1598+
bool emberResult = emberAfContainsAttribute(path.mEndpointId, path.mClusterId, path.mAttributeId);
1599+
1600+
// Ensure that Provider interface and ember are IDENTICAL in attribute location (i.e. "check" mode)
1601+
VerifyOrDie(providerResult == emberResult);
15971602
#endif
15981603

15991604
return GetDataModelProvider()
@@ -1615,7 +1620,7 @@ void InteractionModelEngine::RemoveDuplicateConcreteAttributePath(SingleLinkedLi
16151620

16161621
// skip all wildcard paths and invalid concrete attribute
16171622
if (path1->mValue.IsWildcardPath() ||
1618-
!IsExistingAttributePath(
1623+
!IsExistentAttributePath(
16191624
ConcreteAttributePath(path1->mValue.mEndpointId, path1->mValue.mClusterId, path1->mValue.mAttributeId)))
16201625
{
16211626
prev = path1;
@@ -1766,7 +1771,49 @@ void InteractionModelEngine::DispatchCommand(CommandHandlerImpl & apCommandObj,
17661771

17671772
Protocols::InteractionModel::Status InteractionModelEngine::CommandExists(const ConcreteCommandPath & aCommandPath)
17681773
{
1774+
#if CHIP_CONFIG_USE_DATA_MODEL_INTERFACE
1775+
auto provider = GetDataModelProvider();
1776+
if (provider->GetAcceptedCommandInfo(aCommandPath).has_value())
1777+
{
1778+
#if CHIP_CONFIG_USE_EMBER_DATA_MODEL
1779+
VerifyOrDie(ServerClusterCommandExists(aCommandPath) == Protocols::InteractionModel::Status::Success);
1780+
#endif
1781+
return Protocols::InteractionModel::Status::Success;
1782+
}
1783+
1784+
// We failed, figure out why ...
1785+
//
1786+
if (provider->GetClusterInfo(aCommandPath).has_value())
1787+
{
1788+
#if CHIP_CONFIG_USE_EMBER_DATA_MODEL
1789+
VerifyOrDie(ServerClusterCommandExists(aCommandPath) == Protocols::InteractionModel::Status::UnsupportedCommand);
1790+
#endif
1791+
return Protocols::InteractionModel::Status::UnsupportedCommand; // cluster exists, so command is invalid
1792+
}
1793+
1794+
// At this point either cluster or endpoint does not exist. If we find the endpoint, then the cluster
1795+
// is invalid
1796+
for (EndpointId endpoint = provider->FirstEndpoint(); endpoint != kInvalidEndpointId;
1797+
endpoint = provider->NextEndpoint(endpoint))
1798+
{
1799+
if (endpoint == aCommandPath.mEndpointId)
1800+
{
1801+
#if CHIP_CONFIG_USE_EMBER_DATA_MODEL
1802+
VerifyOrDie(ServerClusterCommandExists(aCommandPath) == Protocols::InteractionModel::Status::UnsupportedCluster);
1803+
#endif
1804+
// endpoint exists, so cluster is invalid
1805+
return Protocols::InteractionModel::Status::UnsupportedCluster;
1806+
}
1807+
}
1808+
1809+
// endpoint not found
1810+
#if CHIP_CONFIG_USE_EMBER_DATA_MODEL
1811+
VerifyOrDie(ServerClusterCommandExists(aCommandPath) == Protocols::InteractionModel::Status::UnsupportedEndpoint);
1812+
#endif
1813+
return Protocols::InteractionModel::Status::UnsupportedEndpoint;
1814+
#else
17691815
return ServerClusterCommandExists(aCommandPath);
1816+
#endif
17701817
}
17711818

17721819
DataModel::Provider * InteractionModelEngine::SetDataModelProvider(DataModel::Provider * model)

src/app/InteractionModelEngine.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -508,6 +508,7 @@ class InteractionModelEngine : public Messaging::UnsolicitedMessageHandler,
508508

509509
void DispatchCommand(CommandHandlerImpl & apCommandObj, const ConcreteCommandPath & aCommandPath,
510510
TLV::TLVReader & apPayload) override;
511+
511512
Protocols::InteractionModel::Status CommandExists(const ConcreteCommandPath & aCommandPath) override;
512513

513514
bool HasActiveRead();
@@ -615,7 +616,7 @@ class InteractionModelEngine : public Messaging::UnsolicitedMessageHandler,
615616
/**
616617
* Check if the given attribute path is a valid path in the data model provider.
617618
*/
618-
bool IsExistingAttributePath(const ConcreteAttributePath & path);
619+
bool IsExistentAttributePath(const ConcreteAttributePath & path);
619620

620621
static void ResumeSubscriptionsTimerCallback(System::Layer * apSystemLayer, void * apAppState);
621622

src/app/tests/TestAclAttribute.cpp

+20-2
Original file line numberDiff line numberDiff line change
@@ -41,11 +41,27 @@
4141
#include <messaging/Flags.h>
4242
#include <protocols/interaction_model/Constants.h>
4343

44-
#include <type_traits>
45-
4644
namespace {
4745
using namespace chip;
4846
using namespace chip::Access;
47+
using namespace chip::Test;
48+
49+
const MockNodeConfig & TestMockNodeConfig()
50+
{
51+
using namespace chip::app;
52+
using namespace chip::app::Clusters::Globals::Attributes;
53+
54+
// clang-format off
55+
static const MockNodeConfig config({
56+
MockEndpointConfig(kTestEndpointId, {
57+
MockClusterConfig(kTestClusterId, {
58+
ClusterRevision::Id, FeatureMap::Id, 1, 2
59+
}),
60+
}),
61+
});
62+
// clang-format on
63+
return config;
64+
}
4965

5066
class TestAccessControlDelegate : public AccessControl::Delegate
5167
{
@@ -119,10 +135,12 @@ class TestAclAttribute : public Test::AppContext
119135
Access::GetAccessControl().Finish();
120136
Access::GetAccessControl().Init(GetTestAccessControlDelegate(), gDeviceTypeResolver);
121137
mOldProvider = InteractionModelEngine::GetInstance()->SetDataModelProvider(&TestImCustomDataModel::Instance());
138+
chip::Test::SetMockNodeConfig(TestMockNodeConfig());
122139
}
123140

124141
void TearDown() override
125142
{
143+
chip::Test::ResetMockNodeConfig();
126144
AppContext::TearDown();
127145
InteractionModelEngine::GetInstance()->SetDataModelProvider(mOldProvider);
128146
}

src/app/tests/TestCommandInteraction.cpp

+64
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
#include <app/AppConfig.h>
3232
#include <app/CommandHandlerImpl.h>
3333
#include <app/InteractionModelEngine.h>
34+
#include <app/data-model-provider/ActionReturnStatus.h>
3435
#include <app/data-model/Encode.h>
3536
#include <app/tests/AppTestContext.h>
3637
#include <app/tests/test-interaction-model-api.h>
@@ -96,6 +97,33 @@ class SimpleTLVPayload : public app::DataModel::EncodableToTLV
9697
}
9798
};
9899

100+
const chip::Test::MockNodeConfig & TestMockNodeConfig()
101+
{
102+
using namespace chip::app;
103+
using namespace chip::Test;
104+
using namespace chip::app::Clusters::Globals::Attributes;
105+
106+
// clang-format off
107+
static const MockNodeConfig config({
108+
MockEndpointConfig(chip::kTestEndpointId, {
109+
MockClusterConfig(Clusters::Identify::Id, {
110+
ClusterRevision::Id, FeatureMap::Id,
111+
},
112+
{}, // events
113+
{
114+
kTestCommandIdWithData,
115+
kTestCommandIdNoData,
116+
kTestCommandIdCommandSpecificResponse,
117+
kTestCommandIdFillResponseMessage,
118+
}, // accepted commands
119+
{} // generated commands
120+
),
121+
}),
122+
});
123+
// clang-format on
124+
return config;
125+
}
126+
99127
} // namespace
100128

101129
namespace app {
@@ -354,9 +382,42 @@ class MockCommandHandlerCallback : public CommandHandlerImpl::Callback
354382
int onFinalCalledTimes = 0;
355383
} mockCommandHandlerDelegate;
356384

385+
class TestCommandInteractionModel : public TestImCustomDataModel
386+
{
387+
public:
388+
static TestCommandInteractionModel * Instance()
389+
{
390+
static TestCommandInteractionModel instance;
391+
return &instance;
392+
}
393+
394+
TestCommandInteractionModel() {}
395+
396+
std::optional<DataModel::ActionReturnStatus> Invoke(const DataModel::InvokeRequest & request,
397+
chip::TLV::TLVReader & input_arguments, CommandHandler * handler)
398+
{
399+
DispatchSingleClusterCommand(request.path, input_arguments, handler);
400+
return std::nullopt; // handler status is set by the dispatch
401+
}
402+
};
403+
357404
class TestCommandInteraction : public chip::Test::AppContext
358405
{
359406
public:
407+
void SetUp() override
408+
{
409+
AppContext::SetUp();
410+
mOldProvider = InteractionModelEngine::GetInstance()->SetDataModelProvider(TestCommandInteractionModel::Instance());
411+
chip::Test::SetMockNodeConfig(TestMockNodeConfig());
412+
}
413+
414+
void TearDown() override
415+
{
416+
chip::Test::ResetMockNodeConfig();
417+
InteractionModelEngine::GetInstance()->SetDataModelProvider(mOldProvider);
418+
AppContext::TearDown();
419+
}
420+
360421
static size_t GetNumActiveCommandResponderObjects()
361422
{
362423
return chip::app::InteractionModelEngine::GetInstance()->mCommandResponderObjs.Allocated();
@@ -423,6 +484,9 @@ class TestCommandInteraction : public chip::Test::AppContext
423484
static void FillCurrentInvokeResponseBuffer(CommandHandlerImpl * apCommandHandler,
424485
const ConcreteCommandPath & aRequestCommandPath, uint32_t aSizeToLeaveInBuffer);
425486
static void ValidateCommandHandlerEncodeInvokeResponseMessage(bool aNeedStatusCode);
487+
488+
protected:
489+
chip::app::DataModel::Provider * mOldProvider = nullptr;
426490
};
427491

428492
class TestExchangeDelegate : public Messaging::ExchangeDelegate

0 commit comments

Comments
 (0)