functions_evloop.h 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #ifndef NETDATA_FUNCTIONS_EVLOOP_H
  3. #define NETDATA_FUNCTIONS_EVLOOP_H
  4. #include "../libnetdata.h"
  5. #define MAX_FUNCTION_PARAMETERS 1024
  6. #define PLUGINSD_KEYWORD_CHART "CHART"
  7. #define PLUGINSD_KEYWORD_CHART_DEFINITION_END "CHART_DEFINITION_END"
  8. #define PLUGINSD_KEYWORD_DIMENSION "DIMENSION"
  9. #define PLUGINSD_KEYWORD_BEGIN "BEGIN"
  10. #define PLUGINSD_KEYWORD_SET "SET"
  11. #define PLUGINSD_KEYWORD_END "END"
  12. #define PLUGINSD_KEYWORD_FLUSH "FLUSH"
  13. #define PLUGINSD_KEYWORD_DISABLE "DISABLE"
  14. #define PLUGINSD_KEYWORD_VARIABLE "VARIABLE"
  15. #define PLUGINSD_KEYWORD_LABEL "LABEL"
  16. #define PLUGINSD_KEYWORD_OVERWRITE "OVERWRITE"
  17. #define PLUGINSD_KEYWORD_CLABEL "CLABEL"
  18. #define PLUGINSD_KEYWORD_CLABEL_COMMIT "CLABEL_COMMIT"
  19. #define PLUGINSD_KEYWORD_FUNCTION "FUNCTION"
  20. #define PLUGINSD_KEYWORD_FUNCTION_PAYLOAD "FUNCTION_PAYLOAD"
  21. #define PLUGINSD_KEYWORD_FUNCTION_PAYLOAD_END "FUNCTION_PAYLOAD_END"
  22. #define PLUGINSD_KEYWORD_FUNCTION_CANCEL "FUNCTION_CANCEL"
  23. #define PLUGINSD_KEYWORD_FUNCTION_PROGRESS "FUNCTION_PROGRESS"
  24. #define PLUGINSD_KEYWORD_FUNCTION_RESULT_BEGIN "FUNCTION_RESULT_BEGIN"
  25. #define PLUGINSD_KEYWORD_FUNCTION_RESULT_END "FUNCTION_RESULT_END"
  26. #define PLUGINSD_KEYWORD_CONFIG "CONFIG"
  27. #define PLUGINSD_KEYWORD_CONFIG_ACTION_CREATE "create"
  28. #define PLUGINSD_KEYWORD_CONFIG_ACTION_DELETE "delete"
  29. #define PLUGINSD_KEYWORD_CONFIG_ACTION_STATUS "status"
  30. #define PLUGINSD_FUNCTION_CONFIG "config"
  31. #define PLUGINSD_KEYWORD_REPLAY_CHART "REPLAY_CHART"
  32. #define PLUGINSD_KEYWORD_REPLAY_BEGIN "RBEGIN"
  33. #define PLUGINSD_KEYWORD_REPLAY_SET "RSET"
  34. #define PLUGINSD_KEYWORD_REPLAY_RRDDIM_STATE "RDSTATE"
  35. #define PLUGINSD_KEYWORD_REPLAY_RRDSET_STATE "RSSTATE"
  36. #define PLUGINSD_KEYWORD_REPLAY_END "REND"
  37. #define PLUGINSD_KEYWORD_BEGIN_V2 "BEGIN2"
  38. #define PLUGINSD_KEYWORD_SET_V2 "SET2"
  39. #define PLUGINSD_KEYWORD_END_V2 "END2"
  40. #define PLUGINSD_KEYWORD_HOST_DEFINE "HOST_DEFINE"
  41. #define PLUGINSD_KEYWORD_HOST_DEFINE_END "HOST_DEFINE_END"
  42. #define PLUGINSD_KEYWORD_HOST_LABEL "HOST_LABEL"
  43. #define PLUGINSD_KEYWORD_HOST "HOST"
  44. #define PLUGINSD_KEYWORD_EXIT "EXIT"
  45. #define PLUGINSD_KEYWORD_SLOT "SLOT" // to change the length of this, update pluginsd_extract_chart_slot() too
  46. #define PLUGINS_FUNCTIONS_TIMEOUT_DEFAULT 10 // seconds
  47. typedef void (*functions_evloop_worker_execute_t)(const char *transaction, char *function, usec_t *stop_monotonic_ut, bool *cancelled, BUFFER *payload, const char *source, void *data);
  48. struct functions_evloop_worker_job;
  49. struct functions_evloop_globals *functions_evloop_init(size_t worker_threads, const char *tag, netdata_mutex_t *stdout_mutex, bool *plugin_should_exit);
  50. void functions_evloop_add_function(struct functions_evloop_globals *wg, const char *function, functions_evloop_worker_execute_t cb, time_t default_timeout, void *data);
  51. void functions_evloop_cancel_threads(struct functions_evloop_globals *wg);
  52. #define FUNCTIONS_EXTENDED_TIME_ON_PROGRESS_UT (10 * USEC_PER_SEC)
  53. static inline void functions_stop_monotonic_update_on_progress(usec_t *stop_monotonic_ut) {
  54. usec_t now_ut = now_monotonic_usec();
  55. if(now_ut + FUNCTIONS_EXTENDED_TIME_ON_PROGRESS_UT > *stop_monotonic_ut) {
  56. nd_log(NDLS_DAEMON, NDLP_DEBUG, "Extending function timeout due to PROGRESS update...");
  57. __atomic_store_n(stop_monotonic_ut, now_ut + FUNCTIONS_EXTENDED_TIME_ON_PROGRESS_UT, __ATOMIC_RELAXED);
  58. }
  59. else
  60. nd_log(NDLS_DAEMON, NDLP_DEBUG, "Received PROGRESS update...");
  61. }
  62. #define pluginsd_function_result_begin_to_buffer(wb, transaction, code, content_type, expires) \
  63. buffer_sprintf(wb \
  64. , PLUGINSD_KEYWORD_FUNCTION_RESULT_BEGIN " \"%s\" %d \"%s\" %ld\n" \
  65. , (transaction) ? (transaction) : "" \
  66. , (int)(code) \
  67. , (content_type) ? (content_type) : "" \
  68. , (long int)(expires) \
  69. )
  70. #define pluginsd_function_result_end_to_buffer(wb) \
  71. buffer_strcat(wb, "\n" PLUGINSD_KEYWORD_FUNCTION_RESULT_END "\n")
  72. #define pluginsd_function_result_begin_to_stdout(transaction, code, content_type, expires) \
  73. fprintf(stdout \
  74. , PLUGINSD_KEYWORD_FUNCTION_RESULT_BEGIN " \"%s\" %d \"%s\" %ld\n" \
  75. , (transaction) ? (transaction) : "" \
  76. , (int)(code) \
  77. , (content_type) ? (content_type) : "" \
  78. , (long int)(expires) \
  79. )
  80. #define pluginsd_function_result_end_to_stdout() \
  81. fprintf(stdout, "\n" PLUGINSD_KEYWORD_FUNCTION_RESULT_END "\n")
  82. static inline void pluginsd_function_json_error_to_stdout(const char *transaction, int code, const char *msg) {
  83. char buffer[PLUGINSD_LINE_MAX + 1];
  84. json_escape_string(buffer, msg, PLUGINSD_LINE_MAX);
  85. pluginsd_function_result_begin_to_stdout(transaction, code, "application/json", now_realtime_sec());
  86. fprintf(stdout, "{\"status\":%d,\"error_message\":\"%s\"}", code, buffer);
  87. pluginsd_function_result_end_to_stdout();
  88. fflush(stdout);
  89. }
  90. static inline void pluginsd_function_result_to_stdout(const char *transaction, int code, const char *content_type, time_t expires, BUFFER *result) {
  91. pluginsd_function_result_begin_to_stdout(transaction, code, content_type, expires);
  92. fwrite(buffer_tostring(result), buffer_strlen(result), 1, stdout);
  93. pluginsd_function_result_end_to_stdout();
  94. fflush(stdout);
  95. }
  96. static inline void pluginsd_function_progress_to_stdout(const char *transaction, size_t done, size_t all) {
  97. fprintf(stdout, PLUGINSD_KEYWORD_FUNCTION_PROGRESS " '%s' %zu %zu\n",
  98. transaction, done, all);
  99. fflush(stdout);
  100. }
  101. void functions_evloop_dyncfg_add(struct functions_evloop_globals *wg, const char *id, const char *path, DYNCFG_STATUS status, DYNCFG_TYPE type, DYNCFG_SOURCE_TYPE source_type, const char *source, DYNCFG_CMDS cmds, dyncfg_cb_t cb, void *data);
  102. void functions_evloop_dyncfg_del(struct functions_evloop_globals *wg, const char *id);
  103. void functions_evloop_dyncfg_status(struct functions_evloop_globals *wg, const char *id, DYNCFG_STATUS status);
  104. #endif //NETDATA_FUNCTIONS_EVLOOP_H