Skip to content

Commit dca82c4

Browse files
committed
MQTT client demo builds and links - but it is still using shared objects instead of static libs
1 parent d708d37 commit dca82c4

File tree

3 files changed

+185
-6
lines changed

3 files changed

+185
-6
lines changed

CMakeLists.txt

+14-4
Original file line numberDiff line numberDiff line change
@@ -15,9 +15,18 @@ add_executable(mqtt_pi_client ${SOURCE_FILES})
1515
target_include_directories(mqtt_pi_client
1616
PUBLIC
1717
.
18-
#include
19-
#ports/posix
20-
#source/
18+
paho.mqtt.c/src
19+
paho.mqtt.cpp/src
20+
)
21+
22+
find_library(PAHO_MQTT_C_LIB_PRO NAMES paho-mqtt3as libpaho-mqtt3as HINTS "${CMAKE_CURRENT_SOURCE_DIR}/build/paho.mqtt.c/src")
23+
find_library(PAHO_MQTT_CPP_LIB_PRO NAMES paho-mqttpp3 libpaho-mqttpp3 HINTS "${CMAKE_CURRENT_SOURCE_DIR}/build/paho.mqtt.cpp/src")
24+
25+
26+
target_link_libraries(mqtt_pi_client
27+
PUBLIC
28+
${PAHO_MQTT_C_LIB_PRO}
29+
${PAHO_MQTT_CPP_LIB_PRO}
2130
)
2231

2332
# Build Paho MQTT C as lib to be used by Paho MQTT C++
@@ -28,11 +37,12 @@ set(PAHO_HIGH_PERFORMANCE TRUE)
2837
add_subdirectory(paho.mqtt.c)
2938

3039
# Specify where the MQTT C libs/headers can be found
31-
#list(APPEND CMAKE_PREFIX_PATH "${CMAKE_CURRENT_SOURCE_DIR}/build/paho.mqtt.c/src")
40+
list(APPEND CMAKE_PREFIX_PATH "${CMAKE_CURRENT_SOURCE_DIR}/build/paho.mqtt.c/src")
3241
set(PAHO_MQTT_C_LIBRARIES ${CMAKE_CURRENT_SOURCE_DIR}/build/paho.mqtt.c/src/libpaho-mqtt3cs.so) #TODO: Determine what lib we actually want to link to - there are three variants
3342
set(PAHO_MQTT_C_INCLUDE_DIRS ${CMAKE_CURRENT_SOURCE_DIR}/paho.mqtt.c/src)
3443

3544
# Many of the above Paho MQTT C settings apply to C++ as well
45+
list(APPEND CMAKE_PREFIX_PATH "${CMAKE_CURRENT_SOURCE_DIR}/build/paho.mqtt.cpp/src")
3646
add_subdirectory(paho.mqtt.cpp)
3747

3848
set(CPACK_PROJECT_NAME ${PROJECT_NAME})

main.cpp

+170-1
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,179 @@
11
#include <iostream>
22
#include "project_config.hpp"
33

4+
#include "mqtt/async_client.h"
5+
46
using std::cout;
57
using std::endl;
68

