mongodb.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #define EXPORTING_INTERNALS
  3. #include "mongodb.h"
  4. #define CONFIG_FILE_LINE_MAX ((CONFIG_MAX_NAME + CONFIG_MAX_VALUE + 1024) * 2)
  5. /**
  6. * Initialize MongoDB connector specific data, including a ring buffer
  7. *
  8. * @param instance an instance data structure.
  9. * @return Returns 0 on success, 1 on failure.
  10. */
  11. int mongodb_init(struct instance *instance)
  12. {
  13. struct mongodb_specific_config *connector_specific_config = instance->config.connector_specific_config;
  14. mongoc_uri_t *uri;
  15. bson_error_t bson_error;
  16. if (unlikely(!connector_specific_config->collection || !*connector_specific_config->collection)) {
  17. error("EXPORTING: collection name is a mandatory MongoDB parameter, but it is not configured");
  18. return 1;
  19. }
  20. uri = mongoc_uri_new_with_error(instance->config.destination, &bson_error);
  21. if (unlikely(!uri)) {
  22. error(
  23. "EXPORTING: failed to parse URI: %s. Error message: %s", instance->config.destination, bson_error.message);
  24. return 1;
  25. }
  26. int32_t socket_timeout =
  27. mongoc_uri_get_option_as_int32(uri, MONGOC_URI_SOCKETTIMEOUTMS, instance->config.timeoutms);
  28. if (!mongoc_uri_set_option_as_int32(uri, MONGOC_URI_SOCKETTIMEOUTMS, socket_timeout)) {
  29. error("EXPORTING: failed to set %s to the value %d", MONGOC_URI_SOCKETTIMEOUTMS, socket_timeout);
  30. return 1;
  31. };
  32. struct mongodb_specific_data *connector_specific_data =
  33. (struct mongodb_specific_data *)instance->connector_specific_data;
  34. connector_specific_data->client = mongoc_client_new_from_uri(uri);
  35. if (unlikely(!connector_specific_data->client)) {
  36. error("EXPORTING: failed to create a new client");
  37. return 1;
  38. }
  39. if (!mongoc_client_set_appname(connector_specific_data->client, "netdata")) {
  40. error("EXPORTING: failed to set client appname");
  41. };
  42. connector_specific_data->collection = mongoc_client_get_collection(
  43. connector_specific_data->client, connector_specific_config->database, connector_specific_config->collection);
  44. mongoc_uri_destroy(uri);
  45. // create a ring buffer
  46. struct bson_buffer *first_buffer = NULL;
  47. if (instance->config.buffer_on_failures < 2)
  48. instance->config.buffer_on_failures = 1;
  49. else
  50. instance->config.buffer_on_failures -= 1;
  51. for (int i = 0; i < instance->config.buffer_on_failures; i++) {
  52. struct bson_buffer *current_buffer = callocz(1, sizeof(struct bson_buffer));
  53. if (!connector_specific_data->first_buffer)
  54. first_buffer = current_buffer;
  55. else
  56. current_buffer->next = connector_specific_data->first_buffer;
  57. connector_specific_data->first_buffer = current_buffer;
  58. }
  59. first_buffer->next = connector_specific_data->first_buffer;
  60. connector_specific_data->last_buffer = connector_specific_data->first_buffer;
  61. return 0;
  62. }
  63. /**
  64. * Initialize a MongoDB connector instance
  65. *
  66. * @param instance an instance data structure.
  67. * @return Returns 0 on success, 1 on failure.
  68. */
  69. int init_mongodb_instance(struct instance *instance)
  70. {
  71. instance->worker = mongodb_connector_worker;
  72. instance->start_batch_formatting = NULL;
  73. instance->start_host_formatting = format_host_labels_json_plaintext;
  74. instance->start_chart_formatting = NULL;
  75. if (EXPORTING_OPTIONS_DATA_SOURCE(instance->config.options) == EXPORTING_SOURCE_DATA_AS_COLLECTED)
  76. instance->metric_formatting = format_dimension_collected_json_plaintext;
  77. else
  78. instance->metric_formatting = format_dimension_stored_json_plaintext;
  79. instance->end_chart_formatting = NULL;
  80. instance->variables_formatting = NULL;
  81. instance->end_host_formatting = flush_host_labels;
  82. instance->end_batch_formatting = format_batch_mongodb;
  83. instance->prepare_header = NULL;
  84. instance->check_response = NULL;
  85. instance->buffer = (void *)buffer_create(0, &netdata_buffers_statistics.buffers_exporters);
  86. if (!instance->buffer) {
  87. error("EXPORTING: cannot create buffer for MongoDB exporting connector instance %s", instance->config.name);
  88. return 1;
  89. }
  90. if (uv_mutex_init(&instance->mutex))
  91. return 1;
  92. if (uv_cond_init(&instance->cond_var))
  93. return 1;
  94. struct mongodb_specific_data *connector_specific_data = callocz(1, sizeof(struct mongodb_specific_data));
  95. instance->connector_specific_data = (void *)connector_specific_data;
  96. instance->config.timeoutms =
  97. (instance->config.update_every >= 2) ? (instance->engine->config.update_every * MSEC_PER_SEC - 500) : 1000;
  98. if (!instance->engine->mongoc_initialized) {
  99. mongoc_init();
  100. instance->engine->mongoc_initialized = 1;
  101. }
  102. if (unlikely(mongodb_init(instance))) {
  103. error("EXPORTING: cannot initialize MongoDB exporting connector");
  104. return 1;
  105. }
  106. return 0;
  107. }
  108. /**
  109. * Free an array of BSON structures
  110. *
  111. * @param insert an array of documents.
  112. * @param documents_inserted the number of documents inserted.
  113. */
  114. void free_bson(bson_t **insert, size_t documents_inserted)
  115. {
  116. size_t i;
  117. for (i = 0; i < documents_inserted; i++)
  118. bson_destroy(insert[i]);
  119. freez(insert);
  120. }
  121. /**
  122. * Format a batch for the MongoDB connector
  123. *
  124. * @param instance an instance data structure.
  125. * @return Returns 0 on success, 1 on failure.
  126. */
  127. int format_batch_mongodb(struct instance *instance)
  128. {
  129. struct mongodb_specific_data *connector_specific_data =
  130. (struct mongodb_specific_data *)instance->connector_specific_data;
  131. struct stats *stats = &instance->stats;
  132. bson_t **insert = connector_specific_data->last_buffer->insert;
  133. if (insert) {
  134. // ring buffer is full, reuse the oldest element
  135. connector_specific_data->first_buffer = connector_specific_data->first_buffer->next;
  136. free_bson(insert, connector_specific_data->last_buffer->documents_inserted);
  137. connector_specific_data->total_documents_inserted -= connector_specific_data->last_buffer->documents_inserted;
  138. stats->buffered_bytes -= connector_specific_data->last_buffer->buffered_bytes;
  139. }
  140. insert = callocz((size_t)stats->buffered_metrics, sizeof(bson_t *));
  141. connector_specific_data->last_buffer->insert = insert;
  142. BUFFER *buffer = (BUFFER *)instance->buffer;
  143. char *start = (char *)buffer_tostring(buffer);
  144. char *end = start;
  145. size_t documents_inserted = 0;
  146. while (*end && documents_inserted <= (size_t)stats->buffered_metrics) {
  147. while (*end && *end != '\n')
  148. end++;
  149. if (likely(*end)) {
  150. *end = '\0';
  151. end++;
  152. } else {
  153. break;
  154. }
  155. bson_error_t bson_error;
  156. insert[documents_inserted] = bson_new_from_json((const uint8_t *)start, -1, &bson_error);
  157. if (unlikely(!insert[documents_inserted])) {
  158. error(
  159. "EXPORTING: Failed creating a BSON document from a JSON string \"%s\" : %s", start, bson_error.message);
  160. free_bson(insert, documents_inserted);
  161. return 1;
  162. }
  163. start = end;
  164. documents_inserted++;
  165. }
  166. stats->buffered_bytes += connector_specific_data->last_buffer->buffered_bytes = buffer_strlen(buffer);
  167. buffer_flush(buffer);
  168. // The stats->buffered_metrics is used in the MongoDB batch formatting as a variable for the number
  169. // of metrics, added in the current iteration, so we are clearing it here. We will use the
  170. // connector_specific_data->total_documents_inserted in the worker to show the statistics.
  171. stats->buffered_metrics = 0;
  172. connector_specific_data->total_documents_inserted += documents_inserted;
  173. connector_specific_data->last_buffer->documents_inserted = documents_inserted;
  174. connector_specific_data->last_buffer = connector_specific_data->last_buffer->next;
  175. return 0;
  176. }
  177. /**
  178. * Clean a MongoDB connector instance up
  179. *
  180. * @param instance an instance data structure.
  181. */
  182. void mongodb_cleanup(struct instance *instance)
  183. {
  184. info("EXPORTING: cleaning up instance %s ...", instance->config.name);
  185. struct mongodb_specific_data *connector_specific_data =
  186. (struct mongodb_specific_data *)instance->connector_specific_data;
  187. mongoc_collection_destroy(connector_specific_data->collection);
  188. mongoc_client_destroy(connector_specific_data->client);
  189. if (instance->engine->mongoc_initialized) {
  190. mongoc_cleanup();
  191. instance->engine->mongoc_initialized = 0;
  192. }
  193. buffer_free(instance->buffer);
  194. struct bson_buffer *next_buffer = connector_specific_data->first_buffer;
  195. for (int i = 0; i < instance->config.buffer_on_failures; i++) {
  196. struct bson_buffer *current_buffer = next_buffer;
  197. next_buffer = next_buffer->next;
  198. if (current_buffer->insert)
  199. free_bson(current_buffer->insert, current_buffer->documents_inserted);
  200. freez(current_buffer);
  201. }
  202. freez(connector_specific_data);
  203. struct mongodb_specific_config *connector_specific_config =
  204. (struct mongodb_specific_config *)instance->config.connector_specific_config;
  205. freez(connector_specific_config->database);
  206. freez(connector_specific_config->collection);
  207. freez(connector_specific_config);
  208. info("EXPORTING: instance %s exited", instance->config.name);
  209. instance->exited = 1;
  210. return;
  211. }
  212. /**
  213. * MongoDB connector worker
  214. *
  215. * Runs in a separate thread for every instance.
  216. *
  217. * @param instance_p an instance data structure.
  218. */
  219. void mongodb_connector_worker(void *instance_p)
  220. {
  221. struct instance *instance = (struct instance *)instance_p;
  222. #ifdef NETDATA_INTERNAL_CHECKS
  223. struct mongodb_specific_config *connector_specific_config = instance->config.connector_specific_config;
  224. #endif
  225. struct mongodb_specific_data *connector_specific_data =
  226. (struct mongodb_specific_data *)instance->connector_specific_data;
  227. while (!instance->engine->exit) {
  228. struct stats *stats = &instance->stats;
  229. uv_mutex_lock(&instance->mutex);
  230. if (!connector_specific_data->first_buffer->insert ||
  231. !connector_specific_data->first_buffer->documents_inserted) {
  232. while (!instance->data_is_ready)
  233. uv_cond_wait(&instance->cond_var, &instance->mutex);
  234. instance->data_is_ready = 0;
  235. }
  236. if (unlikely(instance->engine->exit)) {
  237. uv_mutex_unlock(&instance->mutex);
  238. break;
  239. }
  240. // reset the monitoring chart counters
  241. stats->received_bytes =
  242. stats->sent_bytes =
  243. stats->sent_metrics =
  244. stats->lost_metrics =
  245. stats->receptions =
  246. stats->transmission_successes =
  247. stats->transmission_failures =
  248. stats->data_lost_events =
  249. stats->lost_bytes =
  250. stats->reconnects = 0;
  251. bson_t **insert = connector_specific_data->first_buffer->insert;
  252. size_t documents_inserted = connector_specific_data->first_buffer->documents_inserted;
  253. size_t buffered_bytes = connector_specific_data->first_buffer->buffered_bytes;
  254. connector_specific_data->first_buffer->insert = NULL;
  255. connector_specific_data->first_buffer->documents_inserted = 0;
  256. connector_specific_data->first_buffer->buffered_bytes = 0;
  257. connector_specific_data->first_buffer = connector_specific_data->first_buffer->next;
  258. uv_mutex_unlock(&instance->mutex);
  259. size_t data_size = 0;
  260. for (size_t i = 0; i < documents_inserted; i++) {
  261. data_size += insert[i]->len;
  262. }
  263. debug(
  264. D_EXPORTING,
  265. "EXPORTING: mongodb_insert(): destination = %s, database = %s, collection = %s, data size = %zu",
  266. instance->config.destination,
  267. connector_specific_config->database,
  268. connector_specific_config->collection,
  269. data_size);
  270. if (likely(documents_inserted != 0)) {
  271. bson_error_t bson_error;
  272. if (likely(mongoc_collection_insert_many(
  273. connector_specific_data->collection,
  274. (const bson_t **)insert,
  275. documents_inserted,
  276. NULL,
  277. NULL,
  278. &bson_error))) {
  279. stats->sent_metrics = documents_inserted;
  280. stats->sent_bytes += data_size;
  281. stats->transmission_successes++;
  282. stats->receptions++;
  283. } else {
  284. // oops! we couldn't send (all or some of the) data
  285. error("EXPORTING: %s", bson_error.message);
  286. error(
  287. "EXPORTING: failed to write data to the database '%s'. "
  288. "Willing to write %zu bytes, wrote %zu bytes.",
  289. instance->config.destination, data_size, 0UL);
  290. stats->transmission_failures++;
  291. stats->data_lost_events++;
  292. stats->lost_bytes += buffered_bytes;
  293. stats->lost_metrics += documents_inserted;
  294. }
  295. }
  296. free_bson(insert, documents_inserted);
  297. if (unlikely(instance->engine->exit))
  298. break;
  299. uv_mutex_lock(&instance->mutex);
  300. stats->buffered_metrics = connector_specific_data->total_documents_inserted;
  301. send_internal_metrics(instance);
  302. connector_specific_data->total_documents_inserted -= documents_inserted;
  303. stats->buffered_metrics = 0;
  304. stats->buffered_bytes -= buffered_bytes;
  305. uv_mutex_unlock(&instance->mutex);
  306. #ifdef UNIT_TESTING
  307. return;
  308. #endif
  309. }
  310. mongodb_cleanup(instance);
  311. }