Line data Source code
1 : /**
2 : * @file libhttp.c
3 : * @brief HTTP parsing and building library
4 : *
5 : * @author François Cerbelle (Fanfan), francois@cerbelle.net
6 : *
7 : * @internal
8 : * Created: 15/11/2024
9 : * Revision: none
10 : * Last modified: 2024-11-30 22:15
11 : * Compiler: gcc
12 : * Organization: Cerbelle.net
13 : * Copyright: Copyright (c) 2024, François Cerbelle
14 : *
15 : * This source code is released for free distribution under the terms of the
16 : * GNU General Public License as published by the Free Software Foundation.
17 : */
18 :
19 : #ifdef HAVE_CONFIG_H
20 : #include "config.h"
21 : #endif
22 :
23 : #include "libhttp.h"
24 : #include "sclist.h"
25 : #include "base64.h" /* Base64 encoder and decoder */
26 :
27 : #include <stdlib.h>
28 : #include <stdio.h>
29 : #include <string.h>
30 : #include <assert.h>
31 :
32 : typedef struct HTTP_s {
33 : sclist_t* headers;
34 : char* body;
35 : } HTTP_t;
36 :
37 : typedef struct HTTPHeader_s {
38 : sclistrecord_t* self;
39 : char* name;
40 : char* value;
41 : } HTTPHeader_t;
42 :
43 10 : HTTPHeader_t* HTTPHeader_setname(HTTPHeader_t* header, const char* name) {
44 : char* newname;
45 :
46 10 : assert(header);
47 8 : assert(header->name);
48 8 : assert(header->value);
49 8 : assert(name);
50 :
51 6 : if (name[0]==0) {
52 3 : fprintf(stderr,"ERROR: HTTP_setname empty name\n");
53 3 : return NULL;
54 : }
55 :
56 3 : if (NULL==(newname=realloc(header->name,strlen(name)+1))) {
57 0 : perror("ERROR: HTTPHeader_setname realloc");
58 0 : return NULL;
59 : }
60 3 : header->name = newname;
61 3 : strcpy(header->name,name);
62 3 : return header;
63 : }
64 :
65 231 : char* HTTPHeader_getname(HTTPHeader_t* header) {
66 231 : assert(header);
67 229 : assert(header->name);
68 229 : assert(header->value);
69 :
70 229 : return header->name;
71 : }
72 :
73 10 : HTTPHeader_t* HTTPHeader_setvalue(HTTPHeader_t* header, const char* value) {
74 : char* newvalue;
75 :
76 10 : assert(header);
77 8 : assert(header->name);
78 8 : assert(header->value);
79 8 : assert(value);
80 :
81 6 : if (NULL==(newvalue=realloc(header->value,strlen(value)+1))) {
82 0 : perror("ERROR: HTTPHeader_setvalue realloc");
83 0 : return NULL;
84 : }
85 6 : header->value = newvalue;
86 6 : strcpy(header->value,value);
87 6 : return header;
88 : }
89 :
90 162 : char* HTTPHeader_getvalue(HTTPHeader_t* header) {
91 162 : assert(header);
92 160 : assert(header->name);
93 160 : assert(header->value);
94 :
95 160 : return header->value;
96 : }
97 :
98 204 : static HTTPHeader_t* HTTPHeader_new(const char* name, const char* value) {
99 : HTTPHeader_t* header;
100 :
101 204 : assert(name);
102 204 : assert(value);
103 :
104 204 : if (NULL==(header=malloc(sizeof(struct HTTPHeader_s)))) {
105 0 : perror("ERROR: HTTPHeader_new");
106 0 : return NULL;
107 : }
108 204 : if (NULL==(header->name=strdup(name))) {
109 0 : perror("ERROR: HTTPHeader_new name");
110 0 : free(header);
111 0 : return NULL;
112 : }
113 204 : if (NULL==(header->value=strdup(value))) {
114 0 : perror("ERROR: HTTPHeader_new value");
115 0 : free(header->name);
116 0 : free(header);
117 0 : return NULL;
118 : }
119 204 : header->self = NULL;
120 204 : return header;
121 : }
122 :
123 192 : static void HTTPHeader_del(HTTPHeader_t* header) {
124 192 : assert(header);
125 192 : assert(header->name);
126 192 : assert(header->value);
127 :
128 192 : free(header->name);
129 192 : free(header->value);
130 192 : free(header);
131 192 : }
132 :
133 162 : HTTP_t* HTTP_new() {
134 : HTTP_t* http;
135 :
136 162 : if (NULL==(http=malloc(sizeof(struct HTTP_s)))) {
137 0 : perror("ERROR: HTTP_new HTTP_s");
138 0 : return NULL;
139 : }
140 :
141 162 : if (NULL==(http->body=strdup(""))) {
142 0 : perror("ERROR: HTTP_new body");
143 0 : free(http);
144 0 : return NULL;
145 : };
146 :
147 162 : if (NULL==(http->headers = sclist_new())) {
148 0 : fprintf(stderr,"ERROR: HTTP_new headers\n");
149 0 : free(http->body);
150 0 : free(http);
151 0 : return NULL;
152 : }
153 :
154 162 : return http;
155 : }
156 :
157 140 : void HTTP_del(HTTP_t* http) {
158 : sclistrecord_t* headerlst_entry;
159 :
160 140 : assert(http);
161 138 : assert(http->headers);
162 138 : assert(http->body);
163 :
164 138 : headerlst_entry = sclist_firstrecord(http->headers);
165 318 : while (headerlst_entry) {
166 : HTTPHeader_t* headerlst_value;
167 :
168 : /* Free payload */
169 180 : headerlst_value=sclist_getvalue(headerlst_entry);
170 180 : HTTPHeader_del(headerlst_value);
171 :
172 : /* Remember the entry to delete before moving to the next one */
173 180 : sclistrecord_t* tmp = headerlst_entry;
174 180 : headerlst_entry = sclist_nextrecord(headerlst_entry);
175 180 : sclist_remrecord(http->headers,tmp);
176 : }
177 138 : sclist_del(http->headers);
178 :
179 138 : free(http->body);
180 138 : free(http);
181 138 : }
182 :
183 19 : HTTP_t* HTTP_setbody(HTTP_t* http, const char* body) {
184 : char* newbody;
185 :
186 19 : assert(http);
187 17 : assert(http->headers);
188 17 : assert(http->body);
189 17 : assert(body);
190 :
191 15 : if (NULL==(newbody=realloc(http->body,strlen(body)+1))) {
192 0 : perror("ERROR: HTTP_setbody realloc");
193 0 : return NULL;
194 : }
195 :
196 15 : http->body = newbody;
197 15 : strcpy(http->body, body);
198 15 : return http;
199 : }
200 :
201 17 : char* HTTP_getbody(HTTP_t* http) {
202 17 : assert(http);
203 15 : assert(http->headers);
204 15 : assert(http->body);
205 :
206 15 : return http->body;
207 : }
208 :
209 213 : HTTPHeader_t* HTTP_addheader(HTTP_t* http, const char* name, const char* value) {
210 : HTTPHeader_t* header;
211 :
212 213 : assert(http);
213 211 : assert(http->headers);
214 211 : assert(http->body);
215 211 : assert(name);
216 209 : assert(value);
217 :
218 207 : if (name[0]==0) {
219 3 : fprintf(stderr,"ERROR: HTTP_addheader empty name\n");
220 3 : return NULL;
221 : }
222 :
223 204 : if (NULL==(header = HTTPHeader_new(name,value))) {
224 0 : fprintf(stderr,"ERROR: HTTP_addheader HTTPHeader_new\n");
225 0 : return NULL;
226 : }
227 :
228 204 : if (NULL==(header->self=sclist_addrecord(http->headers,header))) {
229 0 : fprintf(stderr,"ERROR: HTTP_addheader\n");
230 0 : HTTPHeader_del(header);
231 0 : return NULL;
232 : }
233 204 : return header;
234 : }
235 :
236 15 : HTTPHeader_t* HTTP_addbasicauth(HTTP_t* http, const char* login, const char* pass) {
237 : char* auth_encoded;
238 : char* auth;
239 : HTTPHeader_t* header;
240 :
241 15 : assert(http);
242 13 : assert(http->headers);
243 13 : assert(http->body);
244 13 : assert(login);
245 11 : assert(pass);
246 :
247 9 : if (NULL==(auth=malloc(strlen(login)+1+strlen(pass)+1))) {
248 0 : perror("ERROR: HTTP_addbasicauth credentials");
249 0 : return NULL;
250 : }
251 : /* strcpy/strcat expected to be faster than sprintf */
252 9 : strcpy(auth,login);
253 9 : strcat(auth,":");
254 9 : strcat(auth,pass);
255 9 : auth_encoded=base64_encode(auth);
256 9 : free(auth);
257 :
258 9 : if (NULL==(auth=malloc(strlen("Basic ")+strlen(auth_encoded)+1))) {
259 0 : perror("ERROR: HTTP_addbasicauth value");
260 0 : free(auth_encoded);
261 0 : return NULL;
262 : }
263 9 : strcpy(auth,"Basic ");
264 9 : strcat(auth, auth_encoded);
265 9 : free(auth_encoded);
266 :
267 9 : if (NULL==(header=HTTP_addheader(http,"Authorization",auth))) {
268 0 : fprintf(stderr,"ERROR: HTTP_addbasicauth addheader\n");
269 0 : free(auth);
270 0 : return NULL;
271 : }
272 9 : free(auth);
273 9 : return header;
274 : }
275 :
276 84 : HTTPHeader_t* HTTP_firstheader(const HTTP_t* http) {
277 : sclistrecord_t* headerlst_entry;
278 :
279 84 : assert(http);
280 82 : assert(http->headers);
281 82 : assert(http->body);
282 :
283 82 : if (NULL==(headerlst_entry=sclist_firstrecord(http->headers)))
284 11 : return NULL;
285 71 : return sclist_getvalue(headerlst_entry);
286 : }
287 :
288 52 : HTTPHeader_t* HTTP_nextheader(const HTTPHeader_t* header) {
289 : sclistrecord_t* headerlst_entry;
290 :
291 52 : assert(header);
292 48 : assert(header->self);
293 48 : assert(header->name);
294 48 : assert(header->value);
295 :
296 48 : headerlst_entry = header->self;
297 :
298 : /* EOL reached */
299 48 : if (NULL==(headerlst_entry = sclist_nextrecord(headerlst_entry)))
300 21 : return NULL;
301 : else
302 27 : return sclist_getvalue(headerlst_entry);
303 : }
304 :
305 37 : HTTPHeader_t* HTTP_findheader(const HTTPHeader_t* start, const char* name) {
306 : sclistrecord_t* headerlst_entry;
307 :
308 37 : assert(start);
309 35 : assert(start->self);
310 35 : assert(start->name);
311 35 : assert(start->value);
312 35 : assert(name);
313 :
314 33 : headerlst_entry = start->self;
315 :
316 75 : while (headerlst_entry) {
317 : HTTPHeader_t* headerlst_value;
318 : char* header_name;
319 :
320 69 : headerlst_value = sclist_getvalue(headerlst_entry);
321 69 : assert(headerlst_value);
322 69 : header_name = HTTPHeader_getname(headerlst_value);
323 69 : if (0==strcmp(header_name,name))
324 27 : break;
325 42 : headerlst_entry=sclist_nextrecord(headerlst_entry);
326 : }
327 33 : if (!headerlst_entry)
328 : /* Not found */
329 6 : return NULL;
330 : else
331 : /* Found */
332 27 : return sclist_getvalue(headerlst_entry);
333 : }
334 :
335 :
336 19 : HTTP_t* HTTP_remheader(HTTP_t* http, HTTPHeader_t* header) {
337 19 : assert(http);
338 17 : assert(http->headers);
339 17 : assert(http->body);
340 17 : assert(header);
341 15 : assert(header->self);
342 15 : assert(header->name);
343 15 : assert(header->value);
344 :
345 : /* Remove from the header list */
346 15 : if (NULL==sclist_remrecord(http->headers, header->self)) {
347 3 : fprintf(stderr,"ERROR: HTTP_remheader not found\n");
348 3 : return NULL;
349 : }
350 12 : header->self = NULL;
351 12 : HTTPHeader_del(header);
352 :
353 12 : return http;
354 : }
355 :
356 246 : static char* strconcat(char* dst, const char* src) {
357 : char* newdst;
358 :
359 246 : assert(dst);
360 246 : assert(src);
361 :
362 246 : if (NULL==(newdst=realloc(dst,strlen(dst)+strlen(src)+1))) {
363 0 : perror("ERROR: strconcat");
364 0 : return NULL;
365 : }
366 246 : dst = newdst;
367 246 : strcat(dst,src);
368 246 : return dst;
369 : }
370 :
371 28 : char* HTTP_buildheaders(const HTTP_t* http) {
372 28 : char* headers_str=NULL;
373 : sclistrecord_t* headerlst_entry;
374 :
375 28 : assert(http);
376 24 : assert(http->headers);
377 24 : assert(http->body);
378 :
379 24 : if (NULL==(headers_str=strdup(""))) {
380 0 : perror("ERROR: HTTP_buildheaders headers_str");
381 0 : return NULL;
382 : }
383 24 : for (
384 24 : headerlst_entry=sclist_firstrecord(http->headers);
385 48 : headerlst_entry;
386 24 : headerlst_entry = sclist_nextrecord(headerlst_entry)
387 : ) {
388 : HTTPHeader_t* headerlst_value;
389 :
390 24 : headerlst_value = sclist_getvalue(headerlst_entry);
391 24 : assert(headerlst_value);
392 : /** @bug possible memory leak, check strconcat return values */
393 24 : headers_str=strconcat(headers_str,HTTPHeader_getname(headerlst_value));
394 24 : headers_str=strconcat(headers_str,": ");
395 24 : headers_str=strconcat(headers_str,HTTPHeader_getvalue(headerlst_value));
396 24 : headers_str=strconcat(headers_str,"\r\n");
397 : }
398 :
399 24 : return headers_str;
400 : }
401 :
402 : /* @todo redesign headers and implementation to identify different errors */
403 22 : char* HTTP_buildrequest(const HTTP_t* http, const HTTPMethod_t method, const char* uri, const HTTPVersion_t version) {
404 : char* request;
405 : char* newrequest;
406 : char* headers;
407 :
408 22 : assert(uri);
409 :
410 20 : if (NULL==(request=strdup(""))) {
411 0 : perror("ERROR: HTTP_getrequest strdup");
412 0 : return NULL;
413 : }
414 :
415 20 : switch(method) {
416 17 : case HTTPMETHOD_GET:
417 17 : if (NULL==(newrequest=strconcat(request, "GET"))) {
418 0 : fprintf(stderr,"ERROR: HTTP_buildrequest method\n");
419 0 : free(request);
420 0 : return NULL;
421 : }
422 17 : request = newrequest;
423 17 : break;;
424 3 : case HTTPMETHOD_HEAD:
425 : case HTTPMETHOD_POST:
426 : case HTTPMETHOD_PUT:
427 : case HTTPMETHOD_DELETE:
428 : case HTTPMETHOD_CONNECT:
429 : case HTTPMETHOD_OPTIONS:
430 : case HTTPMETHOD_TRACE:
431 : case HTTPMETHOD_PATCH:
432 : case HTTPMETHOD_INVALID:
433 : default:
434 3 : fprintf(stderr,"ERROR: HTTP method not supported\n");
435 3 : free(request);
436 3 : return NULL;
437 : break;;
438 : }
439 :
440 17 : if (uri[0]) {
441 14 : if (NULL==(newrequest=strconcat(request, " "))) {
442 0 : fprintf(stderr,"ERROR: HTTP_buildrequest uri separator\n");
443 0 : free(request);
444 0 : return NULL;
445 : }
446 14 : request = newrequest;
447 14 : if (NULL==(newrequest=strconcat(request, uri))) {
448 0 : fprintf(stderr,"ERROR: HTTP_buildrequest uri\n");
449 0 : free(request);
450 0 : return NULL;
451 : }
452 14 : request = newrequest;
453 : }
454 :
455 17 : if (NULL==(newrequest=strconcat(request, " HTTP/"))) {
456 0 : fprintf(stderr,"ERROR: HTTP_buildrequest HTTP\n");
457 0 : free(request);
458 0 : return NULL;
459 : }
460 17 : request = newrequest;
461 17 : switch(version) {
462 14 : case HTTPVERSION_HTTP11:
463 : case HTTPVERSION_HTTP11b:
464 14 : if (NULL==(newrequest=strconcat(request, "1.1"))) {
465 0 : fprintf(stderr,"ERROR: HTTP_buildrequest Version\n");
466 0 : free(request);
467 0 : return NULL;
468 : }
469 14 : request = newrequest;
470 14 : break;;
471 3 : case HTTPVERSION_HTTP09:
472 : case HTTPVERSION_HTTP10:
473 : case HTTPVERSION_HTTP2:
474 : case HTTPVERSION_HTTP3:
475 : case HTTPVERSION_INVALID:
476 : default:
477 3 : fprintf(stderr,"ERROR: HTTP version not supported\n");
478 3 : free(request);
479 3 : return NULL;
480 : break;;
481 : }
482 14 : if (NULL==(newrequest=strconcat(request, "\r\n"))) {
483 0 : fprintf(stderr,"ERROR: HTTP_buildrequest version separator\n");
484 0 : free(request);
485 0 : return NULL;
486 : }
487 14 : request = newrequest;
488 :
489 14 : if (NULL==(headers=HTTP_buildheaders(http))) {
490 0 : fprintf(stderr,"ERROR: HTTP_buildrequest buildheaders\n");
491 0 : free(request);
492 0 : return NULL;
493 : }
494 :
495 12 : if (NULL==(newrequest=strconcat(request, headers))) {
496 0 : fprintf(stderr,"ERROR: HTTP_buildrequest headers\n");
497 0 : free(request);
498 0 : return NULL;
499 : }
500 12 : request = newrequest;
501 12 : free(headers);
502 :
503 12 : if (NULL==(newrequest=strconcat(request, "\r\n"))) {
504 0 : fprintf(stderr,"ERROR: HTTP_buildrequest header separator\n");
505 0 : free(request);
506 0 : return NULL;
507 : }
508 12 : request = newrequest;
509 :
510 12 : if (NULL==(newrequest=strconcat(request, http->body))) {
511 0 : fprintf(stderr,"ERROR: HTTP_buildrequest body\n");
512 0 : free(request);
513 0 : return NULL;
514 : }
515 12 : request = newrequest;
516 :
517 12 : return request;
518 : }
519 :
520 3 : char* HTTP_buildreply(const HTTP_t* http, const HTTPVersion_t version, const HTTPStatus_t status) {
521 : char* reply;
522 : char* newreply;
523 : char* headers;
524 : char status_str[6];
525 :
526 3 : if (NULL==(reply=strdup(""))) {
527 0 : fprintf(stderr,"ERROR: HTTP_buildreply strdup\n");
528 0 : return NULL;
529 : }
530 :
531 3 : if (NULL==(newreply=strconcat(reply, "HTTP/"))) {
532 0 : fprintf(stderr,"ERROR: HTTP_buildreply HTTP\n");
533 0 : free(reply);
534 0 : return NULL;
535 : }
536 3 : reply = newreply;
537 3 : switch(version) {
538 3 : case HTTPVERSION_HTTP11:
539 : case HTTPVERSION_HTTP11b:
540 3 : if (NULL==(newreply=strconcat(reply, "1.1"))) {
541 0 : fprintf(stderr,"ERROR: HTTP_buildreply Version\n");
542 0 : free(reply);
543 0 : return NULL;
544 : }
545 3 : reply = newreply;
546 3 : break;;
547 0 : case HTTPVERSION_HTTP09:
548 : case HTTPVERSION_HTTP10:
549 : case HTTPVERSION_HTTP2:
550 : case HTTPVERSION_HTTP3:
551 : case HTTPVERSION_INVALID:
552 : default:
553 0 : fprintf(stderr,"ERROR: HTTP version not supported\n");
554 0 : free(reply);
555 0 : return NULL;
556 : break;;
557 : }
558 :
559 3 : snprintf(status_str,6," %d ",status);
560 3 : if (NULL==(newreply=strconcat(reply, status_str))) {
561 0 : fprintf(stderr,"ERROR: HTTP_buildreply status code\n");
562 0 : free(reply);
563 0 : return NULL;
564 : }
565 3 : reply = newreply;
566 :
567 3 : if (NULL==(newreply=strconcat(reply, "Dummy Reason"))) {
568 0 : fprintf(stderr,"ERROR: HTTP_buildreply reason phrase\n");
569 0 : free(reply);
570 0 : return NULL;
571 : }
572 3 : reply = newreply;
573 :
574 3 : if (NULL==(newreply=strconcat(reply, "\r\n"))) {
575 0 : fprintf(stderr,"ERROR: HTTP_buildreply reason separator\n");
576 0 : free(reply);
577 0 : return NULL;
578 : }
579 3 : reply = newreply;
580 :
581 3 : if (NULL==(headers=HTTP_buildheaders(http))) {
582 0 : fprintf(stderr,"ERROR: HTTP_buildreply buildheaders\n");
583 0 : free(reply);
584 0 : return NULL;
585 : }
586 :
587 3 : if (NULL==(newreply=strconcat(reply, headers))) {
588 0 : fprintf(stderr,"ERROR: HTTP_buildreply headers\n");
589 0 : free(reply);
590 0 : return NULL;
591 : }
592 3 : reply = newreply;
593 3 : free(headers);
594 :
595 3 : if (NULL==(newreply=strconcat(reply, "\r\n"))) {
596 0 : fprintf(stderr,"ERROR: HTTP_buildreply header separator\n");
597 0 : free(reply);
598 0 : return NULL;
599 : }
600 3 : reply = newreply;
601 :
602 3 : if (NULL==(newreply=strconcat(reply, http->body))) {
603 0 : fprintf(stderr,"ERROR: HTTP_buildreply body\n");
604 0 : free(reply);
605 0 : return NULL;
606 : }
607 3 : reply = newreply;
608 :
609 3 : return reply;
610 : }
611 :
612 0 : HTTP_t* HTTP_parseheaders(const char* text) {
613 : (void) text;
614 0 : fprintf(stderr,"Not yet implemented\n");
615 0 : abort();
616 : return NULL;
617 : }
618 :
619 0 : void HTTP_parserequest(const char* request, HTTP_t** http, HTTPMethod_t method, char* uri, HTTPVersion_t version) {
620 : (void) request;
621 : (void) http;
622 : (void) method;
623 : (void) uri;
624 : (void) version;
625 0 : fprintf(stderr,"Not yet implemented\n");
626 0 : abort();
627 : }
628 :
629 0 : void HTTP_parsereply(const char* reply, HTTP_t** http, HTTPVersion_t* version, HTTPStatus_t* status) {
630 : (void) reply;
631 : (void) http;
632 : (void) version;
633 : (void) status;
634 0 : fprintf(stderr,"Not yet implemented\n");
635 0 : abort();
636 : }
|