cueOS  2.4
cueOS - Universal Show Control OS for ARM
http_request.c
1/***============================================================================================================================
2 * Depedencies inclusion
3 * Necessary dependencies should be declared here. Header file should contain as little dependecies declarations as possible
4 *=============================================================================================================================*/
5
6#include "altcp.h"
7#include "http_defs.h"
8#include "http_request.h"
9#include "cmsis_os.h"
10#include <string.h>
11
12
13/***============================================================================================================================
14 * Private functions definitions
15 * These functions are only accessible from within the file's scope
16 *=============================================================================================================================*/
17
26static uint8_t _http_request_parse_method(http_request_s *req){
27
28 uint8_t ret = 0;
29
30 for(uint8_t i=HTTP_GET; i<HTTP_DELETE; i++){
31 uint8_t method_len = strlen(http_methods[i]);
32 if(!strncmp(req->raw_data, http_methods[i], method_len - 1)){
33 req->method = i;
34 req->raw_data += method_len + 1;
35 req->raw_len -= method_len - 1;
36 ret = 1;
37 break;
38 }
39 }
40
41 return ret;
42
43}
44
53static uint8_t _http_request_parse_uri(http_request_s *req){
54
55 uint8_t ret = 0;
56
57 req->param_count = 0;
58
59 char *uri_end = lwip_strnstr(req->raw_data + 1, STR_COMMON_SP, req->raw_len);
60 char *arg_start = NULL;
61 char *val_start = NULL;
62 char *val_end = NULL;
63
64 uint8_t uri_len = uri_end - req->raw_data;
65
66 req->uri = pvPortMalloc(uri_len + 1);
67 memmove(req->uri, req->raw_data, uri_len);
68 req->uri[uri_len] = 0;
69
70 char *params = lwip_strnstr(req->uri, STR_COMMON_QMRK, uri_len);
71
72 uint16_t params_len = strlen(params) - 1;
73
74
75 req->raw_data = uri_end + 1;
76 req->raw_len -= uri_len + params_len + 1;
77
78 if(uri_end != NULL){ //TODO: Is this really useful ?
79
80 ret = 1;
81
82 if(params != NULL){
83
84 params++;
85
86 do{
87
88 arg_start = params;
89 val_start = strnstr(arg_start, STR_COMMON_EQUL, params_len);
90 val_end = strnstr(val_start, STR_COMMON_AMPR, params_len);
91 if(val_end == NULL){ val_end = uri_end; }
92
93 if(arg_start != NULL && val_start != NULL){
94
95 uint8_t arg_len = val_start++ - arg_start + 1;
96 uint8_t val_len = val_end - val_start + 1;
97
98 req->params = pvPortRealloc(req->params, sizeof(http_param_s) * (++req->param_count));
99
100 req->params[req->param_count-1].arg = pvPortMalloc(sizeof(char) * arg_len);
101 req->params[req->param_count-1].val = pvPortMalloc(sizeof(char) * val_len);
102
103 memmove(req->params[req->param_count-1].arg, arg_start, arg_len);
104 memmove(req->params[req->param_count-1].val, val_start, val_len);
105
106 req->params[req->param_count-1].arg[arg_len - 1] = 0; //Terminating string
107 req->params[req->param_count-1].val[val_len - 1] = 0; //Terminating string
108
109 params = val_end + 1;
110
111 }
112
113 }while(arg_start != NULL && val_start != NULL && val_end != NULL);
114
115 }
116 }
117
118 return ret;
119
120}
121
122
131static uint8_t _http_request_parse_http_version(http_request_s *req){
132
133 uint8_t ret = 0;
134
135 char *http_version_end = strnstr(req->raw_data, STR_COMMON_CRLF, req->raw_len);
136 uint8_t http_version_len = http_version_end - req->raw_data;
137
138 for(uint8_t i=HTTP_VERSION_0_9; i<HTTP_VERSION_2_0; i++){
139 if(!strncmp(req->raw_data, http_versions[i], http_version_len)){
140 req->http_version = i;
141 req->raw_data += http_version_len + 2;
142 req->raw_len -= http_version_len - 2;
143 ret = 1;
144 break;
145 }
146 }
147
148 return ret;
149
150}
151
160static uint8_t _http_request_parse_headers(http_request_s *req){
161
162 uint8_t ret = 1;
163
164 char *content_length_index = lwip_strnstr(req->raw_data, http_header_field_str[HTTP_HEADER_FIELD_CONTENT_LENGTH], req->raw_len);
165
166 if(content_length_index != NULL){
167
168 char *content_length_value = lwip_strnstr(content_length_index, " ", req->raw_len) + 1;
169 char *content_length_value_end = lwip_strnstr(content_length_value, STR_COMMON_CRLF, req->raw_len);
170
171 uint8_t content_length_str_len = content_length_value_end - content_length_value;
172
173 char *content_length_value_str[content_length_str_len];
174 memmove(content_length_value_str, content_length_value, content_length_str_len);
175 req->content_length = atoi(content_length_value);
176
177 }
178
179 return ret;
180
181}
182
191static uint8_t _http_request_parse_content(http_request_s *req){
192
193 uint8_t ret = 1;
194
195 if(req->content_length > 0){
196
197 char *content_data = lwip_strnstr(req->raw_data, STR_COMMON_CRLF STR_COMMON_CRLF, req->raw_len) + strlen(STR_COMMON_CRLF STR_COMMON_CRLF);
198
199 req->content = pvPortMalloc(req->content_length + 1);
200 memcpy(req->content, content_data, req->content_length);
201 req->content[req->content_length] = 0;
202
203 }
204
205 return ret; //TODO: maybe return error if content is too big ?
206
207}
208
209
210/***============================================================================================================================
211 * Public functions definitions
212 * These functions can be accessed outside of the file's scope
213 * @see http_request.h for declarations
214 *=============================================================================================================================*/
215
225
226 http_request_s *req = pvPortMalloc(sizeof(http_request_s));
227
228 req->res = http_response_new();
229 req->router = router;
230 req->method = HTTP_GET;
231 req->content = NULL;
232 req->uri = NULL;
233 req->params = NULL;
234 req->retry_count = 0;
235 req->param_count = 0;
236 req->content_length = 0;
237
238 return req;
239
240}
241
248
250
251 for(uint8_t i=0; i<req->param_count; i++){
252 vPortFree(req->params[i].arg);
253 vPortFree(req->params[i].val);
254 }
255
256 if(req->params != NULL){
257 vPortFree(req->params);
258 }
259
260 if(req->uri != NULL){
261 vPortFree(req->uri);
262 }
263
264 if(req->content !=NULL){
265 vPortFree(req->content);
266 }
267
268 req->params = NULL;
269 req->router = NULL;
270 req->content = 0;
271 req->uri = NULL;
272 req->params = NULL;
273 req->retry_count = 0;
274 req->param_count = 0;
275 req->content_length = 0;
276 req->raw_len = 0;
277 req->raw_data = NULL;
278
279 vPortFree(req);
280
281}
282
293uint8_t http_request_parse(http_request_s *req, struct pbuf *p){
294
295 uint8_t ret = 0;
296 req->raw_data = p->payload;
297 req->raw_len = p->tot_len;
298
299 pbuf_ref(p);
300
301 if(lwip_strnstr(req->raw_data , STR_COMMON_CRLF STR_COMMON_CRLF, req->raw_len) != NULL){
302 if(!_http_request_parse_method(req)){
303 }else if(!_http_request_parse_uri(req)){
304 }else if(!_http_request_parse_http_version(req)){
305 }else if(!_http_request_parse_headers(req)){
306 }else if(!_http_request_parse_content(req)){
307 }else{
308 ret = 1; //TODO: maybe use error handler to handle internal errors
309 }
310 }
311
312 pbuf_free(p);
313
314 return ret;
315
316}
#define STR_COMMON_CRLF
Definition: http_defs.h:12
#define STR_COMMON_QMRK
Definition: http_defs.h:13
static char *const http_header_field_str[]
list of HTTP header fields strings.
Definition: http_defs.h:103
static const char *const http_methods[]
list of HTTP method strings.
Definition: http_defs.h:124
#define STR_COMMON_SP
Definition: http_defs.h:11
static const char *const http_versions[]
list of HTTP versions strings.
Definition: http_defs.h:135
#define STR_COMMON_AMPR
Definition: http_defs.h:15
#define STR_COMMON_EQUL
Definition: http_defs.h:14
@ HTTP_DELETE
Definition: http_defs.h:67
@ HTTP_GET
Definition: http_defs.h:63
@ HTTP_VERSION_0_9
Definition: http_defs.h:74
@ HTTP_VERSION_2_0
Definition: http_defs.h:76
@ HTTP_HEADER_FIELD_CONTENT_LENGTH
Definition: http_defs.h:44
void http_request_free(http_request_s *req)
Safely frees an HTTP request instance.
Definition: http_request.c:247
http_request_s * http_request_new(router_fn router)
Creates a new HTTP request instance.
Definition: http_request.c:224
void(* router_fn)(http_request_s *req)
Definition: http_request.h:23
uint8_t http_request_parse(http_request_s *req, struct pbuf *p)
Extracts and parses HTTP method headers and content from received data and calls the router function ...
Definition: http_request.c:293
http_response_s * http_response_new(void)
Creates a new HTTP response instance.
Definition: http_response.c:43
void http_response_free(http_response_s *res)
Safely frees an HTTP response instance.
Definition: http_response.c:61
HTTP uri parameter object.
Definition: http_request.h:35
HTTP request object.
Definition: http_request.h:44
uint8_t param_count
Definition: http_request.h:51
http_version_e http_version
Definition: http_request.h:50
http_method_e method
Definition: http_request.h:49
http_param_s * params
Definition: http_request.h:55
uint16_t content_length
Definition: http_request.h:53
char * content
Definition: http_request.h:48
http_response_s * res
Definition: http_request.h:56
router_fn router
Definition: http_request.h:54
uint8_t retry_count
Definition: http_request.h:52
char * raw_data
Definition: http_request.h:45
uint16_t raw_len
Definition: http_request.h:46