Use the GTest portability headers
[folly.git] / folly / experimental / bser / test / BserTest.cpp
1 /*
2  * Copyright 2016 Facebook, Inc.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *   http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 #include <folly/experimental/bser/Bser.h>
17 #include <folly/String.h>
18 #include <folly/portability/GTest.h>
19
20 using folly::dynamic;
21
22 static const dynamic roundtrips[] = {
23     1,
24     std::numeric_limits<int8_t>::max(),
25     std::numeric_limits<int16_t>::max(),
26     std::numeric_limits<int32_t>::max(),
27     std::numeric_limits<int64_t>::max(),
28     std::numeric_limits<int8_t>::min(),
29     std::numeric_limits<int16_t>::min(),
30     std::numeric_limits<int32_t>::min(),
31     std::numeric_limits<int64_t>::min(),
32     bool(true),
33     bool(false),
34     nullptr,
35     1.5,
36     "hello",
37     {1, 2, 3},
38     dynamic::object("key", "value")("otherkey", "otherval"),
39 };
40
41 // Here's a blob from the watchman test suite
42 const uint8_t template_blob[] =
43     "\x00\x01\x03\x28"
44     "\x0b\x00\x03\x02\x02\x03\x04\x6e\x61\x6d\x65\x02"
45     "\x03\x03\x61\x67\x65\x03\x03\x02\x03\x04\x66\x72"
46     "\x65\x64\x03\x14\x02\x03\x04\x70\x65\x74\x65\x03"
47     "\x1e\x0c\x03\x19";
48
49 // and here's what it represents
50 static const dynamic template_dynamic = {
51     dynamic::object("name", "fred")("age", 20),
52     dynamic::object("name", "pete")("age", 30),
53     dynamic::object("name", nullptr)("age", 25),
54 };
55
56 TEST(Bser, RoundTrip) {
57   dynamic decoded(nullptr);
58   folly::fbstring str;
59
60   for (const auto& dyn : roundtrips) {
61     try {
62       str = folly::bser::toBser(dyn, folly::bser::serialization_opts());
63       decoded = folly::bser::parseBser(str);
64
65       EXPECT_EQ(decoded, dyn);
66     } catch (const std::exception& err) {
67       LOG(ERROR) << err.what() << "\nInput: " << dyn.typeName() << ": " << dyn
68                  << " decoded back as " << decoded.typeName() << ": " << decoded
69                  << "\n" << folly::hexDump(str.data(), str.size());
70       throw;
71     }
72   }
73 }
74
75 TEST(Bser, Template) {
76   dynamic decoded(nullptr);
77   folly::fbstring str;
78   // Decode the template value provided from elsewhere
79   decoded = folly::bser::parseBser(
80       folly::ByteRange(template_blob, sizeof(template_blob) - 1));
81   EXPECT_EQ(decoded, template_dynamic)
82       << "Didn't load template value."
83          "\nInput: " << template_dynamic.typeName() << ": " << template_dynamic
84       << " decoded back as " << decoded.typeName() << ": " << decoded << "\n"
85       << folly::hexDump(template_blob, sizeof(template_blob) - 1);
86
87   // Now check that we can generate this same data representation
88   folly::bser::serialization_opts opts;
89   folly::bser::serialization_opts::TemplateMap templates = {
90       std::make_pair(&decoded, folly::dynamic{"name", "age"})};
91   opts.templates = templates;
92
93   str = folly::bser::toBser(decoded, opts);
94   EXPECT_EQ(folly::ByteRange((const uint8_t*)str.data(), str.size()),
95             folly::ByteRange(template_blob, sizeof(template_blob) - 1))
96       << "Expected:\n"
97       << folly::hexDump(template_blob, sizeof(template_blob) - 1) << "\nGot:\n"
98       << folly::hexDump(str.data(), str.size());
99 }
100
101 TEST(Bser, PduLength) {
102   EXPECT_THROW([] {
103     // Try to decode PDU for a short buffer that doesn't even have the
104     // complete length available
105     auto buf = folly::IOBuf::wrapBuffer(template_blob, 3);
106     auto len = folly::bser::decodePduLength(&*buf);
107     LOG(ERROR) << "managed to return a length, but only had 3 bytes";
108   }(), std::out_of_range);
109
110   auto buf = folly::IOBuf::wrapBuffer(template_blob, sizeof(template_blob));
111   auto len = folly::bser::decodePduLength(&*buf);
112   EXPECT_EQ(len, 44) << "PduLength should be 44, got " << len;
113 }
114
115 /* vim:ts=2:sw=2:et:
116  */