1
0
mirror of https://github.com/eclipse/paho.mqtt.cpp.git synced 2025-05-10 03:39:07 +08:00

Added support for persistence encoding/decoding.

This commit is contained in:
fpagliughi 2020-12-06 10:04:36 -05:00
parent d67a92dbce
commit 22a85aa7c5
7 changed files with 229 additions and 66 deletions

View File

@ -35,101 +35,70 @@ namespace mqtt {
// Constructors
async_client::async_client(const string& serverURI, const string& clientId,
const string& persistDir)
: async_client(serverURI, clientId, 0, persistDir)
const string& persistDir,
ipersistence_encoder* encoder /*=nullptr*/)
: async_client(serverURI, clientId, 0, persistDir, encoder)
{
}
async_client::async_client(const string& serverURI, const string& clientId,
iclient_persistence* persistence /*=nullptr*/)
: async_client(serverURI, clientId, 0, persistence)
iclient_persistence* persistence /*=nullptr*/,
ipersistence_encoder* encoder /*=nullptr*/)
: async_client(serverURI, clientId, 0, persistence, encoder)
{
}
async_client::async_client(const string& serverURI, const string& clientId,
int maxBufferedMessages, const string& persistDir)
: serverURI_(serverURI), clientId_(clientId), mqttVersion_(MQTTVERSION_DEFAULT),
persist_(nullptr), userCallback_(nullptr)
int maxBufferedMessages, const string& persistDir,
ipersistence_encoder* encoder /*=nullptr*/)
: serverURI_(serverURI), clientId_(clientId),
mqttVersion_(MQTTVERSION_DEFAULT), userCallback_(nullptr)
{
create_options opts;
if (maxBufferedMessages != 0) {
opts.set_send_while_disconnected(true);
opts.set_max_buffered_messages(maxBufferedMessages);
}
create_options opts(maxBufferedMessages);
int rc = MQTTAsync_createWithOptions(&cli_, serverURI.c_str(), clientId.c_str(),
MQTTCLIENT_PERSISTENCE_DEFAULT,
const_cast<char*>(persistDir.c_str()),
&opts.opts_);
if (rc != 0)
throw exception(rc);
persistence_encoder(encoder);
}
async_client::async_client(const string& serverURI, const string& clientId,
int maxBufferedMessages, iclient_persistence* persistence /*=nullptr*/)
: serverURI_(serverURI), clientId_(clientId), mqttVersion_(MQTTVERSION_DEFAULT),
persist_(nullptr), userCallback_(nullptr)
int maxBufferedMessages,
iclient_persistence* persistence /*=nullptr*/,
ipersistence_encoder* encoder /*=nullptr*/)
: async_client(serverURI, clientId,
create_options(maxBufferedMessages),
persistence, encoder)
{
create_options opts;
if (maxBufferedMessages != 0) {
opts.set_send_while_disconnected(true);
opts.set_max_buffered_messages(maxBufferedMessages);
}
int rc = MQTTASYNC_SUCCESS;
if (!persistence) {
rc = MQTTAsync_createWithOptions(&cli_, serverURI.c_str(), clientId.c_str(),
MQTTCLIENT_PERSISTENCE_NONE, nullptr,
&opts.opts_);
}
else {
persist_.reset(new MQTTClient_persistence {
persistence,
&iclient_persistence::persistence_open,
&iclient_persistence::persistence_close,
&iclient_persistence::persistence_put,
&iclient_persistence::persistence_get,
&iclient_persistence::persistence_remove,
&iclient_persistence::persistence_keys,
&iclient_persistence::persistence_clear,
&iclient_persistence::persistence_containskey
});
rc = MQTTAsync_createWithOptions(&cli_, serverURI.c_str(), clientId.c_str(),
MQTTCLIENT_PERSISTENCE_USER, persist_.get(),
&opts.opts_);
}
if (rc != 0)
throw exception(rc);
}
async_client::async_client(const string& serverURI, const string& clientId,
const create_options& opts,
const string& persistDir)
const string& persistDir,
ipersistence_encoder* encoder /*=nullptr*/)
: serverURI_(serverURI), clientId_(clientId),
mqttVersion_(opts.opts_.MQTTVersion),
persist_(nullptr), userCallback_(nullptr)
mqttVersion_(opts.opts_.MQTTVersion), userCallback_(nullptr)
{
int rc = MQTTAsync_createWithOptions(&cli_, serverURI.c_str(), clientId.c_str(),
MQTTCLIENT_PERSISTENCE_DEFAULT,
const_cast<char*>(persistDir.c_str()),
const_cast<MQTTAsync_createOptions*>(&opts.opts_));
if (rc != 0)
throw exception(rc);
persistence_encoder(encoder);
}
async_client::async_client(const string& serverURI, const string& clientId,
const create_options& opts,
iclient_persistence* persistence /*=nullptr*/)
iclient_persistence* persistence /*=nullptr*/,
ipersistence_encoder* encoder /*=nullptr*/)
: serverURI_(serverURI), clientId_(clientId),
mqttVersion_(opts.opts_.MQTTVersion),
persist_(nullptr), userCallback_(nullptr)
mqttVersion_(opts.opts_.MQTTVersion), userCallback_(nullptr)
{
int rc = MQTTASYNC_SUCCESS;
@ -154,6 +123,9 @@ async_client::async_client(const string& serverURI, const string& clientId,
rc = MQTTAsync_createWithOptions(&cli_, serverURI.c_str(), clientId.c_str(),
MQTTCLIENT_PERSISTENCE_USER, persist_.get(),
const_cast<MQTTAsync_createOptions*>(&opts.opts_));
if (rc == 0)
persistence_encoder(encoder);
}
if (rc != 0)
throw exception(rc);
@ -350,6 +322,13 @@ void async_client::remove_token(token* tok)
}
}
void async_client::persistence_encoder(ipersistence_encoder* encoder)
{
if (encoder && cli_) {
MQTTAsync_setBeforePersistenceWrite(cli_, encoder, &ipersistence_encoder::before_write);
MQTTAsync_setAfterPersistenceRead(cli_, encoder, &ipersistence_encoder::after_read);
}
}
// --------------------------------------------------------------------------
// Callback management

