http_client.h 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. /** @file
  2. * @brief HTTP client API
  3. *
  4. * An API for applications do HTTP requests
  5. */
  6. /*
  7. * Copyright (c) 2019 Intel Corporation
  8. *
  9. * SPDX-License-Identifier: Apache-2.0
  10. */
  11. #ifndef ZEPHYR_INCLUDE_NET_HTTP_CLIENT_H_
  12. #define ZEPHYR_INCLUDE_NET_HTTP_CLIENT_H_
  13. /**
  14. * @brief HTTP client API
  15. * @defgroup http_client HTTP client API
  16. * @ingroup networking
  17. * @{
  18. */
  19. #include <kernel.h>
  20. #include <net/net_ip.h>
  21. #include <net/http_parser.h>
  22. #ifdef __cplusplus
  23. extern "C" {
  24. #endif
  25. #if !defined(HTTP_CRLF)
  26. #define HTTP_CRLF "\r\n"
  27. #endif
  28. #if !defined(HTTP_STATUS_STR_SIZE)
  29. #define HTTP_STATUS_STR_SIZE 32
  30. #endif
  31. /* Is there more data to come */
  32. enum http_final_call {
  33. HTTP_DATA_MORE = 0,
  34. HTTP_DATA_FINAL = 1,
  35. };
  36. struct http_request;
  37. struct http_response;
  38. /**
  39. * @typedef http_payload_cb_t
  40. * @brief Callback used when data needs to be sent to the server.
  41. *
  42. * @param sock Socket id of the connection
  43. * @param req HTTP request information
  44. * @param user_data User specified data specified in http_client_req()
  45. *
  46. * @return >=0 amount of data sent, in this case http_client_req() should
  47. * continue sending data,
  48. * <0 if http_client_req() should return the error code to the
  49. * caller.
  50. */
  51. typedef int (*http_payload_cb_t)(int sock,
  52. struct http_request *req,
  53. void *user_data);
  54. /**
  55. * @typedef http_header_cb_t
  56. * @brief Callback can be used if application wants to construct additional
  57. * HTTP headers when the HTTP request is sent. Usage of this is optional.
  58. *
  59. * @param sock Socket id of the connection
  60. * @param req HTTP request information
  61. * @param user_data User specified data specified in http_client_req()
  62. *
  63. * @return >=0 amount of data sent, in this case http_client_req() should
  64. * continue sending data,
  65. * <0 if http_client_req() should return the error code to the
  66. * caller.
  67. */
  68. typedef int (*http_header_cb_t)(int sock,
  69. struct http_request *req,
  70. void *user_data);
  71. /**
  72. * @typedef http_response_cb_t
  73. * @brief Callback used when data is received from the server.
  74. *
  75. * @param rsp HTTP response information
  76. * @param final_data Does this data buffer contain all the data or
  77. * is there still more data to come.
  78. * @param user_data User specified data specified in http_client_req()
  79. */
  80. typedef void (*http_response_cb_t)(struct http_response *rsp,
  81. enum http_final_call final_data,
  82. void *user_data);
  83. /**
  84. * HTTP response from the server.
  85. */
  86. struct http_response {
  87. /** HTTP parser settings for the application usage */
  88. const struct http_parser_settings *http_cb;
  89. /** User provided HTTP response callback which is
  90. * called when a response is received to a sent HTTP
  91. * request.
  92. */
  93. http_response_cb_t cb;
  94. /** Where the body starts */
  95. uint8_t *body_start;
  96. /** Where the response is stored, this is to be
  97. * provided by the user.
  98. */
  99. uint8_t *recv_buf;
  100. /** Response buffer maximum length */
  101. size_t recv_buf_len;
  102. /** Length of the data in the result buf. If the value
  103. * is larger than recv_buf_len, then it means that
  104. * the data is truncated and could not be fully copied
  105. * into recv_buf. This can only happen if the user
  106. * did not set the response callback. If the callback
  107. * is set, then the HTTP client API will call response
  108. * callback many times so that all the data is
  109. * delivered to the user.
  110. */
  111. size_t data_len;
  112. /** HTTP Content-Length field value */
  113. size_t content_length;
  114. /** Content length parsed. This should be the same as
  115. * the content_length field if parsing was ok.
  116. */
  117. size_t processed;
  118. /* https://tools.ietf.org/html/rfc7230#section-3.1.2
  119. * The status-code element is a 3-digit integer code
  120. *
  121. * The reason-phrase element exists for the sole
  122. * purpose of providing a textual description
  123. * associated with the numeric status code. A client
  124. * SHOULD ignore the reason-phrase content.
  125. */
  126. char http_status[HTTP_STATUS_STR_SIZE];
  127. /** Numeric HTTP status code which corresponds to the
  128. * textual description.
  129. */
  130. uint16_t http_status_code;
  131. uint8_t cl_present : 1;
  132. uint8_t body_found : 1;
  133. uint8_t message_complete : 1;
  134. };
  135. /** HTTP client internal data that the application should not touch
  136. */
  137. struct http_client_internal_data {
  138. /** Work for handling timeout */
  139. struct k_work_delayable work;
  140. /** HTTP parser context */
  141. struct http_parser parser;
  142. /** HTTP parser settings */
  143. struct http_parser_settings parser_settings;
  144. /** HTTP response specific data (filled by http_client_req() when
  145. * data is received)
  146. */
  147. struct http_response response;
  148. /** User data */
  149. void *user_data;
  150. /** HTTP socket */
  151. int sock;
  152. /** Request timeout */
  153. k_timeout_t timeout;
  154. };
  155. /**
  156. * HTTP client request. This contains all the data that is needed when doing
  157. * a HTTP request.
  158. */
  159. struct http_request {
  160. /** HTTP client request internal data */
  161. struct http_client_internal_data internal;
  162. /* User should fill in following parameters */
  163. /** The HTTP method: GET, HEAD, OPTIONS, POST, ... */
  164. enum http_method method;
  165. /** User supplied callback function to call when response is
  166. * received.
  167. */
  168. http_response_cb_t response;
  169. /** User supplied list of HTTP callback functions if the
  170. * calling application wants to know the parsing status or the HTTP
  171. * fields. This is optional and normally not needed.
  172. */
  173. const struct http_parser_settings *http_cb;
  174. /** User supplied buffer where received data is stored */
  175. uint8_t *recv_buf;
  176. /** Length of the user supplied receive buffer */
  177. size_t recv_buf_len;
  178. /** The URL for this request, for example: /index.html */
  179. const char *url;
  180. /** The HTTP protocol, for example "HTTP/1.1" */
  181. const char *protocol;
  182. /** The HTTP header fields (application specific)
  183. * The Content-Type may be specified here or in the next field.
  184. * Depending on your application, the Content-Type may vary, however
  185. * some header fields may remain constant through the application's
  186. * life cycle. This is a NULL terminated list of header fields.
  187. */
  188. const char **header_fields;
  189. /** The value of the Content-Type header field, may be NULL */
  190. const char *content_type_value;
  191. /** Hostname to be used in the request */
  192. const char *host;
  193. /** Port number to be used in the request */
  194. const char *port;
  195. /** User supplied callback function to call when payload
  196. * needs to be sent. This can be NULL in which case the payload field
  197. * in http_request is used. The idea of this payload callback is to
  198. * allow user to send more data that is practical to store in allocated
  199. * memory.
  200. */
  201. http_payload_cb_t payload_cb;
  202. /** Payload, may be NULL */
  203. const char *payload;
  204. /** Payload length is used to calculate Content-Length. Set to 0
  205. * for chunked transfers.
  206. */
  207. size_t payload_len;
  208. /** User supplied callback function to call when optional headers need
  209. * to be sent. This can be NULL, in which case the optional_headers
  210. * field in http_request is used. The idea of this optional_headers
  211. * callback is to allow user to send more HTTP header data that is
  212. * practical to store in allocated memory.
  213. */
  214. http_header_cb_t optional_headers_cb;
  215. /** A NULL terminated list of any optional headers that
  216. * should be added to the HTTP request. May be NULL.
  217. * If the optional_headers_cb is specified, then this field is ignored.
  218. * Note that there are two similar fields that contain headers,
  219. * the header_fields above and this optional_headers. This is done
  220. * like this to support Websocket use case where Websocket will use
  221. * header_fields variable and any optional application specific
  222. * headers will be placed into this field.
  223. */
  224. const char **optional_headers;
  225. };
  226. /**
  227. * @brief Do a HTTP request. The callback is called when data is received
  228. * from the HTTP server. The caller must have created a connection to the
  229. * server before calling this function so connect() call must have be done
  230. * successfully for the socket.
  231. *
  232. * @param sock Socket id of the connection.
  233. * @param req HTTP request information
  234. * @param timeout Max timeout to wait for the data. The timeout value cannot be
  235. * 0 as there would be no time to receive the data.
  236. * The timeout value is in milliseconds.
  237. * @param user_data User specified data that is passed to the callback.
  238. *
  239. * @return <0 if error, >=0 amount of data sent to the server
  240. */
  241. int http_client_req(int sock, struct http_request *req,
  242. int32_t timeout, void *user_data);
  243. #ifdef __cplusplus
  244. }
  245. #endif
  246. /**
  247. * @}
  248. */
  249. #endif /* ZEPHYR_INCLUDE_NET_HTTP_CLIENT_H_ */