| /* |
| * |
| * Copyright (c) 2023 Project CHIP Authors |
| * |
| * 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 <app-common/zap-generated/cluster-objects.h> |
| #include <app/data-model/Decode.h> |
| #include <app/data-model/Encode.h> |
| #include <lib/core/TLVDebug.h> |
| #include <lib/core/TLVReader.h> |
| #include <lib/support/UnitTestRegistration.h> |
| #include <lib/support/jsontlv/JsonToTlv.h> |
| #include <lib/support/jsontlv/TextFormat.h> |
| #include <lib/support/jsontlv/TlvToJson.h> |
| #include <nlunit-test.h> |
| |
| namespace { |
| |
| using namespace chip::Encoding; |
| using namespace chip; |
| using namespace chip::app; |
| |
| constexpr uint32_t kImplicitProfileId = 0x1234; |
| |
| uint8_t gBuf1[1024]; |
| uint8_t gBuf2[1024]; |
| TLV::TLVWriter gWriter1; |
| TLV::TLVWriter gWriter2; |
| nlTestSuite * gSuite; |
| |
| void SetupWriters() |
| { |
| gWriter1.Init(gBuf1); |
| gWriter1.ImplicitProfileId = kImplicitProfileId; |
| |
| gWriter2.Init(gBuf2); |
| gWriter2.ImplicitProfileId = kImplicitProfileId; |
| } |
| |
| void PrintBytes(const uint8_t * buf, uint32_t len) |
| { |
| for (uint32_t i = 0; i < len; i++) |
| { |
| printf("%02X ", buf[i]); |
| } |
| printf("\n"); |
| } |
| |
| bool MatchWriter1and2() |
| { |
| auto matches = |
| (gWriter1.GetLengthWritten() == gWriter2.GetLengthWritten()) && (memcmp(gBuf1, gBuf2, gWriter1.GetLengthWritten()) == 0); |
| |
| if (!matches) |
| { |
| printf("Didn't match!\n"); |
| printf("Reference:\n"); |
| PrintBytes(gBuf1, gWriter1.GetLengthWritten()); |
| |
| printf("Generated:\n"); |
| PrintBytes(gBuf2, gWriter2.GetLengthWritten()); |
| } |
| |
| return matches; |
| } |
| |
| template <typename T> |
| void ConvertJsonToTlvAndValidate(T val, const std::string & jsonString) |
| { |
| CHIP_ERROR err; |
| TLV::TLVType container; |
| |
| SetupWriters(); |
| |
| err = gWriter1.StartContainer(TLV::AnonymousTag(), TLV::kTLVType_Structure, container); |
| NL_TEST_ASSERT(gSuite, err == CHIP_NO_ERROR); |
| |
| err = DataModel::Encode(gWriter1, TLV::ContextTag(1), val); |
| NL_TEST_ASSERT(gSuite, err == CHIP_NO_ERROR); |
| |
| err = gWriter1.EndContainer(container); |
| NL_TEST_ASSERT(gSuite, err == CHIP_NO_ERROR); |
| |
| err = gWriter1.Finalize(); |
| NL_TEST_ASSERT(gSuite, err == CHIP_NO_ERROR); |
| |
| err = JsonToTlv(jsonString, gWriter2); |
| NL_TEST_ASSERT(gSuite, err == CHIP_NO_ERROR); |
| |
| NL_TEST_ASSERT(gSuite, MatchWriter1and2()); |
| } |
| |
| void TestConverter(nlTestSuite * inSuite, void * inContext) |
| { |
| std::string jsonString; |
| |
| gSuite = inSuite; |
| |
| jsonString = "{\n" |
| " \"1:UINT\" : 30\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(static_cast<uint32_t>(30), jsonString); |
| |
| jsonString = "{\n" |
| " \"1:INT\" : -30\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(static_cast<int32_t>(-30), jsonString); |
| |
| jsonString = "{\n" |
| " \"1:BOOL\" : false\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(false, jsonString); |
| |
| jsonString = "{\n" |
| " \"1:BOOL\" : true\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(true, jsonString); |
| |
| jsonString = "{\n" |
| " \"1:FLOAT\" : 1.0\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(static_cast<float>(1.0), jsonString); |
| |
| jsonString = "{\n" |
| " \"1:DOUBLE\" : 1.0\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(static_cast<double>(1.0), jsonString); |
| |
| jsonString = "{\n" |
| " \"1:STRING\" : \"hello\"\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate("hello"_span, jsonString); |
| |
| // Validated using https://base64.guru/converter/encode/hex |
| const uint8_t byteBuf[] = { 0x01, 0x02, 0x03, 0x04, 0xff, 0xfe, 0x99, 0x88, 0xdd, 0xcd }; |
| ByteSpan byteSpan(byteBuf); |
| jsonString = "{\n" |
| " \"1:BYTES\" : \"AQIDBP/+mYjdzQ==\"\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(byteSpan, jsonString); |
| |
| DataModel::Nullable<uint8_t> nullValue; |
| jsonString = "{\n" |
| " \"1:NULL\" : null\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(nullValue, jsonString); |
| |
| Clusters::UnitTesting::Structs::SimpleStruct::Type structVal; |
| structVal.a = 20; |
| structVal.b = true; |
| structVal.d = byteBuf; |
| structVal.e = "hello"_span; |
| structVal.g = static_cast<float>(1.0); |
| structVal.h = static_cast<double>(1.0); |
| |
| jsonString = "{\n" |
| " \"1:STRUCT\" : {\n" |
| " \"0:UINT\" : 20,\n" |
| " \"1:BOOL\" : true,\n" |
| " \"2:UINT\" : 0,\n" |
| " \"3:BYTES\" : \"AQIDBP/+mYjdzQ==\",\n" |
| " \"4:STRING\" : \"hello\",\n" |
| " \"5:UINT\" : 0,\n" |
| " \"6:FLOAT\" : 1.0,\n" |
| " \"7:DOUBLE\" : 1.0\n" |
| " }\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(structVal, jsonString); |
| |
| uint8_t int8uListData[] = { 1, 2, 3, 4 }; |
| DataModel::List<uint8_t> int8uList; |
| |
| int8uList = int8uListData; |
| jsonString = "{\n" |
| " \"1:ARRAY-UINT\" : [ 1, 2, 3, 4 ]\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(int8uList, jsonString); |
| |
| int8uList = {}; |
| jsonString = "{\n" |
| " \"1:ARRAY-?\" : [ ]\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(int8uList, jsonString); |
| |
| DataModel::Nullable<DataModel::List<uint8_t>> nullValueList; |
| jsonString = "{\n" |
| " \"1:NULL\" : null\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(nullValueList, jsonString); |
| |
| Clusters::UnitTesting::Structs::SimpleStruct::Type structListData[2] = { structVal, structVal }; |
| DataModel::List<Clusters::UnitTesting::Structs::SimpleStruct::Type> structList; |
| |
| structList = structListData; |
| jsonString = "{\n" |
| " \"1:ARRAY-STRUCT\" : [\n" |
| " {\n" |
| " \"0:UINT\" : 20,\n" |
| " \"1:BOOL\" : true,\n" |
| " \"2:UINT\" : 0,\n" |
| " \"3:BYTES\" : \"AQIDBP/+mYjdzQ==\",\n" |
| " \"4:STRING\" : \"hello\",\n" |
| " \"5:UINT\" : 0,\n" |
| " \"6:FLOAT\" : 1.0,\n" |
| " \"7:DOUBLE\" : 1.0\n" |
| " },\n" |
| " {\n" |
| " \"0:UINT\" : 20,\n" |
| " \"1:BOOL\" : true,\n" |
| " \"2:UINT\" : 0,\n" |
| " \"3:BYTES\" : \"AQIDBP/+mYjdzQ==\",\n" |
| " \"4:STRING\" : \"hello\",\n" |
| " \"5:UINT\" : 0,\n" |
| " \"6:FLOAT\" : 1.0,\n" |
| " \"7:DOUBLE\" : 1.0\n" |
| " }\n" |
| " ]\n" |
| "}\n"; |
| ConvertJsonToTlvAndValidate(structList, jsonString); |
| } |
| |
| void Test32BitConvert(nlTestSuite * inSuite, void * inContext) |
| { |
| // JSON TLV format explicitly wants to support 32-bit integer preservation. |
| // |
| // This is to support encode/decode of a format like: |
| // { "123456:BOOL" : true } to be a compact way of encoding |
| // "attribute id 123456 has value true" |
| // |
| // Such an encoding is NOT part of the matter spec, so best-effort is done here: |
| // - low ids are encoded as context tags (this is in the spec for any structure encoding) |
| // - large ids are encoded as implicit tags (NOT used in spec as spec never has such high ids) |
| TLV::TLVReader reader; |
| TLV::TLVType tlvType; |
| int32_t value = 0; |
| |
| // convert a simple single value |
| { |
| SetupWriters(); |
| JsonToTlv("{\"1:INT\": 321}", gWriter1); |
| NL_TEST_ASSERT(inSuite, gWriter1.Finalize() == CHIP_NO_ERROR); |
| |
| reader.Init(gBuf1, gWriter1.GetLengthWritten()); |
| reader.ImplicitProfileId = kImplicitProfileId; |
| |
| NL_TEST_ASSERT(inSuite, reader.Next(TLV::AnonymousTag()) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.GetType() == TLV::kTLVType_Structure); |
| NL_TEST_ASSERT(inSuite, reader.EnterContainer(tlvType) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.Next(TLV::ContextTag(1)) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.GetType() == TLV::kTLVType_SignedInteger); |
| NL_TEST_ASSERT(inSuite, reader.Get(value) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, value == 321); |
| NL_TEST_ASSERT(inSuite, reader.Next() == CHIP_END_OF_TLV); |
| NL_TEST_ASSERT(inSuite, reader.ExitContainer(tlvType) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.Next() == CHIP_END_OF_TLV); |
| } |
| |
| // convert a single value that is larger than 8 bit |
| { |
| SetupWriters(); |
| JsonToTlv("{\"1234:INT\": 321}", gWriter1); |
| NL_TEST_ASSERT(inSuite, gWriter1.Finalize() == CHIP_NO_ERROR); |
| |
| reader.Init(gBuf1, gWriter1.GetLengthWritten()); |
| reader.ImplicitProfileId = kImplicitProfileId; |
| |
| NL_TEST_ASSERT(inSuite, reader.Next(TLV::AnonymousTag()) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.GetType() == TLV::kTLVType_Structure); |
| NL_TEST_ASSERT(inSuite, reader.EnterContainer(tlvType) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.Next(TLV::ProfileTag(kImplicitProfileId, 1234)) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.GetType() == TLV::kTLVType_SignedInteger); |
| NL_TEST_ASSERT(inSuite, reader.Get(value) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, value == 321); |
| NL_TEST_ASSERT(inSuite, reader.Next() == CHIP_END_OF_TLV); |
| NL_TEST_ASSERT(inSuite, reader.ExitContainer(tlvType) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.Next() == CHIP_END_OF_TLV); |
| } |
| |
| // Convert to a full 32-bit value, unsigned |
| { |
| SetupWriters(); |
| JsonToTlv("{\"4275878552:INT\": 321}", gWriter1); |
| NL_TEST_ASSERT(inSuite, gWriter1.Finalize() == CHIP_NO_ERROR); |
| |
| reader.Init(gBuf1, gWriter1.GetLengthWritten()); |
| reader.ImplicitProfileId = kImplicitProfileId; |
| |
| NL_TEST_ASSERT(inSuite, reader.Next(TLV::AnonymousTag()) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.GetType() == TLV::kTLVType_Structure); |
| NL_TEST_ASSERT(inSuite, reader.EnterContainer(tlvType) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.Next(TLV::ProfileTag(kImplicitProfileId, 0xFEDCBA98u)) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.GetType() == TLV::kTLVType_SignedInteger); |
| NL_TEST_ASSERT(inSuite, reader.Get(value) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, value == 321); |
| NL_TEST_ASSERT(inSuite, reader.Next() == CHIP_END_OF_TLV); |
| NL_TEST_ASSERT(inSuite, reader.ExitContainer(tlvType) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, reader.Next() == CHIP_END_OF_TLV); |
| } |
| |
| // FIXME: implement |
| } |
| |
| int Initialize(void * apSuite) |
| { |
| VerifyOrReturnError(chip::Platform::MemoryInit() == CHIP_NO_ERROR, FAILURE); |
| return SUCCESS; |
| } |
| |
| int Finalize(void * aContext) |
| { |
| chip::Platform::MemoryShutdown(); |
| return SUCCESS; |
| } |
| |
| // clang-format off |
| const nlTest sTests[] = |
| { |
| NL_TEST_DEF("TestConverter", TestConverter), |
| NL_TEST_DEF("Test32BitConvert", Test32BitConvert), |
| NL_TEST_SENTINEL() |
| }; |
| // clang-format on |
| |
| } // namespace |
| |
| int TestJsonToTlv() |
| { |
| nlTestSuite theSuite = { "JsonToTlv", sTests, Initialize, Finalize }; |
| nlTestRunner(&theSuite, nullptr); |
| return nlTestRunnerStats(&theSuite); |
| } |
| |
| CHIP_REGISTER_TEST_SUITE(TestJsonToTlv) |