View File

@ -24,6 +24,15 @@ namespace mqtt {
const MQTTAsync_createOptions create_options::DFLT_C_STRUCT =
MQTTAsync_createOptions_initializer5;
create_options::create_options(int maxBufferedMessages) : create_options()
{
if (maxBufferedMessages != 0) {
opts_.sendWhileDisconnected = to_int(true);
opts_.maxBufferedMessages = maxBufferedMessages;
}
}
/////////////////////////////////////////////////////////////////////////////
} // end namespace mqtt

View File

@ -145,6 +145,58 @@ int iclient_persistence::persistence_containskey(void* handle, char* key)
return MQTTCLIENT_PERSISTENCE_ERROR;
}
/////////////////////////////////////////////////////////////////////////////
// Encoder
int ipersistence_encoder::before_write(void* context, int nbuf, char* bufs[], int buflens[])
{
try {
if (context && nbuf > 0 && bufs && buflens) {
std::vector<string_view> vec;
auto n = size_t(nbuf);
vec.reserve(n);
for (size_t i=0; i<n; ++i)
vec.push_back(string_view(bufs[i], buflens[i]));
static_cast<ipersistence_encoder*>(context)->encode(&vec[0], n);
for (size_t i=0; i<n; ++i) {
if (bufs[i] != vec[i].data()) {
MQTTAsync_free(bufs[i]);
bufs[i] = const_cast<char*>(vec[i].data());
}
buflens[i] = vec[i].size();
}
return MQTTASYNC_SUCCESS;
}
}
catch (...) {}
return MQTTCLIENT_PERSISTENCE_ERROR;
}
int ipersistence_encoder::after_read(void* context, char** buf, int* buflen)
{
try {
if (context && buf && *buf && buflen && *buflen > 0) {
string_view sv(*buf, *buflen);
static_cast<ipersistence_encoder*>(context)->decode(sv);
if (*buf != sv.data()) {
MQTTAsync_free(*buf);
*buf = const_cast<char*>(sv.data());
}
*buflen = sv.size();
return MQTTASYNC_SUCCESS;
}
}
catch (...) {}
return MQTTCLIENT_PERSISTENCE_ERROR;
}
/////////////////////////////////////////////////////////////////////////////
// end namespace mqtt
}

