web_client.h 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #ifndef NETDATA_WEB_CLIENT_H
  3. #define NETDATA_WEB_CLIENT_H 1
  4. #include "libnetdata/libnetdata.h"
  5. extern int web_enable_gzip, web_gzip_level, web_gzip_strategy;
  6. #define HTTP_REQ_MAX_HEADER_FETCH_TRIES 100
  7. extern int respect_web_browser_do_not_track_policy;
  8. extern char *web_x_frame_options;
  9. typedef enum web_client_mode {
  10. WEB_CLIENT_MODE_GET = 0,
  11. WEB_CLIENT_MODE_POST = 1,
  12. WEB_CLIENT_MODE_FILECOPY = 2,
  13. WEB_CLIENT_MODE_OPTIONS = 3,
  14. WEB_CLIENT_MODE_STREAM = 4,
  15. } WEB_CLIENT_MODE;
  16. typedef enum {
  17. HTTP_VALIDATION_OK,
  18. HTTP_VALIDATION_NOT_SUPPORTED,
  19. HTTP_VALIDATION_TOO_MANY_READ_RETRIES,
  20. HTTP_VALIDATION_EXCESS_REQUEST_DATA,
  21. HTTP_VALIDATION_MALFORMED_URL,
  22. HTTP_VALIDATION_INCOMPLETE,
  23. #ifdef ENABLE_HTTPS
  24. HTTP_VALIDATION_REDIRECT
  25. #endif
  26. } HTTP_VALIDATION;
  27. typedef enum web_client_flags {
  28. WEB_CLIENT_FLAG_DEAD = 1 << 1, // if set, this client is dead
  29. WEB_CLIENT_FLAG_KEEPALIVE = 1 << 2, // if set, the web client will be re-used
  30. WEB_CLIENT_FLAG_WAIT_RECEIVE = 1 << 3, // if set, we are waiting more input data
  31. WEB_CLIENT_FLAG_WAIT_SEND = 1 << 4, // if set, we have data to send to the client
  32. WEB_CLIENT_FLAG_DO_NOT_TRACK = 1 << 5, // if set, we should not set cookies on this client
  33. WEB_CLIENT_FLAG_TRACKING_REQUIRED = 1 << 6, // if set, we need to send cookies
  34. WEB_CLIENT_FLAG_TCP_CLIENT = 1 << 7, // if set, the client is using a TCP socket
  35. WEB_CLIENT_FLAG_UNIX_CLIENT = 1 << 8, // if set, the client is using a UNIX socket
  36. WEB_CLIENT_FLAG_DONT_CLOSE_SOCKET = 1 << 9, // don't close the socket when cleaning up (static-threaded web server)
  37. WEB_CLIENT_CHUNKED_TRANSFER = 1 << 10, // chunked transfer (used with zlib compression)
  38. WEB_CLIENT_FLAG_SSL_WAIT_RECEIVE = 1 << 11, // if set, we are waiting more input data from an ssl conn
  39. WEB_CLIENT_FLAG_SSL_WAIT_SEND = 1 << 12, // if set, we have data to send to the client from an ssl conn
  40. WEB_CLIENT_FLAG_PROXY_HTTPS = 1 << 13, // if set, the client reaches us via an https proxy
  41. } WEB_CLIENT_FLAGS;
  42. #define web_client_flag_check(w, flag) ((w)->flags & (flag))
  43. #define web_client_flag_set(w, flag) (w)->flags |= flag
  44. #define web_client_flag_clear(w, flag) (w)->flags &= ~flag
  45. #define WEB_CLIENT_IS_DEAD(w) web_client_flag_set(w, WEB_CLIENT_FLAG_DEAD)
  46. #define web_client_check_dead(w) web_client_flag_check(w, WEB_CLIENT_FLAG_DEAD)
  47. #define web_client_has_keepalive(w) web_client_flag_check(w, WEB_CLIENT_FLAG_KEEPALIVE)
  48. #define web_client_enable_keepalive(w) web_client_flag_set(w, WEB_CLIENT_FLAG_KEEPALIVE)
  49. #define web_client_disable_keepalive(w) web_client_flag_clear(w, WEB_CLIENT_FLAG_KEEPALIVE)
  50. #define web_client_has_donottrack(w) web_client_flag_check(w, WEB_CLIENT_FLAG_DO_NOT_TRACK)
  51. #define web_client_enable_donottrack(w) web_client_flag_set(w, WEB_CLIENT_FLAG_DO_NOT_TRACK)
  52. #define web_client_disable_donottrack(w) web_client_flag_clear(w, WEB_CLIENT_FLAG_DO_NOT_TRACK)
  53. #define web_client_has_tracking_required(w) web_client_flag_check(w, WEB_CLIENT_FLAG_TRACKING_REQUIRED)
  54. #define web_client_enable_tracking_required(w) web_client_flag_set(w, WEB_CLIENT_FLAG_TRACKING_REQUIRED)
  55. #define web_client_disable_tracking_required(w) web_client_flag_clear(w, WEB_CLIENT_FLAG_TRACKING_REQUIRED)
  56. #define web_client_has_wait_receive(w) web_client_flag_check(w, WEB_CLIENT_FLAG_WAIT_RECEIVE)
  57. #define web_client_enable_wait_receive(w) web_client_flag_set(w, WEB_CLIENT_FLAG_WAIT_RECEIVE)
  58. #define web_client_disable_wait_receive(w) web_client_flag_clear(w, WEB_CLIENT_FLAG_WAIT_RECEIVE)
  59. #define web_client_has_wait_send(w) web_client_flag_check(w, WEB_CLIENT_FLAG_WAIT_SEND)
  60. #define web_client_enable_wait_send(w) web_client_flag_set(w, WEB_CLIENT_FLAG_WAIT_SEND)
  61. #define web_client_disable_wait_send(w) web_client_flag_clear(w, WEB_CLIENT_FLAG_WAIT_SEND)
  62. #define web_client_has_ssl_wait_receive(w) web_client_flag_check(w, WEB_CLIENT_FLAG_SSL_WAIT_RECEIVE)
  63. #define web_client_enable_ssl_wait_receive(w) web_client_flag_set(w, WEB_CLIENT_FLAG_SSL_WAIT_RECEIVE)
  64. #define web_client_disable_ssl_wait_receive(w) web_client_flag_clear(w, WEB_CLIENT_FLAG_SSL_WAIT_RECEIVE)
  65. #define web_client_has_ssl_wait_send(w) web_client_flag_check(w, WEB_CLIENT_FLAG_SSL_WAIT_SEND)
  66. #define web_client_enable_ssl_wait_send(w) web_client_flag_set(w, WEB_CLIENT_FLAG_SSL_WAIT_SEND)
  67. #define web_client_disable_ssl_wait_send(w) web_client_flag_clear(w, WEB_CLIENT_FLAG_SSL_WAIT_SEND)
  68. #define web_client_set_tcp(w) web_client_flag_set(w, WEB_CLIENT_FLAG_TCP_CLIENT)
  69. #define web_client_set_unix(w) web_client_flag_set(w, WEB_CLIENT_FLAG_UNIX_CLIENT)
  70. #define web_client_check_unix(w) web_client_flag_check(w, WEB_CLIENT_FLAG_UNIX_CLIENT)
  71. #define web_client_check_tcp(w) web_client_flag_check(w, WEB_CLIENT_FLAG_TCP_CLIENT)
  72. #define web_client_is_corkable(w) web_client_flag_check(w, WEB_CLIENT_FLAG_TCP_CLIENT)
  73. #define NETDATA_WEB_REQUEST_URL_SIZE 65536 // static allocation
  74. #define NETDATA_WEB_RESPONSE_ZLIB_CHUNK_SIZE 16384
  75. #define NETDATA_WEB_RESPONSE_HEADER_INITIAL_SIZE 4096
  76. #define NETDATA_WEB_RESPONSE_INITIAL_SIZE 8192
  77. #define NETDATA_WEB_REQUEST_INITIAL_SIZE 8192
  78. #define NETDATA_WEB_REQUEST_MAX_SIZE 65536
  79. #define NETDATA_WEB_DECODED_URL_INITIAL_SIZE 512
  80. struct response {
  81. BUFFER *header; // our response header
  82. BUFFER *header_output; // internal use
  83. BUFFER *data; // our response data buffer
  84. short int code; // the HTTP response code
  85. bool has_cookies;
  86. size_t rlen; // if non-zero, the excepted size of ifd (input of firecopy)
  87. size_t sent; // current data length sent to output
  88. bool zoutput; // if set to 1, web_client_send() will send compressed data
  89. bool zinitialized;
  90. z_stream zstream; // zlib stream for sending compressed output to client
  91. size_t zsent; // the compressed bytes we have sent to the client
  92. size_t zhave; // the compressed bytes that we have received from zlib
  93. Bytef zbuffer[NETDATA_WEB_RESPONSE_ZLIB_CHUNK_SIZE]; // temporary buffer for storing compressed output
  94. };
  95. struct web_client;
  96. typedef bool (*web_client_interrupt_t)(struct web_client *, void *data);
  97. struct web_client {
  98. unsigned long long id;
  99. size_t use_count;
  100. WEB_CLIENT_FLAGS flags; // status flags for the client
  101. WEB_CLIENT_MODE mode; // the operational mode of the client
  102. WEB_CLIENT_ACL acl; // the access list of the client
  103. int port_acl; // the operations permitted on the port the client connected to
  104. size_t header_parse_tries;
  105. size_t header_parse_last_size;
  106. bool tcp_cork;
  107. int ifd;
  108. int ofd;
  109. char client_ip[INET6_ADDRSTRLEN]; // Defined buffer sizes include null-terminators
  110. char client_port[NI_MAXSERV];
  111. char client_host[NI_MAXHOST];
  112. BUFFER *url_as_received; // the entire URL as received, used for logging - DO NOT MODIFY
  113. BUFFER *url_path_decoded; // the path, decoded - it is incrementally parsed and altered
  114. BUFFER *url_query_string_decoded; // the query string, decoded - it is incrementally parsed and altered
  115. // THESE NEED TO BE FREED
  116. char *auth_bearer_token; // the Bearer auth token (if sent)
  117. char *server_host; // the Host: header
  118. char *forwarded_host; // the X-Forwarded-For: header
  119. char *origin; // the Origin: header
  120. char *user_agent; // the User-Agent: header
  121. char *post_payload; // when this request is a POST, this has the payload
  122. size_t post_payload_size; // the size of the buffer allocated for the payload
  123. // the actual contents may be less than the size
  124. // STATIC-THREADED WEB SERVER MEMBERS
  125. size_t pollinfo_slot; // POLLINFO slot of the web client
  126. size_t pollinfo_filecopy_slot; // POLLINFO slot of the file read
  127. #ifdef ENABLE_HTTPS
  128. NETDATA_SSL ssl;
  129. #endif
  130. struct { // A callback to check if the query should be interrupted / stopped
  131. web_client_interrupt_t callback;
  132. void *callback_data;
  133. } interrupt;
  134. struct {
  135. size_t received_bytes;
  136. size_t sent_bytes;
  137. size_t *memory_accounting; // temporary pointer for constructor to use
  138. } statistics;
  139. struct {
  140. usec_t timeout_ut; // timeout if set, or zero
  141. struct timeval tv_in; // request received
  142. struct timeval tv_ready; // request processed - response ready
  143. struct timeval tv_timeout_last_checkpoint; // last checkpoint
  144. } timings;
  145. struct {
  146. struct web_client *prev;
  147. struct web_client *next;
  148. } cache;
  149. struct response response;
  150. };
  151. int web_client_permission_denied(struct web_client *w);
  152. ssize_t web_client_send(struct web_client *w);
  153. ssize_t web_client_receive(struct web_client *w);
  154. ssize_t web_client_read_file(struct web_client *w);
  155. void web_client_process_request(struct web_client *w);
  156. void web_client_request_done(struct web_client *w);
  157. void buffer_data_options2string(BUFFER *wb, uint32_t options);
  158. int mysendfile(struct web_client *w, char *filename);
  159. void web_client_build_http_header(struct web_client *w);
  160. char *strip_control_characters(char *url);
  161. void web_client_zero(struct web_client *w);
  162. struct web_client *web_client_create(size_t *statistics_memory_accounting);
  163. void web_client_free(struct web_client *w);
  164. #include "web/api/web_api_v1.h"
  165. #include "web/api/web_api_v2.h"
  166. #include "daemon/common.h"
  167. void web_client_decode_path_and_query_string(struct web_client *w, const char *path_and_query_string);
  168. int web_client_api_request(RRDHOST *host, struct web_client *w, char *url_path_fragment);
  169. const char *web_content_type_to_string(HTTP_CONTENT_TYPE content_type);
  170. void web_client_enable_deflate(struct web_client *w, int gzip);
  171. int web_client_api_request_with_node_selection(RRDHOST *host, struct web_client *w, char *decoded_url_path);
  172. void web_client_timeout_checkpoint_init(struct web_client *w);
  173. void web_client_timeout_checkpoint_set(struct web_client *w, int timeout_ms);
  174. usec_t web_client_timeout_checkpoint(struct web_client *w);
  175. bool web_client_timeout_checkpoint_and_check(struct web_client *w, usec_t *usec_since_last_checkpoint);
  176. usec_t web_client_timeout_checkpoint_response_ready(struct web_client *w, usec_t *usec_since_last_checkpoint);
  177. #endif