blob: 975d6206ecb1b244ab356c5f462a8ff7a775401f [file] [log] [blame]
/*
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 "amqp_0_10/unit_test.h"
#include "amqp_0_10/allSegmentTypes.h"
#include "qpid/framing/AMQFrame.h"
#include "qpid/framing/Buffer.h"
#include "qpid/amqp_0_10/Packer.h"
#include "qpid/amqp_0_10/built_in_types.h"
#include "qpid/amqp_0_10/Codec.h"
#include "qpid/amqp_0_10/specification.h"
#include "qpid/amqp_0_10/ControlHolder.h"
#include "qpid/amqp_0_10/Struct32.h"
#include "qpid/amqp_0_10/FrameHeader.h"
#include "qpid/amqp_0_10/Map.h"
#include "qpid/amqp_0_10/Unit.h"
#include "allSegmentTypes.h"
#include <boost/test/test_case_template.hpp>
#include <boost/type_traits/is_arithmetic.hpp>
#include <boost/utility/enable_if.hpp>
#include <boost/optional.hpp>
#include <boost/mpl/vector.hpp>
#include <boost/mpl/back_inserter.hpp>
#include <boost/mpl/copy.hpp>
#include <boost/mpl/empty_sequence.hpp>
#include <boost/current_function.hpp>
#include <iterator>
#include <string>
#include <sstream>
#include <iostream>
#include <netinet/in.h>
// Missing operators needed for tests.
namespace boost {
template <class T, size_t N>
std::ostream& operator<<(std::ostream& out, const array<T,N>& a) {
std::ostream_iterator<T> o(out, " ");
std::copy(a.begin(), a.end(), o);
return out;
}
} // boost
QPID_AUTO_TEST_SUITE(SerializeTestSuite)
using namespace std;
namespace mpl=boost::mpl;
using namespace qpid::amqp_0_10;
using qpid::framing::in_place;
template <class A, class B> struct concat2 { typedef typename mpl::copy<B, typename mpl::back_inserter<A> >::type type; };
template <class A, class B, class C> struct concat3 { typedef typename concat2<A, typename concat2<B, C>::type>::type type; };
template <class A, class B, class C, class D> struct concat4 { typedef typename concat2<A, typename concat3<B, C, D>::type>::type type; };
typedef mpl::vector<Boolean, Char, Int32, Int64, Int8, Uint16, CharUtf32, Uint32, Uint64, Bin8, Uint8>::type IntegralTypes;
typedef mpl::vector<Bin1024, Bin128, Bin16, Bin256, Bin32, Bin40, Bin512, Bin64, Bin72>::type BinTypes;
typedef mpl::vector<Double, Float>::type FloatTypes;
typedef mpl::vector<SequenceNo, Uuid, Datetime, Dec32, Dec64> FixedSizeClassTypes;
typedef mpl::vector<Map, Vbin8, Str8Latin, Str8, Str8Utf16, Vbin16, Str16Latin, Str16, Str16Utf16, Vbin32> VariableSizeTypes;
typedef concat4<IntegralTypes, BinTypes, FloatTypes, FixedSizeClassTypes>::type FixedSizeTypes;
typedef concat2<FixedSizeTypes, VariableSizeTypes>::type AllTypes;
// TODO aconway 2008-02-20: should test 64 bit integrals for order also.
QPID_AUTO_TEST_CASE(testNetworkByteOrder) {
string data;
uint32_t l = 0x11223344;
Codec::encode(std::back_inserter(data))(l);
uint32_t enc=reinterpret_cast<const uint32_t&>(*data.data());
uint32_t l2 = ntohl(enc);
BOOST_CHECK_EQUAL(l, l2);
data.clear();
uint16_t s = 0x1122;
Codec::encode(std::back_inserter(data))(s);
uint32_t s2 = ntohs(*reinterpret_cast<const uint32_t*>(data.data()));
BOOST_CHECK_EQUAL(s, s2);
}
QPID_AUTO_TEST_CASE(testSetLimit) {
typedef Codec::Encoder<back_insert_iterator<string> > Encoder;
string data;
Encoder encode(back_inserter(data), 3);
encode('1')('2')('3');
try {
encode('4');
BOOST_FAIL("Expected exception");
} catch (...) {} // FIXME aconway 2008-04-03: catch proper exception
BOOST_CHECK_EQUAL(data, "123");
}
QPID_AUTO_TEST_CASE(testScopedLimit) {
typedef Codec::Encoder<back_insert_iterator<string> > Encoder;
string data;
Encoder encode(back_inserter(data), 10);
encode(Str8("123")); // 4 bytes
{
Encoder::ScopedLimit l(encode, 3);
encode('a')('b')('c');
try {
encode('d');
BOOST_FAIL("Expected exception");
} catch(...) {} // FIXME aconway 2008-04-03: catch proper exception
}
BOOST_CHECK_EQUAL(data, "\003123abc");
encode('x')('y')('z');
try {
encode('!');
BOOST_FAIL("Expected exception");
} catch(...) {} // FIXME aconway 2008-04-03: catch proper exception
BOOST_CHECK_EQUAL(data.size(), 10u);
}
// Assign test values to the various types.
void testValue(bool& b) { b = true; }
void testValue(Bit&) { }
template <class T> typename boost::enable_if<boost::is_arithmetic<T> >::type testValue(T& n) { n=42; }
void testValue(CharUtf32& c) { c = 43; }
void testValue(long long& l) { l = 0x012345; }
void testValue(Datetime& dt) { dt = qpid::sys::now(); }
void testValue(Uuid& uuid) { uuid=Uuid(true); }
template <class E, class M> void testValue(Decimal<E,M>& d) { d.exponent=2; d.mantissa=0x1122; }
void testValue(SequenceNo& s) { s = 42; }
template <size_t N> void testValue(Bin<N>& a) { a.assign(42); }
template <class T, class S, int Unique> void testValue(SerializableString<T, S, Unique>& s) {
char msg[]="foobar";
s.assign(msg, msg+sizeof(msg));
}
void testValue(Str16& s) { s = "the quick brown fox jumped over the lazy dog"; }
void testValue(Str8& s) { s = "foobar"; }
void testValue(Map& m) { m["s"] = Str8("foobar"); m["b"] = true; m["c"] = uint16_t(42); }
//typedef mpl::vector<Str8, Str16>::type TestTypes;
/*BOOST_AUTO_TEST_CASE_TEMPLATE(testEncodeDecode, T, AllTypes)
{
string data;
T t;
testValue(t);
Codec::encode(std::back_inserter(data))(t);
BOOST_CHECK_EQUAL(Codec::size(t), data.size());
T t2;
Codec::decode(data.begin())(t2);
BOOST_CHECK_EQUAL(t,t2);
}
*/
struct TestMe {
bool encoded, decoded;
char value;
TestMe(char v) : encoded(), decoded(), value(v) {}
template <class S> void encode(S& s) const {
const_cast<TestMe*>(this)->encoded=true; s(value);
}
template <class S> void decode(S& s) { decoded=true; s(value); }
template <class S> void serialize(S& s) { s.split(*this); }
};
QPID_AUTO_TEST_CASE(testSplit) {
string data;
TestMe t1('x');
Codec::encode(std::back_inserter(data))(t1);
BOOST_CHECK(t1.encoded);
BOOST_CHECK(!t1.decoded);
BOOST_CHECK_EQUAL(data, "x");
TestMe t2('y');
Codec::decode(data.begin())(t2);
BOOST_CHECK(!t2.encoded);
BOOST_CHECK(t2.decoded);
BOOST_CHECK_EQUAL(t2.value, 'x');
}
QPID_AUTO_TEST_CASE(testControlEncodeDecode) {
string data;
Control::Holder h(in_place<connection::Tune>(1,2,3,4));
Codec::encode(std::back_inserter(data))(h);
BOOST_CHECK_EQUAL(data.size(), Codec::size(h));
Codec::Decoder<string::iterator> decode(data.begin());
Control::Holder h2;
decode(h2);
BOOST_REQUIRE(h2.get());
BOOST_CHECK_EQUAL(h2.get()->getClassCode(), connection::CODE);
BOOST_CHECK_EQUAL(h2.get()->getCode(), uint8_t(connection::Tune::CODE));
connection::Tune& tune=static_cast<connection::Tune&>(*h2.get());
BOOST_CHECK_EQUAL(tune.channelMax, 1u);
BOOST_CHECK_EQUAL(tune.maxFrameSize, 2u);
BOOST_CHECK_EQUAL(tune.heartbeatMin, 3u);
BOOST_CHECK_EQUAL(tune.heartbeatMax, 4u);
}
QPID_AUTO_TEST_CASE(testStruct32) {
message::DeliveryProperties dp;
dp.priority=message::MEDIUM;
dp.routingKey="foo";
Struct32 s(dp);
string data;
Codec::encode(back_inserter(data))(s);
uint32_t structSize; // Starts with size
Codec::decode(data.begin())(structSize);
BOOST_CHECK_EQUAL(structSize, Codec::size(dp) + 2); // +2 for code
BOOST_CHECK_EQUAL(structSize, data.size()-4); // encoded body
BOOST_CHECK_EQUAL(data.size(), Codec::size(s));
Struct32 s2;
Codec::decode(data.begin())(s2);
message::DeliveryProperties* dp2 = s2.getIf<message::DeliveryProperties>();
BOOST_REQUIRE(dp2);
BOOST_CHECK_EQUAL(dp2->priority, message::MEDIUM);
BOOST_CHECK_EQUAL(dp2->routingKey, "foo");
}
QPID_AUTO_TEST_CASE(testStruct32Unknown) {
// Verify we can recode an unknown struct unchanged.
Struct32 s;
string data;
Codec::encode(back_inserter(data))(uint32_t(10));
data.append(10, 'X');
Codec::decode(data.begin())(s);
string data2;
Codec::encode(back_inserter(data2))(s);
BOOST_CHECK_EQUAL(data.size(), data2.size());
BOOST_CHECK_EQUAL(data, data2);
}
struct DummyPacked {
static const uint8_t PACK=1;
boost::optional<char> i, j;
char k;
Bit l,m;
DummyPacked(char a=0, char b=0, char c=0) : i(a), j(b), k(c), l(), m() {}
template <class S> void serialize(S& s) { s(i)(j)(k)(l)(m); }
};
Packer<DummyPacked> serializable(DummyPacked& d) { return Packer<DummyPacked>(d); }
QPID_AUTO_TEST_CASE(testPackBits) {
DummyPacked d('a','b','c');
BOOST_CHECK_EQUAL(packBits(d), 7u);
d.j = boost::none;
BOOST_CHECK_EQUAL(packBits(d), 5u);
d.m = true;
BOOST_CHECK_EQUAL(packBits(d), 0x15u);
}
QPID_AUTO_TEST_CASE(testPacked) {
string data;
Codec::encode(back_inserter(data))('a')(boost::optional<char>('b'))(boost::optional<char>())('c');
BOOST_CHECK_EQUAL(data, "abc");
data.clear();
DummyPacked dummy('a','b','c');
Codec::encode(back_inserter(data))(dummy);
BOOST_CHECK_EQUAL(data.size(), 4u);
BOOST_CHECK_EQUAL(data, string("\007abc"));
data.clear();
dummy.i = boost::none;
Codec::encode(back_inserter(data))(dummy);
BOOST_CHECK_EQUAL(data, string("\6bc"));
data.clear();
const char* missing = "\5xy";
Codec::decode(missing)(dummy);
BOOST_CHECK(dummy.i);
BOOST_CHECK_EQUAL(*dummy.i, 'x');
BOOST_CHECK(!dummy.j);
BOOST_CHECK_EQUAL(dummy.k, 'y');
}
QPID_AUTO_TEST_CASE(testUnitControl) {
string data;
Control::Holder h(in_place<connection::Tune>(1,2,3,4));
Codec::encode(std::back_inserter(data))(h);
Unit unit(FrameHeader(FIRST_FRAME|LAST_FRAME, CONTROL));
Codec::decode(data.begin())(unit);
BOOST_REQUIRE(unit.get<ControlHolder>());
string data2;
Codec::encode(back_inserter(data2))(unit);
BOOST_CHECK_EQUAL(data, data2);
}
QPID_AUTO_TEST_CASE(testArray) {
ArrayDomain<char> a;
a.resize(3, 'x');
string data;
Codec::encode(back_inserter(data))(a);
ArrayDomain<char> b;
Codec::decode(data.begin())(b);
BOOST_CHECK_EQUAL(b.size(), 3u);
string data3;
Codec::encode(back_inserter(data3))(a);
BOOST_CHECK_EQUAL(data, data3);
Array x;
Codec::decode(data.begin())(x);
BOOST_CHECK_EQUAL(x.size(), 3u);
BOOST_CHECK_EQUAL(x[0].size(), 1u);
BOOST_CHECK_EQUAL(*x[0].begin(), 'x');
BOOST_CHECK_EQUAL(*x[2].begin(), 'x');
string data2;
Codec::encode(back_inserter(data2))(x);
BOOST_CHECK_EQUAL(data,data2);
}
QPID_AUTO_TEST_CASE(testStruct) {
string data;
message::DeliveryProperties dp;
BOOST_CHECK(!dp.discardUnroutable);
dp.immediate = true;
dp.redelivered = false;
dp.priority = message::MEDIUM;
dp.exchange = "foo";
Codec::encode(back_inserter(data))(dp);
// Skip 4 bytes size, little-endian decode for pack bits.
uint16_t encodedBits=uint8_t(data[5]);
encodedBits <<= 8;
encodedBits += uint8_t(data[4]);
BOOST_CHECK_EQUAL(encodedBits, packBits(dp));
data.clear();
Struct32 h(dp);
Codec::encode(back_inserter(data))(h);
Struct32 h2;
Codec::decode(data.begin())(h2);
BOOST_CHECK_EQUAL(h2.getClassCode(), Uint8(message::DeliveryProperties::CLASS_CODE));
BOOST_CHECK_EQUAL(h2.getCode(), Uint8(message::DeliveryProperties::CODE));
message::DeliveryProperties* dp2 =
dynamic_cast<message::DeliveryProperties*>(h2.get());
BOOST_CHECK(dp2);
BOOST_CHECK(!dp2->discardUnroutable);
BOOST_CHECK(dp2->immediate);
BOOST_CHECK(!dp2->redelivered);
BOOST_CHECK_EQUAL(dp2->priority, message::MEDIUM);
BOOST_CHECK_EQUAL(dp2->exchange, "foo");
}
struct RecodeUnit {
template <class T>
void operator() (const T& t) {
BOOST_MESSAGE(BOOST_CURRENT_FUNCTION << " called with: " << t);
using qpid::framing::Buffer;
using qpid::framing::AMQFrame;
session::Header sh;
BOOST_CHECK_EQUAL(Codec::size(sh), 2u);
// Encode unit.
Unit u(t);
string data;
Codec::encode(back_inserter(data))(u.getHeader())(u);
data.push_back(char(0xCE)); // Preview end-of-frame
// Decode AMQFrame
Buffer buf(&data[0], data.size());
AMQFrame f;
f.decode(buf);
BOOST_MESSAGE("AMQFrame decoded: " << f);
// Encode AMQFrame
string data2(f.size(), ' ');
Buffer buf2(&data2[0], data.size());
f.encode(buf2);
// Verify encoded by unit == encoded by AMQFrame
BOOST_CHECK_MESSAGE(data == data2, BOOST_CURRENT_FUNCTION);
// Decode unit
// FIXME aconway 2008-04-15: must set limit to decode a header.
Codec::Decoder<string::iterator> decode(data2.begin(), data2.size()-1);
FrameHeader h;
decode(h);
BOOST_CHECK_EQUAL(u.getHeader(), h);
Unit u2(h);
decode(u2);
// Re-encode unit
string data3;
Codec::encode(back_inserter(data3))(u2.getHeader())(u2);
data3.push_back(char(0xCE)); // Preview end-of-frame
BOOST_CHECK_MESSAGE(data3 == data2, BOOST_CURRENT_FUNCTION);
}
};
QPID_AUTO_TEST_CASE(testSerializeAllSegmentTypes) {
RecodeUnit recode;
allSegmentTypes(recode);
}
QPID_AUTO_TEST_SUITE_END()