blob: df0684e832c98d0bc7c60d8f428f2b852a180093 [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 "qpid/Exception.h"
#include "qpid/broker/Exchange.h"
#include "qpid/broker/Queue.h"
#include "qpid/broker/DeliverableMessage.h"
#include "qpid/broker/DirectExchange.h"
#include "qpid/broker/ExchangeRegistry.h"
#include "qpid/broker/FanOutExchange.h"
#include "qpid/broker/HeadersExchange.h"
#include "qpid/broker/TopicExchange.h"
#include "qpid/framing/reply_exceptions.h"
#include "unit_test.h"
#include <iostream>
#include "MessageUtils.h"
using std::string;
using namespace qpid::broker;
using namespace qpid::framing;
using namespace qpid::sys;
using namespace qpid;
namespace qpid {
namespace tests {
QPID_AUTO_TEST_SUITE(ExchangeTestSuite)
QPID_AUTO_TEST_CASE(testMe)
{
Queue::shared_ptr queue(new Queue("queue", true));
Queue::shared_ptr queue2(new Queue("queue2", true));
TopicExchange topic("topic");
topic.bind(queue, "abc", 0);
topic.bind(queue2, "abc", 0);
DirectExchange direct("direct");
direct.bind(queue, "abc", 0);
direct.bind(queue2, "abc", 0);
queue.reset();
queue2.reset();
DeliverableMessage msg(MessageUtils::createMessage("exchange", "abc"), 0);
topic.route(msg);
direct.route(msg);
}
QPID_AUTO_TEST_CASE(testIsBound)
{
Queue::shared_ptr a(new Queue("a", true));
Queue::shared_ptr b(new Queue("b", true));
Queue::shared_ptr c(new Queue("c", true));
Queue::shared_ptr d(new Queue("d", true));
string k1("abc");
string k2("def");
string k3("xyz");
FanOutExchange fanout("fanout");
BOOST_CHECK(fanout.bind(a, "", 0));
BOOST_CHECK(fanout.bind(b, "", 0));
BOOST_CHECK(fanout.bind(c, "", 0));
BOOST_CHECK(fanout.isBound(a, 0, 0));
BOOST_CHECK(fanout.isBound(b, 0, 0));
BOOST_CHECK(fanout.isBound(c, 0, 0));
BOOST_CHECK(!fanout.isBound(d, 0, 0));
DirectExchange direct("direct");
BOOST_CHECK(direct.bind(a, k1, 0));
BOOST_CHECK(direct.bind(a, k3, 0));
BOOST_CHECK(direct.bind(b, k2, 0));
BOOST_CHECK(direct.bind(c, k1, 0));
BOOST_CHECK(direct.isBound(a, 0, 0));
BOOST_CHECK(direct.isBound(a, &k1, 0));
BOOST_CHECK(direct.isBound(a, &k3, 0));
BOOST_CHECK(!direct.isBound(a, &k2, 0));
BOOST_CHECK(direct.isBound(b, 0, 0));
BOOST_CHECK(direct.isBound(b, &k2, 0));
BOOST_CHECK(direct.isBound(c, &k1, 0));
BOOST_CHECK(!direct.isBound(d, 0, 0));
BOOST_CHECK(!direct.isBound(d, &k1, 0));
BOOST_CHECK(!direct.isBound(d, &k2, 0));
BOOST_CHECK(!direct.isBound(d, &k3, 0));
TopicExchange topic("topic");
BOOST_CHECK(topic.bind(a, k1, 0));
BOOST_CHECK(topic.bind(a, k3, 0));
BOOST_CHECK(topic.bind(b, k2, 0));
BOOST_CHECK(topic.bind(c, k1, 0));
BOOST_CHECK(topic.isBound(a, 0, 0));
BOOST_CHECK(topic.isBound(a, &k1, 0));
BOOST_CHECK(topic.isBound(a, &k3, 0));
BOOST_CHECK(!topic.isBound(a, &k2, 0));
BOOST_CHECK(topic.isBound(b, 0, 0));
BOOST_CHECK(topic.isBound(b, &k2, 0));
BOOST_CHECK(topic.isBound(c, &k1, 0));
BOOST_CHECK(!topic.isBound(d, 0, 0));
BOOST_CHECK(!topic.isBound(d, &k1, 0));
BOOST_CHECK(!topic.isBound(d, &k2, 0));
BOOST_CHECK(!topic.isBound(d, &k3, 0));
HeadersExchange headers("headers");
FieldTable args1;
args1.setString("x-match", "all");
args1.setString("a", "A");
args1.setInt("b", 1);
FieldTable args2;
args2.setString("x-match", "any");
args2.setString("a", "A");
args2.setInt("b", 1);
FieldTable args3;
args3.setString("x-match", "any");
args3.setString("c", "C");
args3.setInt("b", 6);
headers.bind(a, "", &args1);
headers.bind(a, "other", &args3);//need to use different binding key to correctly identify second binding
headers.bind(b, "", &args2);
headers.bind(c, "", &args1);
BOOST_CHECK(headers.isBound(a, 0, 0));
BOOST_CHECK(headers.isBound(a, 0, &args1));
BOOST_CHECK(headers.isBound(a, 0, &args3));
BOOST_CHECK(!headers.isBound(a, 0, &args2));
BOOST_CHECK(headers.isBound(b, 0, 0));
BOOST_CHECK(headers.isBound(b, 0, &args2));
BOOST_CHECK(headers.isBound(c, 0, &args1));
BOOST_CHECK(!headers.isBound(d, 0, 0));
BOOST_CHECK(!headers.isBound(d, 0, &args1));
BOOST_CHECK(!headers.isBound(d, 0, &args2));
BOOST_CHECK(!headers.isBound(d, 0, &args3));
}
QPID_AUTO_TEST_CASE(testDeleteGetAndRedeclare)
{
ExchangeRegistry exchanges;
exchanges.declare("my-exchange", "direct", false, false, FieldTable());
exchanges.destroy("my-exchange");
try {
exchanges.get("my-exchange");
} catch (const NotFoundException&) {}
std::pair<Exchange::shared_ptr, bool> response = exchanges.declare("my-exchange", "direct", false, false, FieldTable());
BOOST_CHECK_EQUAL(string("direct"), response.first->getType());
}
QPID_AUTO_TEST_CASE(testSequenceOptions)
{
FieldTable args;
args.setInt("qpid.msg_sequence",1);
char* buff = new char[10000];
framing::Buffer buffer(buff,10000);
{
DirectExchange direct("direct1", false, false, args);
DeliverableMessage msg1(MessageUtils::createMessage("e", "abc"), 0);
DeliverableMessage msg2(MessageUtils::createMessage("e", "abc"), 0);
DeliverableMessage msg3(MessageUtils::createMessage("e", "abc"), 0);
direct.route(msg1);
direct.route(msg2);
direct.route(msg3);
BOOST_CHECK_EQUAL(1, msg1.getMessage().getAnnotation("qpid.msg_sequence").asInt64());
BOOST_CHECK_EQUAL(2, msg2.getMessage().getAnnotation("qpid.msg_sequence").asInt64());
BOOST_CHECK_EQUAL(3, msg3.getMessage().getAnnotation("qpid.msg_sequence").asInt64());
FanOutExchange fanout("fanout1", false, false, args);
HeadersExchange header("headers1", false, false, args);
TopicExchange topic ("topic1", false, false, args);
// check other exchanges, that they preroute
DeliverableMessage msg4(MessageUtils::createMessage("e", "abc"), 0);
DeliverableMessage msg5(MessageUtils::createMessage("e", "abc"), 0);
DeliverableMessage msg6(MessageUtils::createMessage("e", "abc"), 0);
fanout.route(msg4);
BOOST_CHECK_EQUAL(1, msg4.getMessage().getAnnotation("qpid.msg_sequence").asInt64());
header.route(msg5);
BOOST_CHECK_EQUAL(1, msg5.getMessage().getAnnotation("qpid.msg_sequence").asInt64());
topic.route(msg6);
BOOST_CHECK_EQUAL(1, msg6.getMessage().getAnnotation("qpid.msg_sequence").asInt64());
direct.encode(buffer);
}
{
ExchangeRegistry exchanges;
buffer.reset();
DirectExchange::shared_ptr exch_dec = Exchange::decode(exchanges, buffer);
DeliverableMessage msg1(MessageUtils::createMessage("e", "abc"), 0);
exch_dec->route(msg1);
BOOST_CHECK_EQUAL(4, msg1.getMessage().getAnnotation("qpid.msg_sequence").asInt64());
}
delete [] buff;
}
QPID_AUTO_TEST_CASE(testIVEOption)
{
FieldTable args;
args.setInt("qpid.ive",1);
DirectExchange direct("direct1", false, false, args);
FanOutExchange fanout("fanout1", false, false, args);
HeadersExchange header("headers1", false, false, args);
TopicExchange topic ("topic1", false, false, args);
qpid::types::Variant::Map properties;
properties["routing-key"] = "abc";
properties["a"] = "abc";
Message msg1 = MessageUtils::createMessage(properties, "my-message", "direct1");
DeliverableMessage dmsg1(msg1, 0);
FieldTable args2;
args2.setString("x-match", "any");
args2.setString("a", "abc");
direct.route(dmsg1);
fanout.route(dmsg1);
header.route(dmsg1);
topic.route(dmsg1);
Queue::shared_ptr queue(new Queue("queue", true));
Queue::shared_ptr queue1(new Queue("queue1", true));
Queue::shared_ptr queue2(new Queue("queue2", true));
Queue::shared_ptr queue3(new Queue("queue3", true));
BOOST_CHECK(direct.bind(queue, "abc", 0));
BOOST_CHECK(fanout.bind(queue1, "abc", 0));
BOOST_CHECK(header.bind(queue2, "", &args2));
BOOST_CHECK(topic.bind(queue3, "abc", 0));
BOOST_CHECK_EQUAL(1u,queue->getMessageCount());
BOOST_CHECK_EQUAL(1u,queue1->getMessageCount());
BOOST_CHECK_EQUAL(1u,queue2->getMessageCount());
BOOST_CHECK_EQUAL(1u,queue3->getMessageCount());
}
QPID_AUTO_TEST_SUITE_END()
}} // namespace qpid::tests