http_parser.h 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. /* SPDX-License-Identifier: MIT */
  2. /* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a copy
  5. * of this software and associated documentation files (the "Software"), to
  6. * deal in the Software without restriction, including without limitation the
  7. * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. * sell copies of the Software, and to permit persons to whom the Software is
  9. * furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. * IN THE SOFTWARE.
  21. */
  22. #ifndef ZEPHYR_INCLUDE_NET_HTTP_PARSER_H_
  23. #define ZEPHYR_INCLUDE_NET_HTTP_PARSER_H_
  24. /* Also update SONAME in the Makefile whenever you change these. */
  25. #define HTTP_PARSER_VERSION_MAJOR 2
  26. #define HTTP_PARSER_VERSION_MINOR 7
  27. #define HTTP_PARSER_VERSION_PATCH 1
  28. #include <sys/types.h>
  29. #if defined(_WIN32) && !defined(__MINGW32__) && \
  30. (!defined(_MSC_VER) || _MSC_VER < 1600) && !defined(__WINE__)
  31. #include <BaseTsd.h>
  32. #include <stddef.h>
  33. typedef __int8 int8_t;
  34. typedef unsigned __int8 uint8_t;
  35. typedef __int16 int16_t;
  36. typedef unsigned __int16 uint16_t;
  37. typedef __int32 int32_t;
  38. typedef unsigned __int32 uint32_t;
  39. typedef __int64 int64_t;
  40. typedef unsigned __int64 uint64_t;
  41. #else
  42. #include <zephyr/types.h>
  43. #include <stddef.h>
  44. #endif
  45. #include <net/http_parser_state.h>
  46. #include <net/http_parser_url.h>
  47. #ifdef __cplusplus
  48. extern "C" {
  49. #endif
  50. /* Maximium header size allowed. If the macro is not defined
  51. * before including this header then the default is used. To
  52. * change the maximum header size, define the macro in the build
  53. * environment (e.g. -DHTTP_MAX_HEADER_SIZE=<value>). To remove
  54. * the effective limit on the size of the header, define the macro
  55. * to a very large number (e.g. -DHTTP_MAX_HEADER_SIZE=0x7fffffff)
  56. */
  57. #ifndef HTTP_MAX_HEADER_SIZE
  58. # define HTTP_MAX_HEADER_SIZE (80 * 1024)
  59. #endif
  60. struct http_parser;
  61. struct http_parser_settings;
  62. /* Callbacks should return non-zero to indicate an error. The parser will
  63. * then halt execution.
  64. *
  65. * The one exception is on_headers_complete. In a HTTP_RESPONSE parser
  66. * returning '1' from on_headers_complete will tell the parser that it
  67. * should not expect a body. This is used when receiving a response to a
  68. * HEAD request which may contain 'Content-Length' or 'Transfer-Encoding:
  69. * chunked' headers that indicate the presence of a body.
  70. *
  71. * Returning `2` from on_headers_complete will tell parser that it should not
  72. * expect neither a body nor any further responses on this connection. This is
  73. * useful for handling responses to a CONNECT request which may not contain
  74. * `Upgrade` or `Connection: upgrade` headers.
  75. *
  76. * http_data_cb does not return data chunks. It will be called arbitrarily
  77. * many times for each string. E.G. you might get 10 callbacks for "on_url"
  78. * each providing just a few characters more data.
  79. */
  80. typedef int (*http_data_cb)(struct http_parser *, const char *at,
  81. size_t length);
  82. typedef int (*http_cb)(struct http_parser *);
  83. enum http_method {
  84. HTTP_DELETE = 0,
  85. HTTP_GET = 1,
  86. HTTP_HEAD = 2,
  87. HTTP_POST = 3,
  88. HTTP_PUT = 4,
  89. HTTP_CONNECT = 5,
  90. HTTP_OPTIONS = 6,
  91. HTTP_TRACE = 7,
  92. HTTP_COPY = 8,
  93. HTTP_LOCK = 9,
  94. HTTP_MKCOL = 10,
  95. HTTP_MOVE = 11,
  96. HTTP_PROPFIND = 12,
  97. HTTP_PROPPATCH = 13,
  98. HTTP_SEARCH = 14,
  99. HTTP_UNLOCK = 15,
  100. HTTP_BIND = 16,
  101. HTTP_REBIND = 17,
  102. HTTP_UNBIND = 18,
  103. HTTP_ACL = 19,
  104. HTTP_REPORT = 20,
  105. HTTP_MKACTIVITY = 21,
  106. HTTP_CHECKOUT = 22,
  107. HTTP_MERGE = 23,
  108. HTTP_MSEARCH = 24,
  109. HTTP_NOTIFY = 25,
  110. HTTP_SUBSCRIBE = 26,
  111. HTTP_UNSUBSCRIBE = 27,
  112. HTTP_PATCH = 28,
  113. HTTP_PURGE = 29,
  114. HTTP_MKCALENDAR = 30,
  115. HTTP_LINK = 31,
  116. HTTP_UNLINK = 32
  117. };
  118. enum http_parser_type { HTTP_REQUEST, HTTP_RESPONSE, HTTP_BOTH };
  119. /* Flag values for http_parser.flags field */
  120. enum flags {
  121. F_CHUNKED = 1 << 0,
  122. F_CONNECTION_KEEP_ALIVE = 1 << 1,
  123. F_CONNECTION_CLOSE = 1 << 2,
  124. F_CONNECTION_UPGRADE = 1 << 3,
  125. F_TRAILING = 1 << 4,
  126. F_UPGRADE = 1 << 5,
  127. F_SKIPBODY = 1 << 6,
  128. F_CONTENTLENGTH = 1 << 7
  129. };
  130. enum http_errno {
  131. HPE_OK,
  132. HPE_CB_message_begin,
  133. HPE_CB_url,
  134. HPE_CB_header_field,
  135. HPE_CB_header_value,
  136. HPE_CB_headers_complete,
  137. HPE_CB_body,
  138. HPE_CB_message_complete,
  139. HPE_CB_status,
  140. HPE_CB_chunk_header,
  141. HPE_CB_chunk_complete,
  142. HPE_INVALID_EOF_STATE,
  143. HPE_HEADER_OVERFLOW,
  144. HPE_CLOSED_CONNECTION,
  145. HPE_INVALID_VERSION,
  146. HPE_INVALID_STATUS,
  147. HPE_INVALID_METHOD,
  148. HPE_INVALID_URL,
  149. HPE_INVALID_HOST,
  150. HPE_INVALID_PORT,
  151. HPE_INVALID_PATH,
  152. HPE_INVALID_QUERY_STRING,
  153. HPE_INVALID_FRAGMENT,
  154. HPE_LF_EXPECTED,
  155. HPE_INVALID_HEADER_TOKEN,
  156. HPE_INVALID_CONTENT_LENGTH,
  157. HPE_UNEXPECTED_CONTENT_LENGTH,
  158. HPE_INVALID_CHUNK_SIZE,
  159. HPE_INVALID_CONSTANT,
  160. HPE_INVALID_INTERNAL_STATE,
  161. HPE_STRICT,
  162. HPE_PAUSED,
  163. HPE_UNKNOWN
  164. };
  165. /* Get an http_errno value from an http_parser */
  166. #define HTTP_PARSER_ERRNO(p) ((enum http_errno) (p)->http_errno)
  167. struct http_parser {
  168. /** PRIVATE **/
  169. unsigned int type : 2; /* enum http_parser_type */
  170. unsigned int flags : 8; /* F_xxx values from 'flags' enum;
  171. * semi-public
  172. */
  173. unsigned int state : 7; /* enum state from http_parser.c */
  174. unsigned int header_state : 7; /* enum header_state from http_parser.c
  175. */
  176. unsigned int index : 7; /* index into current matcher */
  177. unsigned int lenient_http_headers : 1;
  178. uint32_t nread; /* # bytes read in various scenarios */
  179. uint64_t content_length; /* # bytes in body (0 if no Content-Length
  180. * header)
  181. */
  182. /** READ-ONLY **/
  183. unsigned short http_major;
  184. unsigned short http_minor;
  185. unsigned int status_code : 16; /* responses only */
  186. unsigned int method : 8; /* requests only */
  187. unsigned int http_errno : 7;
  188. /* 1 = Upgrade header was present and the parser has exited because of
  189. * that.
  190. * 0 = No upgrade header present.
  191. * Should be checked when http_parser_execute() returns in addition to
  192. * error checking.
  193. */
  194. unsigned int upgrade : 1;
  195. /** PUBLIC **/
  196. void *data; /* A pointer to get hook to the "connection" or "socket"
  197. * object
  198. */
  199. /* Remote socket address of http connection, where parser can initiate
  200. * replies if necessary.
  201. */
  202. const struct sockaddr *addr;
  203. };
  204. struct http_parser_settings {
  205. http_cb on_message_begin;
  206. http_data_cb on_url;
  207. http_data_cb on_status;
  208. http_data_cb on_header_field;
  209. http_data_cb on_header_value;
  210. http_cb on_headers_complete;
  211. http_data_cb on_body;
  212. http_cb on_message_complete;
  213. /* When on_chunk_header is called, the current chunk length is stored
  214. * in parser->content_length.
  215. */
  216. http_cb on_chunk_header;
  217. http_cb on_chunk_complete;
  218. };
  219. /* Returns the library version. Bits 16-23 contain the major version number,
  220. * bits 8-15 the minor version number and bits 0-7 the patch level.
  221. * Usage example:
  222. *
  223. * unsigned long version = http_parser_version();
  224. * unsigned major = (version >> 16) & 255;
  225. * unsigned minor = (version >> 8) & 255;
  226. * unsigned patch = version & 255;
  227. * printf("http_parser v%u.%u.%u\n", major, minor, patch);
  228. */
  229. unsigned long http_parser_version(void);
  230. void http_parser_init(struct http_parser *parser, enum http_parser_type type);
  231. /* Initialize http_parser_settings members to 0
  232. */
  233. void http_parser_settings_init(struct http_parser_settings *settings);
  234. /* Executes the parser. Returns number of parsed bytes. Sets
  235. * `parser->http_errno` on error.
  236. */
  237. size_t http_parser_execute(struct http_parser *parser,
  238. const struct http_parser_settings *settings,
  239. const char *data, size_t len);
  240. /* If http_should_keep_alive() in the on_headers_complete or
  241. * on_message_complete callback returns 0, then this should be
  242. * the last message on the connection.
  243. * If you are the server, respond with the "Connection: close" header.
  244. * If you are the client, close the connection.
  245. */
  246. int http_should_keep_alive(const struct http_parser *parser);
  247. /* Returns a string version of the HTTP method. */
  248. const char *http_method_str(enum http_method m);
  249. /* Return a string name of the given error */
  250. const char *http_errno_name(enum http_errno err);
  251. /* Return a string description of the given error */
  252. const char *http_errno_description(enum http_errno err);
  253. /* Pause or un-pause the parser; a nonzero value pauses */
  254. void http_parser_pause(struct http_parser *parser, int paused);
  255. /* Checks if this is the final chunk of the body. */
  256. int http_body_is_final(const struct http_parser *parser);
  257. #ifdef __cplusplus
  258. }
  259. #endif
  260. #endif