View File

@ -154,6 +154,9 @@ private:
throw exception(rc);
}
/** Installs a persistence encoder/decoder */
void persistence_encoder(ipersistence_encoder* encoder);
public:
/**
* Create an async_client that can be used to communicate with an MQTT
@ -164,10 +167,13 @@ public:
* @param clientId a client identifier that is unique on the server
* being connected to
* @param persistDir The directory to use for persistence data
* @param encoder An object to encode and decode the persistence data.
* @throw exception if an argument is invalid
*/
async_client(const string& serverURI, const string& clientId,
const string& persistDir);
const string& persistDir,
ipersistence_encoder* encoder=nullptr);
/**
* Create an async_client that can be used to communicate with an MQTT
* server.
@ -179,10 +185,12 @@ public:
* being connected to
* @param persistence The user persistence structure. If this is null,
* then no persistence is used.
* @param encoder An object to encode and decode the persistence data.
* @throw exception if an argument is invalid
*/
async_client(const string& serverURI, const string& clientId,
iclient_persistence* persistence=nullptr);
iclient_persistence* persistence=nullptr,
ipersistence_encoder* encoder=nullptr);
/**
* Create an async_client that can be used to communicate with an MQTT
* server, which allows for off-line message buffering.
@ -194,10 +202,12 @@ public:
* @param maxBufferedMessages the maximum number of messages allowed to
* be buffered while not connected
* @param persistDir The directory to use for persistence data
* @param encoder An object to encode and decode the persistence data.
* @throw exception if an argument is invalid
*/
async_client(const string& serverURI, const string& clientId,
int maxBufferedMessages, const string& persistDir);
int maxBufferedMessages, const string& persistDir,
ipersistence_encoder* encoder=nullptr);
/**
* Create an async_client that can be used to communicate with an MQTT
* server, which allows for off-line message buffering.
@ -211,11 +221,13 @@ public:
* be buffered while not connected
* @param persistence The user persistence structure. If this is null,
* then no persistence is used.
* @param encoder An object to encode and decode the persistence data.
* @throw exception if an argument is invalid
*/
async_client(const string& serverURI, const string& clientId,
int maxBufferedMessages, iclient_persistence* persistence=nullptr);
int maxBufferedMessages,
iclient_persistence* persistence=nullptr,
ipersistence_encoder* encoder=nullptr);
/**
* Create an async_client that can be used to communicate with an MQTT
* server, which allows for off-line message buffering.
@ -226,10 +238,12 @@ public:
* being connected to
* @param opts The create options
* @param persistDir The directory to use for persistence data
* @param encoder An object to encode and decode the persistence data.
* @throw exception if an argument is invalid
*/
async_client(const string& serverURI, const string& clientId,
const create_options& opts, const string& persistDir);
const create_options& opts, const string& persistDir,
ipersistence_encoder* encoder=nullptr);
/**
* Create an async_client that can be used to communicate with an MQTT
* server, which allows for off-line message buffering.
@ -243,11 +257,13 @@ public:
* be buffered while not connected
* @param persistence The user persistence structure. If this is null,
* then no persistence is used.
* @param encoder An object to encode and decode the persistence data.
* @throw exception if an argument is invalid
*/
async_client(const string& serverURI, const string& clientId,
const create_options& opts,
iclient_persistence* persistence=nullptr);
iclient_persistence* persistence=nullptr,
ipersistence_encoder* encoder=nullptr);
/**
* Destructor
*/

View File

