| /* |
| * |
| * Copyright (c) 2020 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. |
| */ |
| |
| /** |
| * @file |
| * This file implements a process to effect a functional test for |
| * the Message Header class within the transport layer |
| * |
| */ |
| #include <support/CodeUtils.h> |
| #include <support/ErrorStr.h> |
| #include <support/UnitTestRegistration.h> |
| #include <transport/raw/MessageHeader.h> |
| |
| #include <nlunit-test.h> |
| |
| namespace { |
| |
| using namespace chip; |
| |
| void TestPacketHeaderInitialState(nlTestSuite * inSuite, void * inContext) |
| { |
| PacketHeader header; |
| |
| NL_TEST_ASSERT(inSuite, !header.IsSecureSessionControlMsg()); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 0); |
| NL_TEST_ASSERT(inSuite, header.GetEncryptionKeyID() == 0); |
| NL_TEST_ASSERT(inSuite, !header.GetDestinationNodeId().HasValue()); |
| NL_TEST_ASSERT(inSuite, !header.GetSourceNodeId().HasValue()); |
| } |
| |
| void TestPayloadHeaderInitialState(nlTestSuite * inSuite, void * inContext) |
| { |
| PayloadHeader header; |
| |
| NL_TEST_ASSERT(inSuite, header.GetMessageType() == 0); |
| NL_TEST_ASSERT(inSuite, header.GetExchangeID() == 0); |
| NL_TEST_ASSERT(inSuite, header.GetProtocolID() == 0); |
| NL_TEST_ASSERT(inSuite, !header.GetVendorId().HasValue()); |
| } |
| |
| void TestPacketHeaderEncodeDecode(nlTestSuite * inSuite, void * inContext) |
| { |
| PacketHeader header; |
| uint8_t buffer[64]; |
| uint16_t encodeLen; |
| uint16_t decodeLen; |
| |
| header.SetMessageId(123).SetPayloadLength(16); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| // change it to verify decoding |
| header.SetMessageId(222).SetSourceNodeId(1).SetDestinationNodeId(2); |
| |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encodeLen == decodeLen); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 123); |
| NL_TEST_ASSERT(inSuite, !header.GetDestinationNodeId().HasValue()); |
| |
| header.SetSourceNodeId(55); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| // change it to verify decoding |
| header.SetMessageId(222).SetSourceNodeId(1).SetDestinationNodeId(2); |
| |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encodeLen == decodeLen); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 123); |
| NL_TEST_ASSERT(inSuite, !header.GetDestinationNodeId().HasValue()); |
| NL_TEST_ASSERT(inSuite, header.GetSourceNodeId() == Optional<uint64_t>::Value(55)); |
| |
| header.ClearSourceNodeId().SetDestinationNodeId(11); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| // change it to verify decoding |
| header.SetMessageId(222).SetSourceNodeId(1).SetDestinationNodeId(2); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encodeLen == decodeLen); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 123); |
| NL_TEST_ASSERT(inSuite, header.GetDestinationNodeId() == Optional<uint64_t>::Value(11)); |
| NL_TEST_ASSERT(inSuite, !header.GetSourceNodeId().HasValue()); |
| |
| header.SetMessageId(234).SetSourceNodeId(77).SetDestinationNodeId(88); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| // change it to verify decoding |
| header.SetMessageId(222).SetSourceNodeId(1).SetDestinationNodeId(2); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encodeLen == decodeLen); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 234); |
| NL_TEST_ASSERT(inSuite, header.GetDestinationNodeId() == Optional<uint64_t>::Value(88)); |
| NL_TEST_ASSERT(inSuite, header.GetSourceNodeId() == Optional<uint64_t>::Value(77)); |
| |
| header.SetMessageId(234).SetSourceNodeId(77).SetDestinationNodeId(88).SetSecureSessionControlMsg(true); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| // change it to verify decoding |
| header.SetMessageId(222).SetSourceNodeId(1).SetDestinationNodeId(2); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 234); |
| NL_TEST_ASSERT(inSuite, header.GetDestinationNodeId() == Optional<uint64_t>::Value(88)); |
| NL_TEST_ASSERT(inSuite, header.GetSourceNodeId() == Optional<uint64_t>::Value(77)); |
| NL_TEST_ASSERT(inSuite, header.IsSecureSessionControlMsg()); |
| |
| header.SetMessageId(234).SetSourceNodeId(77).SetDestinationNodeId(88).SetEncryptionKeyID(2); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| // change it to verify decoding |
| header.SetMessageId(222).SetSourceNodeId(1).SetDestinationNodeId(2); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 234); |
| NL_TEST_ASSERT(inSuite, header.GetDestinationNodeId() == Optional<uint64_t>::Value(88)); |
| NL_TEST_ASSERT(inSuite, header.GetSourceNodeId() == Optional<uint64_t>::Value(77)); |
| NL_TEST_ASSERT(inSuite, header.GetEncryptionKeyID() == 2); |
| |
| header.SetMessageId(234).SetSourceNodeId(77).SetDestinationNodeId(88); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| // change it to verify decoding |
| header.SetMessageId(222).SetSourceNodeId(1).SetDestinationNodeId(2); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, header.GetMessageId() == 234); |
| NL_TEST_ASSERT(inSuite, header.GetDestinationNodeId() == Optional<uint64_t>::Value(88)); |
| NL_TEST_ASSERT(inSuite, header.GetSourceNodeId() == Optional<uint64_t>::Value(77)); |
| } |
| |
| void TestPayloadHeaderEncodeDecode(nlTestSuite * inSuite, void * inContext) |
| { |
| PayloadHeader header; |
| uint8_t buffer[64]; |
| uint16_t encodeLen; |
| uint16_t decodeLen; |
| |
| header.SetMessageType(112).SetExchangeID(2233); |
| NL_TEST_ASSERT(inSuite, !header.GetVendorId().HasValue()); |
| |
| header.SetMessageType(112).SetExchangeID(2233).SetProtocolID(1221).SetInitiator(true); |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| header.SetMessageType(221).SetExchangeID(3322).SetProtocolID(4567); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encodeLen == decodeLen); |
| NL_TEST_ASSERT(inSuite, header.GetMessageType() == 112); |
| NL_TEST_ASSERT(inSuite, header.GetExchangeID() == 2233); |
| NL_TEST_ASSERT(inSuite, header.GetProtocolID() == 1221); |
| NL_TEST_ASSERT(inSuite, !header.GetVendorId().HasValue()); |
| NL_TEST_ASSERT(inSuite, header.IsInitiator()); |
| |
| header.SetMessageType(112).SetExchangeID(2233).SetProtocolID(1221); |
| header.SetVendorId(6789); |
| |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, sizeof(buffer), &encodeLen) == CHIP_NO_ERROR); |
| |
| header.SetMessageType(111).SetExchangeID(222); |
| |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encodeLen == decodeLen); |
| NL_TEST_ASSERT(inSuite, header.GetExchangeID() == 2233); |
| NL_TEST_ASSERT(inSuite, header.GetVendorId() == Optional<uint16_t>::Value(6789)); |
| |
| header.SetMessageType(221).SetExchangeID(3322).SetProtocolID(4567); |
| header.SetVendorId(8976); |
| |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, sizeof(buffer), &decodeLen) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encodeLen == decodeLen); |
| NL_TEST_ASSERT(inSuite, header.GetExchangeID() == 2233); |
| NL_TEST_ASSERT(inSuite, header.GetProtocolID() == 1221); |
| NL_TEST_ASSERT(inSuite, header.GetVendorId() == Optional<uint16_t>::Value(6789)); |
| } |
| |
| void TestPacketHeaderEncodeDecodeBounds(nlTestSuite * inSuite, void * inContext) |
| { |
| PacketHeader header; |
| uint8_t buffer[64]; |
| uint16_t unusedLen; |
| |
| for (uint16_t shortLen = 0; shortLen < 10; shortLen++) |
| { |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| } |
| |
| // Now check that with 10 bytes we can successfully encode a |
| // default-constructed PacketHeader. |
| static const size_t minLen = 10; |
| uint16_t encoded_len; |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, minLen, &encoded_len) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encoded_len == minLen); |
| // Verify that decoding at any smaller length fails. |
| for (uint16_t shortLen = 0; shortLen < encoded_len; shortLen++) |
| { |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| } |
| uint16_t decoded_len; |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, encoded_len, &decoded_len) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, decoded_len == encoded_len); |
| |
| // Now test encoding/decoding with a source node id present. |
| header.SetSourceNodeId(1); |
| for (uint16_t shortLen = minLen; shortLen < minLen + 8; shortLen++) |
| { |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| } |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, minLen + 8, &encoded_len) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encoded_len == minLen + 8); |
| for (uint16_t shortLen = 0; shortLen < encoded_len; shortLen++) |
| { |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| } |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, encoded_len, &decoded_len) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, decoded_len == encoded_len); |
| |
| // Now test encoding/decoding with a source and destination node id present. |
| header.SetDestinationNodeId(1); |
| for (uint16_t shortLen = minLen; shortLen < minLen + 16; shortLen++) |
| { |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| } |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, minLen + 16, &encoded_len) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, encoded_len == minLen + 16); |
| for (uint16_t shortLen = 0; shortLen < encoded_len; shortLen++) |
| { |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| } |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, encoded_len, &decoded_len) == CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, decoded_len == encoded_len); |
| } |
| |
| void TestPayloadHeaderEncodeDecodeBounds(nlTestSuite * inSuite, void * inContext) |
| { |
| PayloadHeader header; |
| uint8_t buffer[64]; |
| uint16_t unusedLen; |
| |
| for (uint16_t shortLen = 0; shortLen < 6; shortLen++) |
| { |
| NL_TEST_ASSERT(inSuite, header.Encode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| NL_TEST_ASSERT(inSuite, header.Decode(buffer, shortLen, &unusedLen) != CHIP_NO_ERROR); |
| } |
| } |
| |
| } // namespace |
| |
| // clang-format off |
| static const nlTest sTests[] = |
| { |
| NL_TEST_DEF("PacketInitialState", TestPacketHeaderInitialState), |
| NL_TEST_DEF("PayloadInitialState", TestPayloadHeaderInitialState), |
| NL_TEST_DEF("PacketEncodeDecode", TestPacketHeaderEncodeDecode), |
| NL_TEST_DEF("PayloadEncodeDecode", TestPayloadHeaderEncodeDecode), |
| NL_TEST_DEF("PacketEncodeDecodeBounds", TestPacketHeaderEncodeDecodeBounds), |
| NL_TEST_DEF("PayloadEncodeDecodeBounds", TestPayloadHeaderEncodeDecodeBounds), |
| NL_TEST_SENTINEL() |
| }; |
| // clang-format on |
| |
| int TestMessageHeader(void) |
| { |
| nlTestSuite theSuite = { "Transport-MessageHeader", &sTests[0], nullptr, nullptr }; |
| nlTestRunner(&theSuite, nullptr); |
| return nlTestRunnerStats(&theSuite); |
| } |
| |
| CHIP_REGISTER_TEST_SUITE(TestMessageHeader) |