9+
const std::string SERVER_ADDRESS("tcp:broker.hivemq.com:1883");
10+
const std::string CLIENT_ID("totally_random_guy");
11+
const std::string TOPIC("goofygumdrop");
12+
13+
const int QOS = 1;
14+
const int N_RETRY_ATTEMPTS = 5;
15+
16+
// Callbacks for the success or failures of requested actions.
17+
// This could be used to initiate further action, but here we just log the
18+
// results to the console.
19+
20+
class action_listener : public virtual mqtt::iaction_listener
21+
{
22+
std::string name_;
23+
24+
void on_failure(const mqtt::token& tok) override {
25+
std::cout << name_ << " failure";
26+
if (tok.get_message_id() != 0)
27+
std::cout << " for token: [" << tok.get_message_id() << "]" << std::endl;
28+
std::cout << std::endl;
29+
}
30+
31+
void on_success(const mqtt::token& tok) override {
32+
std::cout << name_ << " success";
33+
if (tok.get_message_id() != 0)
34+
std::cout << " for token: [" << tok.get_message_id() << "]" << std::endl;
35+
auto top = tok.get_topics();
36+
if (top && !top->empty())
37+
std::cout << "\ttoken topic: '" << (*top)[0] << "', ..." << std::endl;
38+
std::cout << std::endl;
39+
}
40+
41+
public:
42+
action_listener(const std::string& name) : name_(name) {}
43+
};
44+
45+
/////////////////////////////////////////////////////////////////////////////
46+
47+
/**
48+
* Local callback & listener class for use with the client connection.
49+
* This is primarily intended to receive messages, but it will also monitor
50+
* the connection to the broker. If the connection is lost, it will attempt
51+
* to restore the connection and re-subscribe to the topic.
52+
*/
53+
class callback : public virtual mqtt::callback,
54+
public virtual mqtt::iaction_listener
55+
56+
{
57+
// Counter for the number of connection retries
58+
int nretry_;
59+
// The MQTT client
60+
mqtt::async_client& cli_;
61+
// Options to use if we need to reconnect
62+
mqtt::connect_options& connOpts_;
63+
// An action listener to display the result of actions.
64+
action_listener subListener_;
65+
66+
// This deomonstrates manually reconnecting to the broker by calling
67+
// connect() again. This is a possibility for an application that keeps
68+
// a copy of it's original connect_options, or if the app wants to
69+
// reconnect with different options.
70+
// Another way this can be done manually, if using the same options, is
71+
// to just call the async_client::reconnect() method.
72+
void reconnect() {
73+
std::this_thread::sleep_for(std::chrono::milliseconds(2500));
74+
try {
75+
cli_.connect(connOpts_, nullptr, *this);
76+
}
77+
catch (const mqtt::exception& exc) {
78+
std::cerr << "Error: " << exc.what() << std::endl;
79+
exit(1);
80+
}
81+
}
82+
83+
// Re-connection failure
84+
void on_failure(const mqtt::token& tok) override {
85+
std::cout << "Connection attempt failed" << std::endl;
86+
if (++nretry_ > N_RETRY_ATTEMPTS)
87+
exit(1);
88+
reconnect();
89+
}
90+
91+
// (Re)connection success
92+
// Either this or connected() can be used for callbacks.
93+
void on_success(const mqtt::token& tok) override {}
94+
95+
// (Re)connection success
96+
void connected(const std::string& cause) override {
97+
std::cout << "\nConnection success" << std::endl;
98+
std::cout << "\nSubscribing to topic '" << TOPIC << "'\n"
99+
<< "\tfor client " << CLIENT_ID
100+
<< " using QoS" << QOS << "\n"
101+
<< "\nPress Q<Enter> to quit\n" << std::endl;
102+
103+
cli_.subscribe(TOPIC, QOS, nullptr, subListener_);
104+
}
105+
106+
// Callback for when the connection is lost.
107+
// This will initiate the attempt to manually reconnect.
108+
void connection_lost(const std::string& cause) override {
109+
std::cout << "\nConnection lost" << std::endl;
110+
if (!cause.empty())
111+
std::cout << "\tcause: " << cause << std::endl;
112+
113+
std::cout << "Reconnecting..." << std::endl;
114+
nretry_ = 0;
115+
reconnect();
116+
}
117+
118+
// Callback for when a message arrives.
119+
void message_arrived(mqtt::const_message_ptr msg) override {
120+
std::cout << "Message arrived" << std::endl;
121+
std::cout << "\ttopic: '" << msg->get_topic() << "'" << std::endl;
122+
std::cout << "\tpayload: '" << msg->to_string() << "'\n" << std::endl;
123+
}
124+
125+
void delivery_complete(mqtt::delivery_token_ptr token) override {}
126+
127+
public:
128+
callback(mqtt::async_client& cli, mqtt::connect_options& connOpts)
129+
: nretry_(0), cli_(cli), connOpts_(connOpts), subListener_("Subscription") {}
130+
};
131+
7132
int main(int, char**) {
8133
cout << "Project name: " << PROJECT_NAME << " Version: " << PROJECT_VER << endl;
9-
cout << "Just adding some more text for another breakpoint location" << endl;
134+
135+
// A subscriber often wants the server to remember its messages when its
136+
// disconnected. In that case, it needs a unique ClientID and a
137+
// non-clean session.
138+
139+
mqtt::async_client cli(SERVER_ADDRESS, CLIENT_ID);
140+
141+
mqtt::connect_options connOpts;
142+
connOpts.set_clean_session(false);
143+
144+
// Install the callback(s) before connecting.
145+
callback cb(cli, connOpts);
146+
cli.set_callback(cb);
147+
148+
// Start the connection.
149+
// When completed, the callback will subscribe to topic.
150+
151+
try {
152+
std::cout << "Connecting to the MQTT server..." << std::flush;
153+
cli.connect(connOpts, nullptr, cb);
154+
}
155+
catch (const mqtt::exception& exc) {
156+
std::cerr << "\nERROR: Unable to connect to MQTT server: '"
157+
<< SERVER_ADDRESS << "'" << exc << std::endl;
158+
return 1;
159+
}
160+
161+
// Just block till user tells us to quit.
162+
163+
while (std::tolower(std::cin.get()) != 'q')
164+
;
165+
166+
// Disconnect
167+
168+
try {
169+
std::cout << "\nDisconnecting from the MQTT server..." << std::flush;
170+
cli.disconnect()->wait();
171+
std::cout << "OK" << std::endl;
172+
}
173+
catch (const mqtt::exception& exc) {
174+
std::cerr << exc << std::endl;
175+
return 1;
176+
}
177+
178+
return 0;
10179
}

toolchain.cmake

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,6 @@ MESSAGE(STATUS "Toolchain build type:${CMAKE_BUILD_TYPE}")
2121
SET(CMAKE_C_FLAGS ${CMAKE_CXX_FLAGS})
2222
SET(CMAKE_EXE_LINKER_FLAGS "--sysroot=${sysroot_target}")
2323
set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
24-
set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
24+
set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY BOTH)
2525
set(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)
2626
set(CMAKE_FIND_ROOT_PATH_MODE_PACKAGE ONLY)

0 commit comments

Comments
 (0)