@ -56,6 +56,12 @@ public:
* Default set of client create options.
*/
create_options() : opts_(DFLT_C_STRUCT) {}
/**
* Default create options, but with off-line buffering enabled.
* @param maxBufferedMessages the maximum number of messages allowed to
* be buffered while not connected
*/
explicit create_options(int maxBufferedMessages);
/**
* Gets whether the client will accept message to publish while
* disconnected.

View File

@ -32,6 +32,24 @@
namespace mqtt {
/**
* Allocate memory for use with user persistence.
*
* @param n The number of bytes for the buffer.
* @return A pointer to the allocated memory
*/
inline char* persistence_malloc(size_t n) {
return static_cast<char*>(MQTTAsync_malloc(n));
}
/**
* Frees memory allocated with @ref persistence_malloc
* @param p Pointer to a buffer obtained by persistence_malloc.
*/
inline void persistence_free(char* p) {
MQTTAsync_free(p);
}
/////////////////////////////////////////////////////////////////////////////
/**
@ -128,8 +146,56 @@ using iclient_persistence_ptr = iclient_persistence::ptr_t;
/** Smart/shared pointer to a persistence client */
using const_iclient_persistence_ptr = iclient_persistence::const_ptr_t;
/////////////////////////////////////////////////////////////////////////////
/**
* Interface for objects to encode and decode data going into the
* persistence store.
*
* This is typically used to encrypt the data before writing to
* persistence, and then decrypt it when reading it back from persistence.
*/
class ipersistence_encoder
{
friend class async_client;
/** Callbacks from the C library */
static int before_write(void* context, int bufcount, char* buffers[], int buflens[]);
static int after_read(void* context, char** buffer, int* buflen);
public:
/**
* Virtual destructor.
*/
virtual ~ipersistence_encoder() {}
/**
* Callback to let the application encode data before writing it to
* persistence.
*
* This is called just prior to writing the data to persistence. If a
* buffer needs to grow, the application can call @ref
* persistence_malloc to get a new buffer, and then update the pointer
* and side of the buffer. It *should not* free the old buffer. That is
* done automatically.
*
* @param bufs The data buffers that need to be encoded.
* @param n The number of buffers
*/
virtual void encode(string_view bufs[], size_t n) =0;
/**
* Callback to let the application decode data after it is retrieved
* from persistence.
*
* @param buffers The data buffers that need to be decoded.
* @param n The number of buffers
*/
virtual void decode(string_view& buf) =0;
};
/////////////////////////////////////////////////////////////////////////////
// end namespace mqtt
}
#endif // __mqtt_iclient_persistence_h

View File

@ -60,6 +60,7 @@ using namespace std;
using namespace std::chrono;
const std::string DFLT_ADDRESS { "tcp://localhost:1883" };
const std::string CLIENT_ID { "paho-cpp-data-publish" };
const string TOPIC { "data/rand" };
const int QOS = 1;
@ -72,11 +73,45 @@ const string PERSIST_DIR { "data-persist" };
/////////////////////////////////////////////////////////////////////////////
class persistence_encoder : virtual public mqtt::ipersistence_encoder
{
/**
* Callback to let the application encode data before writing it to
* persistence.
*/
void encode(mqtt::string_view bufs[], size_t n) override {
cout << "Encoding " << n << " buffers" << endl;
auto sz = bufs[0].size();
auto buf = mqtt::persistence_malloc(sz+6);
strcpy(buf, "bubba");
memcpy(buf+6, bufs[0].data(), sz);
bufs[0] = mqtt::string_view(buf, n+6);
}
/**
* Callback to let the application decode data after it is retrieved
* from persistence.
*
* @param buffers The data buffers that need to be decoded.
* @param n The number of buffers
*/
void decode(mqtt::string_view& buf) override {
cout << "Decoding buffer: " << buf.data() << endl;
auto n = buf.size();
auto newBuf = mqtt::persistence_malloc(n-6);
memcpy(newBuf, buf.data(), n-6);
buf = mqtt::string_view(newBuf, n-6);
}
};
/////////////////////////////////////////////////////////////////////////////
int main(int argc, char* argv[])
{
string address = (argc > 1) ? string(argv[1]) : DFLT_ADDRESS;
mqtt::async_client cli(address, "", MAX_BUFFERED_MSGS, PERSIST_DIR);
persistence_encoder encoder;
mqtt::async_client cli(address, CLIENT_ID, MAX_BUFFERED_MSGS,
PERSIST_DIR, &encoder);
mqtt::connect_options connOpts;
connOpts.set_keep_alive_interval(MAX_BUFFERED_MSGS * PERIOD);