diff options
Diffstat (limited to 'cpp/src')
-rw-r--r-- | cpp/src/Makefile.am | 1 | ||||
-rw-r--r-- | cpp/src/qpid/broker/ConnectionHandler.cpp | 3 | ||||
-rw-r--r-- | cpp/src/qpid/broker/SemanticHandler.cpp | 14 | ||||
-rw-r--r-- | cpp/src/qpid/broker/SessionHandler.cpp | 7 | ||||
-rw-r--r-- | cpp/src/qpid/broker/SessionHandler.h | 2 | ||||
-rw-r--r-- | cpp/src/qpid/client/ExecutionHandler.cpp | 10 | ||||
-rw-r--r-- | cpp/src/qpid/client/ExecutionHandler.h | 6 | ||||
-rw-r--r-- | cpp/src/qpid/framing/AMQMethodBody.cpp | 15 | ||||
-rw-r--r-- | cpp/src/qpid/framing/AMQMethodBody.h | 4 | ||||
-rw-r--r-- | cpp/src/qpid/framing/FrameDefaultVisitor.h | 3 | ||||
-rw-r--r-- | cpp/src/qpid/framing/Invoker.h | 86 | ||||
-rw-r--r-- | cpp/src/qpid/framing/StructHelper.h | 13 |
12 files changed, 110 insertions, 54 deletions
diff --git a/cpp/src/Makefile.am b/cpp/src/Makefile.am index 7a7a0b879a..7f644a3c8f 100644 --- a/cpp/src/Makefile.am +++ b/cpp/src/Makefile.am @@ -341,6 +341,7 @@ nobase_include_HEADERS = \ qpid/framing/HandlerUpdater.h \ qpid/framing/HeaderProperties.h \ qpid/framing/InitiationHandler.h \ + qpid/framing/Invoker.h \ qpid/framing/InputHandler.h \ qpid/framing/MethodContent.h \ qpid/framing/MethodHolder.h \ diff --git a/cpp/src/qpid/broker/ConnectionHandler.cpp b/cpp/src/qpid/broker/ConnectionHandler.cpp index a769d05470..f697986194 100644 --- a/cpp/src/qpid/broker/ConnectionHandler.cpp +++ b/cpp/src/qpid/broker/ConnectionHandler.cpp @@ -23,6 +23,7 @@ #include "ConnectionHandler.h" #include "Connection.h" #include "qpid/framing/ConnectionStartBody.h" +#include "qpid/framing/ServerInvoker.h" using namespace qpid; using namespace qpid::broker; @@ -44,7 +45,7 @@ void ConnectionHandler::handle(framing::AMQFrame& frame) { AMQMethodBody* method=frame.getBody()->getMethod(); try{ - if (!method->invoke(handler.get())) + if (!invoke(*handler.get(), *method)) throw ConnectionException(503, "Class can't be accessed over channel 0"); }catch(ConnectionException& e){ handler->client.close(e.code, e.toString(), method->amqpClassId(), method->amqpMethodId()); diff --git a/cpp/src/qpid/broker/SemanticHandler.cpp b/cpp/src/qpid/broker/SemanticHandler.cpp index dc5407be99..8535dc6a60 100644 --- a/cpp/src/qpid/broker/SemanticHandler.cpp +++ b/cpp/src/qpid/broker/SemanticHandler.cpp @@ -28,7 +28,7 @@ #include "Connection.h" #include "qpid/framing/ExecutionCompleteBody.h" #include "qpid/framing/ExecutionResultBody.h" -#include "qpid/framing/InvocationVisitor.h" +#include "qpid/framing/ServerInvoker.h" #include <boost/format.hpp> #include <boost/bind.hpp> @@ -121,14 +121,13 @@ void SemanticHandler::handleCommand(framing::AMQMethodBody* method) { SequenceNumber id = incoming.next(); BrokerAdapter adapter(state); - InvocationVisitor v(&adapter); - method->accept(v); + Invoker::Result invoker = invoke(adapter, *method); incoming.complete(id); - if (!v.wasHandled()) { + if (!invoker.wasHandled()) { throw ConnectionException(540, "Not implemented"); - } else if (v.hasResult()) { - session.getProxy().getExecution().result(id.getValue(), v.getResult()); + } else if (invoker.hasResult()) { + session.getProxy().getExecution().result(id.getValue(), invoker.getResult()); } if (method->isSync()) { incoming.sync(id); @@ -139,9 +138,8 @@ void SemanticHandler::handleCommand(framing::AMQMethodBody* method) void SemanticHandler::handleL3(framing::AMQMethodBody* method) { - if (!method->invoke(this)) { + if (!invoke(*this, *method)) throw ConnectionException(540, "Not implemented"); - } } void SemanticHandler::handleContent(AMQFrame& frame) diff --git a/cpp/src/qpid/broker/SessionHandler.cpp b/cpp/src/qpid/broker/SessionHandler.cpp index d7308572f9..ed092d6a05 100644 --- a/cpp/src/qpid/broker/SessionHandler.cpp +++ b/cpp/src/qpid/broker/SessionHandler.cpp @@ -23,6 +23,7 @@ #include "Connection.h" #include "qpid/framing/reply_exceptions.h" #include "qpid/framing/constants.h" +#include "qpid/framing/ServerInvoker.h" #include "qpid/log/Statement.h" namespace qpid { @@ -48,10 +49,10 @@ void SessionHandler::handleIn(AMQFrame& f) { // state. This is a temporary state after we have sent a channel // exception, where extra frames might arrive that should be // ignored. - // - AMQMethodBody* m=f.getMethod(); + // + AMQMethodBody* m = f.getBody()->getMethod(); try { - if (m && m->invoke(this)) + if (m && invoke(*this, *m)) return; else if (session.get()) session->in(f); diff --git a/cpp/src/qpid/broker/SessionHandler.h b/cpp/src/qpid/broker/SessionHandler.h index aec3731dc0..51a65e3092 100644 --- a/cpp/src/qpid/broker/SessionHandler.h +++ b/cpp/src/qpid/broker/SessionHandler.h @@ -41,7 +41,7 @@ class SessionState; * association between the channel and a session. */ class SessionHandler : public framing::FrameHandler::InOutHandler, - private framing::AMQP_ServerOperations::SessionHandler, + public framing::AMQP_ServerOperations::SessionHandler, private boost::noncopyable { public: diff --git a/cpp/src/qpid/client/ExecutionHandler.cpp b/cpp/src/qpid/client/ExecutionHandler.cpp index 25813dd623..e4edece414 100644 --- a/cpp/src/qpid/client/ExecutionHandler.cpp +++ b/cpp/src/qpid/client/ExecutionHandler.cpp @@ -25,6 +25,7 @@ #include "qpid/framing/MessageTransferBody.h" #include "qpid/framing/AMQP_HighestVersion.h" #include "qpid/framing/all_method_bodies.h" +#include "qpid/framing/ServerInvoker.h" using namespace qpid::client; using namespace qpid::framing; @@ -49,20 +50,13 @@ bool isContentFrame(AMQFrame& frame) return type == HEADER_BODY || type == CONTENT_BODY || isMessageMethod(body); } -bool invoke(AMQBody* body, Invocable* target) -{ - AMQMethodBody* method=body->getMethod(); - return method && method->invoke(target); -} - ExecutionHandler::ExecutionHandler(uint64_t _maxFrameSize) : version(framing::highestProtocolVersion), maxFrameSize(_maxFrameSize) {} //incoming: void ExecutionHandler::handle(AMQFrame& frame) { - AMQBody* body = frame.getBody(); - if (!invoke(body, this)) { + if (!invoke(*this, *frame.getBody())) { if (!arriving) { arriving = FrameSet::shared_ptr(new FrameSet(++incomingCounter)); } diff --git a/cpp/src/qpid/client/ExecutionHandler.h b/cpp/src/qpid/client/ExecutionHandler.h index 7205983cf7..0999540909 100644 --- a/cpp/src/qpid/client/ExecutionHandler.h +++ b/cpp/src/qpid/client/ExecutionHandler.h @@ -38,9 +38,9 @@ namespace qpid { namespace client { class ExecutionHandler : - private framing::AMQP_ServerOperations::ExecutionHandler, - public framing::FrameHandler, - public Execution + public framing::AMQP_ServerOperations::ExecutionHandler, + public framing::FrameHandler, + public Execution { framing::SequenceNumber incomingCounter; framing::AccumulatedAck incomingCompletionStatus; diff --git a/cpp/src/qpid/framing/AMQMethodBody.cpp b/cpp/src/qpid/framing/AMQMethodBody.cpp index 48b50763fc..924d906d43 100644 --- a/cpp/src/qpid/framing/AMQMethodBody.cpp +++ b/cpp/src/qpid/framing/AMQMethodBody.cpp @@ -19,25 +19,10 @@ * */ #include "AMQMethodBody.h" -#include "qpid/framing/InvocationVisitor.h" namespace qpid { namespace framing { AMQMethodBody::~AMQMethodBody() {} -void AMQMethodBody::invoke(AMQP_ServerOperations& ops) -{ - InvocationVisitor v(&ops); - accept(v); - assert(v.wasHandled()); -} - -bool AMQMethodBody::invoke(Invocable* invocable) -{ - InvocationVisitor v(invocable); - accept(v); - return v.wasHandled(); -} - }} // namespace qpid::framing diff --git a/cpp/src/qpid/framing/AMQMethodBody.h b/cpp/src/qpid/framing/AMQMethodBody.h index 09a5ea4f00..9f64fd1690 100644 --- a/cpp/src/qpid/framing/AMQMethodBody.h +++ b/cpp/src/qpid/framing/AMQMethodBody.h @@ -35,7 +35,6 @@ namespace framing { class Buffer; class AMQP_ServerOperations; -class Invocable; class MethodBodyConstVisitor; class AMQMethodBody : public AMQBody { @@ -53,9 +52,6 @@ class AMQMethodBody : public AMQBody { virtual bool resultExpected() const = 0; virtual bool responseExpected() const = 0; - void invoke(AMQP_ServerOperations&); - bool invoke(Invocable*); - template <class T> bool isA() const { return amqpClassId()==T::CLASS_ID && amqpMethodId()==T::METHOD_ID; } diff --git a/cpp/src/qpid/framing/FrameDefaultVisitor.h b/cpp/src/qpid/framing/FrameDefaultVisitor.h index a826d69b3d..93a5204308 100644 --- a/cpp/src/qpid/framing/FrameDefaultVisitor.h +++ b/cpp/src/qpid/framing/FrameDefaultVisitor.h @@ -43,7 +43,8 @@ class AMQHeartbeatBody; * for any non-overridden visit functions. * */ -struct FrameDefaultVisitor : public AMQBodyConstVisitor, public MethodBodyDefaultVisitor +struct FrameDefaultVisitor : public AMQBodyConstVisitor, + protected MethodBodyDefaultVisitor { virtual void defaultVisit(const AMQBody&) = 0; diff --git a/cpp/src/qpid/framing/Invoker.h b/cpp/src/qpid/framing/Invoker.h new file mode 100644 index 0000000000..e6467ab3c4 --- /dev/null +++ b/cpp/src/qpid/framing/Invoker.h @@ -0,0 +1,86 @@ +#ifndef QPID_FRAMING_INVOKER_H +#define QPID_FRAMING_INVOKER_H + +/* + * + * 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/framing/AMQMethodBody.h" +#include "qpid/framing/MethodBodyDefaultVisitor.h" +#include "qpid/framing/StructHelper.h" + +#include <boost/optional.hpp> + +namespace qpid { +namespace framing { + +class AMQMethodBody; + +/** + * Base class for invoker visitors. + */ +class Invoker: public MethodBodyDefaultVisitor, protected StructHelper +{ + public: + struct Result { + public: + Result() : handled(false) {} + const std::string& getResult() const { return result; } + const bool hasResult() const { return !result.empty(); } + bool wasHandled() const { return handled; } + operator bool() const { return handled; } + + std::string result; + bool handled; + }; + + void defaultVisit(const AMQMethodBody&) {} + Result getResult() const { return result; } + + protected: + Result result; +}; + +/** + * Invoke on an invocable object. + * Invocable classes must provide a nested type Invoker. + */ +template <class Invocable> +Invoker::Result invoke(Invocable& target, const AMQMethodBody& body) { + typename Invocable::Invoker invoker(target); + body.accept(invoker); + return invoker.getResult(); +} + +/** + * Invoke on an invocable object. + * Invocable classes must provide a nested type Invoker. + */ +template <class Invocable> +Invoker::Result invoke(Invocable& target, const AMQBody& body) { + typename Invocable::Invoker invoker(target); + const AMQMethodBody* method = body.getMethod(); + if (method) + method->accept(invoker); + return invoker.getResult(); +} + +}} // namespace qpid::framing + +#endif /*!QPID_FRAMING_INVOKER_H*/ diff --git a/cpp/src/qpid/framing/StructHelper.h b/cpp/src/qpid/framing/StructHelper.h index 6b111e1f9e..7fc1d2e22b 100644 --- a/cpp/src/qpid/framing/StructHelper.h +++ b/cpp/src/qpid/framing/StructHelper.h @@ -35,21 +35,14 @@ public: template <class T> void encode(const T t, std::string& data) { uint32_t size = t.size() + 2/*type*/; - char* bytes = static_cast<char*>(::alloca(size)); - Buffer wbuffer(bytes, size); + data.resize(size); + Buffer wbuffer(const_cast<char*>(data.data()), size); wbuffer.putShort(T::TYPE); t.encode(wbuffer); - - Buffer rbuffer(bytes, size); - rbuffer.getRawData(data, size); } template <class T> void decode(T& t, const std::string& data) { - char* bytes = static_cast<char*>(::alloca(data.length())); - Buffer wbuffer(bytes, data.length()); - wbuffer.putRawData(data); - - Buffer rbuffer(bytes, data.length()); + Buffer rbuffer(const_cast<char*>(data.data()), data.length()); uint16_t type = rbuffer.getShort(); if (type == T::TYPE) { t.decode(rbuffer); |