buffer.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "../libnetdata.h"
  3. static inline void buffer_overflow_init(BUFFER *b)
  4. {
  5. b->buffer[b->size] = '\0';
  6. strcpy(&b->buffer[b->size + 1], BUFFER_OVERFLOW_EOF);
  7. }
  8. void buffer_reset(BUFFER *wb) {
  9. buffer_flush(wb);
  10. wb->content_type = CT_TEXT_PLAIN;
  11. wb->options = 0;
  12. wb->date = 0;
  13. wb->expires = 0;
  14. buffer_no_cacheable(wb);
  15. buffer_overflow_check(wb);
  16. }
  17. const char *buffer_tostring(BUFFER *wb)
  18. {
  19. buffer_need_bytes(wb, 1);
  20. wb->buffer[wb->len] = '\0';
  21. buffer_overflow_check(wb);
  22. return(wb->buffer);
  23. }
  24. void buffer_char_replace(BUFFER *wb, char from, char to) {
  25. char *s = wb->buffer, *end = &wb->buffer[wb->len];
  26. while(s != end) {
  27. if(*s == from) *s = to;
  28. s++;
  29. }
  30. buffer_overflow_check(wb);
  31. }
  32. void buffer_print_sn_flags(BUFFER *wb, SN_FLAGS flags, bool send_anomaly_bit) {
  33. if(unlikely(flags == SN_EMPTY_SLOT)) {
  34. buffer_fast_strcat(wb, "E", 1);
  35. return;
  36. }
  37. size_t printed = 0;
  38. if(likely(send_anomaly_bit && (flags & SN_FLAG_NOT_ANOMALOUS))) {
  39. buffer_fast_strcat(wb, "A", 1);
  40. printed++;
  41. }
  42. if(unlikely(flags & SN_FLAG_RESET)) {
  43. buffer_fast_strcat(wb, "R", 1);
  44. printed++;
  45. }
  46. if(!printed)
  47. buffer_fast_strcat(wb, "''", 2);
  48. }
  49. void buffer_strcat_htmlescape(BUFFER *wb, const char *txt)
  50. {
  51. while(*txt) {
  52. switch(*txt) {
  53. case '&': buffer_strcat(wb, "&"); break;
  54. case '<': buffer_strcat(wb, "&lt;"); break;
  55. case '>': buffer_strcat(wb, "&gt;"); break;
  56. case '"': buffer_strcat(wb, "&quot;"); break;
  57. case '/': buffer_strcat(wb, "&#x2F;"); break;
  58. case '\'': buffer_strcat(wb, "&#x27;"); break;
  59. default: {
  60. buffer_need_bytes(wb, 1);
  61. wb->buffer[wb->len++] = *txt;
  62. }
  63. }
  64. txt++;
  65. }
  66. buffer_overflow_check(wb);
  67. }
  68. void buffer_snprintf(BUFFER *wb, size_t len, const char *fmt, ...)
  69. {
  70. if(unlikely(!fmt || !*fmt)) return;
  71. buffer_need_bytes(wb, len + 1);
  72. va_list args;
  73. va_start(args, fmt);
  74. wb->len += vsnprintfz(&wb->buffer[wb->len], len, fmt, args);
  75. va_end(args);
  76. buffer_overflow_check(wb);
  77. // the buffer is \0 terminated by vsnprintfz
  78. }
  79. void buffer_vsprintf(BUFFER *wb, const char *fmt, va_list args)
  80. {
  81. if(unlikely(!fmt || !*fmt)) return;
  82. size_t wrote = 0, need = 2, space_remaining = 0;
  83. do {
  84. need += space_remaining * 2;
  85. netdata_log_debug(D_WEB_BUFFER, "web_buffer_sprintf(): increasing web_buffer at position %zu, size = %zu, by %zu bytes (wrote = %zu)\n", wb->len, wb->size, need, wrote);
  86. buffer_need_bytes(wb, need);
  87. space_remaining = wb->size - wb->len - 1;
  88. wrote = (size_t) vsnprintfz(&wb->buffer[wb->len], space_remaining, fmt, args);
  89. } while(wrote >= space_remaining);
  90. wb->len += wrote;
  91. // the buffer is \0 terminated by vsnprintf
  92. }
  93. void buffer_sprintf(BUFFER *wb, const char *fmt, ...)
  94. {
  95. if(unlikely(!fmt || !*fmt)) return;
  96. va_list args;
  97. size_t wrote = 0, need = 2, space_remaining = 0;
  98. do {
  99. need += space_remaining * 2;
  100. netdata_log_debug(D_WEB_BUFFER, "web_buffer_sprintf(): increasing web_buffer at position %zu, size = %zu, by %zu bytes (wrote = %zu)\n", wb->len, wb->size, need, wrote);
  101. buffer_need_bytes(wb, need);
  102. space_remaining = wb->size - wb->len - 1;
  103. va_start(args, fmt);
  104. wrote = (size_t) vsnprintfz(&wb->buffer[wb->len], space_remaining, fmt, args);
  105. va_end(args);
  106. } while(wrote >= space_remaining);
  107. wb->len += wrote;
  108. // the buffer is \0 terminated by vsnprintf
  109. }
  110. // generate a javascript date, the fastest possible way...
  111. void buffer_jsdate(BUFFER *wb, int year, int month, int day, int hours, int minutes, int seconds)
  112. {
  113. // 10 20 30 = 35
  114. // 01234567890123456789012345678901234
  115. // Date(2014,04,01,03,28,20)
  116. buffer_need_bytes(wb, 30);
  117. char *b = &wb->buffer[wb->len], *p;
  118. unsigned int *q = (unsigned int *)b;
  119. #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
  120. *q++ = 0x65746144; // "Date" backwards.
  121. #else
  122. *q++ = 0x44617465; // "Date"
  123. #endif
  124. p = (char *)q;
  125. *p++ = '(';
  126. *p++ = '0' + year / 1000; year %= 1000;
  127. *p++ = '0' + year / 100; year %= 100;
  128. *p++ = '0' + year / 10;
  129. *p++ = '0' + year % 10;
  130. *p++ = ',';
  131. *p = '0' + month / 10; if (*p != '0') p++;
  132. *p++ = '0' + month % 10;
  133. *p++ = ',';
  134. *p = '0' + day / 10; if (*p != '0') p++;
  135. *p++ = '0' + day % 10;
  136. *p++ = ',';
  137. *p = '0' + hours / 10; if (*p != '0') p++;
  138. *p++ = '0' + hours % 10;
  139. *p++ = ',';
  140. *p = '0' + minutes / 10; if (*p != '0') p++;
  141. *p++ = '0' + minutes % 10;
  142. *p++ = ',';
  143. *p = '0' + seconds / 10; if (*p != '0') p++;
  144. *p++ = '0' + seconds % 10;
  145. unsigned short *r = (unsigned short *)p;
  146. #if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
  147. *r++ = 0x0029; // ")\0" backwards.
  148. #else
  149. *r++ = 0x2900; // ")\0"
  150. #endif
  151. wb->len += (size_t)((char *)r - b - 1);
  152. // terminate it
  153. wb->buffer[wb->len] = '\0';
  154. buffer_overflow_check(wb);
  155. }
  156. // generate a date, the fastest possible way...
  157. void buffer_date(BUFFER *wb, int year, int month, int day, int hours, int minutes, int seconds)
  158. {
  159. // 10 20 30 = 35
  160. // 01234567890123456789012345678901234
  161. // 2014-04-01 03:28:20
  162. buffer_need_bytes(wb, 36);
  163. char *b = &wb->buffer[wb->len];
  164. char *p = b;
  165. *p++ = '0' + year / 1000; year %= 1000;
  166. *p++ = '0' + year / 100; year %= 100;
  167. *p++ = '0' + year / 10;
  168. *p++ = '0' + year % 10;
  169. *p++ = '-';
  170. *p++ = '0' + month / 10;
  171. *p++ = '0' + month % 10;
  172. *p++ = '-';
  173. *p++ = '0' + day / 10;
  174. *p++ = '0' + day % 10;
  175. *p++ = ' ';
  176. *p++ = '0' + hours / 10;
  177. *p++ = '0' + hours % 10;
  178. *p++ = ':';
  179. *p++ = '0' + minutes / 10;
  180. *p++ = '0' + minutes % 10;
  181. *p++ = ':';
  182. *p++ = '0' + seconds / 10;
  183. *p++ = '0' + seconds % 10;
  184. *p = '\0';
  185. wb->len += (size_t)(p - b);
  186. // terminate it
  187. wb->buffer[wb->len] = '\0';
  188. buffer_overflow_check(wb);
  189. }
  190. BUFFER *buffer_create(size_t size, size_t *statistics)
  191. {
  192. BUFFER *b;
  193. netdata_log_debug(D_WEB_BUFFER, "Creating new web buffer of size %zu.", size);
  194. b = callocz(1, sizeof(BUFFER));
  195. b->buffer = mallocz(size + sizeof(BUFFER_OVERFLOW_EOF) + 2);
  196. b->buffer[0] = '\0';
  197. b->size = size;
  198. b->content_type = CT_TEXT_PLAIN;
  199. b->statistics = statistics;
  200. buffer_no_cacheable(b);
  201. buffer_overflow_init(b);
  202. buffer_overflow_check(b);
  203. if(b->statistics)
  204. __atomic_add_fetch(b->statistics, b->size + sizeof(BUFFER) + sizeof(BUFFER_OVERFLOW_EOF) + 2, __ATOMIC_RELAXED);
  205. return(b);
  206. }
  207. void buffer_free(BUFFER *b) {
  208. if(unlikely(!b)) return;
  209. buffer_overflow_check(b);
  210. netdata_log_debug(D_WEB_BUFFER, "Freeing web buffer of size %zu.", b->size);
  211. if(b->statistics)
  212. __atomic_sub_fetch(b->statistics, b->size + sizeof(BUFFER) + sizeof(BUFFER_OVERFLOW_EOF) + 2, __ATOMIC_RELAXED);
  213. freez(b->buffer);
  214. freez(b);
  215. }
  216. void buffer_increase(BUFFER *b, size_t free_size_required) {
  217. buffer_overflow_check(b);
  218. size_t left = b->size - b->len;
  219. if(left >= free_size_required) return;
  220. size_t wanted = free_size_required - left;
  221. size_t minimum = WEB_DATA_LENGTH_INCREASE_STEP;
  222. if(minimum > wanted) wanted = minimum;
  223. size_t optimal = (b->size > 5*1024*1024) ? b->size / 2 : b->size;
  224. if(optimal > wanted) wanted = optimal;
  225. netdata_log_debug(D_WEB_BUFFER, "Increasing data buffer from size %zu to %zu.", b->size, b->size + wanted);
  226. b->buffer = reallocz(b->buffer, b->size + wanted + sizeof(BUFFER_OVERFLOW_EOF) + 2);
  227. b->size += wanted;
  228. if(b->statistics)
  229. __atomic_add_fetch(b->statistics, wanted, __ATOMIC_RELAXED);
  230. buffer_overflow_init(b);
  231. buffer_overflow_check(b);
  232. }
  233. // ----------------------------------------------------------------------------
  234. void buffer_json_initialize(BUFFER *wb, const char *key_quote, const char *value_quote, int depth,
  235. bool add_anonymous_object, BUFFER_JSON_OPTIONS options) {
  236. strncpyz(wb->json.key_quote, key_quote, BUFFER_QUOTE_MAX_SIZE);
  237. strncpyz(wb->json.value_quote, value_quote, BUFFER_QUOTE_MAX_SIZE);
  238. wb->json.options = options;
  239. wb->json.depth = (int8_t)(depth - 1);
  240. _buffer_json_depth_push(wb, BUFFER_JSON_OBJECT);
  241. if(add_anonymous_object)
  242. buffer_fast_strcat(wb, "{", 1);
  243. wb->content_type = CT_APPLICATION_JSON;
  244. buffer_no_cacheable(wb);
  245. }
  246. void buffer_json_finalize(BUFFER *wb) {
  247. while(wb->json.depth >= 0) {
  248. switch(wb->json.stack[wb->json.depth].type) {
  249. case BUFFER_JSON_OBJECT:
  250. buffer_json_object_close(wb);
  251. break;
  252. case BUFFER_JSON_ARRAY:
  253. buffer_json_array_close(wb);
  254. break;
  255. default:
  256. internal_fatal(true, "BUFFER: unknown json member type in stack");
  257. break;
  258. }
  259. }
  260. if(!(wb->json.options & BUFFER_JSON_OPTIONS_MINIFY))
  261. buffer_fast_strcat(wb, "\n", 1);
  262. }
  263. // ----------------------------------------------------------------------------
  264. const char hex_digits[16] = "0123456789ABCDEF";
  265. const char base64_digits[64] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
  266. unsigned char hex_value_from_ascii[256];
  267. unsigned char base64_value_from_ascii[256];
  268. __attribute__((constructor)) void initialize_ascii_maps(void) {
  269. for(size_t i = 0 ; i < 256 ; i++) {
  270. hex_value_from_ascii[i] = 255;
  271. base64_value_from_ascii[i] = 255;
  272. }
  273. for(size_t i = 0; i < 16 ; i++)
  274. hex_value_from_ascii[(int)hex_digits[i]] = i;
  275. for(size_t i = 0; i < 64 ; i++)
  276. base64_value_from_ascii[(int)base64_digits[i]] = i;
  277. }
  278. // ----------------------------------------------------------------------------
  279. // unit test
  280. static int buffer_expect(BUFFER *wb, const char *expected) {
  281. const char *generated = buffer_tostring(wb);
  282. if(strcmp(generated, expected) != 0) {
  283. netdata_log_error("BUFFER: mismatch.\nGenerated:\n%s\nExpected:\n%s\n",
  284. generated, expected);
  285. return 1;
  286. }
  287. return 0;
  288. }
  289. static int buffer_uint64_roundtrip(BUFFER *wb, NUMBER_ENCODING encoding, uint64_t value, const char *expected) {
  290. int errors = 0;
  291. buffer_flush(wb);
  292. buffer_print_uint64_encoded(wb, encoding, value);
  293. if(expected)
  294. errors += buffer_expect(wb, expected);
  295. uint64_t v = str2ull_encoded(buffer_tostring(wb));
  296. if(v != value) {
  297. netdata_log_error("BUFFER: string '%s' does resolves to %llu, expected %llu",
  298. buffer_tostring(wb), (unsigned long long)v, (unsigned long long)value);
  299. errors++;
  300. }
  301. buffer_flush(wb);
  302. return errors;
  303. }
  304. static int buffer_int64_roundtrip(BUFFER *wb, NUMBER_ENCODING encoding, int64_t value, const char *expected) {
  305. int errors = 0;
  306. buffer_flush(wb);
  307. buffer_print_int64_encoded(wb, encoding, value);
  308. if(expected)
  309. errors += buffer_expect(wb, expected);
  310. int64_t v = str2ll_encoded(buffer_tostring(wb));
  311. if(v != value) {
  312. netdata_log_error("BUFFER: string '%s' does resolves to %lld, expected %lld",
  313. buffer_tostring(wb), (long long)v, (long long)value);
  314. errors++;
  315. }
  316. buffer_flush(wb);
  317. return errors;
  318. }
  319. static int buffer_double_roundtrip(BUFFER *wb, NUMBER_ENCODING encoding, NETDATA_DOUBLE value, const char *expected) {
  320. int errors = 0;
  321. buffer_flush(wb);
  322. buffer_print_netdata_double_encoded(wb, encoding, value);
  323. if(expected)
  324. errors += buffer_expect(wb, expected);
  325. NETDATA_DOUBLE v = str2ndd_encoded(buffer_tostring(wb), NULL);
  326. if(v != value) {
  327. netdata_log_error("BUFFER: string '%s' does resolves to %.12f, expected %.12f",
  328. buffer_tostring(wb), v, value);
  329. errors++;
  330. }
  331. buffer_flush(wb);
  332. return errors;
  333. }
  334. int buffer_unittest(void) {
  335. int errors = 0;
  336. BUFFER *wb = buffer_create(0, NULL);
  337. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 0, "0");
  338. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_HEX, 0, "0x0");
  339. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_BASE64, 0, "#A");
  340. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 1676071986, "1676071986");
  341. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_HEX, 1676071986, "0x63E6D432");
  342. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_BASE64, 1676071986, "#Bj5tQy");
  343. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 18446744073709551615ULL, "18446744073709551615");
  344. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_HEX, 18446744073709551615ULL, "0xFFFFFFFFFFFFFFFF");
  345. buffer_uint64_roundtrip(wb, NUMBER_ENCODING_BASE64, 18446744073709551615ULL, "#P//////////");
  346. buffer_int64_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 0, "0");
  347. buffer_int64_roundtrip(wb, NUMBER_ENCODING_HEX, 0, "0x0");
  348. buffer_int64_roundtrip(wb, NUMBER_ENCODING_BASE64, 0, "#A");
  349. buffer_int64_roundtrip(wb, NUMBER_ENCODING_DECIMAL, -1676071986, "-1676071986");
  350. buffer_int64_roundtrip(wb, NUMBER_ENCODING_HEX, -1676071986, "-0x63E6D432");
  351. buffer_int64_roundtrip(wb, NUMBER_ENCODING_BASE64, -1676071986, "-#Bj5tQy");
  352. buffer_int64_roundtrip(wb, NUMBER_ENCODING_DECIMAL, (int64_t)-9223372036854775807ULL, "-9223372036854775807");
  353. buffer_int64_roundtrip(wb, NUMBER_ENCODING_HEX, (int64_t)-9223372036854775807ULL, "-0x7FFFFFFFFFFFFFFF");
  354. buffer_int64_roundtrip(wb, NUMBER_ENCODING_BASE64, (int64_t)-9223372036854775807ULL, "-#H//////////");
  355. buffer_double_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 0, "0");
  356. buffer_double_roundtrip(wb, NUMBER_ENCODING_HEX, 0, "%0");
  357. buffer_double_roundtrip(wb, NUMBER_ENCODING_BASE64, 0, "@A");
  358. buffer_double_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 1.5, "1.5");
  359. buffer_double_roundtrip(wb, NUMBER_ENCODING_HEX, 1.5, "%3FF8000000000000");
  360. buffer_double_roundtrip(wb, NUMBER_ENCODING_BASE64, 1.5, "@D/4AAAAAAAA");
  361. buffer_double_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 1.23e+14, "123000000000000");
  362. buffer_double_roundtrip(wb, NUMBER_ENCODING_HEX, 1.23e+14, "%42DBF78AD3AC0000");
  363. buffer_double_roundtrip(wb, NUMBER_ENCODING_BASE64, 1.23e+14, "@ELb94rTrAAA");
  364. buffer_double_roundtrip(wb, NUMBER_ENCODING_DECIMAL, 9.12345678901234567890123456789e+45, "9.123456789012346128e+45");
  365. buffer_double_roundtrip(wb, NUMBER_ENCODING_HEX, 9.12345678901234567890123456789e+45, "%497991C25C9E4309");
  366. buffer_double_roundtrip(wb, NUMBER_ENCODING_BASE64, 9.12345678901234567890123456789e+45, "@El5kcJcnkMJ");
  367. buffer_flush(wb);
  368. {
  369. char buf[1024 + 1];
  370. for(size_t i = 0; i < 1024 ;i++)
  371. buf[i] = (char)(i % 26) + 'A';
  372. buf[1024] = '\0';
  373. buffer_strcat(wb, buf);
  374. errors += buffer_expect(wb, buf);
  375. }
  376. buffer_flush(wb);
  377. buffer_json_initialize(wb, "\"", "\"", 0, true, BUFFER_JSON_OPTIONS_DEFAULT);
  378. buffer_json_finalize(wb);
  379. errors += buffer_expect(wb, "{\n}\n");
  380. buffer_flush(wb);
  381. buffer_json_initialize(wb, "\"", "\"", 0, true, BUFFER_JSON_OPTIONS_DEFAULT);
  382. buffer_json_member_add_string(wb, "hello", "world");
  383. buffer_json_member_add_string(wb, "alpha", "this: \" is a double quote");
  384. buffer_json_member_add_object(wb, "object1");
  385. buffer_json_member_add_string(wb, "hello", "world");
  386. buffer_json_finalize(wb);
  387. errors += buffer_expect(wb, "{\n \"hello\":\"world\",\n \"alpha\":\"this: \\\" is a double quote\",\n \"object1\":{\n \"hello\":\"world\"\n }\n}\n");
  388. buffer_free(wb);
  389. return errors;
  390. }
  391. #ifdef ENABLE_H2O
  392. h2o_iovec_t buffer_to_h2o_iovec(BUFFER *wb) {
  393. h2o_iovec_t ret;
  394. ret.base = wb->buffer;
  395. ret.len = wb->len;
  396. return ret;
  397. }
  398. #endif