blob: aa515d07c285995bb73c528de20ee68ab611d148 [file] [log] [blame]
/*
* Copyright (c) 2025 Project CHIP Authors
* All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <pw_unit_test/framework.h>
#include <app/clusters/user-label-server/UserLabelCluster.h>
#include <app/server-cluster/AttributeListBuilder.h>
#include <app/server-cluster/testing/AttributeTesting.h>
#include <app/server-cluster/testing/ClusterTester.h>
#include <app/server-cluster/testing/TestServerClusterContext.h>
#include <clusters/UserLabel/Attributes.h>
#include <clusters/UserLabel/Enums.h>
#include <clusters/UserLabel/Metadata.h>
#include <clusters/UserLabel/Structs.h>
#include <platform/DeviceInfoProvider.h>
namespace {
using namespace chip;
using namespace chip::app;
using namespace chip::app::Clusters;
using namespace chip::app::Clusters::UserLabel;
using namespace chip::app::Clusters::UserLabel::Attributes;
using namespace chip::Testing;
// Mock DeviceInfoProvider for testing
class MockDeviceInfoProvider : public DeviceLayer::DeviceInfoProvider
{
public:
MockDeviceInfoProvider() = default;
~MockDeviceInfoProvider() override = default;
FixedLabelIterator * IterateFixedLabel(EndpointId endpoint) override { return nullptr; }
UserLabelIterator * IterateUserLabel(EndpointId endpoint) override { return nullptr; }
SupportedCalendarTypesIterator * IterateSupportedCalendarTypes() override { return nullptr; }
SupportedLocalesIterator * IterateSupportedLocales() override { return nullptr; }
protected:
// Simple no-op implementations - we only need these to return success
// so that the cluster's validation logic can be tested
CHIP_ERROR SetUserLabelLength(EndpointId endpoint, size_t val) override { return CHIP_NO_ERROR; }
CHIP_ERROR GetUserLabelLength(EndpointId endpoint, size_t & val) override
{
val = 0;
return CHIP_NO_ERROR;
}
CHIP_ERROR SetUserLabelAt(EndpointId endpoint, size_t index, const UserLabelType & userLabel) override { return CHIP_NO_ERROR; }
CHIP_ERROR DeleteUserLabelAt(EndpointId endpoint, size_t index) override { return CHIP_NO_ERROR; }
};
struct TestUserLabelCluster : public ::testing::Test
{
static void SetUpTestSuite() { ASSERT_EQ(chip::Platform::MemoryInit(), CHIP_NO_ERROR); }
static void TearDownTestSuite() { chip::Platform::MemoryShutdown(); }
void SetUp() override
{
DeviceLayer::SetDeviceInfoProvider(&mDeviceInfoProvider);
ASSERT_EQ(userLabel.Startup(testContext.Get()), CHIP_NO_ERROR);
}
void TearDown() override
{
userLabel.Shutdown(ClusterShutdownType::kClusterShutdown);
DeviceLayer::SetDeviceInfoProvider(nullptr);
}
TestUserLabelCluster() : userLabel(kRootEndpointId) {}
TestServerClusterContext testContext;
UserLabelCluster userLabel;
MockDeviceInfoProvider mDeviceInfoProvider;
};
} // namespace
TEST_F(TestUserLabelCluster, AttributeTest)
{
ReadOnlyBufferBuilder<DataModel::AttributeEntry> attributes;
ASSERT_EQ(userLabel.Attributes(ConcreteClusterPath(kRootEndpointId, UserLabel::Id), attributes), CHIP_NO_ERROR);
ReadOnlyBufferBuilder<DataModel::AttributeEntry> expected;
AttributeListBuilder listBuilder(expected);
ASSERT_EQ(listBuilder.Append(Span(UserLabel::Attributes::kMandatoryMetadata), {}), CHIP_NO_ERROR);
ASSERT_TRUE(chip::Testing::EqualAttributeSets(attributes.TakeBuffer(), expected.TakeBuffer()));
}
TEST_F(TestUserLabelCluster, ReadAttributeTest)
{
ClusterTester tester(userLabel);
uint16_t revision{};
ASSERT_EQ(tester.ReadAttribute(Globals::Attributes::ClusterRevision::Id, revision), CHIP_NO_ERROR);
uint32_t features{};
ASSERT_EQ(tester.ReadAttribute(FeatureMap::Id, features), CHIP_NO_ERROR);
DataModel::DecodableList<Structs::LabelStruct::DecodableType> labelList;
ASSERT_EQ(tester.ReadAttribute(LabelList::Id, labelList), CHIP_NO_ERROR);
auto it = labelList.begin();
while (it.Next())
{
ASSERT_GT(it.GetValue().label.size(), 0u);
}
}
TEST_F(TestUserLabelCluster, WriteValidLabelListTest)
{
ClusterTester tester(userLabel);
Structs::LabelStruct::Type labels[] = {
{ .label = "room"_span, .value = "bedroom 2"_span },
{ .label = "orientation"_span, .value = "North"_span },
};
ASSERT_EQ(tester.WriteAttribute(LabelList::Id, DataModel::List(labels)), CHIP_NO_ERROR);
}
TEST_F(TestUserLabelCluster, WriteLabelWithLabelTooLongTest)
{
ClusterTester tester(userLabel);
constexpr auto tooLongLabel = "this_label_is_way_too_long"_span;
static_assert(tooLongLabel.size() > UserLabelCluster::kMaxLabelSize);
Structs::LabelStruct::Type labels[] = {
{ .label = tooLongLabel, .value = "value"_span },
};
ASSERT_EQ(tester.WriteAttribute(LabelList::Id, DataModel::List(labels)), CHIP_IM_GLOBAL_STATUS(ConstraintError));
}
TEST_F(TestUserLabelCluster, WriteLabelWithValueTooLongTest)
{
ClusterTester tester(userLabel);
constexpr auto tooLongValue = "this_value_is_way_too_long"_span;
static_assert(tooLongValue.size() > UserLabelCluster::kMaxValueSize);
Structs::LabelStruct::Type labels[] = {
{ .label = "room"_span, .value = tooLongValue },
};
ASSERT_EQ(tester.WriteAttribute(LabelList::Id, DataModel::List(labels)), CHIP_IM_GLOBAL_STATUS(ConstraintError));
}
TEST_F(TestUserLabelCluster, WriteEmptyLabelsTest)
{
ClusterTester tester(userLabel);
Structs::LabelStruct::Type labels[] = {
{ .label = ""_span, .value = ""_span }, // empty label and value are allowed per spec
};
ASSERT_EQ(tester.WriteAttribute(LabelList::Id, DataModel::List(labels)), CHIP_NO_ERROR);
}
TEST_F(TestUserLabelCluster, WriteMaxSizeLabelListTest)
{
ClusterTester tester(userLabel);
std::array<Structs::LabelStruct::Type, chip::DeviceLayer::kMaxUserLabelListLength> labels;
for (size_t i = 0; i < labels.size(); i++)
{
labels[i].label = "label"_span;
labels[i].value = "value"_span;
}
ASSERT_EQ(tester.WriteAttribute(LabelList::Id, DataModel::List(labels.data(), labels.size())), CHIP_NO_ERROR);
}