dyncfg.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "../libnetdata.h"
  3. // ----------------------------------------------------------------------------
  4. static struct {
  5. DYNCFG_TYPE type;
  6. const char *name;
  7. } dyncfg_types[] = {
  8. { .type = DYNCFG_TYPE_SINGLE, .name = "single" },
  9. { .type = DYNCFG_TYPE_TEMPLATE, .name = "template" },
  10. { .type = DYNCFG_TYPE_JOB, .name = "job" },
  11. };
  12. DYNCFG_TYPE dyncfg_type2id(const char *type) {
  13. if(!type || !*type)
  14. return DYNCFG_TYPE_SINGLE;
  15. size_t entries = sizeof(dyncfg_types) / sizeof(dyncfg_types[0]);
  16. for(size_t i = 0; i < entries ;i++) {
  17. if(strcmp(dyncfg_types[i].name, type) == 0)
  18. return dyncfg_types[i].type;
  19. }
  20. return DYNCFG_TYPE_SINGLE;
  21. }
  22. const char *dyncfg_id2type(DYNCFG_TYPE type) {
  23. size_t entries = sizeof(dyncfg_types) / sizeof(dyncfg_types[0]);
  24. for(size_t i = 0; i < entries ;i++) {
  25. if(type == dyncfg_types[i].type)
  26. return dyncfg_types[i].name;
  27. }
  28. return "single";
  29. }
  30. // ----------------------------------------------------------------------------
  31. static struct {
  32. DYNCFG_SOURCE_TYPE source_type;
  33. const char *name;
  34. } dyncfg_source_types[] = {
  35. { .source_type = DYNCFG_SOURCE_TYPE_INTERNAL, .name = "internal" },
  36. { .source_type = DYNCFG_SOURCE_TYPE_STOCK, .name = "stock" },
  37. { .source_type = DYNCFG_SOURCE_TYPE_USER, .name = "user" },
  38. { .source_type = DYNCFG_SOURCE_TYPE_DYNCFG, .name = "dyncfg" },
  39. { .source_type = DYNCFG_SOURCE_TYPE_DISCOVERED, .name = "discovered" },
  40. };
  41. DYNCFG_SOURCE_TYPE dyncfg_source_type2id(const char *source_type) {
  42. if(!source_type || !*source_type)
  43. return DYNCFG_SOURCE_TYPE_INTERNAL;
  44. size_t entries = sizeof(dyncfg_source_types) / sizeof(dyncfg_source_types[0]);
  45. for(size_t i = 0; i < entries ;i++) {
  46. if(strcmp(dyncfg_source_types[i].name, source_type) == 0)
  47. return dyncfg_source_types[i].source_type;
  48. }
  49. return DYNCFG_SOURCE_TYPE_INTERNAL;
  50. }
  51. const char *dyncfg_id2source_type(DYNCFG_SOURCE_TYPE source_type) {
  52. size_t entries = sizeof(dyncfg_source_types) / sizeof(dyncfg_source_types[0]);
  53. for(size_t i = 0; i < entries ;i++) {
  54. if(source_type == dyncfg_source_types[i].source_type)
  55. return dyncfg_source_types[i].name;
  56. }
  57. return "internal";
  58. }
  59. // ----------------------------------------------------------------------------
  60. static struct {
  61. DYNCFG_STATUS status;
  62. const char *name;
  63. } dyncfg_statuses[] = {
  64. { .status = DYNCFG_STATUS_NONE, .name = "none" },
  65. { .status = DYNCFG_STATUS_ACCEPTED, .name = "accepted" },
  66. { .status = DYNCFG_STATUS_RUNNING, .name = "running" },
  67. { .status = DYNCFG_STATUS_FAILED, .name = "failed" },
  68. { .status = DYNCFG_STATUS_DISABLED, .name = "disabled" },
  69. { .status = DYNCFG_STATUS_ORPHAN, .name = "orphan" },
  70. { .status = DYNCFG_STATUS_INCOMPLETE, .name = "incomplete" },
  71. };
  72. DYNCFG_STATUS dyncfg_status2id(const char *status) {
  73. if(!status || !*status)
  74. return DYNCFG_STATUS_NONE;
  75. size_t entries = sizeof(dyncfg_statuses) / sizeof(dyncfg_statuses[0]);
  76. for(size_t i = 0; i < entries ;i++) {
  77. if(strcmp(dyncfg_statuses[i].name, status) == 0)
  78. return dyncfg_statuses[i].status;
  79. }
  80. return DYNCFG_STATUS_NONE;
  81. }
  82. const char *dyncfg_id2status(DYNCFG_STATUS status) {
  83. size_t entries = sizeof(dyncfg_statuses) / sizeof(dyncfg_statuses[0]);
  84. for(size_t i = 0; i < entries ;i++) {
  85. if(status == dyncfg_statuses[i].status)
  86. return dyncfg_statuses[i].name;
  87. }
  88. return "none";
  89. }
  90. // ----------------------------------------------------------------------------
  91. static struct {
  92. DYNCFG_CMDS cmd;
  93. const char *name;
  94. } cmd_map[] = {
  95. { .cmd = DYNCFG_CMD_GET, .name = "get" },
  96. { .cmd = DYNCFG_CMD_SCHEMA, .name = "schema" },
  97. { .cmd = DYNCFG_CMD_UPDATE, .name = "update" },
  98. { .cmd = DYNCFG_CMD_ADD, .name = "add" },
  99. { .cmd = DYNCFG_CMD_TEST, .name = "test" },
  100. { .cmd = DYNCFG_CMD_REMOVE, .name = "remove" },
  101. { .cmd = DYNCFG_CMD_ENABLE, .name = "enable" },
  102. { .cmd = DYNCFG_CMD_DISABLE, .name = "disable" },
  103. { .cmd = DYNCFG_CMD_RESTART, .name = "restart" }
  104. };
  105. const char *dyncfg_id2cmd_one(DYNCFG_CMDS cmd) {
  106. for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
  107. if(cmd == cmd_map[i].cmd)
  108. return cmd_map[i].name;
  109. }
  110. return NULL;
  111. }
  112. DYNCFG_CMDS dyncfg_cmds2id(const char *cmds) {
  113. if(!cmds || !*cmds)
  114. return DYNCFG_CMD_NONE;
  115. DYNCFG_CMDS result = DYNCFG_CMD_NONE;
  116. const char *p = cmds;
  117. size_t len, i;
  118. while (*p) {
  119. // Skip any leading spaces
  120. while (*p == ' ') p++;
  121. // Find the end of the current word
  122. const char *end = p;
  123. while (*end && *end != ' ') end++;
  124. len = end - p;
  125. // Compare with known commands
  126. for (i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
  127. if (strncmp(p, cmd_map[i].name, len) == 0 && cmd_map[i].name[len] == '\0') {
  128. result |= cmd_map[i].cmd;
  129. break;
  130. }
  131. }
  132. // Move to the next word
  133. p = end;
  134. }
  135. return result;
  136. }
  137. void dyncfg_cmds2fp(DYNCFG_CMDS cmds, FILE *fp) {
  138. for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
  139. if(cmds & cmd_map[i].cmd)
  140. fprintf(fp, "%s ", cmd_map[i].name);
  141. }
  142. }
  143. void dyncfg_cmds2json_array(DYNCFG_CMDS cmds, const char *key, BUFFER *wb) {
  144. buffer_json_member_add_array(wb, key);
  145. for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
  146. if(cmds & cmd_map[i].cmd)
  147. buffer_json_add_array_item_string(wb, cmd_map[i].name);
  148. }
  149. buffer_json_array_close(wb);
  150. }
  151. void dyncfg_cmds2buffer(DYNCFG_CMDS cmds, BUFFER *wb) {
  152. size_t added = 0;
  153. for (size_t i = 0; i < sizeof(cmd_map) / sizeof(cmd_map[0]); i++) {
  154. if(cmds & cmd_map[i].cmd) {
  155. if(added)
  156. buffer_fast_strcat(wb, " ", 1);
  157. buffer_strcat(wb, cmd_map[i].name);
  158. added++;
  159. }
  160. }
  161. }
  162. // ----------------------------------------------------------------------------
  163. bool dyncfg_is_valid_id(const char *id) {
  164. const char *s = id;
  165. while(*s) {
  166. if(isspace(*s) || *s == '\'') return false;
  167. s++;
  168. }
  169. return true;
  170. }
  171. char *dyncfg_escape_id_for_filename(const char *id) {
  172. if (id == NULL) return NULL;
  173. // Allocate memory for the worst case, where every character is escaped.
  174. char *escaped = mallocz(strlen(id) * 3 + 1); // Each char can become '%XX', plus '\0'
  175. if (!escaped) return NULL;
  176. const char *src = id;
  177. char *dest = escaped;
  178. while (*src) {
  179. if (*src == '/' || isspace(*src) || !isprint(*src)) {
  180. sprintf(dest, "%%%02X", (unsigned char)*src);
  181. dest += 3;
  182. } else {
  183. *dest++ = *src;
  184. }
  185. src++;
  186. }
  187. *dest = '\0';
  188. return escaped;
  189. }
  190. // ----------------------------------------------------------------------------
  191. int dyncfg_default_response(BUFFER *wb, int code, const char *msg) {
  192. buffer_flush(wb);
  193. wb->content_type = CT_APPLICATION_JSON;
  194. wb->expires = now_realtime_sec();
  195. buffer_json_initialize(wb, "\"", "\"", 0, true, BUFFER_JSON_OPTIONS_MINIFY);
  196. buffer_json_member_add_uint64(wb, "status", code);
  197. buffer_json_member_add_string(wb, "message", msg);
  198. buffer_json_finalize(wb);
  199. return code;
  200. }
  201. int dyncfg_node_find_and_call(DICTIONARY *dyncfg_nodes, const char *transaction, const char *function,
  202. usec_t *stop_monotonic_ut, bool *cancelled,
  203. BUFFER *payload, const char *source, BUFFER *result) {
  204. if(!function || !*function)
  205. return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "command received is empty");
  206. char buf[strlen(function) + 1];
  207. memcpy(buf, function, sizeof(buf));
  208. char *words[MAX_FUNCTION_PARAMETERS]; // an array of pointers for the words in this line
  209. size_t num_words = quoted_strings_splitter_pluginsd(buf, words, MAX_FUNCTION_PARAMETERS);
  210. const char *id = get_word(words, num_words, 1);
  211. const char *action = get_word(words, num_words, 2);
  212. if(!id || !*id)
  213. return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "dyncfg node: id is missing from the request");
  214. if(!action || !*action)
  215. return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "dyncfg node: action is missing from the request");
  216. DYNCFG_CMDS cmd = dyncfg_cmds2id(action);
  217. if(cmd == DYNCFG_CMD_NONE)
  218. return dyncfg_default_response(result, HTTP_RESP_BAD_REQUEST, "dyncfg node: action given in request is unknown");
  219. const DICTIONARY_ITEM *item = dictionary_get_and_acquire_item(dyncfg_nodes, id);
  220. if(!item)
  221. return dyncfg_default_response(result, HTTP_RESP_NOT_FOUND, "dyncfg node: id is not found");
  222. struct dyncfg_node *df = dictionary_acquired_item_value(item);
  223. buffer_flush(result);
  224. result->content_type = CT_APPLICATION_JSON;
  225. int code = df->cb(transaction, id, cmd, payload, stop_monotonic_ut, cancelled, result, source, df->data);
  226. if(!result->expires)
  227. result->expires = now_realtime_sec();
  228. if(!buffer_tostring(result))
  229. dyncfg_default_response(result, code, "");
  230. dictionary_acquired_item_release(dyncfg_nodes, item);
  231. return code;
  232. }