pxmlw6n2f/Gazebo_Distributed_TCP/gazebo/transport/CallbackHelper.hh

230 lines
7.8 KiB
C++

/*
* Copyright (C) 2012 Open Source Robotics Foundation
*
* Licensed 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.
*
*/
#ifndef _CALLBACKHELPER_HH_
#define _CALLBACKHELPER_HH_
#include <google/protobuf/message.h>
#include <boost/function.hpp>
#include <boost/shared_ptr.hpp>
#include <vector>
#include <string>
#include <mutex>
#include "gazebo/common/Console.hh"
#include "gazebo/msgs/msgs.hh"
#include "gazebo/common/Exception.hh"
#include "gazebo/transport/TransportTypes.hh"
#include "gazebo/util/system.hh"
namespace gazebo
{
namespace transport
{
/// \addtogroup gazebo_transport Transport
/// \{
/// \class CallbackHelper CallbackHelper.hh transport/transport.hh
/// \brief A helper class to handle callbacks when messages arrive
class GZ_TRANSPORT_VISIBLE CallbackHelper
{
/// \brief Constructor
/// \param[in] _latching Set to true to make the callback helper
/// latching.
public: CallbackHelper(bool _latching = false);
/// \brief Destructor
public: virtual ~CallbackHelper();
/// \brief Get the typename of the message that is handled
/// \return String representation of the message type
public: virtual std::string GetMsgType() const;
/// \brief Process new incoming data
/// \param[in] _newdata Incoming data to be processed
/// \return true if successfully processed; false otherwise
/// \param[in] _cb If non-null, callback to be invoked which signals
/// that transmission is complete.
/// \param[in] _id ID associated with the message data.
public: virtual bool HandleData(const std::string &_newdata,
boost::function<void(uint32_t)> _cb, uint32_t _id) = 0;
/// \brief Process new incoming message
/// \param[in] _newMsg Incoming message to be processed
/// \return true if successfully processed; false otherwise
public: virtual bool HandleMessage(MessagePtr _newMsg) = 0;
/// \brief Is the callback local?
/// \return true if the callback is local, false if the callback
/// is tied to a remote connection
public: virtual bool IsLocal() const = 0;
/// \brief Is the callback latching?
/// \return true if the callback is latching, false otherwise
public: bool GetLatching() const;
/// \brief Set whether this callback is latching.
/// This function should only be used by the Transport library.
/// \param[in] _latch False to turn off latching.
public: void SetLatching(bool _latch);
/// \brief Get the unique ID of this callback.
/// \return The unique ID of this callback.
public: unsigned int GetId() const;
/// \brief True means that the callback helper will get the last
/// published message on the topic.
protected: bool latching;
/// \brief Mutex to protect the latching variable.
protected: mutable std::mutex latchingMutex;
/// \brief A counter to generate the unique id of this callback.
private: static unsigned int idCounter;
/// \brief The unique id of this callback.
private: unsigned int id;
};
/// \brief boost shared pointer to transport::CallbackHelper
typedef boost::shared_ptr<CallbackHelper> CallbackHelperPtr;
/// \class CallbackHelperT CallbackHelper.hh transport/transport.hh
/// \brief Callback helper Template
template<class M>
class CallbackHelperT : public CallbackHelper
{
/// \brief Constructor
/// \param[in] _cb boost function to call on incoming messages
/// \param[in] _latching Set to true to make the callback helper
/// latching.
public: CallbackHelperT(const boost::function<
void (const boost::shared_ptr<M const> &)> &_cb,
bool _latching = false)
: CallbackHelper(_latching), callback(_cb)
{
// Just some code to make sure we have a google protobuf.
/*M test;
google::protobuf::Message *m;
if ((m =dynamic_cast<google::protobuf::Message*>(&test))
== NULL)
gzthrow("Message type must be a google::protobuf type\n");
*/
}
// documentation inherited
public: std::string GetMsgType() const
{
M test;
google::protobuf::Message *m;
if ((m = dynamic_cast<google::protobuf::Message*>(&test))
== NULL)
gzthrow("Message type must be a google::protobuf type\n");
return m->GetTypeName();
}
// documentation inherited
public: virtual bool HandleData(const std::string &_newdata,
boost::function<void(uint32_t)> _cb, uint32_t _id)
{
this->SetLatching(false);
boost::shared_ptr<M> m(new M);
m->ParseFromString(_newdata);
this->callback(m);
if (!_cb.empty())
_cb(_id);
return true;
}
// documentation inherited
public: virtual bool HandleMessage(MessagePtr _newMsg)
{
this->SetLatching(false);
this->callback(boost::dynamic_pointer_cast<M>(_newMsg));
return true;
}
// documentation inherited
public: virtual bool IsLocal() const
{
return true;
}
private: boost::function<void (const boost::shared_ptr<M const> &)>
callback;
};
/// \class RawCallbackHelper RawCallbackHelper.hh transport/transport.hh
/// \brief Used to connect publishers to subscribers, where the
/// subscriber wants the raw data from the publisher. Raw means that the
/// data has not been converted into a protobuf message.
class GZ_TRANSPORT_VISIBLE RawCallbackHelper : public CallbackHelper
{
/// \brief Constructor
/// \param[in] _cb boost function to call on incoming messages
/// \param[in] _latching Set to true to make the callback helper
/// latching.
public: RawCallbackHelper(
const boost::function<void (const std::string &)> &_cb,
bool _latching = false)
: CallbackHelper(_latching), callback(_cb)
{
}
// documentation inherited
public: std::string GetMsgType() const
{
return "raw";
}
// documentation inherited
public: virtual bool HandleData(const std::string &_newdata,
boost::function<void(uint32_t)> _cb, uint32_t _id)
{
this->SetLatching(false);
this->callback(_newdata);
if (!_cb.empty())
_cb(_id);
return true;
}
// documentation inherited
public: virtual bool HandleMessage(MessagePtr _newMsg)
{
this->SetLatching(false);
std::string data;
_newMsg->SerializeToString(&data);
this->callback(data);
return true;
}
// documentation inherited
public: virtual bool IsLocal() const
{
return true;
}
private: boost::function<void (const std::string &)> callback;
};
/// \}
}
}
#endif