analytics.c 39 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "common.h"
  3. #include "buildinfo.h"
  4. struct analytics_data analytics_data;
  5. extern void analytics_exporting_connectors (BUFFER *b);
  6. extern void analytics_exporting_connectors_ssl (BUFFER *b);
  7. extern void analytics_build_info (BUFFER *b);
  8. extern int aclk_connected;
  9. struct collector {
  10. const char *plugin;
  11. const char *module;
  12. };
  13. struct array_printer {
  14. int c;
  15. BUFFER *both;
  16. };
  17. /*
  18. * Debug logging
  19. */
  20. void analytics_log_data(void)
  21. {
  22. debug(D_ANALYTICS, "NETDATA_CONFIG_STREAM_ENABLED : [%s]", analytics_data.netdata_config_stream_enabled);
  23. debug(D_ANALYTICS, "NETDATA_CONFIG_MEMORY_MODE : [%s]", analytics_data.netdata_config_memory_mode);
  24. debug(D_ANALYTICS, "NETDATA_CONFIG_EXPORTING_ENABLED : [%s]", analytics_data.netdata_config_exporting_enabled);
  25. debug(D_ANALYTICS, "NETDATA_EXPORTING_CONNECTORS : [%s]", analytics_data.netdata_exporting_connectors);
  26. debug(D_ANALYTICS, "NETDATA_ALLMETRICS_PROMETHEUS_USED : [%s]", analytics_data.netdata_allmetrics_prometheus_used);
  27. debug(D_ANALYTICS, "NETDATA_ALLMETRICS_SHELL_USED : [%s]", analytics_data.netdata_allmetrics_shell_used);
  28. debug(D_ANALYTICS, "NETDATA_ALLMETRICS_JSON_USED : [%s]", analytics_data.netdata_allmetrics_json_used);
  29. debug(D_ANALYTICS, "NETDATA_DASHBOARD_USED : [%s]", analytics_data.netdata_dashboard_used);
  30. debug(D_ANALYTICS, "NETDATA_COLLECTORS : [%s]", analytics_data.netdata_collectors);
  31. debug(D_ANALYTICS, "NETDATA_COLLECTORS_COUNT : [%s]", analytics_data.netdata_collectors_count);
  32. debug(D_ANALYTICS, "NETDATA_BUILDINFO : [%s]", analytics_data.netdata_buildinfo);
  33. debug(D_ANALYTICS, "NETDATA_CONFIG_PAGE_CACHE_SIZE : [%s]", analytics_data.netdata_config_page_cache_size);
  34. debug(D_ANALYTICS, "NETDATA_CONFIG_MULTIDB_DISK_QUOTA : [%s]", analytics_data.netdata_config_multidb_disk_quota);
  35. debug(D_ANALYTICS, "NETDATA_CONFIG_HTTPS_ENABLED : [%s]", analytics_data.netdata_config_https_enabled);
  36. debug(D_ANALYTICS, "NETDATA_CONFIG_WEB_ENABLED : [%s]", analytics_data.netdata_config_web_enabled);
  37. debug(D_ANALYTICS, "NETDATA_CONFIG_RELEASE_CHANNEL : [%s]", analytics_data.netdata_config_release_channel);
  38. debug(D_ANALYTICS, "NETDATA_MIRRORED_HOST_COUNT : [%s]", analytics_data.netdata_mirrored_host_count);
  39. debug(D_ANALYTICS, "NETDATA_MIRRORED_HOSTS_REACHABLE : [%s]", analytics_data.netdata_mirrored_hosts_reachable);
  40. debug(D_ANALYTICS, "NETDATA_MIRRORED_HOSTS_UNREACHABLE : [%s]", analytics_data.netdata_mirrored_hosts_unreachable);
  41. debug(D_ANALYTICS, "NETDATA_NOTIFICATION_METHODS : [%s]", analytics_data.netdata_notification_methods);
  42. debug(D_ANALYTICS, "NETDATA_ALARMS_NORMAL : [%s]", analytics_data.netdata_alarms_normal);
  43. debug(D_ANALYTICS, "NETDATA_ALARMS_WARNING : [%s]", analytics_data.netdata_alarms_warning);
  44. debug(D_ANALYTICS, "NETDATA_ALARMS_CRITICAL : [%s]", analytics_data.netdata_alarms_critical);
  45. debug(D_ANALYTICS, "NETDATA_CHARTS_COUNT : [%s]", analytics_data.netdata_charts_count);
  46. debug(D_ANALYTICS, "NETDATA_METRICS_COUNT : [%s]", analytics_data.netdata_metrics_count);
  47. debug(D_ANALYTICS, "NETDATA_CONFIG_IS_PARENT : [%s]", analytics_data.netdata_config_is_parent);
  48. debug(D_ANALYTICS, "NETDATA_CONFIG_HOSTS_AVAILABLE : [%s]", analytics_data.netdata_config_hosts_available);
  49. debug(D_ANALYTICS, "NETDATA_HOST_CLOUD_AVAILABLE : [%s]", analytics_data.netdata_host_cloud_available);
  50. debug(D_ANALYTICS, "NETDATA_HOST_ACLK_AVAILABLE : [%s]", analytics_data.netdata_host_aclk_available);
  51. debug(D_ANALYTICS, "NETDATA_HOST_ACLK_PROTOCOL : [%s]", analytics_data.netdata_host_aclk_protocol);
  52. debug(D_ANALYTICS, "NETDATA_HOST_ACLK_IMPLEMENTATION : [%s]", analytics_data.netdata_host_aclk_implementation);
  53. debug(D_ANALYTICS, "NETDATA_HOST_AGENT_CLAIMED : [%s]", analytics_data.netdata_host_agent_claimed);
  54. debug(D_ANALYTICS, "NETDATA_HOST_CLOUD_ENABLED : [%s]", analytics_data.netdata_host_cloud_enabled);
  55. debug(D_ANALYTICS, "NETDATA_CONFIG_HTTPS_AVAILABLE : [%s]", analytics_data.netdata_config_https_available);
  56. debug(D_ANALYTICS, "NETDATA_INSTALL_TYPE : [%s]", analytics_data.netdata_install_type);
  57. debug(D_ANALYTICS, "NETDATA_PREBUILT_DISTRO : [%s]", analytics_data.netdata_prebuilt_distro);
  58. debug(D_ANALYTICS, "NETDATA_CONFIG_IS_PRIVATE_REGISTRY : [%s]", analytics_data.netdata_config_is_private_registry);
  59. debug(D_ANALYTICS, "NETDATA_CONFIG_USE_PRIVATE_REGISTRY: [%s]", analytics_data.netdata_config_use_private_registry);
  60. debug(D_ANALYTICS, "NETDATA_CONFIG_OOM_SCORE : [%s]", analytics_data.netdata_config_oom_score);
  61. }
  62. /*
  63. * Free data
  64. */
  65. void analytics_free_data(void)
  66. {
  67. freez(analytics_data.netdata_config_stream_enabled);
  68. freez(analytics_data.netdata_config_memory_mode);
  69. freez(analytics_data.netdata_config_exporting_enabled);
  70. freez(analytics_data.netdata_exporting_connectors);
  71. freez(analytics_data.netdata_allmetrics_prometheus_used);
  72. freez(analytics_data.netdata_allmetrics_shell_used);
  73. freez(analytics_data.netdata_allmetrics_json_used);
  74. freez(analytics_data.netdata_dashboard_used);
  75. freez(analytics_data.netdata_collectors);
  76. freez(analytics_data.netdata_collectors_count);
  77. freez(analytics_data.netdata_buildinfo);
  78. freez(analytics_data.netdata_config_page_cache_size);
  79. freez(analytics_data.netdata_config_multidb_disk_quota);
  80. freez(analytics_data.netdata_config_https_enabled);
  81. freez(analytics_data.netdata_config_web_enabled);
  82. freez(analytics_data.netdata_config_release_channel);
  83. freez(analytics_data.netdata_mirrored_host_count);
  84. freez(analytics_data.netdata_mirrored_hosts_reachable);
  85. freez(analytics_data.netdata_mirrored_hosts_unreachable);
  86. freez(analytics_data.netdata_notification_methods);
  87. freez(analytics_data.netdata_alarms_normal);
  88. freez(analytics_data.netdata_alarms_warning);
  89. freez(analytics_data.netdata_alarms_critical);
  90. freez(analytics_data.netdata_charts_count);
  91. freez(analytics_data.netdata_metrics_count);
  92. freez(analytics_data.netdata_config_is_parent);
  93. freez(analytics_data.netdata_config_hosts_available);
  94. freez(analytics_data.netdata_host_cloud_available);
  95. freez(analytics_data.netdata_host_aclk_available);
  96. freez(analytics_data.netdata_host_aclk_protocol);
  97. freez(analytics_data.netdata_host_aclk_implementation);
  98. freez(analytics_data.netdata_host_agent_claimed);
  99. freez(analytics_data.netdata_host_cloud_enabled);
  100. freez(analytics_data.netdata_config_https_available);
  101. freez(analytics_data.netdata_install_type);
  102. freez(analytics_data.netdata_config_is_private_registry);
  103. freez(analytics_data.netdata_config_use_private_registry);
  104. freez(analytics_data.netdata_config_oom_score);
  105. freez(analytics_data.netdata_prebuilt_distro);
  106. }
  107. /*
  108. * Set a numeric/boolean data with a value
  109. */
  110. void analytics_set_data(char **name, char *value)
  111. {
  112. if (*name) {
  113. analytics_data.data_length -= strlen(*name);
  114. freez(*name);
  115. }
  116. *name = strdupz(value);
  117. analytics_data.data_length += strlen(*name);
  118. }
  119. /*
  120. * Set a string data with a value
  121. */
  122. void analytics_set_data_str(char **name, char *value)
  123. {
  124. size_t value_string_len;
  125. if (*name) {
  126. analytics_data.data_length -= strlen(*name);
  127. freez(*name);
  128. }
  129. value_string_len = strlen(value) + 4;
  130. *name = mallocz(sizeof(char) * value_string_len);
  131. snprintfz(*name, value_string_len - 1, "\"%s\"", value);
  132. analytics_data.data_length += strlen(*name);
  133. }
  134. /*
  135. * Log hits on the allmetrics page, with prometheus parameter
  136. */
  137. void analytics_log_prometheus(void)
  138. {
  139. if (netdata_anonymous_statistics_enabled == 1 && likely(analytics_data.prometheus_hits < ANALYTICS_MAX_PROMETHEUS_HITS)) {
  140. analytics_data.prometheus_hits++;
  141. char b[21];
  142. snprintfz(b, 20, "%zu", analytics_data.prometheus_hits);
  143. analytics_set_data(&analytics_data.netdata_allmetrics_prometheus_used, b);
  144. }
  145. }
  146. /*
  147. * Log hits on the allmetrics page, with shell parameter (or default)
  148. */
  149. void analytics_log_shell(void)
  150. {
  151. if (netdata_anonymous_statistics_enabled == 1 && likely(analytics_data.shell_hits < ANALYTICS_MAX_SHELL_HITS)) {
  152. analytics_data.shell_hits++;
  153. char b[21];
  154. snprintfz(b, 20, "%zu", analytics_data.shell_hits);
  155. analytics_set_data(&analytics_data.netdata_allmetrics_shell_used, b);
  156. }
  157. }
  158. /*
  159. * Log hits on the allmetrics page, with json parameter
  160. */
  161. void analytics_log_json(void)
  162. {
  163. if (netdata_anonymous_statistics_enabled == 1 && likely(analytics_data.json_hits < ANALYTICS_MAX_JSON_HITS)) {
  164. analytics_data.json_hits++;
  165. char b[21];
  166. snprintfz(b, 20, "%zu", analytics_data.json_hits);
  167. analytics_set_data(&analytics_data.netdata_allmetrics_json_used, b);
  168. }
  169. }
  170. /*
  171. * Log hits on the dashboard, (when calling HELLO).
  172. */
  173. void analytics_log_dashboard(void)
  174. {
  175. if (netdata_anonymous_statistics_enabled == 1 && likely(analytics_data.dashboard_hits < ANALYTICS_MAX_DASHBOARD_HITS)) {
  176. analytics_data.dashboard_hits++;
  177. char b[21];
  178. snprintfz(b, 20, "%zu", analytics_data.dashboard_hits);
  179. analytics_set_data(&analytics_data.netdata_dashboard_used, b);
  180. }
  181. }
  182. /*
  183. * Called when setting the oom score
  184. */
  185. void analytics_report_oom_score(long long int score){
  186. char b[21];
  187. snprintfz(b, 20, "%lld", score);
  188. analytics_set_data(&analytics_data.netdata_config_oom_score, b);
  189. }
  190. void analytics_mirrored_hosts(void)
  191. {
  192. RRDHOST *host;
  193. size_t count = 0;
  194. size_t reachable = 0;
  195. size_t unreachable = 0;
  196. char b[21];
  197. rrd_rdlock();
  198. rrdhost_foreach_read(host)
  199. {
  200. if (rrdhost_flag_check(host, RRDHOST_FLAG_ARCHIVED))
  201. continue;
  202. ((host == localhost || !rrdhost_flag_check(host, RRDHOST_FLAG_ORPHAN)) ? reachable++ : unreachable++);
  203. count++;
  204. }
  205. rrd_unlock();
  206. snprintfz(b, 20, "%zu", count);
  207. analytics_set_data(&analytics_data.netdata_mirrored_host_count, b);
  208. snprintfz(b, 20, "%zu", reachable);
  209. analytics_set_data(&analytics_data.netdata_mirrored_hosts_reachable, b);
  210. snprintfz(b, 20, "%zu", unreachable);
  211. analytics_set_data(&analytics_data.netdata_mirrored_hosts_unreachable, b);
  212. }
  213. void analytics_exporters(void)
  214. {
  215. //when no exporters are available, an empty string will be sent
  216. //decide if something else is more suitable (but probably not null)
  217. BUFFER *bi = buffer_create(1000, NULL);
  218. analytics_exporting_connectors(bi);
  219. analytics_set_data_str(&analytics_data.netdata_exporting_connectors, (char *)buffer_tostring(bi));
  220. buffer_free(bi);
  221. }
  222. int collector_counter_callb(const DICTIONARY_ITEM *item __maybe_unused, void *entry, void *data) {
  223. struct array_printer *ap = (struct array_printer *)data;
  224. struct collector *col = (struct collector *)entry;
  225. BUFFER *bt = ap->both;
  226. if (likely(ap->c)) {
  227. buffer_strcat(bt, ",");
  228. }
  229. buffer_strcat(bt, "{");
  230. buffer_strcat(bt, " \"plugin\": \"");
  231. buffer_strcat(bt, col->plugin);
  232. buffer_strcat(bt, "\", \"module\":\"");
  233. buffer_strcat(bt, col->module);
  234. buffer_strcat(bt, "\" }");
  235. (ap->c)++;
  236. return 0;
  237. }
  238. /*
  239. * Create a JSON array of available collectors, same as in api/v1/info
  240. */
  241. void analytics_collectors(void)
  242. {
  243. RRDSET *st;
  244. DICTIONARY *dict = dictionary_create(DICT_OPTION_SINGLE_THREADED);
  245. char name[500];
  246. BUFFER *bt = buffer_create(1000, NULL);
  247. rrdset_foreach_read(st, localhost) {
  248. if(!rrdset_is_available_for_viewers(st))
  249. continue;
  250. struct collector col = {
  251. .plugin = rrdset_plugin_name(st),
  252. .module = rrdset_module_name(st)
  253. };
  254. snprintfz(name, 499, "%s:%s", col.plugin, col.module);
  255. dictionary_set(dict, name, &col, sizeof(struct collector));
  256. }
  257. rrdset_foreach_done(st);
  258. struct array_printer ap;
  259. ap.c = 0;
  260. ap.both = bt;
  261. dictionary_walkthrough_read(dict, collector_counter_callb, &ap);
  262. dictionary_destroy(dict);
  263. analytics_set_data(&analytics_data.netdata_collectors, (char *)buffer_tostring(ap.both));
  264. {
  265. char b[21];
  266. snprintfz(b, 20, "%d", ap.c);
  267. analytics_set_data(&analytics_data.netdata_collectors_count, b);
  268. }
  269. buffer_free(bt);
  270. }
  271. /*
  272. * Run alarm-notify.sh script using the dump_methods parameter
  273. * SEND_CUSTOM is always available
  274. */
  275. void analytics_alarms_notifications(void)
  276. {
  277. char *script;
  278. script = mallocz(
  279. sizeof(char) * (strlen(netdata_configured_primary_plugins_dir) + strlen("alarm-notify.sh dump_methods") + 2));
  280. sprintf(script, "%s/%s", netdata_configured_primary_plugins_dir, "alarm-notify.sh");
  281. if (unlikely(access(script, R_OK) != 0)) {
  282. info("Alarm notify script %s not found.", script);
  283. freez(script);
  284. return;
  285. }
  286. strcat(script, " dump_methods");
  287. pid_t command_pid;
  288. debug(D_ANALYTICS, "Executing %s", script);
  289. BUFFER *b = buffer_create(1000, NULL);
  290. int cnt = 0;
  291. FILE *fp_child_input;
  292. FILE *fp_child_output = netdata_popen(script, &command_pid, &fp_child_input);
  293. if (fp_child_output) {
  294. char line[200 + 1];
  295. while (fgets(line, 200, fp_child_output) != NULL) {
  296. char *end = line;
  297. while (*end && *end != '\n')
  298. end++;
  299. *end = '\0';
  300. if (likely(cnt))
  301. buffer_strcat(b, "|");
  302. buffer_strcat(b, line);
  303. cnt++;
  304. }
  305. netdata_pclose(fp_child_input, fp_child_output, command_pid);
  306. }
  307. freez(script);
  308. analytics_set_data_str(&analytics_data.netdata_notification_methods, (char *)buffer_tostring(b));
  309. buffer_free(b);
  310. }
  311. static void analytics_get_install_type(struct rrdhost_system_info *system_info)
  312. {
  313. if (system_info->install_type == NULL) {
  314. analytics_set_data_str(&analytics_data.netdata_install_type, "unknown");
  315. } else {
  316. analytics_set_data_str(&analytics_data.netdata_install_type, system_info->install_type);
  317. }
  318. if (system_info->prebuilt_dist != NULL) {
  319. analytics_set_data_str(&analytics_data.netdata_prebuilt_distro, system_info->prebuilt_dist);
  320. }
  321. }
  322. /*
  323. * Pick up if https is actually used
  324. */
  325. void analytics_https(void)
  326. {
  327. BUFFER *b = buffer_create(30, NULL);
  328. #ifdef ENABLE_HTTPS
  329. analytics_exporting_connectors_ssl(b);
  330. buffer_strcat(b, netdata_ssl_client_ctx && rrdhost_flag_check(localhost, RRDHOST_FLAG_RRDPUSH_SENDER_CONNECTED) && localhost->sender->ssl.flags == NETDATA_SSL_HANDSHAKE_COMPLETE ? "streaming|" : "|");
  331. buffer_strcat(b, netdata_ssl_srv_ctx ? "web" : "");
  332. #else
  333. buffer_strcat(b, "||");
  334. #endif
  335. analytics_set_data_str(&analytics_data.netdata_config_https_available, (char *)buffer_tostring(b));
  336. buffer_free(b);
  337. }
  338. void analytics_charts(void)
  339. {
  340. RRDSET *st;
  341. size_t c = 0;
  342. rrdset_foreach_read(st, localhost)
  343. if(rrdset_is_available_for_viewers(st)) c++;
  344. rrdset_foreach_done(st);
  345. analytics_data.charts_count = c;
  346. {
  347. char b[21];
  348. snprintfz(b, 20, "%zu", c);
  349. analytics_set_data(&analytics_data.netdata_charts_count, b);
  350. }
  351. }
  352. void analytics_metrics(void)
  353. {
  354. RRDSET *st;
  355. size_t dimensions = 0;
  356. rrdset_foreach_read(st, localhost) {
  357. if (rrdset_is_available_for_viewers(st)) {
  358. RRDDIM *rd;
  359. rrddim_foreach_read(rd, st) {
  360. if (rrddim_option_check(rd, RRDDIM_OPTION_HIDDEN) || rrddim_flag_check(rd, RRDDIM_FLAG_OBSOLETE))
  361. continue;
  362. dimensions++;
  363. }
  364. rrddim_foreach_done(rd);
  365. }
  366. }
  367. rrdset_foreach_done(st);
  368. analytics_data.metrics_count = dimensions;
  369. {
  370. char b[21];
  371. snprintfz(b, 20, "%zu", dimensions);
  372. analytics_set_data(&analytics_data.netdata_metrics_count, b);
  373. }
  374. }
  375. void analytics_alarms(void)
  376. {
  377. size_t alarm_warn = 0, alarm_crit = 0, alarm_normal = 0;
  378. char b[21];
  379. RRDCALC *rc;
  380. foreach_rrdcalc_in_rrdhost_read(localhost, rc) {
  381. if (unlikely(!rc->rrdset || !rc->rrdset->last_collected_time.tv_sec))
  382. continue;
  383. switch (rc->status) {
  384. case RRDCALC_STATUS_WARNING:
  385. alarm_warn++;
  386. break;
  387. case RRDCALC_STATUS_CRITICAL:
  388. alarm_crit++;
  389. break;
  390. default:
  391. alarm_normal++;
  392. }
  393. }
  394. foreach_rrdcalc_in_rrdhost_done(rc);
  395. snprintfz(b, 20, "%zu", alarm_normal);
  396. analytics_set_data(&analytics_data.netdata_alarms_normal, b);
  397. snprintfz(b, 20, "%zu", alarm_warn);
  398. analytics_set_data(&analytics_data.netdata_alarms_warning, b);
  399. snprintfz(b, 20, "%zu", alarm_crit);
  400. analytics_set_data(&analytics_data.netdata_alarms_critical, b);
  401. }
  402. /*
  403. * Misc attributes to get (run from start)
  404. */
  405. void analytics_misc(void)
  406. {
  407. #ifdef ENABLE_ACLK
  408. analytics_set_data(&analytics_data.netdata_host_cloud_available, "true");
  409. analytics_set_data_str(&analytics_data.netdata_host_aclk_implementation, "Next Generation");
  410. #else
  411. analytics_set_data(&analytics_data.netdata_host_cloud_available, "false");
  412. analytics_set_data_str(&analytics_data.netdata_host_aclk_implementation, "");
  413. #endif
  414. analytics_data.exporting_enabled = appconfig_get_boolean(&exporting_config, CONFIG_SECTION_EXPORTING, "enabled", CONFIG_BOOLEAN_NO);
  415. analytics_set_data(&analytics_data.netdata_config_exporting_enabled, analytics_data.exporting_enabled ? "true" : "false");
  416. analytics_set_data(&analytics_data.netdata_config_is_private_registry, "false");
  417. analytics_set_data(&analytics_data.netdata_config_use_private_registry, "false");
  418. if (strcmp(
  419. config_get(CONFIG_SECTION_REGISTRY, "registry to announce", "https://registry.my-netdata.io"),
  420. "https://registry.my-netdata.io"))
  421. analytics_set_data(&analytics_data.netdata_config_use_private_registry, "true");
  422. //do we need both registry to announce and enabled to indicate that this is a private registry ?
  423. if (config_get_boolean(CONFIG_SECTION_REGISTRY, "enabled", CONFIG_BOOLEAN_NO) &&
  424. web_server_mode != WEB_SERVER_MODE_NONE)
  425. analytics_set_data(&analytics_data.netdata_config_is_private_registry, "true");
  426. }
  427. void analytics_aclk(void)
  428. {
  429. #ifdef ENABLE_ACLK
  430. if (aclk_connected) {
  431. analytics_set_data(&analytics_data.netdata_host_aclk_available, "true");
  432. analytics_set_data_str(&analytics_data.netdata_host_aclk_protocol, "New");
  433. }
  434. else
  435. #endif
  436. analytics_set_data(&analytics_data.netdata_host_aclk_available, "false");
  437. }
  438. /*
  439. * Get the meta data, called from the thread once after the original delay
  440. * These are values that won't change during agent runtime, and therefore
  441. * don't try to read them on each META event send
  442. */
  443. void analytics_gather_immutable_meta_data(void)
  444. {
  445. analytics_misc();
  446. analytics_exporters();
  447. analytics_https();
  448. }
  449. /*
  450. * Get the meta data, called from the thread on every heartbeat, and right before the EXIT event
  451. * These are values that can change between agent restarts, and therefore
  452. * try to read them on each META event send
  453. */
  454. void analytics_gather_mutable_meta_data(void)
  455. {
  456. analytics_collectors();
  457. analytics_alarms();
  458. analytics_charts();
  459. analytics_metrics();
  460. analytics_aclk();
  461. analytics_mirrored_hosts();
  462. analytics_alarms_notifications();
  463. analytics_set_data(
  464. &analytics_data.netdata_config_is_parent, (rrdhost_hosts_available() > 1 || configured_as_parent()) ? "true" : "false");
  465. char *claim_id = get_agent_claimid();
  466. analytics_set_data(&analytics_data.netdata_host_agent_claimed, claim_id ? "true" : "false");
  467. freez(claim_id);
  468. {
  469. char b[21];
  470. snprintfz(b, 20, "%zu", analytics_data.prometheus_hits);
  471. analytics_set_data(&analytics_data.netdata_allmetrics_prometheus_used, b);
  472. snprintfz(b, 20, "%zu", analytics_data.shell_hits);
  473. analytics_set_data(&analytics_data.netdata_allmetrics_shell_used, b);
  474. snprintfz(b, 20, "%zu", analytics_data.json_hits);
  475. analytics_set_data(&analytics_data.netdata_allmetrics_json_used, b);
  476. snprintfz(b, 20, "%zu", analytics_data.dashboard_hits);
  477. analytics_set_data(&analytics_data.netdata_dashboard_used, b);
  478. snprintfz(b, 20, "%zu", rrdhost_hosts_available());
  479. analytics_set_data(&analytics_data.netdata_config_hosts_available, b);
  480. }
  481. }
  482. void analytics_main_cleanup(void *ptr)
  483. {
  484. struct netdata_static_thread *static_thread = (struct netdata_static_thread *)ptr;
  485. static_thread->enabled = NETDATA_MAIN_THREAD_EXITING;
  486. debug(D_ANALYTICS, "Cleaning up...");
  487. analytics_free_data();
  488. static_thread->enabled = NETDATA_MAIN_THREAD_EXITED;
  489. }
  490. /*
  491. * The analytics thread. Sleep for ANALYTICS_INIT_SLEEP_SEC,
  492. * gather the data, and then go to a loop where every ANALYTICS_HEARTBEAT
  493. * it will send a new META event after gathering data that could be changed
  494. * while the agent is running
  495. */
  496. void *analytics_main(void *ptr)
  497. {
  498. netdata_thread_cleanup_push(analytics_main_cleanup, ptr);
  499. unsigned int sec = 0;
  500. heartbeat_t hb;
  501. heartbeat_init(&hb);
  502. usec_t step_ut = USEC_PER_SEC;
  503. debug(D_ANALYTICS, "Analytics thread starts");
  504. //first delay after agent start
  505. while (service_running(SERVICE_ANALYTICS) && likely(sec <= ANALYTICS_INIT_SLEEP_SEC)) {
  506. heartbeat_next(&hb, step_ut);
  507. sec++;
  508. }
  509. if (unlikely(!service_running(SERVICE_ANALYTICS)))
  510. goto cleanup;
  511. analytics_gather_immutable_meta_data();
  512. analytics_gather_mutable_meta_data();
  513. send_statistics("META_START", "-", "-");
  514. analytics_log_data();
  515. sec = 0;
  516. while (1) {
  517. heartbeat_next(&hb, step_ut * 2);
  518. sec += 2;
  519. if (unlikely(!service_running(SERVICE_ANALYTICS)))
  520. break;
  521. if (likely(sec < ANALYTICS_HEARTBEAT))
  522. continue;
  523. analytics_gather_mutable_meta_data();
  524. send_statistics("META", "-", "-");
  525. analytics_log_data();
  526. sec = 0;
  527. }
  528. cleanup:
  529. netdata_thread_cleanup_pop(1);
  530. return NULL;
  531. }
  532. static const char *verify_required_directory(const char *dir)
  533. {
  534. if (chdir(dir) == -1)
  535. fatal("Cannot change directory to '%s'", dir);
  536. DIR *d = opendir(dir);
  537. if (!d)
  538. fatal("Cannot examine the contents of directory '%s'", dir);
  539. closedir(d);
  540. return dir;
  541. }
  542. /*
  543. * This is called after the rrdinit
  544. * These values will be sent on the START event
  545. */
  546. void set_late_global_environment(struct rrdhost_system_info *system_info)
  547. {
  548. analytics_set_data(&analytics_data.netdata_config_stream_enabled, default_rrdpush_enabled ? "true" : "false");
  549. analytics_set_data_str(&analytics_data.netdata_config_memory_mode, (char *)rrd_memory_mode_name(default_rrd_memory_mode));
  550. #ifdef DISABLE_CLOUD
  551. analytics_set_data(&analytics_data.netdata_host_cloud_enabled, "false");
  552. #else
  553. analytics_set_data(
  554. &analytics_data.netdata_host_cloud_enabled,
  555. appconfig_get_boolean(&cloud_config, CONFIG_SECTION_GLOBAL, "enabled", CONFIG_BOOLEAN_YES) ? "true" : "false");
  556. #endif
  557. #ifdef ENABLE_DBENGINE
  558. {
  559. char b[16];
  560. snprintfz(b, 15, "%d", default_rrdeng_page_cache_mb);
  561. analytics_set_data(&analytics_data.netdata_config_page_cache_size, b);
  562. snprintfz(b, 15, "%d", default_multidb_disk_quota_mb);
  563. analytics_set_data(&analytics_data.netdata_config_multidb_disk_quota, b);
  564. }
  565. #endif
  566. #ifdef ENABLE_HTTPS
  567. analytics_set_data(&analytics_data.netdata_config_https_enabled, "true");
  568. #else
  569. analytics_set_data(&analytics_data.netdata_config_https_enabled, "false");
  570. #endif
  571. if (web_server_mode == WEB_SERVER_MODE_NONE)
  572. analytics_set_data(&analytics_data.netdata_config_web_enabled, "false");
  573. else
  574. analytics_set_data(&analytics_data.netdata_config_web_enabled, "true");
  575. analytics_set_data_str(&analytics_data.netdata_config_release_channel, (char *)get_release_channel());
  576. {
  577. BUFFER *bi = buffer_create(1000, NULL);
  578. analytics_build_info(bi);
  579. analytics_set_data_str(&analytics_data.netdata_buildinfo, (char *)buffer_tostring(bi));
  580. buffer_free(bi);
  581. }
  582. analytics_get_install_type(system_info);
  583. }
  584. void get_system_timezone(void)
  585. {
  586. // avoid flood calls to stat(/etc/localtime)
  587. // http://stackoverflow.com/questions/4554271/how-to-avoid-excessive-stat-etc-localtime-calls-in-strftime-on-linux
  588. const char *tz = getenv("TZ");
  589. if (!tz || !*tz)
  590. setenv("TZ", config_get(CONFIG_SECTION_ENV_VARS, "TZ", ":/etc/localtime"), 0);
  591. char buffer[FILENAME_MAX + 1] = "";
  592. const char *timezone = NULL;
  593. ssize_t ret;
  594. // use the TZ variable
  595. if (tz && *tz && *tz != ':') {
  596. timezone = tz;
  597. info("TIMEZONE: using TZ variable '%s'", timezone);
  598. }
  599. // use the contents of /etc/timezone
  600. if (!timezone && !read_file("/etc/timezone", buffer, FILENAME_MAX)) {
  601. timezone = buffer;
  602. info("TIMEZONE: using the contents of /etc/timezone");
  603. }
  604. // read the link /etc/localtime
  605. if (!timezone) {
  606. ret = readlink("/etc/localtime", buffer, FILENAME_MAX);
  607. if (ret > 0) {
  608. buffer[ret] = '\0';
  609. char *cmp = "/usr/share/zoneinfo/";
  610. size_t cmp_len = strlen(cmp);
  611. char *s = strstr(buffer, cmp);
  612. if (s && s[cmp_len]) {
  613. timezone = &s[cmp_len];
  614. info("TIMEZONE: using the link of /etc/localtime: '%s'", timezone);
  615. }
  616. } else
  617. buffer[0] = '\0';
  618. }
  619. // find the timezone from strftime()
  620. if (!timezone) {
  621. time_t t;
  622. struct tm *tmp, tmbuf;
  623. t = now_realtime_sec();
  624. tmp = localtime_r(&t, &tmbuf);
  625. if (tmp != NULL) {
  626. if (strftime(buffer, FILENAME_MAX, "%Z", tmp) == 0)
  627. buffer[0] = '\0';
  628. else {
  629. buffer[FILENAME_MAX] = '\0';
  630. timezone = buffer;
  631. info("TIMEZONE: using strftime(): '%s'", timezone);
  632. }
  633. }
  634. }
  635. if (timezone && *timezone) {
  636. // make sure it does not have illegal characters
  637. // info("TIMEZONE: fixing '%s'", timezone);
  638. size_t len = strlen(timezone);
  639. char tmp[len + 1];
  640. char *d = tmp;
  641. *d = '\0';
  642. while (*timezone) {
  643. if (isalnum(*timezone) || *timezone == '_' || *timezone == '/')
  644. *d++ = *timezone++;
  645. else
  646. timezone++;
  647. }
  648. *d = '\0';
  649. strncpyz(buffer, tmp, len);
  650. timezone = buffer;
  651. info("TIMEZONE: fixed as '%s'", timezone);
  652. }
  653. if (!timezone || !*timezone)
  654. timezone = "unknown";
  655. netdata_configured_timezone = config_get(CONFIG_SECTION_GLOBAL, "timezone", timezone);
  656. //get the utc offset, and the timezone as returned by strftime
  657. //will be sent to the cloud
  658. //Note: This will need an agent restart to get new offset on time change (dst, etc).
  659. {
  660. time_t t;
  661. struct tm *tmp, tmbuf;
  662. char zone[FILENAME_MAX + 1];
  663. char sign[2], hh[3], mm[3];
  664. t = now_realtime_sec();
  665. tmp = localtime_r(&t, &tmbuf);
  666. if (tmp != NULL) {
  667. if (strftime(zone, FILENAME_MAX, "%Z", tmp) == 0) {
  668. netdata_configured_abbrev_timezone = strdupz("UTC");
  669. } else
  670. netdata_configured_abbrev_timezone = strdupz(zone);
  671. if (strftime(zone, FILENAME_MAX, "%z", tmp) == 0) {
  672. netdata_configured_utc_offset = 0;
  673. } else {
  674. sign[0] = zone[0] == '-' || zone[0] == '+' ? zone[0] : '0';
  675. sign[1] = '\0';
  676. hh[0] = isdigit(zone[1]) ? zone[1] : '0';
  677. hh[1] = isdigit(zone[2]) ? zone[2] : '0';
  678. hh[2] = '\0';
  679. mm[0] = isdigit(zone[3]) ? zone[3] : '0';
  680. mm[1] = isdigit(zone[4]) ? zone[4] : '0';
  681. mm[2] = '\0';
  682. netdata_configured_utc_offset = (str2i(hh) * 3600) + (str2i(mm) * 60);
  683. netdata_configured_utc_offset =
  684. sign[0] == '-' ? -netdata_configured_utc_offset : netdata_configured_utc_offset;
  685. }
  686. } else {
  687. netdata_configured_abbrev_timezone = strdupz("UTC");
  688. netdata_configured_utc_offset = 0;
  689. }
  690. }
  691. }
  692. void set_global_environment()
  693. {
  694. {
  695. char b[16];
  696. snprintfz(b, 15, "%d", default_rrd_update_every);
  697. setenv("NETDATA_UPDATE_EVERY", b, 1);
  698. }
  699. setenv("NETDATA_VERSION", program_version, 1);
  700. setenv("NETDATA_HOSTNAME", netdata_configured_hostname, 1);
  701. setenv("NETDATA_CONFIG_DIR", verify_required_directory(netdata_configured_user_config_dir), 1);
  702. setenv("NETDATA_USER_CONFIG_DIR", verify_required_directory(netdata_configured_user_config_dir), 1);
  703. setenv("NETDATA_STOCK_CONFIG_DIR", verify_required_directory(netdata_configured_stock_config_dir), 1);
  704. setenv("NETDATA_PLUGINS_DIR", verify_required_directory(netdata_configured_primary_plugins_dir), 1);
  705. setenv("NETDATA_WEB_DIR", verify_required_directory(netdata_configured_web_dir), 1);
  706. setenv("NETDATA_CACHE_DIR", verify_required_directory(netdata_configured_cache_dir), 1);
  707. setenv("NETDATA_LIB_DIR", verify_required_directory(netdata_configured_varlib_dir), 1);
  708. setenv("NETDATA_LOCK_DIR", netdata_configured_lock_dir, 1);
  709. setenv("NETDATA_LOG_DIR", verify_required_directory(netdata_configured_log_dir), 1);
  710. setenv("HOME", verify_required_directory(netdata_configured_home_dir), 1);
  711. setenv("NETDATA_HOST_PREFIX", netdata_configured_host_prefix, 1);
  712. {
  713. BUFFER *user_plugins_dirs = buffer_create(FILENAME_MAX, NULL);
  714. for (size_t i = 1; i < PLUGINSD_MAX_DIRECTORIES && plugin_directories[i]; i++) {
  715. if (i > 1)
  716. buffer_strcat(user_plugins_dirs, " ");
  717. buffer_strcat(user_plugins_dirs, plugin_directories[i]);
  718. }
  719. setenv("NETDATA_USER_PLUGINS_DIRS", buffer_tostring(user_plugins_dirs), 1);
  720. buffer_free(user_plugins_dirs);
  721. }
  722. analytics_data.data_length = 0;
  723. analytics_set_data(&analytics_data.netdata_config_stream_enabled, "null");
  724. analytics_set_data(&analytics_data.netdata_config_memory_mode, "null");
  725. analytics_set_data(&analytics_data.netdata_config_exporting_enabled, "null");
  726. analytics_set_data(&analytics_data.netdata_exporting_connectors, "null");
  727. analytics_set_data(&analytics_data.netdata_allmetrics_prometheus_used, "null");
  728. analytics_set_data(&analytics_data.netdata_allmetrics_shell_used, "null");
  729. analytics_set_data(&analytics_data.netdata_allmetrics_json_used, "null");
  730. analytics_set_data(&analytics_data.netdata_dashboard_used, "null");
  731. analytics_set_data(&analytics_data.netdata_collectors, "null");
  732. analytics_set_data(&analytics_data.netdata_collectors_count, "null");
  733. analytics_set_data(&analytics_data.netdata_buildinfo, "null");
  734. analytics_set_data(&analytics_data.netdata_config_page_cache_size, "null");
  735. analytics_set_data(&analytics_data.netdata_config_multidb_disk_quota, "null");
  736. analytics_set_data(&analytics_data.netdata_config_https_enabled, "null");
  737. analytics_set_data(&analytics_data.netdata_config_web_enabled, "null");
  738. analytics_set_data(&analytics_data.netdata_config_release_channel, "null");
  739. analytics_set_data(&analytics_data.netdata_mirrored_host_count, "null");
  740. analytics_set_data(&analytics_data.netdata_mirrored_hosts_reachable, "null");
  741. analytics_set_data(&analytics_data.netdata_mirrored_hosts_unreachable, "null");
  742. analytics_set_data(&analytics_data.netdata_notification_methods, "null");
  743. analytics_set_data(&analytics_data.netdata_alarms_normal, "null");
  744. analytics_set_data(&analytics_data.netdata_alarms_warning, "null");
  745. analytics_set_data(&analytics_data.netdata_alarms_critical, "null");
  746. analytics_set_data(&analytics_data.netdata_charts_count, "null");
  747. analytics_set_data(&analytics_data.netdata_metrics_count, "null");
  748. analytics_set_data(&analytics_data.netdata_config_is_parent, "null");
  749. analytics_set_data(&analytics_data.netdata_config_hosts_available, "null");
  750. analytics_set_data(&analytics_data.netdata_host_cloud_available, "null");
  751. analytics_set_data(&analytics_data.netdata_host_aclk_implementation, "null");
  752. analytics_set_data(&analytics_data.netdata_host_aclk_available, "null");
  753. analytics_set_data(&analytics_data.netdata_host_aclk_protocol, "null");
  754. analytics_set_data(&analytics_data.netdata_host_agent_claimed, "null");
  755. analytics_set_data(&analytics_data.netdata_host_cloud_enabled, "null");
  756. analytics_set_data(&analytics_data.netdata_config_https_available, "null");
  757. analytics_set_data(&analytics_data.netdata_install_type, "null");
  758. analytics_set_data(&analytics_data.netdata_config_is_private_registry, "null");
  759. analytics_set_data(&analytics_data.netdata_config_use_private_registry, "null");
  760. analytics_set_data(&analytics_data.netdata_config_oom_score, "null");
  761. analytics_set_data(&analytics_data.netdata_prebuilt_distro, "null");
  762. analytics_data.prometheus_hits = 0;
  763. analytics_data.shell_hits = 0;
  764. analytics_data.json_hits = 0;
  765. analytics_data.dashboard_hits = 0;
  766. analytics_data.charts_count = 0;
  767. analytics_data.metrics_count = 0;
  768. analytics_data.exporting_enabled = false;
  769. char *default_port = appconfig_get(&netdata_config, CONFIG_SECTION_WEB, "default port", NULL);
  770. int clean = 0;
  771. if (!default_port) {
  772. default_port = strdupz("19999");
  773. clean = 1;
  774. }
  775. setenv("NETDATA_LISTEN_PORT", default_port, 1);
  776. if (clean)
  777. freez(default_port);
  778. // set the path we need
  779. char path[1024 + 1], *p = getenv("PATH");
  780. if (!p)
  781. p = "/bin:/usr/bin";
  782. snprintfz(path, 1024, "%s:%s", p, "/sbin:/usr/sbin:/usr/local/bin:/usr/local/sbin");
  783. setenv("PATH", config_get(CONFIG_SECTION_ENV_VARS, "PATH", path), 1);
  784. // python options
  785. p = getenv("PYTHONPATH");
  786. if (!p)
  787. p = "";
  788. setenv("PYTHONPATH", config_get(CONFIG_SECTION_ENV_VARS, "PYTHONPATH", p), 1);
  789. // disable buffering for python plugins
  790. setenv("PYTHONUNBUFFERED", "1", 1);
  791. // switch to standard locale for plugins
  792. setenv("LC_ALL", "C", 1);
  793. }
  794. void send_statistics(const char *action, const char *action_result, const char *action_data)
  795. {
  796. static char *as_script;
  797. if (netdata_anonymous_statistics_enabled == -1) {
  798. char *optout_file = mallocz(
  799. sizeof(char) *
  800. (strlen(netdata_configured_user_config_dir) + strlen(".opt-out-from-anonymous-statistics") + 2));
  801. sprintf(optout_file, "%s/%s", netdata_configured_user_config_dir, ".opt-out-from-anonymous-statistics");
  802. if (likely(access(optout_file, R_OK) != 0)) {
  803. as_script = mallocz(
  804. sizeof(char) *
  805. (strlen(netdata_configured_primary_plugins_dir) + strlen("anonymous-statistics.sh") + 2));
  806. sprintf(as_script, "%s/%s", netdata_configured_primary_plugins_dir, "anonymous-statistics.sh");
  807. if (unlikely(access(as_script, R_OK) != 0)) {
  808. netdata_anonymous_statistics_enabled = 0;
  809. info("Anonymous statistics script %s not found.", as_script);
  810. freez(as_script);
  811. } else {
  812. netdata_anonymous_statistics_enabled = 1;
  813. }
  814. } else {
  815. netdata_anonymous_statistics_enabled = 0;
  816. as_script = NULL;
  817. }
  818. freez(optout_file);
  819. }
  820. if (!netdata_anonymous_statistics_enabled)
  821. return;
  822. if (!action)
  823. return;
  824. if (!action_result)
  825. action_result = "";
  826. if (!action_data)
  827. action_data = "";
  828. char *command_to_run = mallocz(
  829. sizeof(char) * (strlen(action) + strlen(action_result) + strlen(action_data) + strlen(as_script) +
  830. analytics_data.data_length + (ANALYTICS_NO_OF_ITEMS * 3) + 15));
  831. pid_t command_pid;
  832. sprintf(
  833. command_to_run,
  834. "%s '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' '%s' ",
  835. as_script,
  836. action,
  837. action_result,
  838. action_data,
  839. analytics_data.netdata_config_stream_enabled,
  840. analytics_data.netdata_config_memory_mode,
  841. analytics_data.netdata_config_exporting_enabled,
  842. analytics_data.netdata_exporting_connectors,
  843. analytics_data.netdata_allmetrics_prometheus_used,
  844. analytics_data.netdata_allmetrics_shell_used,
  845. analytics_data.netdata_allmetrics_json_used,
  846. analytics_data.netdata_dashboard_used,
  847. analytics_data.netdata_collectors,
  848. analytics_data.netdata_collectors_count,
  849. analytics_data.netdata_buildinfo,
  850. analytics_data.netdata_config_page_cache_size,
  851. analytics_data.netdata_config_multidb_disk_quota,
  852. analytics_data.netdata_config_https_enabled,
  853. analytics_data.netdata_config_web_enabled,
  854. analytics_data.netdata_config_release_channel,
  855. analytics_data.netdata_mirrored_host_count,
  856. analytics_data.netdata_mirrored_hosts_reachable,
  857. analytics_data.netdata_mirrored_hosts_unreachable,
  858. analytics_data.netdata_notification_methods,
  859. analytics_data.netdata_alarms_normal,
  860. analytics_data.netdata_alarms_warning,
  861. analytics_data.netdata_alarms_critical,
  862. analytics_data.netdata_charts_count,
  863. analytics_data.netdata_metrics_count,
  864. analytics_data.netdata_config_is_parent,
  865. analytics_data.netdata_config_hosts_available,
  866. analytics_data.netdata_host_cloud_available,
  867. analytics_data.netdata_host_aclk_available,
  868. analytics_data.netdata_host_aclk_protocol,
  869. analytics_data.netdata_host_aclk_implementation,
  870. analytics_data.netdata_host_agent_claimed,
  871. analytics_data.netdata_host_cloud_enabled,
  872. analytics_data.netdata_config_https_available,
  873. analytics_data.netdata_install_type,
  874. analytics_data.netdata_config_is_private_registry,
  875. analytics_data.netdata_config_use_private_registry,
  876. analytics_data.netdata_config_oom_score,
  877. analytics_data.netdata_prebuilt_distro);
  878. info("%s '%s' '%s' '%s'", as_script, action, action_result, action_data);
  879. FILE *fp_child_input;
  880. FILE *fp_child_output = netdata_popen(command_to_run, &command_pid, &fp_child_input);
  881. if (fp_child_output) {
  882. char buffer[4 + 1];
  883. char *s = fgets(buffer, 4, fp_child_output);
  884. int exit_code = netdata_pclose(fp_child_input, fp_child_output, command_pid);
  885. if (exit_code)
  886. error("Execution of anonymous statistics script returned %d.", exit_code);
  887. if (s && strncmp(buffer, "200", 3))
  888. error("Execution of anonymous statistics script returned http code %s.", buffer);
  889. } else {
  890. error("Failed to run anonymous statistics script %s.", as_script);
  891. }
  892. freez(command_to_run);
  893. }