--- /dev/null
+/******************************************************************************
+ * Icinga 2 *
+ * Copyright (C) 2012-2018 Icinga Development Team (https://www.icinga.com/) *
+ * *
+ * This program is free software; you can redistribute it and/or *
+ * modify it under the terms of the GNU General Public License *
+ * as published by the Free Software Foundation; either version 2 *
+ * of the License, or (at your option) any later version. *
+ * *
+ * This program is distributed in the hope that it will be useful, *
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of *
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
+ * GNU General Public License for more details. *
+ * *
+ * You should have received a copy of the GNU General Public License *
+ * along with this program; if not, write to the Free Software Foundation *
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
+ ******************************************************************************/
+
+#include "base/object-packer.hpp"
+#include "base/value.hpp"
+#include "base/string.hpp"
+#include "base/array.hpp"
+#include "base/dictionary.hpp"
+#include <BoostTestTargetConfig.h>
+#include <climits>
+#include <initializer_list>
+#include <iomanip>
+#include <strstream>
+
+using namespace icinga;
+
+#if CHAR_MIN != 0
+union CharU2SConverter
+{
+ CharU2SConverter()
+ {
+ s = 0;
+ }
+
+ unsigned char u;
+ signed char s;
+};
+#endif
+
+/**
+ * Avoid implementation-defined overflows during unsigned to signed casts
+ */
+static inline char UIntToByte(unsigned i)
+{
+#if CHAR_MIN == 0
+ return i;
+#else
+ CharU2SConverter converter;
+
+ converter.u = i;
+ return converter.s;
+#endif
+}
+
+#if CHAR_MIN != 0
+union CharS2UConverter
+{
+ CharS2UConverter()
+ {
+ u = 0;
+ }
+
+ unsigned char u;
+ signed char s;
+};
+#endif
+
+/**
+ * Avoid implementation-defined underflows during signed to unsigned casts
+ */
+static inline unsigned ByteToUInt(char c)
+{
+#if CHAR_MIN == 0
+ return c;
+#else
+ CharS2UConverter converter;
+
+ converter.s = c;
+ return converter.u;
+#endif
+}
+
+/**
+ * Compare the expected output with the actual output
+ */
+static inline bool ComparePackObjectResult(const String& actualOutput, const std::initializer_list<int>& out)
+{
+ if (actualOutput.GetLength() != out.size())
+ return false;
+
+ auto actualOutputPos = actualOutput.Begin();
+ for (auto byte : out) {
+ if (*actualOutputPos != UIntToByte(byte))
+ return false;
+
+ ++actualOutputPos;
+ }
+
+ return true;
+}
+
+/**
+ * Pack the given input and compare with the expected output
+ */
+static inline bool AssertPackObjectResult(Value in, std::initializer_list<int> out)
+{
+ auto actualOutput = PackObject(in);
+ bool equal = ComparePackObjectResult(actualOutput, out);
+
+ if (!equal) {
+ std::ostringstream buf;
+ buf << std::setw(2) << std::setfill('0') << std::setbase(16);
+
+ buf << "--- ";
+ for (int c : out) {
+ buf << c;
+ }
+ buf << std::endl;
+
+ buf << "+++ ";
+ for (char c : actualOutput) {
+ buf << ByteToUInt(c);
+ }
+ buf << std::endl;
+
+ BOOST_TEST_MESSAGE(buf.str());
+ }
+
+ return equal;
+}
+
+BOOST_AUTO_TEST_SUITE(object_packer)
+
+BOOST_AUTO_TEST_CASE(pack_null)
+{
+ BOOST_CHECK(AssertPackObjectResult(Empty, {0}));
+}
+
+BOOST_AUTO_TEST_CASE(pack_false)
+{
+ BOOST_CHECK(AssertPackObjectResult(false, {1}));
+}
+
+BOOST_AUTO_TEST_CASE(pack_true)
+{
+ BOOST_CHECK(AssertPackObjectResult(true, {2}));
+}
+
+BOOST_AUTO_TEST_CASE(pack_number)
+{
+ BOOST_CHECK(AssertPackObjectResult(42.125, {
+ // type
+ 3,
+ // IEEE 754
+ 64, 69, 16, 0, 0, 0, 0, 0
+ }));
+}
+
+BOOST_AUTO_TEST_CASE(pack_string)
+{
+ BOOST_CHECK(AssertPackObjectResult(
+ String(
+ // ASCII (1 to 127)
+ "\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0a\x0b\x0c\x0d\x0e\x0f"
+ "\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f"
+ "\x20\x21\x22\x23\x24\x25\x26\x27\x28\x29\x2a\x2b\x2c\x2d\x2e\x2f"
+ "\x30\x31\x32\x33\x34\x35\x36\x37\x38\x39\x3a\x3b\x3c\x3d\x3e\x3f"
+ "\x40\x41\x42\x43\x44\x45\x46\x47\x48\x49\x4a\x4b\x4c\x4d\x4e\x4f"
+ "\x50\x51\x52\x53\x54\x55\x56\x57\x58\x59\x5a\x5b\x5c\x5d\x5e\x5f"
+ "\x60\x61\x62\x63\x64\x65\x66\x67\x68\x69\x6a\x6b\x6c\x6d\x6e\x6f"
+ "\x70\x71\x72\x73\x74\x75\x76\x77\x78\x79\x7a\x7b\x7c\x7d\x7e\x7f"
+ // some keyboard-independent non-ASCII unicode characters
+ "áéíóú"
+ ),
+ {
+ // type
+ 4,
+ // length
+ 0, 0, 0, 0, 0, 0, 0, 137,
+ // ASCII
+ 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
+ 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31,
+ 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47,
+ 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63,
+ 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79,
+ 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95,
+ 96, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111,
+ 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 124, 125, 126, 127,
+ // UTF-8
+ 195, 161, 195, 169, 195, 173, 195, 179, 195, 186
+ }
+ ));
+}
+
+BOOST_AUTO_TEST_CASE(pack_array)
+{
+ BOOST_CHECK(AssertPackObjectResult(
+ (Array::Ptr)new Array({Empty, false, true, 42.125, "foobar"}),
+ {
+ // type
+ 5,
+ // length
+ 0, 0, 0, 0, 0, 0, 0, 5,
+ // Empty
+ 0,
+ // false
+ 1,
+ // true
+ 2,
+ // 42.125
+ 3,
+ 64, 69, 16, 0, 0, 0, 0, 0,
+ // "foobar"
+ 4,
+ 0, 0, 0, 0, 0, 0, 0, 6,
+ 102, 111, 111, 98, 97, 114
+ }
+ ));
+}
+
+BOOST_AUTO_TEST_CASE(pack_object)
+{
+ BOOST_CHECK(AssertPackObjectResult(
+ (Dictionary::Ptr)new Dictionary({
+ {"null", Empty},
+ {"false", false},
+ {"true", true},
+ {"42.125", 42.125},
+ {"foobar", "foobar"},
+ {"[]", (Array::Ptr)new Array()}
+ }),
+ {
+ // type
+ 6,
+ // length
+ 0, 0, 0, 0, 0, 0, 0, 6,
+ // "42.125"
+ 0, 0, 0, 0, 0, 0, 0, 6,
+ 52, 50, 46, 49, 50, 53,
+ // 42.125
+ 3,
+ 64, 69, 16, 0, 0, 0, 0, 0,
+ // "[]"
+ 0, 0, 0, 0, 0, 0, 0, 2,
+ 91, 93,
+ // (Array::Ptr)new Array()
+ 5,
+ 0, 0, 0, 0, 0, 0, 0, 0,
+ // "false"
+ 0, 0, 0, 0, 0, 0, 0, 5,
+ 102, 97, 108, 115, 101,
+ // false
+ 1,
+ // "foobar"
+ 0, 0, 0, 0, 0, 0, 0, 6,
+ 102, 111, 111, 98, 97, 114,
+ // "foobar"
+ 4,
+ 0, 0, 0, 0, 0, 0, 0, 6,
+ 102, 111, 111, 98, 97, 114,
+ // "null"
+ 0, 0, 0, 0, 0, 0, 0, 4,
+ 110, 117, 108, 108,
+ // Empty
+ 0,
+ // "true"
+ 0, 0, 0, 0, 0, 0, 0, 4,
+ 116, 114, 117, 101,
+ // true
+ 2
+ }
+ ));
+}
+
+BOOST_AUTO_TEST_SUITE_END()