Modified M2XStreamClient

Fork of M2XStreamClient-JMF by Jim Flynn

Committer:
JMF
Date:
Sun Sep 18 14:45:31 2016 +0000
Revision:
25:67079c11d1ea
Parent:
24:6a5ba4cfa19a
Changed HTTP/1.0 references to HTTP/1.1  because the server apparently answeres 1.0 requests incorrectly which causes the library to hang.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
jb8414 0:f479e4f4db0e 1 #ifndef M2XStreamClient_h
jb8414 0:f479e4f4db0e 2 #define M2XStreamClient_h
jb8414 0:f479e4f4db0e 3
defmacro 22:4d895e732765 4 #if (!defined(ARDUINO_PLATFORM)) && (!defined(ESP8266_PLATFORM)) && (!defined(MBED_PLATFORM))
defmacro 22:4d895e732765 5 #error "Platform definition is missing!"
jb8414 0:f479e4f4db0e 6 #endif
jb8414 0:f479e4f4db0e 7
defmacro 22:4d895e732765 8 #define M2X_VERSION "2.2.0"
jb8414 0:f479e4f4db0e 9
jb8414 0:f479e4f4db0e 10 #ifdef ARDUINO_PLATFORM
defmacro 22:4d895e732765 11 #include "m2x-arduino.h"
defmacro 22:4d895e732765 12 #endif /* ARDUINO_PLATFORM */
defmacro 22:4d895e732765 13
defmacro 22:4d895e732765 14 #ifdef ESP8266_PLATFORM
defmacro 22:4d895e732765 15 #include "m2x-esp8266.h"
defmacro 22:4d895e732765 16 #endif /* ESP8266_PLATFORM */
jb8414 0:f479e4f4db0e 17
jb8414 0:f479e4f4db0e 18 #ifdef MBED_PLATFORM
defmacro 22:4d895e732765 19 #include "m2x-mbed.h"
defmacro 22:4d895e732765 20 #endif /* MBED_PLATFORM */
defmacro 22:4d895e732765 21
defmacro 22:4d895e732765 22 /* If we don't have DBG defined, provide dump implementation */
defmacro 22:4d895e732765 23 #ifndef DBG
jb8414 0:f479e4f4db0e 24 #define DBG(fmt_, data_)
jb8414 0:f479e4f4db0e 25 #define DBGLN(fmt_, data_)
jb8414 0:f479e4f4db0e 26 #define DBGLNEND
defmacro 22:4d895e732765 27 #endif /* DBG */
jb8414 0:f479e4f4db0e 28
defmacro 22:4d895e732765 29 #define MIN(a, b) (((a) > (b))?(b):(a))
defmacro 22:4d895e732765 30 #define TO_HEX(t_) ((char) (((t_) > 9) ? ((t_) - 10 + 'A') : ((t_) + '0')))
jb8414 0:f479e4f4db0e 31 #define MAX_DOUBLE_DIGITS 7
jb8414 0:f479e4f4db0e 32
defmacro 22:4d895e732765 33 /* For tolower */
defmacro 22:4d895e732765 34 #include <ctype.h>
defmacro 22:4d895e732765 35
jb8414 0:f479e4f4db0e 36 static const int E_OK = 0;
jb8414 0:f479e4f4db0e 37 static const int E_NOCONNECTION = -1;
jb8414 0:f479e4f4db0e 38 static const int E_DISCONNECTED = -2;
jb8414 0:f479e4f4db0e 39 static const int E_NOTREACHABLE = -3;
jb8414 0:f479e4f4db0e 40 static const int E_INVALID = -4;
jb8414 0:f479e4f4db0e 41 static const int E_JSON_INVALID = -5;
citrusbyte 16:7903152de19f 42 static const int E_BUFFER_TOO_SMALL = -6;
citrusbyte 16:7903152de19f 43 static const int E_TIMESTAMP_ERROR = -8;
citrusbyte 16:7903152de19f 44
defmacro 22:4d895e732765 45 static const char* DEFAULT_M2X_HOST = "api-m2x.att.com";
defmacro 22:4d895e732765 46 static const int DEFAULT_M2X_PORT = 80;
defmacro 22:4d895e732765 47
citrusbyte 16:7903152de19f 48 static inline bool m2x_status_is_success(int status) {
citrusbyte 16:7903152de19f 49 return (status == E_OK) || (status >= 200 && status <= 299);
citrusbyte 16:7903152de19f 50 }
citrusbyte 16:7903152de19f 51
citrusbyte 16:7903152de19f 52 static inline bool m2x_status_is_client_error(int status) {
citrusbyte 16:7903152de19f 53 return status >= 400 && status <= 499;
citrusbyte 16:7903152de19f 54 }
citrusbyte 16:7903152de19f 55
citrusbyte 16:7903152de19f 56 static inline bool m2x_status_is_server_error(int status) {
citrusbyte 16:7903152de19f 57 return status >= 500 && status <= 599;
citrusbyte 16:7903152de19f 58 }
citrusbyte 16:7903152de19f 59
citrusbyte 16:7903152de19f 60 static inline bool m2x_status_is_error(int status) {
citrusbyte 16:7903152de19f 61 return m2x_status_is_client_error(status) ||
citrusbyte 16:7903152de19f 62 m2x_status_is_server_error(status);
citrusbyte 16:7903152de19f 63 }
jb8414 0:f479e4f4db0e 64
defmacro 22:4d895e732765 65 // Null Print class used to calculate length to print
defmacro 22:4d895e732765 66 class NullPrint : public Print {
defmacro 22:4d895e732765 67 public:
defmacro 22:4d895e732765 68 size_t counter;
defmacro 22:4d895e732765 69
defmacro 22:4d895e732765 70 virtual size_t write(uint8_t b) {
defmacro 22:4d895e732765 71 counter++;
defmacro 22:4d895e732765 72 return 1;
defmacro 22:4d895e732765 73 }
defmacro 22:4d895e732765 74
defmacro 22:4d895e732765 75 virtual size_t write(const uint8_t* buf, size_t size) {
defmacro 22:4d895e732765 76 counter += size;
defmacro 22:4d895e732765 77 return size;
defmacro 22:4d895e732765 78 }
defmacro 22:4d895e732765 79 };
defmacro 22:4d895e732765 80
defmacro 22:4d895e732765 81 // Encodes and prints string using Percent-encoding specified
defmacro 22:4d895e732765 82 // in RFC 1738, Section 2.2
defmacro 22:4d895e732765 83 static inline int print_encoded_string(Print* print, const char* str) {
defmacro 22:4d895e732765 84 int bytes = 0;
defmacro 22:4d895e732765 85 for (int i = 0; str[i] != 0; i++) {
defmacro 22:4d895e732765 86 if (((str[i] >= 'A') && (str[i] <= 'Z')) ||
defmacro 22:4d895e732765 87 ((str[i] >= 'a') && (str[i] <= 'z')) ||
defmacro 22:4d895e732765 88 ((str[i] >= '0') && (str[i] <= '9')) ||
defmacro 22:4d895e732765 89 (str[i] == '-') || (str[i] == '_') ||
defmacro 22:4d895e732765 90 (str[i] == '.') || (str[i] == '~')) {
defmacro 22:4d895e732765 91 bytes += print->print(str[i]);
defmacro 22:4d895e732765 92 } else {
defmacro 22:4d895e732765 93 // Encode all other characters
defmacro 22:4d895e732765 94 bytes += print->print('%');
defmacro 22:4d895e732765 95 bytes += print->print(TO_HEX(str[i] / 16));
defmacro 22:4d895e732765 96 bytes += print->print(TO_HEX(str[i] % 16));
defmacro 22:4d895e732765 97 }
defmacro 22:4d895e732765 98 }
defmacro 22:4d895e732765 99 return bytes;
defmacro 22:4d895e732765 100 }
defmacro 22:4d895e732765 101
defmacro 22:4d895e732765 102 #ifdef M2X_ENABLE_READER
citrusbyte 10:4ce9eba38dbe 103 /*
citrusbyte 10:4ce9eba38dbe 104 * +type+ indicates the value type: 1 for string, 2 for number
citrusbyte 10:4ce9eba38dbe 105 * NOTE that the value type here only contains a hint on how
citrusbyte 10:4ce9eba38dbe 106 * you can use the value. Even though 2 is returned, the value
citrusbyte 10:4ce9eba38dbe 107 * is still stored in (const char *), and atoi/atof is needed to
citrusbyte 10:4ce9eba38dbe 108 * get the actual value
citrusbyte 10:4ce9eba38dbe 109 */
jb8414 0:f479e4f4db0e 110 typedef void (*stream_value_read_callback)(const char* at,
jb8414 0:f479e4f4db0e 111 const char* value,
jb8414 0:f479e4f4db0e 112 int index,
citrusbyte 10:4ce9eba38dbe 113 void* context,
citrusbyte 10:4ce9eba38dbe 114 int type);
jb8414 0:f479e4f4db0e 115
jb8414 0:f479e4f4db0e 116 typedef void (*location_read_callback)(const char* name,
jb8414 0:f479e4f4db0e 117 double latitude,
jb8414 0:f479e4f4db0e 118 double longitude,
jb8414 0:f479e4f4db0e 119 double elevation,
jb8414 0:f479e4f4db0e 120 const char* timestamp,
jb8414 0:f479e4f4db0e 121 int index,
jb8414 0:f479e4f4db0e 122 void* context);
JMF 23:f32837239193 123
defmacro 22:4d895e732765 124 typedef void (*m2x_command_read_callback)(const char* id,
defmacro 22:4d895e732765 125 const char* name,
defmacro 22:4d895e732765 126 int index,
defmacro 22:4d895e732765 127 void *context);
defmacro 22:4d895e732765 128 #endif /* M2X_ENABLE_READER */
defmacro 22:4d895e732765 129
defmacro 22:4d895e732765 130 typedef void (*m2x_fill_data_callback)(Print *print, void *context);
defmacro 22:4d895e732765 131
jb8414 0:f479e4f4db0e 132 class M2XStreamClient {
jb8414 0:f479e4f4db0e 133 public:
jb8414 0:f479e4f4db0e 134 M2XStreamClient(Client* client,
jb8414 0:f479e4f4db0e 135 const char* key,
defmacro 22:4d895e732765 136 void (* idlefunc)(void) = NULL,
jb8414 0:f479e4f4db0e 137 int case_insensitive = 1,
defmacro 22:4d895e732765 138 const char* host = DEFAULT_M2X_HOST,
defmacro 22:4d895e732765 139 int port = DEFAULT_M2X_PORT,
NetArc 14:205076b587fe 140 const char* path_prefix = NULL);
jb8414 0:f479e4f4db0e 141
citrusbyte 10:4ce9eba38dbe 142 // Push data stream value using PUT request, returns the HTTP status code
defmacro 22:4d895e732765 143 // NOTE: if you want to update by a serial, use "serial/<serial ID>" as
defmacro 22:4d895e732765 144 // the device ID here.
jb8414 0:f479e4f4db0e 145 template <class T>
citrusbyte 13:0d574742208f 146 int updateStreamValue(const char* deviceId, const char* streamName, T value);
jb8414 0:f479e4f4db0e 147
jb8414 0:f479e4f4db0e 148 // Post multiple values to M2X all at once.
citrusbyte 13:0d574742208f 149 // +deviceId+ - id of the device to post values
jb8414 0:f479e4f4db0e 150 // +streamNum+ - Number of streams to post
jb8414 0:f479e4f4db0e 151 // +names+ - Array of stream names, the length of the array should
jb8414 0:f479e4f4db0e 152 // be exactly +streamNum+
jb8414 0:f479e4f4db0e 153 // +counts+ - Array of +streamNum+ length, each item in this array
jb8414 0:f479e4f4db0e 154 // containing the number of values we want to post for each stream
jb8414 0:f479e4f4db0e 155 // +ats+ - Timestamps for each value, the length of this array should
jb8414 0:f479e4f4db0e 156 // be the some of all values in +counts+, for the first +counts[0]+
jb8414 0:f479e4f4db0e 157 // items, the values belong to the first stream, for the following
jb8414 0:f479e4f4db0e 158 // +counts[1]+ number of items, the values belong to the second stream,
citrusbyte 10:4ce9eba38dbe 159 // etc. Notice that timestamps are required here: you must provide
citrusbyte 10:4ce9eba38dbe 160 // a timestamp for each value posted.
jb8414 0:f479e4f4db0e 161 // +values+ - Values to post. This works the same way as +ats+, the
jb8414 0:f479e4f4db0e 162 // first +counts[0]+ number of items contain values to post to the first
jb8414 0:f479e4f4db0e 163 // stream, the succeeding +counts[1]+ number of items contain values
jb8414 0:f479e4f4db0e 164 // for the second stream, etc. The length of this array should be
jb8414 0:f479e4f4db0e 165 // the sum of all values in +counts+ array.
defmacro 22:4d895e732765 166 // NOTE: if you want to update by a serial, use "serial/<serial ID>" as
defmacro 22:4d895e732765 167 // the device ID here.
jb8414 0:f479e4f4db0e 168 template <class T>
citrusbyte 13:0d574742208f 169 int postDeviceUpdates(const char* deviceId, int streamNum,
citrusbyte 13:0d574742208f 170 const char* names[], const int counts[],
citrusbyte 13:0d574742208f 171 const char* ats[], T values[]);
jb8414 0:f479e4f4db0e 172
citrusbyte 19:4dfa28d37b8f 173 // Post multiple values of a single device at once.
citrusbyte 19:4dfa28d37b8f 174 // +deviceId+ - id of the device to post values
citrusbyte 19:4dfa28d37b8f 175 // +streamNum+ - Number of streams to post
citrusbyte 19:4dfa28d37b8f 176 // +names+ - Array of stream names, the length of the array should
citrusbyte 19:4dfa28d37b8f 177 // be exactly +streamNum+
citrusbyte 19:4dfa28d37b8f 178 // +values+ - Array of values to post, the length of the array should
citrusbyte 19:4dfa28d37b8f 179 // be exactly +streamNum+. Notice that the array of +values+ should
citrusbyte 19:4dfa28d37b8f 180 // match the array of +names+, and that the ith value in +values+ is
citrusbyte 19:4dfa28d37b8f 181 // exactly the value to post for the ith stream name in +names+
defmacro 22:4d895e732765 182 // NOTE: if you want to update by a serial, use "serial/<serial ID>" as
defmacro 22:4d895e732765 183 // the device ID here.
citrusbyte 19:4dfa28d37b8f 184 template <class T>
defmacro 22:4d895e732765 185 int postDeviceUpdate(const char* deviceId, int streamNum,
defmacro 22:4d895e732765 186 const char* names[], T values[],
defmacro 22:4d895e732765 187 const char* at = NULL);
citrusbyte 19:4dfa28d37b8f 188
defmacro 22:4d895e732765 189 #ifdef M2X_ENABLE_READER
jb8414 0:f479e4f4db0e 190 // Fetch values for a particular data stream. Since memory is
jb8414 0:f479e4f4db0e 191 // very limited on an Arduino, we cannot parse and get all the
jb8414 0:f479e4f4db0e 192 // data points in memory. Instead, we use callbacks here: whenever
jb8414 0:f479e4f4db0e 193 // a new data point is parsed, we call the callback using the values,
jb8414 0:f479e4f4db0e 194 // after that, the values will be thrown away to make space for new
jb8414 0:f479e4f4db0e 195 // values.
jb8414 0:f479e4f4db0e 196 // Note that you can also pass in a user-specified context in this
jb8414 0:f479e4f4db0e 197 // function, this context will be passed to the callback function
jb8414 0:f479e4f4db0e 198 // each time we get a data point.
jb8414 0:f479e4f4db0e 199 // For each data point, the callback will be called once. The HTTP
jb8414 0:f479e4f4db0e 200 // status code will be returned. And the content is only parsed when
jb8414 0:f479e4f4db0e 201 // the status code is 200.
citrusbyte 13:0d574742208f 202 int listStreamValues(const char* deviceId, const char* streamName,
citrusbyte 13:0d574742208f 203 stream_value_read_callback callback, void* context,
citrusbyte 13:0d574742208f 204 const char* query = NULL);
defmacro 22:4d895e732765 205 #endif /* M2X_ENABLE_READER */
jb8414 0:f479e4f4db0e 206
jb8414 0:f479e4f4db0e 207 // Update datasource location
jb8414 0:f479e4f4db0e 208 // NOTE: On an Arduino Uno and other ATMEGA based boards, double has
jb8414 0:f479e4f4db0e 209 // 4-byte (32 bits) precision, which is the same as float. So there's
jb8414 0:f479e4f4db0e 210 // no natural double-precision floating number on these boards. With
jb8414 0:f479e4f4db0e 211 // a float value, we have a precision of roughly 7 digits, that means
jb8414 0:f479e4f4db0e 212 // either 5 or 6 digits after the floating point. According to wikipedia,
jb8414 0:f479e4f4db0e 213 // a difference of 0.00001 will give us ~1.1132m distance. If this
jb8414 0:f479e4f4db0e 214 // precision is good for you, you can use the double-version we provided
jb8414 0:f479e4f4db0e 215 // here. Otherwise, you may need to use the string-version and do the
jb8414 0:f479e4f4db0e 216 // actual conversion by yourselves.
jb8414 0:f479e4f4db0e 217 // However, with an Arduino Due board, double has 8-bytes (64 bits)
jb8414 0:f479e4f4db0e 218 // precision, which means you are free to use the double-version only
jb8414 0:f479e4f4db0e 219 // without any precision problems.
jb8414 0:f479e4f4db0e 220 // Returned value is the http status code.
defmacro 22:4d895e732765 221 // NOTE: if you want to update by a serial, use "serial/<serial ID>" as
defmacro 22:4d895e732765 222 // the device ID here.
jb8414 0:f479e4f4db0e 223 template <class T>
citrusbyte 13:0d574742208f 224 int updateLocation(const char* deviceId, const char* name,
jb8414 0:f479e4f4db0e 225 T latitude, T longitude, T elevation);
jb8414 0:f479e4f4db0e 226
defmacro 22:4d895e732765 227 #ifdef M2X_ENABLE_READER
citrusbyte 13:0d574742208f 228 // Read location information for a device. Also used callback to process
jb8414 0:f479e4f4db0e 229 // data points for memory reasons. The HTTP status code is returned,
jb8414 0:f479e4f4db0e 230 // response is only parsed when the HTTP status code is 200
citrusbyte 13:0d574742208f 231 int readLocation(const char* deviceId, location_read_callback callback,
jb8414 0:f479e4f4db0e 232 void* context);
JMF 23:f32837239193 233
JMF 23:f32837239193 234 // Delete location information for a device. The HTTP status code is
JMF 23:f32837239193 235 // returned,response is only parsed when the HTTP status code is 200
JMF 24:6a5ba4cfa19a 236 int deleteLocations(const char* deviceId,
JMF 23:f32837239193 237 const char* from, const char* end);
JMF 23:f32837239193 238
defmacro 22:4d895e732765 239 #endif /* M2X_ENABLE_READER */
citrusbyte 10:4ce9eba38dbe 240
citrusbyte 10:4ce9eba38dbe 241 // Delete values from a data stream
citrusbyte 10:4ce9eba38dbe 242 // You will need to provide from and end date/time strings in the ISO8601
citrusbyte 10:4ce9eba38dbe 243 // format "yyyy-mm-ddTHH:MM:SS.SSSZ" where
citrusbyte 10:4ce9eba38dbe 244 // yyyy: the year
citrusbyte 10:4ce9eba38dbe 245 // mm: the month
citrusbyte 10:4ce9eba38dbe 246 // dd: the day
citrusbyte 10:4ce9eba38dbe 247 // HH: the hour (24 hour format)
citrusbyte 10:4ce9eba38dbe 248 // MM: the minute
citrusbyte 10:4ce9eba38dbe 249 // SS.SSS: the seconds (to the millisecond)
citrusbyte 10:4ce9eba38dbe 250 // NOTE: the time is given in Zulu (GMT)
citrusbyte 10:4ce9eba38dbe 251 // M2X will delete all values within the from to end date/time range.
citrusbyte 10:4ce9eba38dbe 252 // The status code is 204 on success and 400 on a bad request (e.g. the
citrusbyte 10:4ce9eba38dbe 253 // timestamp is not in ISO8601 format or the from timestamp is not less than
citrusbyte 10:4ce9eba38dbe 254 // or equal to the end timestamp.
citrusbyte 13:0d574742208f 255 int deleteValues(const char* deviceId, const char* streamName,
citrusbyte 10:4ce9eba38dbe 256 const char* from, const char* end);
citrusbyte 16:7903152de19f 257
defmacro 22:4d895e732765 258 #ifdef M2X_ENABLE_READER
defmacro 22:4d895e732765 259 // Fetch commands available for this device, notice that for memory constraints,
defmacro 22:4d895e732765 260 // we only keep ID and name of the command received here.
defmacro 22:4d895e732765 261 // You can tweak the command receiving via the query parameter.
defmacro 22:4d895e732765 262 int listCommands(const char* deviceId,
defmacro 22:4d895e732765 263 m2x_command_read_callback callback, void* context,
defmacro 22:4d895e732765 264 const char* query = NULL);
defmacro 22:4d895e732765 265 #endif /* M2X_ENABLE_READER */
defmacro 22:4d895e732765 266
defmacro 22:4d895e732765 267 // Mark a command as processed.
defmacro 22:4d895e732765 268 // Link: https://m2x.att.com/developer/documentation/v2/commands#Device-Marks-a-Command-as-Processed
defmacro 22:4d895e732765 269 // To make sure the minimal amount of memory is needed, this API works with
defmacro 22:4d895e732765 270 // a callback function. The callback function is then used to fill the request
defmacro 22:4d895e732765 271 // data, note that in order to correctly set the content length in HTTP header,
defmacro 22:4d895e732765 272 // the callback function will be called twice, the caller must make sure both
defmacro 22:4d895e732765 273 // calls fill the Print object with exactly the same data.
defmacro 22:4d895e732765 274 // If you have a pre-allocated buffer filled with the data to post, you can
defmacro 22:4d895e732765 275 // use markCommandProcessedWithData API below.
defmacro 22:4d895e732765 276 int markCommandProcessed(const char* deviceId, const char* commandId,
defmacro 22:4d895e732765 277 m2x_fill_data_callback callback, void *context);
defmacro 22:4d895e732765 278
defmacro 22:4d895e732765 279 // Mark a command as processed with a data buffer
defmacro 22:4d895e732765 280 // Link: https://m2x.att.com/developer/documentation/v2/commands#Device-Marks-a-Command-as-Processed
defmacro 22:4d895e732765 281 // This is exactly like markCommandProcessed, except that a buffer is use to
defmacro 22:4d895e732765 282 // contain the data to post as the request body.
defmacro 22:4d895e732765 283 int markCommandProcessedWithData(const char* deviceId, const char* commandId,
defmacro 22:4d895e732765 284 const char* data);
defmacro 22:4d895e732765 285
defmacro 22:4d895e732765 286 // Mark a command as rejected.
defmacro 22:4d895e732765 287 // Link: https://m2x.att.com/developer/documentation/v2/commands#Device-Marks-a-Command-as-Rejected
defmacro 22:4d895e732765 288 // To make sure the minimal amount of memory is needed, this API works with
defmacro 22:4d895e732765 289 // a callback function. The callback function is then used to fill the request
defmacro 22:4d895e732765 290 // data, note that in order to correctly set the content length in HTTP header,
defmacro 22:4d895e732765 291 // the callback function will be called twice, the caller must make sure both
defmacro 22:4d895e732765 292 // calls fill the Print object with exactly the same data.
defmacro 22:4d895e732765 293 // If you have a pre-allocated buffer filled with the data to post, you can
defmacro 22:4d895e732765 294 // use markCommandRejectedWithData API below.
defmacro 22:4d895e732765 295 int markCommandRejected(const char* deviceId, const char* commandId,
defmacro 22:4d895e732765 296 m2x_fill_data_callback callback, void *context);
defmacro 22:4d895e732765 297
defmacro 22:4d895e732765 298 // Mark a command as rejected with a data buffer
defmacro 22:4d895e732765 299 // Link: https://m2x.att.com/developer/documentation/v2/commands#Device-Marks-a-Command-as-Rejected
defmacro 22:4d895e732765 300 // This is exactly like markCommandRejected, except that a buffer is use to
defmacro 22:4d895e732765 301 // contain the data to post as the request body.
defmacro 22:4d895e732765 302 int markCommandRejectedWithData(const char* deviceId, const char* commandId,
defmacro 22:4d895e732765 303 const char* data);
defmacro 22:4d895e732765 304
citrusbyte 16:7903152de19f 305 // Fetches current timestamp in seconds from M2X server. Since we
citrusbyte 16:7903152de19f 306 // are using signed 32-bit integer as return value, this will only
citrusbyte 16:7903152de19f 307 // return valid results before 03:14:07 UTC on 19 January 2038. If
citrusbyte 16:7903152de19f 308 // the device is supposed to work after that, this function should
citrusbyte 16:7903152de19f 309 // not be used.
citrusbyte 16:7903152de19f 310 //
citrusbyte 16:7903152de19f 311 // The returned value will contain the status code(positive values)
citrusbyte 16:7903152de19f 312 // or the error code(negative values).
citrusbyte 16:7903152de19f 313 // In case of success, the current timestamp will be filled in the
citrusbyte 16:7903152de19f 314 // +ts+ pointer passed in as argument.
citrusbyte 16:7903152de19f 315 //
citrusbyte 16:7903152de19f 316 // NOTE: although returning uint32_t can give us a larger space,
citrusbyte 16:7903152de19f 317 // we prefer to cope with the unix convention here.
citrusbyte 16:7903152de19f 318 int getTimestamp32(int32_t* ts);
citrusbyte 16:7903152de19f 319
citrusbyte 16:7903152de19f 320 // Fetches current timestamp in seconds from M2X server.
citrusbyte 16:7903152de19f 321 // This function will return the timestamp as an integer literal
citrusbyte 16:7903152de19f 322 // in the provided buffer. Hence there's no problem working after
citrusbyte 16:7903152de19f 323 // 03:14:07 UTC on 19 January 2038. The drawback part here, is that
citrusbyte 16:7903152de19f 324 // you will have to work with 64-bit integer, which is not available
citrusbyte 16:7903152de19f 325 // on certain platform(such as Arduino), a bignum library or alike
citrusbyte 16:7903152de19f 326 // is needed in this case.
citrusbyte 16:7903152de19f 327 //
citrusbyte 16:7903152de19f 328 // Notice +bufferLength+ is supposed to contain the length of the
citrusbyte 16:7903152de19f 329 // buffer when calling this function. It is also the caller's
citrusbyte 16:7903152de19f 330 // responsibility to ensure the buffer is big enough, otherwise
citrusbyte 16:7903152de19f 331 // the library will return an error indicating the buffer is too
citrusbyte 16:7903152de19f 332 // small.
citrusbyte 16:7903152de19f 333 // While this is not accurate all the time, one trick here is to
citrusbyte 16:7903152de19f 334 // pass in 0 as the bufferLength, in which case we will always return
citrusbyte 16:7903152de19f 335 // the buffer-too-small error. However, the correct buffer length
citrusbyte 16:7903152de19f 336 // can be found this way so a secound execution is most likely to work
citrusbyte 16:7903152de19f 337 // (unless we are at the edge of the buffer length increasing, for
citrusbyte 16:7903152de19f 338 // example, when the timestamp jumps from 9999999999 to 10000000000,
citrusbyte 16:7903152de19f 339 // which is highly unlikely to happend). However, given that the
citrusbyte 16:7903152de19f 340 // maximum 64-bit integer can be stored in 19 bytes, there's not
citrusbyte 16:7903152de19f 341 // much need to use this trick.)
citrusbyte 16:7903152de19f 342 //
citrusbyte 16:7903152de19f 343 // The returned value will contain the status code(positive values)
citrusbyte 16:7903152de19f 344 // or the error code(negative values).
citrusbyte 16:7903152de19f 345 // In case of success, the current timestamp will be filled in the
citrusbyte 16:7903152de19f 346 // passed +buffer+ pointer, and the actual used buffer length will
citrusbyte 16:7903152de19f 347 // be returned in +bufferLength+ pointer.
citrusbyte 16:7903152de19f 348 // NOTE: as long as we can read the returned buffer length, it will
citrusbyte 16:7903152de19f 349 // be used to fill in the +bufferLength+ variable even though other
citrusbyte 16:7903152de19f 350 // errors occur(buffer is not enough, network is shutdown before
citrusbyte 16:7903152de19f 351 // reading the whole buffer, etc.)
citrusbyte 16:7903152de19f 352 int getTimestamp(char* buffer, int* bufferLength);
jb8414 0:f479e4f4db0e 353 private:
jb8414 0:f479e4f4db0e 354 Client* _client;
jb8414 0:f479e4f4db0e 355 const char* _key;
jb8414 0:f479e4f4db0e 356 int _case_insensitive;
jb8414 0:f479e4f4db0e 357 const char* _host;
jb8414 0:f479e4f4db0e 358 int _port;
defmacro 22:4d895e732765 359 void (* _idlefunc)(void);
NetArc 14:205076b587fe 360 const char* _path_prefix;
jb8414 0:f479e4f4db0e 361 NullPrint _null_print;
jb8414 0:f479e4f4db0e 362
jb8414 0:f479e4f4db0e 363 // Writes the HTTP header part for updating a stream value
citrusbyte 13:0d574742208f 364 void writePutHeader(const char* deviceId,
citrusbyte 10:4ce9eba38dbe 365 const char* streamName,
citrusbyte 10:4ce9eba38dbe 366 int contentLength);
citrusbyte 10:4ce9eba38dbe 367 // Writes the HTTP header part for deleting stream values
citrusbyte 13:0d574742208f 368 void writeDeleteHeader(const char* deviceId,
citrusbyte 10:4ce9eba38dbe 369 const char* streamName,
citrusbyte 10:4ce9eba38dbe 370 int contentLength);
JMF 23:f32837239193 371 void writeDeleteLocationHeader(const char* deviceId,
JMF 23:f32837239193 372 int contentLength);
JMF 23:f32837239193 373
jb8414 0:f479e4f4db0e 374 // Writes HTTP header lines including M2X API Key, host, content
jb8414 0:f479e4f4db0e 375 // type and content length(if the body exists)
jb8414 0:f479e4f4db0e 376 void writeHttpHeader(int contentLength);
jb8414 0:f479e4f4db0e 377 // Parses HTTP response header and return the content length.
jb8414 0:f479e4f4db0e 378 // Note that this function does not parse all http headers, as long
jb8414 0:f479e4f4db0e 379 // as the content length is found, this function will return
jb8414 0:f479e4f4db0e 380 int readContentLength();
jb8414 0:f479e4f4db0e 381 // Skips all HTTP response header part. Return minus value in case
jb8414 0:f479e4f4db0e 382 // the connection is closed before we got all headers
jb8414 0:f479e4f4db0e 383 int skipHttpHeader();
jb8414 0:f479e4f4db0e 384 // Parses and returns the HTTP status code, note this function will
jb8414 0:f479e4f4db0e 385 // return immediately once it gets the status code
jb8414 0:f479e4f4db0e 386 int readStatusCode(bool closeClient);
jb8414 0:f479e4f4db0e 387 // Waits for a certain string pattern in the HTTP header, and returns
jb8414 0:f479e4f4db0e 388 // once the pattern is found. In the pattern, you can use '*' to denote
jb8414 0:f479e4f4db0e 389 // any character
jb8414 0:f479e4f4db0e 390 int waitForString(const char* str);
jb8414 0:f479e4f4db0e 391 // Closes the connection
jb8414 0:f479e4f4db0e 392 void close();
defmacro 22:4d895e732765 393
defmacro 22:4d895e732765 394 #ifdef M2X_ENABLE_READER
jb8414 0:f479e4f4db0e 395 // Parses JSON response of stream value API, and calls callback function
jb8414 0:f479e4f4db0e 396 // once we get a data point
jb8414 0:f479e4f4db0e 397 int readStreamValue(stream_value_read_callback callback, void* context);
jb8414 0:f479e4f4db0e 398 // Parses JSON response of location API, and calls callback function once
jb8414 0:f479e4f4db0e 399 // we get a data point
jb8414 0:f479e4f4db0e 400 int readLocation(location_read_callback callback, void* context);
defmacro 22:4d895e732765 401 // Parses JSON response of command API, and calls callback function once
defmacro 22:4d895e732765 402 // we get a data point
defmacro 22:4d895e732765 403 int readCommand(m2x_command_read_callback callback, void* context);
defmacro 22:4d895e732765 404 #endif /* M2X_ENABLE_READER */
defmacro 22:4d895e732765 405 };
defmacro 22:4d895e732765 406
defmacro 22:4d895e732765 407
defmacro 22:4d895e732765 408 // A ISO8601 timestamp generation service for M2X.
defmacro 22:4d895e732765 409 // It uses the Time API provided by the M2X server to initialize
defmacro 22:4d895e732765 410 // clock, then uses millis() function provided by Arduino to calculate
defmacro 22:4d895e732765 411 // time advancements so as to reduce API query times.
defmacro 22:4d895e732765 412 //
defmacro 22:4d895e732765 413 // Right now, this service only works with 32-bit timestamp, meaning that
defmacro 22:4d895e732765 414 // this service won't work after 03:14:07 UTC on 19 January 2038. However,
defmacro 22:4d895e732765 415 // a similar service that uses 64-bit timestamp can be implemented following
defmacro 22:4d895e732765 416 // the logic here.
defmacro 22:4d895e732765 417 class TimeService {
defmacro 22:4d895e732765 418 public:
defmacro 22:4d895e732765 419 TimeService(M2XStreamClient* client);
defmacro 22:4d895e732765 420
defmacro 22:4d895e732765 421 // Initialize the time service. Notice the TimeService instance is only
defmacro 22:4d895e732765 422 // working after calling this function successfully.
defmacro 22:4d895e732765 423 int init();
defmacro 22:4d895e732765 424
defmacro 22:4d895e732765 425 // Reset the internal recorded time by calling M2X Time API again. Normally,
defmacro 22:4d895e732765 426 // you don't need to call this manually. TimeService will handle Arduino clock
defmacro 22:4d895e732765 427 // overflow automatically
defmacro 22:4d895e732765 428 int reset();
defmacro 22:4d895e732765 429
defmacro 22:4d895e732765 430 // Fills ISO8601 formatted timestamp into the buffer provided. +length+ should
defmacro 22:4d895e732765 431 // contains the maximum supported length of the buffer when calling. For now,
defmacro 22:4d895e732765 432 // the buffer should be able to store 25 characters for a full ISO8601 formatted
defmacro 22:4d895e732765 433 // timestamp, otherwise, an error will be returned.
defmacro 22:4d895e732765 434 int getTimestamp(char* buffer, int* length);
defmacro 22:4d895e732765 435 private:
defmacro 22:4d895e732765 436 M2XStreamClient* _client;
defmacro 22:4d895e732765 437 int32_t _server_timestamp;
defmacro 22:4d895e732765 438 uint32_t _local_last_milli;
defmacro 22:4d895e732765 439 M2XTimer _timer;
jb8414 0:f479e4f4db0e 440 };
jb8414 0:f479e4f4db0e 441
defmacro 22:4d895e732765 442
defmacro 22:4d895e732765 443 // Implementations
defmacro 22:4d895e732765 444 M2XStreamClient::M2XStreamClient(Client* client,
defmacro 22:4d895e732765 445 const char* key,
defmacro 22:4d895e732765 446 void (* idlefunc)(void),
defmacro 22:4d895e732765 447 int case_insensitive,
defmacro 22:4d895e732765 448 const char* host,
defmacro 22:4d895e732765 449 int port,
defmacro 22:4d895e732765 450 const char* path_prefix) : _client(client),
defmacro 22:4d895e732765 451 _key(key),
defmacro 22:4d895e732765 452 _idlefunc(idlefunc),
defmacro 22:4d895e732765 453 _case_insensitive(case_insensitive),
defmacro 22:4d895e732765 454 _host(host),
defmacro 22:4d895e732765 455 _port(port),
defmacro 22:4d895e732765 456 _path_prefix(path_prefix),
defmacro 22:4d895e732765 457 _null_print() {
defmacro 22:4d895e732765 458 }
defmacro 22:4d895e732765 459
defmacro 22:4d895e732765 460 template <class T>
defmacro 22:4d895e732765 461 int M2XStreamClient::updateStreamValue(const char* deviceId, const char* streamName, T value) {
defmacro 22:4d895e732765 462 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 463 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 464 writePutHeader(deviceId, streamName,
defmacro 22:4d895e732765 465 // for {"value": and }
defmacro 22:4d895e732765 466 _null_print.print(value) + 12);
defmacro 22:4d895e732765 467 _client->print("{\"value\":\"");
defmacro 22:4d895e732765 468 _client->print(value);
defmacro 22:4d895e732765 469 _client->print("\"}");
defmacro 22:4d895e732765 470 } else {
defmacro 22:4d895e732765 471 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 472 return E_NOCONNECTION;
defmacro 22:4d895e732765 473 }
defmacro 22:4d895e732765 474
defmacro 22:4d895e732765 475 return readStatusCode(true);
defmacro 22:4d895e732765 476 }
defmacro 22:4d895e732765 477
defmacro 22:4d895e732765 478 template <class T>
defmacro 22:4d895e732765 479 inline int write_multiple_values(Print* print, int streamNum,
defmacro 22:4d895e732765 480 const char* names[], const int counts[],
defmacro 22:4d895e732765 481 const char* ats[], T values[]) {
defmacro 22:4d895e732765 482 int bytes = 0, value_index = 0;
defmacro 22:4d895e732765 483 bytes += print->print("{\"values\":{");
defmacro 22:4d895e732765 484 for (int i = 0; i < streamNum; i++) {
defmacro 22:4d895e732765 485 bytes += print->print("\"");
defmacro 22:4d895e732765 486 bytes += print->print(names[i]);
defmacro 22:4d895e732765 487 bytes += print->print("\":[");
defmacro 22:4d895e732765 488 for (int j = 0; j < counts[i]; j++) {
defmacro 22:4d895e732765 489 bytes += print->print("{\"timestamp\": \"");
defmacro 22:4d895e732765 490 bytes += print->print(ats[value_index]);
defmacro 22:4d895e732765 491 bytes += print->print("\",\"value\": \"");
defmacro 22:4d895e732765 492 bytes += print->print(values[value_index]);
defmacro 22:4d895e732765 493 bytes += print->print("\"}");
defmacro 22:4d895e732765 494 if (j < counts[i] - 1) { bytes += print->print(","); }
defmacro 22:4d895e732765 495 value_index++;
defmacro 22:4d895e732765 496 }
defmacro 22:4d895e732765 497 bytes += print->print("]");
defmacro 22:4d895e732765 498 if (i < streamNum - 1) { bytes += print->print(","); }
defmacro 22:4d895e732765 499 }
defmacro 22:4d895e732765 500 bytes += print->print("}}");
defmacro 22:4d895e732765 501 return bytes;
defmacro 22:4d895e732765 502 }
defmacro 22:4d895e732765 503
defmacro 22:4d895e732765 504 template <class T>
defmacro 22:4d895e732765 505 int M2XStreamClient::postDeviceUpdates(const char* deviceId, int streamNum,
defmacro 22:4d895e732765 506 const char* names[], const int counts[],
defmacro 22:4d895e732765 507 const char* ats[], T values[]) {
defmacro 22:4d895e732765 508 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 509 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 510 int length = write_multiple_values(&_null_print, streamNum, names,
defmacro 22:4d895e732765 511 counts, ats, values);
defmacro 22:4d895e732765 512 _client->print("POST ");
defmacro 22:4d895e732765 513 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 514 _client->print("/v2/devices/");
defmacro 22:4d895e732765 515 _client->print(deviceId);
JMF 25:67079c11d1ea 516 _client->println("/updates HTTP/1.1");
defmacro 22:4d895e732765 517 writeHttpHeader(length);
defmacro 22:4d895e732765 518 write_multiple_values(_client, streamNum, names, counts, ats, values);
defmacro 22:4d895e732765 519 } else {
defmacro 22:4d895e732765 520 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 521 return E_NOCONNECTION;
defmacro 22:4d895e732765 522 }
defmacro 22:4d895e732765 523 return readStatusCode(true);
defmacro 22:4d895e732765 524 }
defmacro 22:4d895e732765 525
defmacro 22:4d895e732765 526 template <class T>
defmacro 22:4d895e732765 527 inline int write_single_device_values(Print* print, int streamNum,
defmacro 22:4d895e732765 528 const char* names[], T values[],
defmacro 22:4d895e732765 529 const char* at) {
defmacro 22:4d895e732765 530 int bytes = 0;
defmacro 22:4d895e732765 531 bytes += print->print("{\"values\":{");
defmacro 22:4d895e732765 532 for (int i = 0; i < streamNum; i++) {
defmacro 22:4d895e732765 533 bytes += print->print("\"");
defmacro 22:4d895e732765 534 bytes += print->print(names[i]);
defmacro 22:4d895e732765 535 bytes += print->print("\": \"");
defmacro 22:4d895e732765 536 bytes += print->print(values[i]);
defmacro 22:4d895e732765 537 bytes += print->print("\"");
defmacro 22:4d895e732765 538 if (i < streamNum - 1) { bytes += print->print(","); }
defmacro 22:4d895e732765 539 }
defmacro 22:4d895e732765 540 bytes += print->print("}");
defmacro 22:4d895e732765 541 if (at != NULL) {
defmacro 22:4d895e732765 542 bytes += print->print(",\"timestamp\":\"");
defmacro 22:4d895e732765 543 bytes += print->print(at);
defmacro 22:4d895e732765 544 bytes += print->print("\"");
defmacro 22:4d895e732765 545 }
defmacro 22:4d895e732765 546 bytes += print->print("}");
defmacro 22:4d895e732765 547 return bytes;
defmacro 22:4d895e732765 548 }
defmacro 22:4d895e732765 549
defmacro 22:4d895e732765 550 template <class T>
defmacro 22:4d895e732765 551 int M2XStreamClient::postDeviceUpdate(const char* deviceId, int streamNum,
defmacro 22:4d895e732765 552 const char* names[], T values[],
defmacro 22:4d895e732765 553 const char* at) {
defmacro 22:4d895e732765 554 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 555 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 556 int length = write_single_device_values(&_null_print, streamNum, names,
defmacro 22:4d895e732765 557 values, at);
defmacro 22:4d895e732765 558 _client->print("POST ");
defmacro 22:4d895e732765 559 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 560 _client->print("/v2/devices/");
defmacro 22:4d895e732765 561 _client->print(deviceId);
JMF 25:67079c11d1ea 562 _client->println("/update HTTP/1.1");
defmacro 22:4d895e732765 563 writeHttpHeader(length);
defmacro 22:4d895e732765 564 write_single_device_values(_client, streamNum, names, values, at);
defmacro 22:4d895e732765 565 } else {
defmacro 22:4d895e732765 566 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 567 return E_NOCONNECTION;
defmacro 22:4d895e732765 568 }
defmacro 22:4d895e732765 569 return readStatusCode(true);
defmacro 22:4d895e732765 570 }
defmacro 22:4d895e732765 571
defmacro 22:4d895e732765 572 template <class T>
defmacro 22:4d895e732765 573 static int write_location_data(Print* print, const char* name,
defmacro 22:4d895e732765 574 T latitude, T longitude,
defmacro 22:4d895e732765 575 T elevation) {
defmacro 22:4d895e732765 576 int bytes = 0;
defmacro 22:4d895e732765 577 bytes += print->print("{\"name\":\"");
defmacro 22:4d895e732765 578 bytes += print->print(name);
defmacro 22:4d895e732765 579 bytes += print->print("\",\"latitude\":\"");
defmacro 22:4d895e732765 580 bytes += print->print(latitude);
defmacro 22:4d895e732765 581 bytes += print->print("\",\"longitude\":\"");
defmacro 22:4d895e732765 582 bytes += print->print(longitude);
defmacro 22:4d895e732765 583 bytes += print->print("\",\"elevation\":\"");
defmacro 22:4d895e732765 584 bytes += print->print(elevation);
defmacro 22:4d895e732765 585 bytes += print->print("\"}");
defmacro 22:4d895e732765 586 return bytes;
defmacro 22:4d895e732765 587 }
defmacro 22:4d895e732765 588
defmacro 22:4d895e732765 589 static int write_location_data(Print* print, const char* name,
defmacro 22:4d895e732765 590 double latitude, double longitude,
defmacro 22:4d895e732765 591 double elevation) {
defmacro 22:4d895e732765 592 int bytes = 0;
defmacro 22:4d895e732765 593 bytes += print->print("{\"name\":\"");
defmacro 22:4d895e732765 594 bytes += print->print(name);
defmacro 22:4d895e732765 595 bytes += print->print("\",\"latitude\":\"");
defmacro 22:4d895e732765 596 bytes += print->print(latitude, MAX_DOUBLE_DIGITS);
defmacro 22:4d895e732765 597 bytes += print->print("\",\"longitude\":\"");
defmacro 22:4d895e732765 598 bytes += print->print(longitude, MAX_DOUBLE_DIGITS);
defmacro 22:4d895e732765 599 bytes += print->print("\",\"elevation\":\"");
defmacro 22:4d895e732765 600 bytes += print->print(elevation);
defmacro 22:4d895e732765 601 bytes += print->print("\"}");
defmacro 22:4d895e732765 602 return bytes;
defmacro 22:4d895e732765 603 }
defmacro 22:4d895e732765 604
defmacro 22:4d895e732765 605 template <class T>
defmacro 22:4d895e732765 606 int M2XStreamClient::updateLocation(const char* deviceId,
defmacro 22:4d895e732765 607 const char* name,
defmacro 22:4d895e732765 608 T latitude,
defmacro 22:4d895e732765 609 T longitude,
defmacro 22:4d895e732765 610 T elevation) {
defmacro 22:4d895e732765 611 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 612 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 613
defmacro 22:4d895e732765 614 int length = write_location_data(&_null_print, name, latitude, longitude,
defmacro 22:4d895e732765 615 elevation);
defmacro 22:4d895e732765 616 _client->print("PUT ");
defmacro 22:4d895e732765 617 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 618 _client->print("/v2/devices/");
defmacro 22:4d895e732765 619 _client->print(deviceId);
JMF 25:67079c11d1ea 620 _client->println("/location HTTP/1.1");
defmacro 22:4d895e732765 621
defmacro 22:4d895e732765 622 writeHttpHeader(length);
defmacro 22:4d895e732765 623 write_location_data(_client, name, latitude, longitude, elevation);
defmacro 22:4d895e732765 624 } else {
defmacro 22:4d895e732765 625 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 626 return E_NOCONNECTION;
defmacro 22:4d895e732765 627 }
defmacro 22:4d895e732765 628 return readStatusCode(true);
defmacro 22:4d895e732765 629 }
defmacro 22:4d895e732765 630
defmacro 22:4d895e732765 631 static inline int write_delete_values(Print* print, const char* from,
defmacro 22:4d895e732765 632 const char* end) {
defmacro 22:4d895e732765 633 int bytes = 0;
defmacro 22:4d895e732765 634 bytes += print->print("{\"from\":\"");
defmacro 22:4d895e732765 635 bytes += print->print(from);
defmacro 22:4d895e732765 636 bytes += print->print("\",\"end\":\"");
defmacro 22:4d895e732765 637 bytes += print->print(end);
defmacro 22:4d895e732765 638 bytes += print->print("\"}");
defmacro 22:4d895e732765 639 return bytes;
defmacro 22:4d895e732765 640 }
defmacro 22:4d895e732765 641
defmacro 22:4d895e732765 642 int M2XStreamClient::deleteValues(const char* deviceId, const char* streamName,
defmacro 22:4d895e732765 643 const char* from, const char* end) {
defmacro 22:4d895e732765 644 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 645 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 646 int length = write_delete_values(&_null_print, from, end);
defmacro 22:4d895e732765 647 writeDeleteHeader(deviceId, streamName, length);
defmacro 22:4d895e732765 648 write_delete_values(_client, from, end);
defmacro 22:4d895e732765 649 } else {
defmacro 22:4d895e732765 650 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 651 return E_NOCONNECTION;
defmacro 22:4d895e732765 652 }
defmacro 22:4d895e732765 653
defmacro 22:4d895e732765 654 return readStatusCode(true);
defmacro 22:4d895e732765 655 }
defmacro 22:4d895e732765 656
defmacro 22:4d895e732765 657 int M2XStreamClient::markCommandProcessed(const char* deviceId,
defmacro 22:4d895e732765 658 const char* commandId,
defmacro 22:4d895e732765 659 m2x_fill_data_callback callback,
defmacro 22:4d895e732765 660 void *context) {
defmacro 22:4d895e732765 661 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 662 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 663 _null_print.counter = 0;
defmacro 22:4d895e732765 664 callback(&_null_print, context);
defmacro 22:4d895e732765 665 int length = _null_print.counter;
defmacro 22:4d895e732765 666 _client->print("POST ");
defmacro 22:4d895e732765 667 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 668 _client->print("/v2/devices/");
defmacro 22:4d895e732765 669 _client->print(deviceId);
defmacro 22:4d895e732765 670 _client->print("/commands/");
defmacro 22:4d895e732765 671 _client->print(commandId);
defmacro 22:4d895e732765 672 _client->print("/process");
JMF 25:67079c11d1ea 673 _client->println(" HTTP/1.1");
defmacro 22:4d895e732765 674 writeHttpHeader(length);
defmacro 22:4d895e732765 675 callback(_client, context);
defmacro 22:4d895e732765 676 } else {
defmacro 22:4d895e732765 677 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 678 return E_NOCONNECTION;
defmacro 22:4d895e732765 679 }
defmacro 22:4d895e732765 680 return readStatusCode(true);
defmacro 22:4d895e732765 681 }
defmacro 22:4d895e732765 682
defmacro 22:4d895e732765 683 static void m2x_fixed_buffer_filling_callback(Print *print, void *context) {
defmacro 22:4d895e732765 684 if (context) {
defmacro 22:4d895e732765 685 print->print((const char *) context);
defmacro 22:4d895e732765 686 }
defmacro 22:4d895e732765 687 }
defmacro 22:4d895e732765 688
defmacro 22:4d895e732765 689 int M2XStreamClient::markCommandProcessedWithData(const char* deviceId,
defmacro 22:4d895e732765 690 const char* commandId,
defmacro 22:4d895e732765 691 const char* data) {
defmacro 22:4d895e732765 692 return markCommandProcessed(deviceId, commandId,
defmacro 22:4d895e732765 693 m2x_fixed_buffer_filling_callback, (void *) data);
defmacro 22:4d895e732765 694 }
defmacro 22:4d895e732765 695
defmacro 22:4d895e732765 696 int M2XStreamClient::markCommandRejected(const char* deviceId,
defmacro 22:4d895e732765 697 const char* commandId,
defmacro 22:4d895e732765 698 m2x_fill_data_callback callback,
defmacro 22:4d895e732765 699 void *context) {
defmacro 22:4d895e732765 700 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 701 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 702 _null_print.counter = 0;
defmacro 22:4d895e732765 703 callback(&_null_print, context);
defmacro 22:4d895e732765 704 int length = _null_print.counter;
defmacro 22:4d895e732765 705 _client->print("POST ");
defmacro 22:4d895e732765 706 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 707 _client->print("/v2/devices/");
defmacro 22:4d895e732765 708 _client->print(deviceId);
defmacro 22:4d895e732765 709 _client->print("/commands/");
defmacro 22:4d895e732765 710 _client->print(commandId);
defmacro 22:4d895e732765 711 _client->print("/reject");
JMF 25:67079c11d1ea 712 _client->println(" HTTP/1.1");
defmacro 22:4d895e732765 713 writeHttpHeader(length);
defmacro 22:4d895e732765 714 callback(_client, context);
defmacro 22:4d895e732765 715 } else {
defmacro 22:4d895e732765 716 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 717 return E_NOCONNECTION;
defmacro 22:4d895e732765 718 }
defmacro 22:4d895e732765 719 return readStatusCode(true);
defmacro 22:4d895e732765 720 }
defmacro 22:4d895e732765 721
defmacro 22:4d895e732765 722 int M2XStreamClient::markCommandRejectedWithData(const char* deviceId,
defmacro 22:4d895e732765 723 const char* commandId,
defmacro 22:4d895e732765 724 const char* data) {
defmacro 22:4d895e732765 725 return markCommandRejected(deviceId, commandId,
defmacro 22:4d895e732765 726 m2x_fixed_buffer_filling_callback, (void *) data);
defmacro 22:4d895e732765 727 }
defmacro 22:4d895e732765 728
defmacro 22:4d895e732765 729 int M2XStreamClient::getTimestamp32(int32_t *ts) {
defmacro 22:4d895e732765 730 // The maximum value of signed 64-bit integer is 0x7fffffffffffffff,
defmacro 22:4d895e732765 731 // which is 9223372036854775807. It consists of 19 characters, so a
defmacro 22:4d895e732765 732 // buffer of 20 is definitely enough here
defmacro 22:4d895e732765 733 int length = 20;
defmacro 22:4d895e732765 734 char buffer[20];
defmacro 22:4d895e732765 735 int status = getTimestamp(buffer, &length);
defmacro 22:4d895e732765 736 if (status == 200) {
defmacro 22:4d895e732765 737 int32_t result = 0;
defmacro 22:4d895e732765 738 for (int i = 0; i < length; i++) {
defmacro 22:4d895e732765 739 result = result * 10 + (buffer[i] - '0');
defmacro 22:4d895e732765 740 }
defmacro 22:4d895e732765 741 if (ts != NULL) { *ts = result; }
defmacro 22:4d895e732765 742 }
defmacro 22:4d895e732765 743 return status;
defmacro 22:4d895e732765 744 }
defmacro 22:4d895e732765 745
defmacro 22:4d895e732765 746 int M2XStreamClient::getTimestamp(char* buffer, int *bufferLength) {
defmacro 22:4d895e732765 747 if (bufferLength == NULL) { return E_INVALID; }
defmacro 22:4d895e732765 748 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 749 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 750 _client->print("GET ");
defmacro 22:4d895e732765 751 if (_path_prefix) { _client->print(_path_prefix); }
JMF 25:67079c11d1ea 752 _client->println("/v2/time/seconds HTTP/1.1");
defmacro 22:4d895e732765 753
defmacro 22:4d895e732765 754 writeHttpHeader(-1);
defmacro 22:4d895e732765 755 } else {
defmacro 22:4d895e732765 756 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 757 return E_NOCONNECTION;
defmacro 22:4d895e732765 758 }
defmacro 22:4d895e732765 759 int status = readStatusCode(false);
defmacro 22:4d895e732765 760 if (status == 200) {
defmacro 22:4d895e732765 761 int length = readContentLength();
defmacro 22:4d895e732765 762 if (length < 0) {
defmacro 22:4d895e732765 763 close();
defmacro 22:4d895e732765 764 return length;
defmacro 22:4d895e732765 765 }
defmacro 22:4d895e732765 766 if (*bufferLength < length) {
defmacro 22:4d895e732765 767 *bufferLength = length;
defmacro 22:4d895e732765 768 return E_BUFFER_TOO_SMALL;
defmacro 22:4d895e732765 769 }
defmacro 22:4d895e732765 770 *bufferLength = length;
defmacro 22:4d895e732765 771 int index = skipHttpHeader();
defmacro 22:4d895e732765 772 if (index != E_OK) {
defmacro 22:4d895e732765 773 close();
defmacro 22:4d895e732765 774 return index;
defmacro 22:4d895e732765 775 }
defmacro 22:4d895e732765 776 index = 0;
defmacro 22:4d895e732765 777 while (index < length) {
defmacro 22:4d895e732765 778 DBG("%s", "Received Data: ");
defmacro 22:4d895e732765 779 while ((index < length) && _client->available()) {
defmacro 22:4d895e732765 780 buffer[index++] = _client->read();
defmacro 22:4d895e732765 781 DBG("%c", buffer[index - 1]);
defmacro 22:4d895e732765 782 }
defmacro 22:4d895e732765 783 DBGLNEND;
defmacro 22:4d895e732765 784
defmacro 22:4d895e732765 785 if ((!_client->connected()) &&
defmacro 22:4d895e732765 786 (index < length)) {
defmacro 22:4d895e732765 787 close();
defmacro 22:4d895e732765 788 return E_NOCONNECTION;
defmacro 22:4d895e732765 789 }
defmacro 22:4d895e732765 790
defmacro 22:4d895e732765 791 if (_idlefunc!=NULL) {
defmacro 22:4d895e732765 792 _idlefunc();
defmacro 22:4d895e732765 793 } else {
defmacro 22:4d895e732765 794 delay(200);
defmacro 22:4d895e732765 795 }
defmacro 22:4d895e732765 796 }
defmacro 22:4d895e732765 797 }
defmacro 22:4d895e732765 798 close();
defmacro 22:4d895e732765 799 return status;
defmacro 22:4d895e732765 800 }
defmacro 22:4d895e732765 801
defmacro 22:4d895e732765 802
defmacro 22:4d895e732765 803 void M2XStreamClient::writePutHeader(const char* deviceId,
defmacro 22:4d895e732765 804 const char* streamName,
defmacro 22:4d895e732765 805 int contentLength) {
defmacro 22:4d895e732765 806 _client->print("PUT ");
defmacro 22:4d895e732765 807 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 808 _client->print("/v2/devices/");
defmacro 22:4d895e732765 809 _client->print(deviceId);
defmacro 22:4d895e732765 810 _client->print("/streams/");
defmacro 22:4d895e732765 811 print_encoded_string(_client, streamName);
JMF 25:67079c11d1ea 812 _client->println("/value HTTP/1.1");
defmacro 22:4d895e732765 813
defmacro 22:4d895e732765 814 writeHttpHeader(contentLength);
defmacro 22:4d895e732765 815 }
defmacro 22:4d895e732765 816
defmacro 22:4d895e732765 817 void M2XStreamClient::writeDeleteHeader(const char* deviceId,
defmacro 22:4d895e732765 818 const char* streamName,
defmacro 22:4d895e732765 819 int contentLength) {
defmacro 22:4d895e732765 820 _client->print("DELETE ");
defmacro 22:4d895e732765 821 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 822 _client->print("/v2/devices/");
defmacro 22:4d895e732765 823 _client->print(deviceId);
defmacro 22:4d895e732765 824 _client->print("/streams/");
defmacro 22:4d895e732765 825 print_encoded_string(_client, streamName);
defmacro 22:4d895e732765 826 _client->print("/values");
JMF 25:67079c11d1ea 827 _client->println(" HTTP/1.1");
defmacro 22:4d895e732765 828
defmacro 22:4d895e732765 829 writeHttpHeader(contentLength);
defmacro 22:4d895e732765 830 }
defmacro 22:4d895e732765 831
JMF 23:f32837239193 832 void M2XStreamClient::writeDeleteLocationHeader(const char* deviceId,
JMF 23:f32837239193 833 int contentLength) {
JMF 23:f32837239193 834 _client->print("DELETE ");
JMF 23:f32837239193 835 if (_path_prefix) { _client->print(_path_prefix); }
JMF 23:f32837239193 836 _client->print("/v2/devices/");
JMF 23:f32837239193 837 _client->print(deviceId);
JMF 23:f32837239193 838 _client->print("/location/waypoints/");
JMF 25:67079c11d1ea 839 _client->println(" HTTP/1.1");
JMF 23:f32837239193 840
JMF 23:f32837239193 841 writeHttpHeader(contentLength);
JMF 23:f32837239193 842 }
JMF 23:f32837239193 843
JMF 23:f32837239193 844
defmacro 22:4d895e732765 845 void M2XStreamClient::writeHttpHeader(int contentLength) {
defmacro 22:4d895e732765 846 _client->println(USER_AGENT);
defmacro 22:4d895e732765 847 _client->print("X-M2X-KEY: ");
defmacro 22:4d895e732765 848 _client->println(_key);
defmacro 22:4d895e732765 849
defmacro 22:4d895e732765 850 _client->print("Host: ");
defmacro 22:4d895e732765 851 print_encoded_string(_client, _host);
defmacro 22:4d895e732765 852 if (_port != DEFAULT_M2X_PORT) {
defmacro 22:4d895e732765 853 _client->print(":");
defmacro 22:4d895e732765 854 // port is an integer, does not need encoding
defmacro 22:4d895e732765 855 _client->print(_port);
defmacro 22:4d895e732765 856 }
defmacro 22:4d895e732765 857 _client->println();
defmacro 22:4d895e732765 858
defmacro 22:4d895e732765 859 if (contentLength > 0) {
defmacro 22:4d895e732765 860 _client->println("Content-Type: application/json");
defmacro 22:4d895e732765 861 DBG("%s", "Content Length: ");
defmacro 22:4d895e732765 862 DBGLN("%d", contentLength);
defmacro 22:4d895e732765 863
defmacro 22:4d895e732765 864 _client->print("Content-Length: ");
defmacro 22:4d895e732765 865 _client->println(contentLength);
defmacro 22:4d895e732765 866 }
defmacro 22:4d895e732765 867 _client->println();
defmacro 22:4d895e732765 868 }
defmacro 22:4d895e732765 869
defmacro 22:4d895e732765 870 int M2XStreamClient::waitForString(const char* str) {
defmacro 22:4d895e732765 871 int currentIndex = 0;
defmacro 22:4d895e732765 872 if (str[currentIndex] == '\0') return E_OK;
defmacro 22:4d895e732765 873
defmacro 22:4d895e732765 874 while (true) {
defmacro 22:4d895e732765 875 while (_client->available()) {
defmacro 22:4d895e732765 876 char c = _client->read();
defmacro 22:4d895e732765 877 DBG("%c", c);
defmacro 22:4d895e732765 878
defmacro 22:4d895e732765 879 int cmp;
defmacro 22:4d895e732765 880 if (_case_insensitive) {
defmacro 22:4d895e732765 881 cmp = tolower(c) - tolower(str[currentIndex]);
defmacro 22:4d895e732765 882 } else {
defmacro 22:4d895e732765 883 cmp = c - str[currentIndex];
defmacro 22:4d895e732765 884 }
defmacro 22:4d895e732765 885
defmacro 22:4d895e732765 886 if ((str[currentIndex] == '*') || (cmp == 0)) {
defmacro 22:4d895e732765 887 currentIndex++;
defmacro 22:4d895e732765 888 if (str[currentIndex] == '\0') {
defmacro 22:4d895e732765 889 return E_OK;
defmacro 22:4d895e732765 890 }
defmacro 22:4d895e732765 891 } else {
defmacro 22:4d895e732765 892 // start from the beginning
defmacro 22:4d895e732765 893 currentIndex = 0;
defmacro 22:4d895e732765 894 }
defmacro 22:4d895e732765 895 }
defmacro 22:4d895e732765 896
defmacro 22:4d895e732765 897 if (!_client->connected()) {
defmacro 22:4d895e732765 898 DBGLN("%s", "ERROR: The client is disconnected from the server!");
defmacro 22:4d895e732765 899
defmacro 22:4d895e732765 900 close();
defmacro 22:4d895e732765 901 return E_DISCONNECTED;
defmacro 22:4d895e732765 902 }
defmacro 22:4d895e732765 903 if (_idlefunc!=NULL)
defmacro 22:4d895e732765 904 _idlefunc();
defmacro 22:4d895e732765 905 else
defmacro 22:4d895e732765 906 delay(200);
defmacro 22:4d895e732765 907 }
defmacro 22:4d895e732765 908 // never reached here
defmacro 22:4d895e732765 909 return E_NOTREACHABLE;
defmacro 22:4d895e732765 910 }
defmacro 22:4d895e732765 911
defmacro 22:4d895e732765 912 int M2XStreamClient::readStatusCode(bool closeClient) {
defmacro 22:4d895e732765 913 int responseCode = 0;
defmacro 22:4d895e732765 914 int ret = waitForString("HTTP/*.* ");
defmacro 22:4d895e732765 915 if (ret != E_OK) {
defmacro 22:4d895e732765 916 if (closeClient) close();
defmacro 22:4d895e732765 917 return ret;
defmacro 22:4d895e732765 918 }
defmacro 22:4d895e732765 919
defmacro 22:4d895e732765 920 // ret is not needed from here(since it must be E_OK), so we can use it
defmacro 22:4d895e732765 921 // as a regular variable now.
defmacro 22:4d895e732765 922 ret = 0;
defmacro 22:4d895e732765 923 while (true) {
defmacro 22:4d895e732765 924 while (_client->available()) {
defmacro 22:4d895e732765 925 char c = _client->read();
defmacro 22:4d895e732765 926 DBG("%c", c);
defmacro 22:4d895e732765 927
defmacro 22:4d895e732765 928 responseCode = responseCode * 10 + (c - '0');
defmacro 22:4d895e732765 929 ret++;
defmacro 22:4d895e732765 930 if (ret == 3) {
defmacro 22:4d895e732765 931 if (closeClient) close();
defmacro 22:4d895e732765 932 return responseCode;
defmacro 22:4d895e732765 933 }
defmacro 22:4d895e732765 934 }
defmacro 22:4d895e732765 935
defmacro 22:4d895e732765 936 if (!_client->connected()) {
defmacro 22:4d895e732765 937 DBGLN("%s", "ERROR: The client is disconnected from the server!");
defmacro 22:4d895e732765 938
defmacro 22:4d895e732765 939 if (closeClient) close();
defmacro 22:4d895e732765 940 return E_DISCONNECTED;
defmacro 22:4d895e732765 941 }
defmacro 22:4d895e732765 942 if (_idlefunc!=NULL)
defmacro 22:4d895e732765 943 _idlefunc();
defmacro 22:4d895e732765 944 else
defmacro 22:4d895e732765 945 delay(200);
defmacro 22:4d895e732765 946 }
defmacro 22:4d895e732765 947
defmacro 22:4d895e732765 948 // never reached here
defmacro 22:4d895e732765 949 return E_NOTREACHABLE;
defmacro 22:4d895e732765 950 }
defmacro 22:4d895e732765 951
defmacro 22:4d895e732765 952 int M2XStreamClient::readContentLength() {
defmacro 22:4d895e732765 953 int ret = waitForString("Content-Length: ");
defmacro 22:4d895e732765 954 if (ret != E_OK) {
defmacro 22:4d895e732765 955 return ret;
defmacro 22:4d895e732765 956 }
defmacro 22:4d895e732765 957
defmacro 22:4d895e732765 958 // From now on, ret is not needed, we can use it
defmacro 22:4d895e732765 959 // to keep the final result
defmacro 22:4d895e732765 960 ret = 0;
defmacro 22:4d895e732765 961 while (true) {
defmacro 22:4d895e732765 962 while (_client->available()) {
defmacro 22:4d895e732765 963 char c = _client->read();
defmacro 22:4d895e732765 964 DBG("%c", c);
defmacro 22:4d895e732765 965
defmacro 22:4d895e732765 966 if ((c == '\r') || (c == '\n')) {
defmacro 22:4d895e732765 967 return (ret == 0) ? (E_INVALID) : (ret);
defmacro 22:4d895e732765 968 } else {
defmacro 22:4d895e732765 969 ret = ret * 10 + (c - '0');
defmacro 22:4d895e732765 970 }
defmacro 22:4d895e732765 971 }
defmacro 22:4d895e732765 972
defmacro 22:4d895e732765 973 if (!_client->connected()) {
defmacro 22:4d895e732765 974 DBGLN("%s", "ERROR: The client is disconnected from the server!");
defmacro 22:4d895e732765 975
defmacro 22:4d895e732765 976 return E_DISCONNECTED;
defmacro 22:4d895e732765 977 }
defmacro 22:4d895e732765 978 if (_idlefunc!=NULL)
defmacro 22:4d895e732765 979 _idlefunc();
defmacro 22:4d895e732765 980 else
defmacro 22:4d895e732765 981 delay(200);
defmacro 22:4d895e732765 982 }
defmacro 22:4d895e732765 983
defmacro 22:4d895e732765 984 // never reached here
defmacro 22:4d895e732765 985 return E_NOTREACHABLE;
defmacro 22:4d895e732765 986 }
defmacro 22:4d895e732765 987
defmacro 22:4d895e732765 988 int M2XStreamClient::skipHttpHeader() {
defmacro 22:4d895e732765 989 return waitForString("\n\r\n");
defmacro 22:4d895e732765 990 }
defmacro 22:4d895e732765 991
defmacro 22:4d895e732765 992 void M2XStreamClient::close() {
defmacro 22:4d895e732765 993 // Eats up buffered data before closing
defmacro 22:4d895e732765 994 _client->flush();
defmacro 22:4d895e732765 995 _client->stop();
defmacro 22:4d895e732765 996 }
defmacro 22:4d895e732765 997
defmacro 22:4d895e732765 998 static inline int fill_iso8601_timestamp(int32_t seconds, int32_t milli,
defmacro 22:4d895e732765 999 char* buffer, int* length);
defmacro 22:4d895e732765 1000
defmacro 22:4d895e732765 1001 TimeService::TimeService(M2XStreamClient* client) : _client(client) {
defmacro 22:4d895e732765 1002 }
defmacro 22:4d895e732765 1003
defmacro 22:4d895e732765 1004 int TimeService::init() {
defmacro 22:4d895e732765 1005 _timer.start();
defmacro 22:4d895e732765 1006 return reset();
defmacro 22:4d895e732765 1007 }
defmacro 22:4d895e732765 1008
defmacro 22:4d895e732765 1009 int TimeService::reset() {
defmacro 22:4d895e732765 1010 int32_t ts;
defmacro 22:4d895e732765 1011 int status = _client->getTimestamp32(&ts);
defmacro 22:4d895e732765 1012
defmacro 22:4d895e732765 1013 if (m2x_status_is_success(status)) {
defmacro 22:4d895e732765 1014 _server_timestamp = ts;
defmacro 22:4d895e732765 1015 _local_last_milli = _timer.read_ms();
defmacro 22:4d895e732765 1016 }
defmacro 22:4d895e732765 1017
defmacro 22:4d895e732765 1018 return status;
defmacro 22:4d895e732765 1019 }
defmacro 22:4d895e732765 1020
defmacro 22:4d895e732765 1021 int TimeService::getTimestamp(char* buffer, int* length) {
defmacro 22:4d895e732765 1022 uint32_t now = _timer.read_ms();
defmacro 22:4d895e732765 1023 if (now < _local_last_milli) {
defmacro 22:4d895e732765 1024 // In case of a timestamp overflow(happens once every 50 days on
defmacro 22:4d895e732765 1025 // Arduino), we reset the server timestamp recorded.
defmacro 22:4d895e732765 1026 // NOTE: while on an Arduino this might be okay, the situation is worse
defmacro 22:4d895e732765 1027 // on mbed, see the notes in m2x-mbed.h for details
defmacro 22:4d895e732765 1028 int status = reset();
defmacro 22:4d895e732765 1029 if (!m2x_status_is_success(status)) { return status; }
defmacro 22:4d895e732765 1030 now = _timer.read_ms();
defmacro 22:4d895e732765 1031 }
defmacro 22:4d895e732765 1032 if (now < _local_last_milli) {
defmacro 22:4d895e732765 1033 // We have already reseted the timestamp, so this cannot happen
defmacro 22:4d895e732765 1034 // (an HTTP request can take longer than 50 days to finished? You
defmacro 22:4d895e732765 1035 // must be kidding here). Something else must be wrong here
defmacro 22:4d895e732765 1036 return E_TIMESTAMP_ERROR;
defmacro 22:4d895e732765 1037 }
defmacro 22:4d895e732765 1038 uint32_t diff = now - _local_last_milli;
defmacro 22:4d895e732765 1039 _local_last_milli = now;
defmacro 22:4d895e732765 1040 _server_timestamp += (int32_t) (diff / 1000); // Milliseconds to seconds
defmacro 22:4d895e732765 1041 return fill_iso8601_timestamp(_server_timestamp, (int32_t) (diff % 1000),
defmacro 22:4d895e732765 1042 buffer, length);
defmacro 22:4d895e732765 1043 }
defmacro 22:4d895e732765 1044
defmacro 22:4d895e732765 1045 #define SIZE_ISO_8601 25
defmacro 22:4d895e732765 1046 static inline bool is_leap_year(int16_t y) {
defmacro 22:4d895e732765 1047 return ((1970 + y) > 0) &&
defmacro 22:4d895e732765 1048 !((1970 + y) % 4) &&
defmacro 22:4d895e732765 1049 (((1970 + y) % 100) || !((1970 + y) % 400));
defmacro 22:4d895e732765 1050 }
defmacro 22:4d895e732765 1051 static inline int32_t days_in_year(int16_t y) {
defmacro 22:4d895e732765 1052 return is_leap_year(y) ? 366 : 365;
defmacro 22:4d895e732765 1053 }
defmacro 22:4d895e732765 1054 static const uint8_t MONTH_DAYS[]={31,28,31,30,31,30,31,31,30,31,30,31};
defmacro 22:4d895e732765 1055
defmacro 22:4d895e732765 1056 static inline int fill_iso8601_timestamp(int32_t timestamp, int32_t milli,
defmacro 22:4d895e732765 1057 char* buffer, int* length) {
defmacro 22:4d895e732765 1058 int16_t year;
defmacro 22:4d895e732765 1059 int8_t month, month_length;
defmacro 22:4d895e732765 1060 int32_t day;
defmacro 22:4d895e732765 1061 int8_t hour, minute, second;
defmacro 22:4d895e732765 1062
defmacro 22:4d895e732765 1063 if (*length < SIZE_ISO_8601) {
defmacro 22:4d895e732765 1064 *length = SIZE_ISO_8601;
defmacro 22:4d895e732765 1065 return E_BUFFER_TOO_SMALL;
defmacro 22:4d895e732765 1066 }
defmacro 22:4d895e732765 1067
defmacro 22:4d895e732765 1068 second = timestamp % 60;
defmacro 22:4d895e732765 1069 timestamp /= 60; // now it is minutes
defmacro 22:4d895e732765 1070
defmacro 22:4d895e732765 1071 minute = timestamp % 60;
defmacro 22:4d895e732765 1072 timestamp /= 60; // now it is hours
defmacro 22:4d895e732765 1073
defmacro 22:4d895e732765 1074 hour = timestamp % 24;
defmacro 22:4d895e732765 1075 timestamp /= 24; // now it is days
defmacro 22:4d895e732765 1076
defmacro 22:4d895e732765 1077 year = 0;
defmacro 22:4d895e732765 1078 day = 0;
defmacro 22:4d895e732765 1079 while ((day += days_in_year(year)) <= timestamp) {
defmacro 22:4d895e732765 1080 year++;
defmacro 22:4d895e732765 1081 }
defmacro 22:4d895e732765 1082 day -= days_in_year(year);
defmacro 22:4d895e732765 1083 timestamp -= day; // now it is days in this year, starting at 0
defmacro 22:4d895e732765 1084
defmacro 22:4d895e732765 1085 day = 0;
defmacro 22:4d895e732765 1086 month_length = 0;
defmacro 22:4d895e732765 1087 for (month = 0; month < 12; month++) {
defmacro 22:4d895e732765 1088 if (month == 1) {
defmacro 22:4d895e732765 1089 // February
defmacro 22:4d895e732765 1090 month_length = is_leap_year(year) ? 29 : 28;
defmacro 22:4d895e732765 1091 } else {
defmacro 22:4d895e732765 1092 month_length = MONTH_DAYS[month];
defmacro 22:4d895e732765 1093 }
defmacro 22:4d895e732765 1094
defmacro 22:4d895e732765 1095 if (timestamp >= month_length) {
defmacro 22:4d895e732765 1096 timestamp -= month_length;
defmacro 22:4d895e732765 1097 } else {
defmacro 22:4d895e732765 1098 break;
defmacro 22:4d895e732765 1099 }
defmacro 22:4d895e732765 1100 }
defmacro 22:4d895e732765 1101 year = 1970 + year;
defmacro 22:4d895e732765 1102 month++; // offset by 1
defmacro 22:4d895e732765 1103 day = timestamp + 1;
defmacro 22:4d895e732765 1104
defmacro 22:4d895e732765 1105 int i = 0, j = 0;
defmacro 22:4d895e732765 1106
defmacro 22:4d895e732765 1107 // NOTE: It seems the snprintf implementation in Arduino has bugs,
defmacro 22:4d895e732765 1108 // we have to manually piece the string together here.
defmacro 22:4d895e732765 1109 #define INT_TO_STR(v_, width_) \
defmacro 22:4d895e732765 1110 for (j = 0; j < (width_); j++) { \
defmacro 22:4d895e732765 1111 buffer[i + (width_) - 1 - j] = '0' + ((v_) % 10); \
defmacro 22:4d895e732765 1112 (v_) /= 10; \
defmacro 22:4d895e732765 1113 } \
defmacro 22:4d895e732765 1114 i += (width_)
defmacro 22:4d895e732765 1115
defmacro 22:4d895e732765 1116 INT_TO_STR(year, 4);
defmacro 22:4d895e732765 1117 buffer[i++] = '-';
defmacro 22:4d895e732765 1118 INT_TO_STR(month, 2);
defmacro 22:4d895e732765 1119 buffer[i++] = '-';
defmacro 22:4d895e732765 1120 INT_TO_STR(day, 2);
defmacro 22:4d895e732765 1121 buffer[i++] = 'T';
defmacro 22:4d895e732765 1122 INT_TO_STR(hour, 2);
defmacro 22:4d895e732765 1123 buffer[i++] = ':';
defmacro 22:4d895e732765 1124 INT_TO_STR(minute, 2);
defmacro 22:4d895e732765 1125 buffer[i++] = ':';
defmacro 22:4d895e732765 1126 INT_TO_STR(second, 2);
defmacro 22:4d895e732765 1127 buffer[i++] = '.';
defmacro 22:4d895e732765 1128 INT_TO_STR(milli, 3);
defmacro 22:4d895e732765 1129 buffer[i++] = 'Z';
defmacro 22:4d895e732765 1130 buffer[i++] = '\0';
defmacro 22:4d895e732765 1131
defmacro 22:4d895e732765 1132 #undef INT_TO_STR
defmacro 22:4d895e732765 1133
defmacro 22:4d895e732765 1134 *length = i;
defmacro 22:4d895e732765 1135 return E_OK;
defmacro 22:4d895e732765 1136 }
defmacro 22:4d895e732765 1137
defmacro 22:4d895e732765 1138 /* Reader functions */
defmacro 22:4d895e732765 1139 #ifdef M2X_ENABLE_READER
defmacro 22:4d895e732765 1140
defmacro 22:4d895e732765 1141 // Data structures and functions used to parse stream values
defmacro 22:4d895e732765 1142
defmacro 22:4d895e732765 1143 #define STREAM_BUF_LEN 32
defmacro 22:4d895e732765 1144
defmacro 22:4d895e732765 1145 typedef struct {
defmacro 22:4d895e732765 1146 uint8_t state;
defmacro 22:4d895e732765 1147 char at_str[STREAM_BUF_LEN + 1];
defmacro 22:4d895e732765 1148 char value_str[STREAM_BUF_LEN + 1];
defmacro 22:4d895e732765 1149 int index;
defmacro 22:4d895e732765 1150
defmacro 22:4d895e732765 1151 stream_value_read_callback callback;
defmacro 22:4d895e732765 1152 void* context;
defmacro 22:4d895e732765 1153 } stream_parsing_context_state;
defmacro 22:4d895e732765 1154
defmacro 22:4d895e732765 1155 #define WAITING_AT 0x1
defmacro 22:4d895e732765 1156 #define GOT_AT 0x2
defmacro 22:4d895e732765 1157 #define WAITING_VALUE 0x4
defmacro 22:4d895e732765 1158 #define GOT_VALUE 0x8
defmacro 22:4d895e732765 1159
defmacro 22:4d895e732765 1160 #define GOT_STREAM (GOT_AT | GOT_VALUE)
defmacro 22:4d895e732765 1161 #define TEST_GOT_STREAM(state_) (((state_) & GOT_STREAM) == GOT_STREAM)
defmacro 22:4d895e732765 1162
defmacro 22:4d895e732765 1163 #define TEST_IS_AT(state_) (((state_) & (WAITING_AT | GOT_AT)) == WAITING_AT)
defmacro 22:4d895e732765 1164 #define TEST_IS_VALUE(state_) (((state_) & (WAITING_VALUE | GOT_VALUE)) == \
defmacro 22:4d895e732765 1165 WAITING_VALUE)
defmacro 22:4d895e732765 1166
defmacro 22:4d895e732765 1167 static void on_stream_key_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1168 jsonlite_token* token)
defmacro 22:4d895e732765 1169 {
defmacro 22:4d895e732765 1170 stream_parsing_context_state* state =
defmacro 22:4d895e732765 1171 (stream_parsing_context_state*) context->client_state;
defmacro 22:4d895e732765 1172 if (strncmp((const char*) token->start, "timestamp", 9) == 0) {
defmacro 22:4d895e732765 1173 state->state |= WAITING_AT;
defmacro 22:4d895e732765 1174 } else if ((strncmp((const char*) token->start, "value", 5) == 0) &&
defmacro 22:4d895e732765 1175 (token->start[5] != 's')) { // get rid of "values"
defmacro 22:4d895e732765 1176 state->state |= WAITING_VALUE;
defmacro 22:4d895e732765 1177 }
defmacro 22:4d895e732765 1178 }
defmacro 22:4d895e732765 1179
defmacro 22:4d895e732765 1180 static void on_stream_value_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1181 jsonlite_token* token,
defmacro 22:4d895e732765 1182 int type)
defmacro 22:4d895e732765 1183 {
defmacro 22:4d895e732765 1184 stream_parsing_context_state* state =
defmacro 22:4d895e732765 1185 (stream_parsing_context_state*) context->client_state;
defmacro 22:4d895e732765 1186
defmacro 22:4d895e732765 1187 if (TEST_IS_AT(state->state)) {
defmacro 22:4d895e732765 1188 strncpy(state->at_str, (const char*) token->start,
defmacro 22:4d895e732765 1189 MIN(token->end - token->start, STREAM_BUF_LEN));
defmacro 22:4d895e732765 1190 state->at_str[MIN(token->end - token->start, STREAM_BUF_LEN)] = '\0';
defmacro 22:4d895e732765 1191 state->state |= GOT_AT;
defmacro 22:4d895e732765 1192 } else if (TEST_IS_VALUE(state->state)) {
defmacro 22:4d895e732765 1193 strncpy(state->value_str, (const char*) token->start,
defmacro 22:4d895e732765 1194 MIN(token->end - token->start, STREAM_BUF_LEN));
defmacro 22:4d895e732765 1195 state->value_str[MIN(token->end - token->start, STREAM_BUF_LEN)] = '\0';
defmacro 22:4d895e732765 1196 state->state |= GOT_VALUE;
defmacro 22:4d895e732765 1197 }
defmacro 22:4d895e732765 1198
defmacro 22:4d895e732765 1199 if (TEST_GOT_STREAM(state->state)) {
defmacro 22:4d895e732765 1200 state->callback(state->at_str, state->value_str,
defmacro 22:4d895e732765 1201 state->index++, state->context, type);
defmacro 22:4d895e732765 1202 state->state = 0;
defmacro 22:4d895e732765 1203 }
defmacro 22:4d895e732765 1204 }
defmacro 22:4d895e732765 1205
defmacro 22:4d895e732765 1206 static void on_stream_string_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1207 jsonlite_token* token)
defmacro 22:4d895e732765 1208 {
defmacro 22:4d895e732765 1209 on_stream_value_found(context, token, 1);
defmacro 22:4d895e732765 1210 }
defmacro 22:4d895e732765 1211
defmacro 22:4d895e732765 1212 static void on_stream_number_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1213 jsonlite_token* token)
defmacro 22:4d895e732765 1214 {
defmacro 22:4d895e732765 1215 on_stream_value_found(context, token, 2);
defmacro 22:4d895e732765 1216 }
defmacro 22:4d895e732765 1217
defmacro 22:4d895e732765 1218 // Data structures and functions used to parse locations
defmacro 22:4d895e732765 1219
defmacro 22:4d895e732765 1220 #define LOCATION_BUF_LEN 20
defmacro 22:4d895e732765 1221
defmacro 22:4d895e732765 1222 typedef struct {
defmacro 22:4d895e732765 1223 uint16_t state;
defmacro 22:4d895e732765 1224 char name_str[LOCATION_BUF_LEN + 1];
defmacro 22:4d895e732765 1225 double latitude;
defmacro 22:4d895e732765 1226 double longitude;
defmacro 22:4d895e732765 1227 double elevation;
defmacro 22:4d895e732765 1228 char timestamp_str[LOCATION_BUF_LEN + 1];
defmacro 22:4d895e732765 1229 int index;
defmacro 22:4d895e732765 1230
defmacro 22:4d895e732765 1231 location_read_callback callback;
defmacro 22:4d895e732765 1232 void* context;
defmacro 22:4d895e732765 1233 } location_parsing_context_state;
defmacro 22:4d895e732765 1234
defmacro 22:4d895e732765 1235 #define WAITING_NAME 0x1
defmacro 22:4d895e732765 1236 #define WAITING_LATITUDE 0x2
defmacro 22:4d895e732765 1237 #define WAITING_LONGITUDE 0x4
defmacro 22:4d895e732765 1238 #define WAITING_ELEVATION 0x8
defmacro 22:4d895e732765 1239 #define WAITING_TIMESTAMP 0x10
defmacro 22:4d895e732765 1240
defmacro 22:4d895e732765 1241 #define GOT_NAME 0x20
defmacro 22:4d895e732765 1242 #define GOT_LATITUDE 0x40
defmacro 22:4d895e732765 1243 #define GOT_LONGITUDE 0x80
defmacro 22:4d895e732765 1244 #define GOT_ELEVATION 0x100
defmacro 22:4d895e732765 1245 #define GOT_TIMESTAMP 0x200
defmacro 22:4d895e732765 1246
defmacro 22:4d895e732765 1247 #define GOT_LOCATION (GOT_NAME | GOT_LATITUDE | GOT_LONGITUDE | GOT_ELEVATION | GOT_TIMESTAMP)
defmacro 22:4d895e732765 1248 #define TEST_GOT_LOCATION(state_) (((state_) & GOT_LOCATION) == GOT_LOCATION)
defmacro 22:4d895e732765 1249
defmacro 22:4d895e732765 1250 #define TEST_IS_NAME(state_) (((state_) & (WAITING_NAME | GOT_NAME)) == WAITING_NAME)
defmacro 22:4d895e732765 1251 #define TEST_IS_LATITUDE(state_) (((state_) & (WAITING_LATITUDE | GOT_LATITUDE)) \
defmacro 22:4d895e732765 1252 == WAITING_LATITUDE)
defmacro 22:4d895e732765 1253 #define TEST_IS_LONGITUDE(state_) (((state_) & (WAITING_LONGITUDE | GOT_LONGITUDE)) \
defmacro 22:4d895e732765 1254 == WAITING_LONGITUDE)
defmacro 22:4d895e732765 1255 #define TEST_IS_ELEVATION(state_) (((state_) & (WAITING_ELEVATION | GOT_ELEVATION)) \
defmacro 22:4d895e732765 1256 == WAITING_ELEVATION)
defmacro 22:4d895e732765 1257 #define TEST_IS_TIMESTAMP(state_) (((state_) & (WAITING_TIMESTAMP | GOT_TIMESTAMP)) \
defmacro 22:4d895e732765 1258 == WAITING_TIMESTAMP)
defmacro 22:4d895e732765 1259
defmacro 22:4d895e732765 1260 static void on_location_key_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1261 jsonlite_token* token) {
defmacro 22:4d895e732765 1262 location_parsing_context_state* state =
defmacro 22:4d895e732765 1263 (location_parsing_context_state*) context->client_state;
defmacro 22:4d895e732765 1264 if (strncmp((const char*) token->start, "waypoints", 9) == 0) {
defmacro 22:4d895e732765 1265 // only parses those locations in waypoints, skip the outer one
defmacro 22:4d895e732765 1266 state->state = 0;
defmacro 22:4d895e732765 1267 } else if (strncmp((const char*) token->start, "name", 4) == 0) {
defmacro 22:4d895e732765 1268 state->state |= WAITING_NAME;
defmacro 22:4d895e732765 1269 } else if (strncmp((const char*) token->start, "latitude", 8) == 0) {
defmacro 22:4d895e732765 1270 state->state |= WAITING_LATITUDE;
defmacro 22:4d895e732765 1271 } else if (strncmp((const char*) token->start, "longitude", 9) == 0) {
defmacro 22:4d895e732765 1272 state->state |= WAITING_LONGITUDE;
defmacro 22:4d895e732765 1273 } else if (strncmp((const char*) token->start, "elevation", 9) == 0) {
defmacro 22:4d895e732765 1274 state->state |= WAITING_ELEVATION;
defmacro 22:4d895e732765 1275 } else if (strncmp((const char*) token->start, "timestamp", 9) == 0) {
defmacro 22:4d895e732765 1276 state->state |= WAITING_TIMESTAMP;
defmacro 22:4d895e732765 1277 }
defmacro 22:4d895e732765 1278 }
defmacro 22:4d895e732765 1279
defmacro 22:4d895e732765 1280 static void on_location_string_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1281 jsonlite_token* token) {
defmacro 22:4d895e732765 1282 location_parsing_context_state* state =
defmacro 22:4d895e732765 1283 (location_parsing_context_state*) context->client_state;
defmacro 22:4d895e732765 1284
defmacro 22:4d895e732765 1285 if (TEST_IS_NAME(state->state)) {
defmacro 22:4d895e732765 1286 strncpy(state->name_str, (const char*) token->start,
defmacro 22:4d895e732765 1287 MIN(token->end - token->start, LOCATION_BUF_LEN));
defmacro 22:4d895e732765 1288 state->name_str[MIN(token->end - token->start, LOCATION_BUF_LEN)] = '\0';
defmacro 22:4d895e732765 1289 state->state |= GOT_NAME;
defmacro 22:4d895e732765 1290 } else if (TEST_IS_LATITUDE(state->state)) {
defmacro 22:4d895e732765 1291 state->latitude = atof((const char*) token->start);
defmacro 22:4d895e732765 1292 state->state |= GOT_LATITUDE;
defmacro 22:4d895e732765 1293 } else if (TEST_IS_LONGITUDE(state->state)) {
defmacro 22:4d895e732765 1294 state->longitude = atof((const char*) token->start);
defmacro 22:4d895e732765 1295 state->state |= GOT_LONGITUDE;
defmacro 22:4d895e732765 1296 } else if (TEST_IS_ELEVATION(state->state)) {
defmacro 22:4d895e732765 1297 state->elevation = atof((const char*) token->start);
defmacro 22:4d895e732765 1298 state->state |= GOT_ELEVATION;
defmacro 22:4d895e732765 1299 } else if (TEST_IS_TIMESTAMP(state->state)) {
defmacro 22:4d895e732765 1300 strncpy(state->timestamp_str, (const char*) token->start,
defmacro 22:4d895e732765 1301 MIN(token->end - token->start, LOCATION_BUF_LEN));
defmacro 22:4d895e732765 1302 state->timestamp_str[MIN(token->end - token->start, LOCATION_BUF_LEN)] = '\0';
defmacro 22:4d895e732765 1303 state->state |= GOT_TIMESTAMP;
defmacro 22:4d895e732765 1304 }
defmacro 22:4d895e732765 1305
defmacro 22:4d895e732765 1306 if (TEST_GOT_LOCATION(state->state)) {
defmacro 22:4d895e732765 1307 state->callback(state->name_str, state->latitude, state->longitude,
defmacro 22:4d895e732765 1308 state->elevation, state->timestamp_str, state->index++,
defmacro 22:4d895e732765 1309 state->context);
defmacro 22:4d895e732765 1310 state->state = 0;
defmacro 22:4d895e732765 1311 }
defmacro 22:4d895e732765 1312 }
defmacro 22:4d895e732765 1313
defmacro 22:4d895e732765 1314 #ifndef M2X_COMMAND_BUF_LEN
defmacro 22:4d895e732765 1315 #define M2X_COMMAND_BUF_LEN 40
defmacro 22:4d895e732765 1316 #endif /* M2X_COMMAND_BUF_LEN */
defmacro 22:4d895e732765 1317
defmacro 22:4d895e732765 1318 typedef struct {
defmacro 22:4d895e732765 1319 uint8_t state;
defmacro 22:4d895e732765 1320 char id_str[M2X_COMMAND_BUF_LEN + 1];
defmacro 22:4d895e732765 1321 char name_str[M2X_COMMAND_BUF_LEN + 1];
defmacro 22:4d895e732765 1322 int index;
defmacro 22:4d895e732765 1323
defmacro 22:4d895e732765 1324 m2x_command_read_callback callback;
defmacro 22:4d895e732765 1325 void* context;
defmacro 22:4d895e732765 1326 } m2x_command_parsing_context_state;
defmacro 22:4d895e732765 1327
defmacro 22:4d895e732765 1328 #define M2X_COMMAND_WAITING_ID 0x1
defmacro 22:4d895e732765 1329 #define M2X_COMMAND_GOT_ID 0x2
defmacro 22:4d895e732765 1330 #define M2X_COMMAND_WAITING_NAME 0x4
defmacro 22:4d895e732765 1331 #define M2X_COMMAND_GOT_NAME 0x8
defmacro 22:4d895e732765 1332
defmacro 22:4d895e732765 1333 #define M2X_COMMAND_GOT_COMMAND (M2X_COMMAND_GOT_ID | M2X_COMMAND_GOT_NAME)
defmacro 22:4d895e732765 1334 #define M2X_COMMAND_TEST_GOT_COMMAND(state_) \
defmacro 22:4d895e732765 1335 (((state_) & M2X_COMMAND_GOT_COMMAND) == M2X_COMMAND_GOT_COMMAND)
defmacro 22:4d895e732765 1336
defmacro 22:4d895e732765 1337 #define M2X_COMMAND_TEST_IS_ID(state_) \
defmacro 22:4d895e732765 1338 (((state_) & (M2X_COMMAND_WAITING_ID | M2X_COMMAND_GOT_ID)) == \
defmacro 22:4d895e732765 1339 M2X_COMMAND_WAITING_ID)
defmacro 22:4d895e732765 1340 #define M2X_COMMAND_TEST_IS_NAME(state_) \
defmacro 22:4d895e732765 1341 (((state_) & (M2X_COMMAND_WAITING_NAME | M2X_COMMAND_GOT_NAME)) == \
defmacro 22:4d895e732765 1342 M2X_COMMAND_WAITING_NAME)
defmacro 22:4d895e732765 1343
defmacro 22:4d895e732765 1344 static void m2x_on_command_key_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1345 jsonlite_token* token)
defmacro 22:4d895e732765 1346 {
defmacro 22:4d895e732765 1347 m2x_command_parsing_context_state* state =
defmacro 22:4d895e732765 1348 (m2x_command_parsing_context_state*) context->client_state;
defmacro 22:4d895e732765 1349 if (strncmp((const char*) token->start, "id", 2) == 0) {
defmacro 22:4d895e732765 1350 state->state |= M2X_COMMAND_WAITING_ID;
defmacro 22:4d895e732765 1351 } else if (strncmp((const char*) token->start, "name", 4) == 0) {
defmacro 22:4d895e732765 1352 state->state |= M2X_COMMAND_WAITING_NAME;
defmacro 22:4d895e732765 1353 }
defmacro 22:4d895e732765 1354 }
defmacro 22:4d895e732765 1355
defmacro 22:4d895e732765 1356 static void m2x_on_command_value_found(jsonlite_callback_context* context,
defmacro 22:4d895e732765 1357 jsonlite_token* token)
defmacro 22:4d895e732765 1358 {
defmacro 22:4d895e732765 1359 m2x_command_parsing_context_state* state =
defmacro 22:4d895e732765 1360 (m2x_command_parsing_context_state*) context->client_state;
defmacro 22:4d895e732765 1361 if (M2X_COMMAND_TEST_IS_ID(state->state)) {
defmacro 22:4d895e732765 1362 strncpy(state->id_str, (const char*) token->start,
defmacro 22:4d895e732765 1363 MIN(token->end - token->start, M2X_COMMAND_BUF_LEN));
defmacro 22:4d895e732765 1364 state->id_str[MIN(token->end - token->start, M2X_COMMAND_BUF_LEN)] = '\0';
defmacro 22:4d895e732765 1365 state->state |= M2X_COMMAND_GOT_ID;
defmacro 22:4d895e732765 1366 } else if (M2X_COMMAND_TEST_IS_NAME(state->state)) {
defmacro 22:4d895e732765 1367 strncpy(state->name_str, (const char*) token->start,
defmacro 22:4d895e732765 1368 MIN(token->end - token->start, M2X_COMMAND_BUF_LEN));
defmacro 22:4d895e732765 1369 state->name_str[MIN(token->end - token->start, M2X_COMMAND_BUF_LEN)] = '\0';
defmacro 22:4d895e732765 1370 state->state |= M2X_COMMAND_GOT_NAME;
defmacro 22:4d895e732765 1371 }
defmacro 22:4d895e732765 1372
defmacro 22:4d895e732765 1373 if (M2X_COMMAND_TEST_GOT_COMMAND(state->state)) {
defmacro 22:4d895e732765 1374 state->callback(state->id_str, state->name_str,
defmacro 22:4d895e732765 1375 state->index++, state->context);
defmacro 22:4d895e732765 1376 state->state = 0;
defmacro 22:4d895e732765 1377 }
defmacro 22:4d895e732765 1378 }
defmacro 22:4d895e732765 1379
defmacro 22:4d895e732765 1380 int M2XStreamClient::listStreamValues(const char* deviceId, const char* streamName,
defmacro 22:4d895e732765 1381 stream_value_read_callback callback, void* context,
defmacro 22:4d895e732765 1382 const char* query) {
defmacro 22:4d895e732765 1383 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 1384 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 1385 _client->print("GET ");
defmacro 22:4d895e732765 1386 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 1387 _client->print("/v2/devices/");
defmacro 22:4d895e732765 1388 _client->print(deviceId);
defmacro 22:4d895e732765 1389 _client->print("/streams/");
defmacro 22:4d895e732765 1390 print_encoded_string(_client, streamName);
defmacro 22:4d895e732765 1391 _client->print("/values");
defmacro 22:4d895e732765 1392
defmacro 22:4d895e732765 1393 if (query) {
defmacro 22:4d895e732765 1394 if (query[0] != '?') {
defmacro 22:4d895e732765 1395 _client->print('?');
defmacro 22:4d895e732765 1396 }
defmacro 22:4d895e732765 1397 _client->print(query);
defmacro 22:4d895e732765 1398 }
defmacro 22:4d895e732765 1399
JMF 25:67079c11d1ea 1400 _client->println(" HTTP/1.1");
defmacro 22:4d895e732765 1401 writeHttpHeader(-1);
defmacro 22:4d895e732765 1402 } else {
defmacro 22:4d895e732765 1403 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 1404 return E_NOCONNECTION;
defmacro 22:4d895e732765 1405 }
defmacro 22:4d895e732765 1406 int status = readStatusCode(false);
defmacro 22:4d895e732765 1407 if (status == 200) {
defmacro 22:4d895e732765 1408 readStreamValue(callback, context);
defmacro 22:4d895e732765 1409 }
defmacro 22:4d895e732765 1410
defmacro 22:4d895e732765 1411 close();
defmacro 22:4d895e732765 1412 return status;
defmacro 22:4d895e732765 1413 }
defmacro 22:4d895e732765 1414
defmacro 22:4d895e732765 1415 int M2XStreamClient::readLocation(const char* deviceId,
defmacro 22:4d895e732765 1416 location_read_callback callback,
defmacro 22:4d895e732765 1417 void* context) {
defmacro 22:4d895e732765 1418 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 1419 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 1420 _client->print("GET ");
defmacro 22:4d895e732765 1421 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 1422 _client->print("/v2/devices/");
defmacro 22:4d895e732765 1423 _client->print(deviceId);
JMF 25:67079c11d1ea 1424 _client->println("/location HTTP/1.1");
defmacro 22:4d895e732765 1425
defmacro 22:4d895e732765 1426 writeHttpHeader(-1);
defmacro 22:4d895e732765 1427 } else {
defmacro 22:4d895e732765 1428 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 1429 return E_NOCONNECTION;
defmacro 22:4d895e732765 1430 }
defmacro 22:4d895e732765 1431 int status = readStatusCode(false);
defmacro 22:4d895e732765 1432 if (status == 200) {
defmacro 22:4d895e732765 1433 readLocation(callback, context);
defmacro 22:4d895e732765 1434 }
defmacro 22:4d895e732765 1435
defmacro 22:4d895e732765 1436 close();
defmacro 22:4d895e732765 1437 return status;
defmacro 22:4d895e732765 1438 }
defmacro 22:4d895e732765 1439
JMF 23:f32837239193 1440 int M2XStreamClient::deleteLocations(const char* deviceId,
JMF 23:f32837239193 1441 const char* from, const char* end) {
JMF 23:f32837239193 1442 if (_client->connect(_host, _port)) {
JMF 23:f32837239193 1443 DBGLN("%s", "Connected to M2X server!");
JMF 23:f32837239193 1444 int length = write_delete_values(&_null_print, from, end);
JMF 23:f32837239193 1445 writeDeleteLocationHeader(deviceId, length);
JMF 23:f32837239193 1446 write_delete_values(_client, from, end);
JMF 23:f32837239193 1447 } else {
JMF 23:f32837239193 1448 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
JMF 23:f32837239193 1449 return E_NOCONNECTION;
JMF 23:f32837239193 1450 }
JMF 23:f32837239193 1451
JMF 23:f32837239193 1452 return readStatusCode(true);
JMF 23:f32837239193 1453 }
JMF 23:f32837239193 1454
defmacro 22:4d895e732765 1455 int M2XStreamClient::listCommands(const char* deviceId,
defmacro 22:4d895e732765 1456 m2x_command_read_callback callback,
defmacro 22:4d895e732765 1457 void* context,
defmacro 22:4d895e732765 1458 const char* query) {
defmacro 22:4d895e732765 1459 if (_client->connect(_host, _port)) {
defmacro 22:4d895e732765 1460 DBGLN("%s", "Connected to M2X server!");
defmacro 22:4d895e732765 1461 _client->print("GET ");
defmacro 22:4d895e732765 1462 if (_path_prefix) { _client->print(_path_prefix); }
defmacro 22:4d895e732765 1463 _client->print("/v2/devices/");
defmacro 22:4d895e732765 1464 _client->print(deviceId);
defmacro 22:4d895e732765 1465 _client->print("/commands");
defmacro 22:4d895e732765 1466
defmacro 22:4d895e732765 1467 if (query) {
defmacro 22:4d895e732765 1468 if (query[0] != '?') {
defmacro 22:4d895e732765 1469 _client->print('?');
defmacro 22:4d895e732765 1470 }
defmacro 22:4d895e732765 1471 _client->print(query);
defmacro 22:4d895e732765 1472 }
defmacro 22:4d895e732765 1473
JMF 25:67079c11d1ea 1474 _client->println(" HTTP/1.1");
defmacro 22:4d895e732765 1475 writeHttpHeader(-1);
defmacro 22:4d895e732765 1476 } else {
defmacro 22:4d895e732765 1477 DBGLN("%s", "ERROR: Cannot connect to M2X server!");
defmacro 22:4d895e732765 1478 return E_NOCONNECTION;
defmacro 22:4d895e732765 1479 }
defmacro 22:4d895e732765 1480 int status = readStatusCode(false);
defmacro 22:4d895e732765 1481 if (status == 200) {
defmacro 22:4d895e732765 1482 readCommand(callback, context);
defmacro 22:4d895e732765 1483 }
defmacro 22:4d895e732765 1484
defmacro 22:4d895e732765 1485 close();
defmacro 22:4d895e732765 1486 return status;
defmacro 22:4d895e732765 1487 }
defmacro 22:4d895e732765 1488
defmacro 22:4d895e732765 1489
defmacro 22:4d895e732765 1490 int M2XStreamClient::readStreamValue(stream_value_read_callback callback,
defmacro 22:4d895e732765 1491 void* context) {
defmacro 22:4d895e732765 1492 const int BUF_LEN = 64;
defmacro 22:4d895e732765 1493 char buf[BUF_LEN];
defmacro 22:4d895e732765 1494
defmacro 22:4d895e732765 1495 int length = readContentLength();
defmacro 22:4d895e732765 1496 if (length < 0) {
defmacro 22:4d895e732765 1497 close();
defmacro 22:4d895e732765 1498 return length;
defmacro 22:4d895e732765 1499 }
defmacro 22:4d895e732765 1500
defmacro 22:4d895e732765 1501 int index = skipHttpHeader();
defmacro 22:4d895e732765 1502 if (index != E_OK) {
defmacro 22:4d895e732765 1503 close();
defmacro 22:4d895e732765 1504 return index;
defmacro 22:4d895e732765 1505 }
defmacro 22:4d895e732765 1506 index = 0;
defmacro 22:4d895e732765 1507
defmacro 22:4d895e732765 1508 stream_parsing_context_state state;
defmacro 22:4d895e732765 1509 state.state = state.index = 0;
defmacro 22:4d895e732765 1510 state.callback = callback;
defmacro 22:4d895e732765 1511 state.context = context;
defmacro 22:4d895e732765 1512
defmacro 22:4d895e732765 1513 jsonlite_parser_callbacks cbs = jsonlite_default_callbacks;
defmacro 22:4d895e732765 1514 cbs.key_found = on_stream_key_found;
defmacro 22:4d895e732765 1515 cbs.number_found = on_stream_number_found;
defmacro 22:4d895e732765 1516 cbs.string_found = on_stream_string_found;
defmacro 22:4d895e732765 1517 cbs.context.client_state = &state;
defmacro 22:4d895e732765 1518
defmacro 22:4d895e732765 1519 jsonlite_parser p = jsonlite_parser_init(jsonlite_parser_estimate_size(5));
defmacro 22:4d895e732765 1520 jsonlite_parser_set_callback(p, &cbs);
defmacro 22:4d895e732765 1521
defmacro 22:4d895e732765 1522 jsonlite_result result = jsonlite_result_unknown;
defmacro 22:4d895e732765 1523 while (index < length) {
defmacro 22:4d895e732765 1524 int i = 0;
defmacro 22:4d895e732765 1525
defmacro 22:4d895e732765 1526 DBG("%s", "Received Data: ");
defmacro 22:4d895e732765 1527 while ((i < BUF_LEN) && _client->available()) {
defmacro 22:4d895e732765 1528 buf[i++] = _client->read();
defmacro 22:4d895e732765 1529 DBG("%c", buf[i - 1]);
defmacro 22:4d895e732765 1530 }
defmacro 22:4d895e732765 1531 DBGLNEND;
defmacro 22:4d895e732765 1532
defmacro 22:4d895e732765 1533 if ((!_client->connected()) &&
defmacro 22:4d895e732765 1534 (!_client->available()) &&
defmacro 22:4d895e732765 1535 ((index + i) < length)) {
defmacro 22:4d895e732765 1536 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1537 close();
defmacro 22:4d895e732765 1538 return E_NOCONNECTION;
defmacro 22:4d895e732765 1539 }
defmacro 22:4d895e732765 1540
defmacro 22:4d895e732765 1541 result = jsonlite_parser_tokenize(p, buf, i);
defmacro 22:4d895e732765 1542 if ((result != jsonlite_result_ok) &&
defmacro 22:4d895e732765 1543 (result != jsonlite_result_end_of_stream)) {
defmacro 22:4d895e732765 1544 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1545 close();
defmacro 22:4d895e732765 1546 return E_JSON_INVALID;
defmacro 22:4d895e732765 1547 }
defmacro 22:4d895e732765 1548
defmacro 22:4d895e732765 1549 index += i;
defmacro 22:4d895e732765 1550 }
defmacro 22:4d895e732765 1551
defmacro 22:4d895e732765 1552 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1553 close();
defmacro 22:4d895e732765 1554 return (result == jsonlite_result_ok) ? (E_OK) : (E_JSON_INVALID);
defmacro 22:4d895e732765 1555 }
defmacro 22:4d895e732765 1556
defmacro 22:4d895e732765 1557 int M2XStreamClient::readLocation(location_read_callback callback,
defmacro 22:4d895e732765 1558 void* context) {
defmacro 22:4d895e732765 1559 const int BUF_LEN = 40;
defmacro 22:4d895e732765 1560 char buf[BUF_LEN];
defmacro 22:4d895e732765 1561
defmacro 22:4d895e732765 1562 int length = readContentLength();
defmacro 22:4d895e732765 1563 if (length < 0) {
defmacro 22:4d895e732765 1564 close();
defmacro 22:4d895e732765 1565 return length;
defmacro 22:4d895e732765 1566 }
defmacro 22:4d895e732765 1567
defmacro 22:4d895e732765 1568 int index = skipHttpHeader();
defmacro 22:4d895e732765 1569 if (index != E_OK) {
defmacro 22:4d895e732765 1570 close();
defmacro 22:4d895e732765 1571 return index;
defmacro 22:4d895e732765 1572 }
defmacro 22:4d895e732765 1573 index = 0;
defmacro 22:4d895e732765 1574
defmacro 22:4d895e732765 1575 location_parsing_context_state state;
defmacro 22:4d895e732765 1576 state.state = state.index = 0;
defmacro 22:4d895e732765 1577 state.callback = callback;
defmacro 22:4d895e732765 1578 state.context = context;
defmacro 22:4d895e732765 1579
defmacro 22:4d895e732765 1580 jsonlite_parser_callbacks cbs = jsonlite_default_callbacks;
defmacro 22:4d895e732765 1581 cbs.key_found = on_location_key_found;
defmacro 22:4d895e732765 1582 cbs.string_found = on_location_string_found;
defmacro 22:4d895e732765 1583 cbs.context.client_state = &state;
defmacro 22:4d895e732765 1584
defmacro 22:4d895e732765 1585 jsonlite_parser p = jsonlite_parser_init(jsonlite_parser_estimate_size(5));
defmacro 22:4d895e732765 1586 jsonlite_parser_set_callback(p, &cbs);
defmacro 22:4d895e732765 1587
defmacro 22:4d895e732765 1588 jsonlite_result result = jsonlite_result_unknown;
defmacro 22:4d895e732765 1589 while (index < length) {
defmacro 22:4d895e732765 1590 int i = 0;
defmacro 22:4d895e732765 1591
defmacro 22:4d895e732765 1592 DBG("%s", "Received Data: ");
defmacro 22:4d895e732765 1593 while ((i < BUF_LEN) && _client->available()) {
defmacro 22:4d895e732765 1594 buf[i++] = _client->read();
defmacro 22:4d895e732765 1595 DBG("%c", buf[i - 1]);
defmacro 22:4d895e732765 1596 }
defmacro 22:4d895e732765 1597 DBGLNEND;
defmacro 22:4d895e732765 1598
defmacro 22:4d895e732765 1599 if ((!_client->connected()) &&
defmacro 22:4d895e732765 1600 (!_client->available()) &&
defmacro 22:4d895e732765 1601 ((index + i) < length)) {
defmacro 22:4d895e732765 1602 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1603 close();
defmacro 22:4d895e732765 1604 return E_NOCONNECTION;
defmacro 22:4d895e732765 1605 }
defmacro 22:4d895e732765 1606
defmacro 22:4d895e732765 1607 result = jsonlite_parser_tokenize(p, buf, i);
defmacro 22:4d895e732765 1608 if ((result != jsonlite_result_ok) &&
defmacro 22:4d895e732765 1609 (result != jsonlite_result_end_of_stream)) {
defmacro 22:4d895e732765 1610 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1611 close();
defmacro 22:4d895e732765 1612 return E_JSON_INVALID;
defmacro 22:4d895e732765 1613 }
defmacro 22:4d895e732765 1614
defmacro 22:4d895e732765 1615 index += i;
defmacro 22:4d895e732765 1616 }
defmacro 22:4d895e732765 1617
defmacro 22:4d895e732765 1618 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1619 close();
defmacro 22:4d895e732765 1620 return (result == jsonlite_result_ok) ? (E_OK) : (E_JSON_INVALID);
defmacro 22:4d895e732765 1621 }
defmacro 22:4d895e732765 1622
defmacro 22:4d895e732765 1623 int M2XStreamClient::readCommand(m2x_command_read_callback callback,
defmacro 22:4d895e732765 1624 void* context) {
defmacro 22:4d895e732765 1625 const int BUF_LEN = 60;
defmacro 22:4d895e732765 1626 char buf[BUF_LEN];
defmacro 22:4d895e732765 1627
defmacro 22:4d895e732765 1628 int length = readContentLength();
defmacro 22:4d895e732765 1629 if (length < 0) {
defmacro 22:4d895e732765 1630 close();
defmacro 22:4d895e732765 1631 return length;
defmacro 22:4d895e732765 1632 }
defmacro 22:4d895e732765 1633
defmacro 22:4d895e732765 1634 int index = skipHttpHeader();
defmacro 22:4d895e732765 1635 if (index != E_OK) {
defmacro 22:4d895e732765 1636 close();
defmacro 22:4d895e732765 1637 return index;
defmacro 22:4d895e732765 1638 }
defmacro 22:4d895e732765 1639 index = 0;
defmacro 22:4d895e732765 1640
defmacro 22:4d895e732765 1641 m2x_command_parsing_context_state state;
defmacro 22:4d895e732765 1642 state.state = 0;
defmacro 22:4d895e732765 1643 state.index = 0;
defmacro 22:4d895e732765 1644 state.callback = callback;
defmacro 22:4d895e732765 1645 state.context = context;
defmacro 22:4d895e732765 1646
defmacro 22:4d895e732765 1647 jsonlite_parser_callbacks cbs = jsonlite_default_callbacks;
defmacro 22:4d895e732765 1648 cbs.key_found = m2x_on_command_key_found;
defmacro 22:4d895e732765 1649 cbs.string_found = m2x_on_command_value_found;
defmacro 22:4d895e732765 1650 cbs.context.client_state = &state;
defmacro 22:4d895e732765 1651
defmacro 22:4d895e732765 1652 jsonlite_parser p = jsonlite_parser_init(jsonlite_parser_estimate_size(5));
defmacro 22:4d895e732765 1653 jsonlite_parser_set_callback(p, &cbs);
defmacro 22:4d895e732765 1654
defmacro 22:4d895e732765 1655 jsonlite_result result = jsonlite_result_unknown;
defmacro 22:4d895e732765 1656 while (index < length) {
defmacro 22:4d895e732765 1657 int i = 0;
defmacro 22:4d895e732765 1658
defmacro 22:4d895e732765 1659 DBG("%s", "Received Data: ");
defmacro 22:4d895e732765 1660 while ((i < BUF_LEN) && _client->available()) {
defmacro 22:4d895e732765 1661 buf[i++] = _client->read();
defmacro 22:4d895e732765 1662 DBG("%c", buf[i - 1]);
defmacro 22:4d895e732765 1663 }
defmacro 22:4d895e732765 1664 DBGLNEND;
defmacro 22:4d895e732765 1665
defmacro 22:4d895e732765 1666 if ((!_client->connected()) &&
defmacro 22:4d895e732765 1667 (!_client->available()) &&
defmacro 22:4d895e732765 1668 ((index + i) < length)) {
defmacro 22:4d895e732765 1669 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1670 close();
defmacro 22:4d895e732765 1671 return E_NOCONNECTION;
defmacro 22:4d895e732765 1672 }
defmacro 22:4d895e732765 1673
defmacro 22:4d895e732765 1674 result = jsonlite_parser_tokenize(p, buf, i);
defmacro 22:4d895e732765 1675 if ((result != jsonlite_result_ok) &&
defmacro 22:4d895e732765 1676 (result != jsonlite_result_end_of_stream)) {
defmacro 22:4d895e732765 1677 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1678 close();
defmacro 22:4d895e732765 1679 return E_JSON_INVALID;
defmacro 22:4d895e732765 1680 }
defmacro 22:4d895e732765 1681
defmacro 22:4d895e732765 1682 index += i;
defmacro 22:4d895e732765 1683 }
defmacro 22:4d895e732765 1684
defmacro 22:4d895e732765 1685 jsonlite_parser_release(p);
defmacro 22:4d895e732765 1686 close();
defmacro 22:4d895e732765 1687 return (result == jsonlite_result_ok) ? (E_OK) : (E_JSON_INVALID);
defmacro 22:4d895e732765 1688 }
defmacro 22:4d895e732765 1689
defmacro 22:4d895e732765 1690 #endif /* M2X_ENABLE_READER */
jb8414 0:f479e4f4db0e 1691
jb8414 0:f479e4f4db0e 1692 #endif /* M2XStreamClient_h */