aclk_util.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "aclk_util.h"
  3. #include "daemon/common.h"
  4. int aclk_use_new_cloud_arch = 0;
  5. usec_t aclk_session_newarch = 0;
  6. aclk_env_t *aclk_env = NULL;
  7. int chart_batch_id;
  8. aclk_encoding_type_t aclk_encoding_type_t_from_str(const char *str) {
  9. if (!strcmp(str, "json")) {
  10. return ACLK_ENC_JSON;
  11. }
  12. if (!strcmp(str, "proto")) {
  13. return ACLK_ENC_PROTO;
  14. }
  15. return ACLK_ENC_UNKNOWN;
  16. }
  17. aclk_transport_type_t aclk_transport_type_t_from_str(const char *str) {
  18. if (!strcmp(str, "MQTTv3")) {
  19. return ACLK_TRP_MQTT_3_1_1;
  20. }
  21. if (!strcmp(str, "MQTTv5")) {
  22. return ACLK_TRP_MQTT_5;
  23. }
  24. return ACLK_TRP_UNKNOWN;
  25. }
  26. void aclk_transport_desc_t_destroy(aclk_transport_desc_t *trp_desc) {
  27. freez(trp_desc->endpoint);
  28. }
  29. void aclk_env_t_destroy(aclk_env_t *env) {
  30. freez(env->auth_endpoint);
  31. if (env->transports) {
  32. for (size_t i = 0; i < env->transport_count; i++) {
  33. if(env->transports[i]) {
  34. aclk_transport_desc_t_destroy(env->transports[i]);
  35. env->transports[i] = NULL;
  36. }
  37. }
  38. freez(env->transports);
  39. }
  40. if (env->capabilities) {
  41. for (size_t i = 0; i < env->capability_count; i++)
  42. freez(env->capabilities[i]);
  43. freez(env->capabilities);
  44. }
  45. }
  46. int aclk_env_has_capa(const char *capa)
  47. {
  48. for (int i = 0; i < (int) aclk_env->capability_count; i++) {
  49. if (!strcasecmp(capa, aclk_env->capabilities[i]))
  50. return 1;
  51. }
  52. return 0;
  53. }
  54. #ifdef ACLK_LOG_CONVERSATION_DIR
  55. volatile int aclk_conversation_log_counter = 0;
  56. #if !defined(HAVE_C___ATOMIC) || defined(NETDATA_NO_ATOMIC_INSTRUCTIONS)
  57. netdata_mutex_t aclk_conversation_log_mutex = NETDATA_MUTEX_INITIALIZER;
  58. int aclk_get_conv_log_next()
  59. {
  60. int ret;
  61. netdata_mutex_lock(&aclk_conversation_log_mutex);
  62. ret = aclk_conversation_log_counter++;
  63. netdata_mutex_unlock(&aclk_conversation_log_mutex);
  64. return ret;
  65. }
  66. #endif
  67. #endif
  68. #define ACLK_TOPIC_PREFIX "/agent/"
  69. struct aclk_topic {
  70. enum aclk_topics topic_id;
  71. // as received from cloud - we keep this for
  72. // eventual topic list update when claim_id changes
  73. char *topic_recvd;
  74. // constructed topic
  75. char *topic;
  76. };
  77. // This helps to cache finalized topics (assembled with claim_id)
  78. // to not have to alloc or create buffer and construct topic every
  79. // time message is sent as in old ACLK
  80. static struct aclk_topic **aclk_topic_cache = NULL;
  81. static size_t aclk_topic_cache_items = 0;
  82. void free_topic_cache(void)
  83. {
  84. if (aclk_topic_cache) {
  85. for (size_t i = 0; i < aclk_topic_cache_items; i++) {
  86. freez(aclk_topic_cache[i]->topic);
  87. freez(aclk_topic_cache[i]->topic_recvd);
  88. freez(aclk_topic_cache[i]);
  89. }
  90. freez(aclk_topic_cache);
  91. aclk_topic_cache = NULL;
  92. aclk_topic_cache_items = 0;
  93. }
  94. }
  95. #define JSON_TOPIC_KEY_TOPIC "topic"
  96. #define JSON_TOPIC_KEY_NAME "name"
  97. struct topic_name {
  98. enum aclk_topics id;
  99. // cloud name - how is it called
  100. // in answer to /password endpoint
  101. const char *name;
  102. } topic_names[] = {
  103. { .id = ACLK_TOPICID_CHART, .name = "chart" },
  104. { .id = ACLK_TOPICID_ALARMS, .name = "alarms" },
  105. { .id = ACLK_TOPICID_METADATA, .name = "meta" },
  106. { .id = ACLK_TOPICID_COMMAND, .name = "inbox-cmd" },
  107. { .id = ACLK_TOPICID_AGENT_CONN, .name = "agent-connection" },
  108. { .id = ACLK_TOPICID_CMD_NG_V1, .name = "inbox-cmd-v1" },
  109. { .id = ACLK_TOPICID_CREATE_NODE, .name = "create-node-instance" },
  110. { .id = ACLK_TOPICID_NODE_CONN, .name = "node-instance-connection" },
  111. { .id = ACLK_TOPICID_CHART_DIMS, .name = "chart-and-dims-updated" },
  112. { .id = ACLK_TOPICID_CHART_CONFIGS_UPDATED, .name = "chart-configs-updated" },
  113. { .id = ACLK_TOPICID_CHART_RESET, .name = "reset-charts" },
  114. { .id = ACLK_TOPICID_RETENTION_UPDATED, .name = "chart-retention-updated" },
  115. { .id = ACLK_TOPICID_NODE_INFO, .name = "node-instance-info" },
  116. { .id = ACLK_TOPICID_ALARM_LOG, .name = "alarm-log" },
  117. { .id = ACLK_TOPICID_ALARM_HEALTH, .name = "alarm-health" },
  118. { .id = ACLK_TOPICID_ALARM_CONFIG, .name = "alarm-config" },
  119. { .id = ACLK_TOPICID_ALARM_SNAPSHOT, .name = "alarm-snapshot" },
  120. { .id = ACLK_TOPICID_UNKNOWN, .name = NULL }
  121. };
  122. enum aclk_topics compulsory_topics_legacy[] = {
  123. ACLK_TOPICID_CHART,
  124. ACLK_TOPICID_ALARMS,
  125. ACLK_TOPICID_METADATA,
  126. ACLK_TOPICID_COMMAND,
  127. ACLK_TOPICID_UNKNOWN
  128. };
  129. enum aclk_topics compulsory_topics_new_cloud_arch[] = {
  130. // TODO remove old topics once not needed anymore
  131. ACLK_TOPICID_CHART,
  132. ACLK_TOPICID_ALARMS,
  133. ACLK_TOPICID_METADATA,
  134. ACLK_TOPICID_COMMAND,
  135. ACLK_TOPICID_AGENT_CONN,
  136. ACLK_TOPICID_CMD_NG_V1,
  137. ACLK_TOPICID_CREATE_NODE,
  138. ACLK_TOPICID_NODE_CONN,
  139. ACLK_TOPICID_CHART_DIMS,
  140. ACLK_TOPICID_CHART_CONFIGS_UPDATED,
  141. ACLK_TOPICID_CHART_RESET,
  142. ACLK_TOPICID_RETENTION_UPDATED,
  143. ACLK_TOPICID_NODE_INFO,
  144. ACLK_TOPICID_ALARM_LOG,
  145. ACLK_TOPICID_ALARM_HEALTH,
  146. ACLK_TOPICID_ALARM_CONFIG,
  147. ACLK_TOPICID_ALARM_SNAPSHOT,
  148. ACLK_TOPICID_UNKNOWN
  149. };
  150. static enum aclk_topics topic_name_to_id(const char *name) {
  151. struct topic_name *topic = topic_names;
  152. while (topic->name) {
  153. if (!strcmp(topic->name, name)) {
  154. return topic->id;
  155. }
  156. topic++;
  157. }
  158. return ACLK_TOPICID_UNKNOWN;
  159. }
  160. static const char *topic_id_to_name(enum aclk_topics tid) {
  161. struct topic_name *topic = topic_names;
  162. while (topic->name) {
  163. if (topic->id == tid)
  164. return topic->name;
  165. topic++;
  166. }
  167. return "unknown";
  168. }
  169. #define CLAIM_ID_REPLACE_TAG "#{claim_id}"
  170. static void topic_generate_final(struct aclk_topic *t) {
  171. char *dest;
  172. char *replace_tag = strstr(t->topic_recvd, CLAIM_ID_REPLACE_TAG);
  173. if (!replace_tag)
  174. return;
  175. rrdhost_aclk_state_lock(localhost);
  176. if (unlikely(!localhost->aclk_state.claimed_id)) {
  177. error("This should never be called if agent not claimed");
  178. rrdhost_aclk_state_unlock(localhost);
  179. return;
  180. }
  181. t->topic = mallocz(strlen(t->topic_recvd) + 1 - strlen(CLAIM_ID_REPLACE_TAG) + strlen(localhost->aclk_state.claimed_id));
  182. memcpy(t->topic, t->topic_recvd, replace_tag - t->topic_recvd);
  183. dest = t->topic + (replace_tag - t->topic_recvd);
  184. memcpy(dest, localhost->aclk_state.claimed_id, strlen(localhost->aclk_state.claimed_id));
  185. dest += strlen(localhost->aclk_state.claimed_id);
  186. rrdhost_aclk_state_unlock(localhost);
  187. replace_tag += strlen(CLAIM_ID_REPLACE_TAG);
  188. strcpy(dest, replace_tag);
  189. dest += strlen(replace_tag);
  190. *dest = 0;
  191. }
  192. static int topic_cache_add_topic(struct json_object *json, struct aclk_topic *topic)
  193. {
  194. struct json_object_iterator it;
  195. struct json_object_iterator itEnd;
  196. it = json_object_iter_begin(json);
  197. itEnd = json_object_iter_end(json);
  198. while (!json_object_iter_equal(&it, &itEnd)) {
  199. if (!strcmp(json_object_iter_peek_name(&it), JSON_TOPIC_KEY_NAME)) {
  200. if (json_object_get_type(json_object_iter_peek_value(&it)) != json_type_string) {
  201. error("topic dictionary key \"" JSON_TOPIC_KEY_NAME "\" is expected to be json_type_string");
  202. return 1;
  203. }
  204. topic->topic_id = topic_name_to_id(json_object_get_string(json_object_iter_peek_value(&it)));
  205. if (topic->topic_id == ACLK_TOPICID_UNKNOWN) {
  206. debug(D_ACLK, "topic dictionary has unknown topic name \"%s\"", json_object_get_string(json_object_iter_peek_value(&it)));
  207. }
  208. json_object_iter_next(&it);
  209. continue;
  210. }
  211. if (!strcmp(json_object_iter_peek_name(&it), JSON_TOPIC_KEY_TOPIC)) {
  212. if (json_object_get_type(json_object_iter_peek_value(&it)) != json_type_string) {
  213. error("topic dictionary key \"" JSON_TOPIC_KEY_TOPIC "\" is expected to be json_type_string");
  214. return 1;
  215. }
  216. topic->topic_recvd = strdupz(json_object_get_string(json_object_iter_peek_value(&it)));
  217. json_object_iter_next(&it);
  218. continue;
  219. }
  220. error("topic dictionary has Unknown/Unexpected key \"%s\" in topic description. Ignoring!", json_object_iter_peek_name(&it));
  221. json_object_iter_next(&it);
  222. }
  223. if (!topic->topic_recvd) {
  224. error("topic dictionary Missig compulsory key %s", JSON_TOPIC_KEY_TOPIC);
  225. return 1;
  226. }
  227. topic_generate_final(topic);
  228. aclk_topic_cache_items++;
  229. return 0;
  230. }
  231. int aclk_generate_topic_cache(struct json_object *json)
  232. {
  233. json_object *obj;
  234. size_t array_size = json_object_array_length(json);
  235. if (!array_size) {
  236. error("Empty topic list!");
  237. return 1;
  238. }
  239. if (aclk_topic_cache)
  240. free_topic_cache();
  241. aclk_topic_cache = callocz(array_size, sizeof(struct aclk_topic *));
  242. for (size_t i = 0; i < array_size; i++) {
  243. obj = json_object_array_get_idx(json, i);
  244. if (json_object_get_type(obj) != json_type_object) {
  245. error("expected json_type_object");
  246. return 1;
  247. }
  248. aclk_topic_cache[i] = callocz(1, sizeof(struct aclk_topic));
  249. if (topic_cache_add_topic(obj, aclk_topic_cache[i])) {
  250. error("failed to parse topic @idx=%d", (int)i);
  251. return 1;
  252. }
  253. }
  254. enum aclk_topics *compulsory_topics = aclk_use_new_cloud_arch ? compulsory_topics_new_cloud_arch : compulsory_topics_legacy;
  255. for (int i = 0; compulsory_topics[i] != ACLK_TOPICID_UNKNOWN; i++) {
  256. if (!aclk_get_topic(compulsory_topics[i])) {
  257. error("missing compulsory topic \"%s\" in password response from cloud", topic_id_to_name(compulsory_topics[i]));
  258. return 1;
  259. }
  260. }
  261. return 0;
  262. }
  263. /*
  264. * Build a topic based on sub_topic and final_topic
  265. * if the sub topic starts with / assume that is an absolute topic
  266. *
  267. */
  268. const char *aclk_get_topic(enum aclk_topics topic)
  269. {
  270. if (!aclk_topic_cache) {
  271. error("Topic cache not initialized");
  272. return NULL;
  273. }
  274. for (size_t i = 0; i < aclk_topic_cache_items; i++) {
  275. if (aclk_topic_cache[i]->topic_id == topic)
  276. return aclk_topic_cache[i]->topic;
  277. }
  278. error("Unknown topic");
  279. return NULL;
  280. }
  281. /*
  282. * TBEB with randomness
  283. *
  284. * @param reset 1 - to reset the delay,
  285. * 0 - to advance a step and calculate sleep time in ms
  286. * @param min, max in seconds
  287. * @returns delay in ms
  288. *
  289. */
  290. unsigned long int aclk_tbeb_delay(int reset, int base, unsigned long int min, unsigned long int max) {
  291. static int attempt = -1;
  292. if (reset) {
  293. attempt = -1;
  294. return 0;
  295. }
  296. attempt++;
  297. if (attempt == 0) {
  298. srandom(time(NULL));
  299. return 0;
  300. }
  301. unsigned long int delay = pow(base, attempt - 1);
  302. delay *= MSEC_PER_SEC;
  303. delay += (random() % (MAX(1000, delay/2)));
  304. if (delay <= min * MSEC_PER_SEC)
  305. return min;
  306. if (delay >= max * MSEC_PER_SEC)
  307. return max;
  308. return delay;
  309. }
  310. #define HTTP_PROXY_PREFIX "http://"
  311. void aclk_set_proxy(char **ohost, int *port, enum mqtt_wss_proxy_type *type)
  312. {
  313. ACLK_PROXY_TYPE pt;
  314. const char *ptr = aclk_get_proxy(&pt);
  315. char *tmp;
  316. char *host;
  317. if (pt != PROXY_TYPE_HTTP)
  318. return;
  319. *port = 0;
  320. if (!strncmp(ptr, HTTP_PROXY_PREFIX, strlen(HTTP_PROXY_PREFIX)))
  321. ptr += strlen(HTTP_PROXY_PREFIX);
  322. if ((tmp = strchr(ptr, '@')))
  323. ptr = tmp;
  324. if ((tmp = strchr(ptr, '/'))) {
  325. host = mallocz((tmp - ptr) + 1);
  326. memcpy(host, ptr, (tmp - ptr));
  327. host[tmp - ptr] = 0;
  328. } else
  329. host = strdupz(ptr);
  330. if ((tmp = strchr(host, ':'))) {
  331. *tmp = 0;
  332. tmp++;
  333. *port = atoi(tmp);
  334. }
  335. if (*port <= 0 || *port > 65535)
  336. *port = 8080;
  337. *ohost = host;
  338. if (type)
  339. *type = MQTT_WSS_PROXY_HTTP;
  340. }