VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/http-server.cpp@ 94157

Last change on this file since 94157 was 93141, checked in by vboxsync, 3 years ago

IPRT/http-server: Also free pszUrl of RTHTTPSERVERREQ in rtHttpServerReqFree().

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 47.0 KB
Line 
1/* $Id: http-server.cpp 93141 2022-01-07 16:09:28Z vboxsync $ */
2/** @file
3 * Simple HTTP server (RFC 7231) implementation.
4 *
5 * Known limitations so far:
6 * - Only HTTP 1.1.
7 * - Only supports GET + HEAD methods so far.
8 * - Only supports UTF-8 charset.
9 * - Only supports plain text and octet stream MIME types.
10 * - No content compression ("gzip", "x-gzip", ++).
11 * - No caching.
12 * - No redirections (via 302).
13 * - No encryption (TLS).
14 * - No IPv6 support.
15 * - No multi-threading.
16 *
17 * For WebDAV (optional via IPRT_HTTP_WITH_WEBDAV):
18 * - Only OPTIONS + PROPLIST methods are implemented (e.g. simple read-only support).
19 * - No pagination support for directory listings.
20 */
21
22/*
23 * Copyright (C) 2020-2022 Oracle Corporation
24 *
25 * This file is part of VirtualBox Open Source Edition (OSE), as
26 * available from http://www.virtualbox.org. This file is free software;
27 * you can redistribute it and/or modify it under the terms of the GNU
28 * General Public License (GPL) as published by the Free Software
29 * Foundation, in version 2 as it comes in the "COPYING" file of the
30 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
31 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
32 *
33 * The contents of this file may alternatively be used under the terms
34 * of the Common Development and Distribution License Version 1.0
35 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
36 * VirtualBox OSE distribution, in which case the provisions of the
37 * CDDL are applicable instead of those of the GPL.
38 *
39 * You may elect to license modified versions of this file under the
40 * terms and conditions of either the GPL or the CDDL or both.
41 */
42
43
44/*********************************************************************************************************************************
45* Header Files *
46*********************************************************************************************************************************/
47#define LOG_GROUP RTLOGGROUP_HTTP
48#include <iprt/http.h>
49#include <iprt/http-server.h>
50#include "internal/iprt.h"
51#include "internal/magics.h"
52
53#include <iprt/asm.h>
54#include <iprt/assert.h>
55#include <iprt/circbuf.h>
56#include <iprt/ctype.h>
57#include <iprt/err.h>
58#include <iprt/file.h> /* For file mode flags. */
59#include <iprt/getopt.h>
60#include <iprt/mem.h>
61#include <iprt/log.h>
62#include <iprt/path.h>
63#include <iprt/poll.h>
64#include <iprt/socket.h>
65#include <iprt/sort.h>
66#include <iprt/string.h>
67#include <iprt/system.h>
68#include <iprt/tcp.h>
69
70
71/*********************************************************************************************************************************
72* Structures and Typedefs *
73*********************************************************************************************************************************/
74/**
75 * Internal HTTP server instance.
76 */
77typedef struct RTHTTPSERVERINTERNAL
78{
79 /** Magic value. */
80 uint32_t u32Magic;
81 /** Callback table. */
82 RTHTTPSERVERCALLBACKS Callbacks;
83 /** Pointer to TCP server instance. */
84 PRTTCPSERVER pTCPServer;
85 /** Pointer to user-specific data. Optional. */
86 void *pvUser;
87 /** Size of user-specific data. Optional. */
88 size_t cbUser;
89} RTHTTPSERVERINTERNAL;
90/** Pointer to an internal HTTP server instance. */
91typedef RTHTTPSERVERINTERNAL *PRTHTTPSERVERINTERNAL;
92
93
94/*********************************************************************************************************************************
95* Defined Constants And Macros *
96*********************************************************************************************************************************/
97/** Validates a handle and returns VERR_INVALID_HANDLE if not valid. */
98#define RTHTTPSERVER_VALID_RETURN_RC(hHttpServer, a_rc) \
99 do { \
100 AssertPtrReturn((hHttpServer), (a_rc)); \
101 AssertReturn((hHttpServer)->u32Magic == RTHTTPSERVER_MAGIC, (a_rc)); \
102 } while (0)
103
104/** Validates a handle and returns VERR_INVALID_HANDLE if not valid. */
105#define RTHTTPSERVER_VALID_RETURN(hHttpServer) RTHTTPSERVER_VALID_RETURN_RC((hHttpServer), VERR_INVALID_HANDLE)
106
107/** Validates a handle and returns (void) if not valid. */
108#define RTHTTPSERVER_VALID_RETURN_VOID(hHttpServer) \
109 do { \
110 AssertPtrReturnVoid(hHttpServer); \
111 AssertReturnVoid((hHttpServer)->u32Magic == RTHTTPSERVER_MAGIC); \
112 } while (0)
113
114
115/** Handles a HTTP server callback with no arguments and returns. */
116#define RTHTTPSERVER_HANDLE_CALLBACK_RET(a_Name) \
117 do \
118 { \
119 PRTHTTPSERVERCALLBACKS pCallbacks = &pClient->pServer->Callbacks; \
120 if (pCallbacks->a_Name) \
121 { \
122 RTHTTPCALLBACKDATA Data = { &pClient->State }; \
123 return pCallbacks->a_Name(&Data); \
124 } \
125 return VERR_NOT_IMPLEMENTED; \
126 } while (0)
127
128/** Handles a HTTP server callback with no arguments and sets rc accordingly. */
129#define RTHTTPSERVER_HANDLE_CALLBACK(a_Name) \
130 do \
131 { \
132 PRTHTTPSERVERCALLBACKS pCallbacks = &pClient->pServer->Callbacks; \
133 if (pCallbacks->a_Name) \
134 { \
135 RTHTTPCALLBACKDATA Data = { &pClient->State, pClient->pServer->pvUser, pClient->pServer->cbUser }; \
136 rc = pCallbacks->a_Name(&Data); \
137 } \
138 } while (0)
139
140/** Handles a HTTP server callback with arguments and sets rc accordingly. */
141#define RTHTTPSERVER_HANDLE_CALLBACK_VA(a_Name, ...) \
142 do \
143 { \
144 PRTHTTPSERVERCALLBACKS pCallbacks = &pClient->pServer->Callbacks; \
145 if (pCallbacks->a_Name) \
146 { \
147 RTHTTPCALLBACKDATA Data = { &pClient->State, pClient->pServer->pvUser, pClient->pServer->cbUser }; \
148 rc = pCallbacks->a_Name(&Data, __VA_ARGS__); \
149 } \
150 } while (0)
151
152/** Handles a HTTP server callback with arguments and returns. */
153#define RTHTTPSERVER_HANDLE_CALLBACK_VA_RET(a_Name, ...) \
154 do \
155 { \
156 PRTHTTPSERVERCALLBACKS pCallbacks = &pClient->pServer->Callbacks; \
157 if (pCallbacks->a_Name) \
158 { \
159 RTHTTPCALLBACKDATA Data = { &pClient->State, pClient->pServer->pvUser, pClient->pServer->cbUser }; \
160 return pCallbacks->a_Name(&Data, __VA_ARGS__); \
161 } \
162 } while (0)
163
164
165/*********************************************************************************************************************************
166* Structures and Typedefs *
167*********************************************************************************************************************************/
168
169/**
170 * Structure for maintaining an internal HTTP server client.
171 */
172typedef struct RTHTTPSERVERCLIENT
173{
174 /** Pointer to internal server state. */
175 PRTHTTPSERVERINTERNAL pServer;
176 /** Socket handle the client is bound to. */
177 RTSOCKET hSocket;
178 /** Actual client state. */
179 RTHTTPSERVERCLIENTSTATE State;
180} RTHTTPSERVERCLIENT;
181/** Pointer to an internal HTTP server client state. */
182typedef RTHTTPSERVERCLIENT *PRTHTTPSERVERCLIENT;
183
184/** Function pointer declaration for a specific HTTP server method handler. */
185typedef DECLCALLBACKTYPE(int, FNRTHTTPSERVERMETHOD,(PRTHTTPSERVERCLIENT pClient, PRTHTTPSERVERREQ pReq));
186/** Pointer to a FNRTHTTPSERVERMETHOD(). */
187typedef FNRTHTTPSERVERMETHOD *PFNRTHTTPSERVERMETHOD;
188
189/**
190 * Static lookup table for some file extensions <-> MIME type. Add more as needed.
191 * Keep this alphabetical (file extension).
192 */
193static const struct
194{
195 /** File extension. */
196 const char *pszExt;
197 /** MIME type. */
198 const char *pszMIMEType;
199} s_aFileExtMIMEType[] = {
200 { ".arj", "application/x-arj-compressed" },
201 { ".asf", "video/x-ms-asf" },
202 { ".avi", "video/x-msvideo" },
203 { ".bmp", "image/bmp" },
204 { ".css", "text/css" },
205 { ".doc", "application/msword" },
206 { ".exe", "application/octet-stream" },
207 { ".gif", "image/gif" },
208 { ".gz", "application/x-gunzip" },
209 { ".htm", "text/html" },
210 { ".html", "text/html" },
211 { ".ico", "image/x-icon" },
212 { ".js", "application/x-javascript" },
213 { ".json", "text/json" },
214 { ".jpg", "image/jpeg" },
215 { ".jpeg", "image/jpeg" },
216 { ".ogg", "application/ogg" },
217 { ".m3u", "audio/x-mpegurl" },
218 { ".m4v", "video/x-m4v" },
219 { ".mid", "audio/mid" },
220 { ".mov", "video/quicktime" },
221 { ".mp3", "audio/x-mp3" },
222 { ".mp4", "video/mp4" },
223 { ".mpg", "video/mpeg" },
224 { ".mpeg", "video/mpeg" },
225 { ".pdf", "application/pdf" },
226 { ".png", "image/png" },
227 { ".ra", "audio/x-pn-realaudio" },
228 { ".ram", "audio/x-pn-realaudio" },
229 { ".rar", "application/x-arj-compressed" },
230 { ".rtf", "application/rtf" },
231 { ".shtm", "text/html" },
232 { ".shtml", "text/html" },
233 { ".svg", "image/svg+xml" },
234 { ".swf", "application/x-shockwave-flash" },
235 { ".torrent", "application/x-bittorrent" },
236 { ".tar", "application/x-tar" },
237 { ".tgz", "application/x-tar-gz" },
238 { ".ttf", "application/x-font-ttf" },
239 { ".txt", "text/plain" },
240 { ".wav", "audio/x-wav" },
241 { ".webm", "video/webm" },
242 { ".xml", "text/xml" },
243 { ".xls", "application/excel" },
244 { ".xsl", "application/xml" },
245 { ".xslt", "application/xml" },
246 { ".zip", "application/x-zip-compressed" },
247 { NULL, NULL }
248};
249
250
251/*********************************************************************************************************************************
252* Internal Functions *
253*********************************************************************************************************************************/
254
255/** @name Method handlers.
256 * @{
257 */
258static FNRTHTTPSERVERMETHOD rtHttpServerHandleGET;
259static FNRTHTTPSERVERMETHOD rtHttpServerHandleHEAD;
260#ifdef IPRT_HTTP_WITH_WEBDAV
261 static FNRTHTTPSERVERMETHOD rtHttpServerHandleOPTIONS;
262 static FNRTHTTPSERVERMETHOD rtHttpServerHandlePROPFIND;
263#endif
264/** @} */
265
266/**
267 * Structure for maintaining a single method entry for the methods table.
268 */
269typedef struct RTHTTPSERVERMETHOD_ENTRY
270{
271 /** Method ID. */
272 RTHTTPMETHOD enmMethod;
273 /** Function pointer invoked to handle the command. */
274 PFNRTHTTPSERVERMETHOD pfnMethod;
275} RTHTTPSERVERMETHOD_ENTRY;
276/** Pointer to a command entry. */
277typedef RTHTTPSERVERMETHOD_ENTRY *PRTHTTPMETHOD_ENTRY;
278
279
280
281/*********************************************************************************************************************************
282* Global Variables *
283*********************************************************************************************************************************/
284/**
285 * Table of handled methods.
286 */
287static const RTHTTPSERVERMETHOD_ENTRY g_aMethodMap[] =
288{
289 { RTHTTPMETHOD_GET, rtHttpServerHandleGET },
290 { RTHTTPMETHOD_HEAD, rtHttpServerHandleHEAD },
291#ifdef IPRT_HTTP_WITH_WEBDAV
292 { RTHTTPMETHOD_OPTIONS, rtHttpServerHandleOPTIONS },
293 { RTHTTPMETHOD_PROPFIND, rtHttpServerHandlePROPFIND },
294#endif
295 { RTHTTPMETHOD_END, NULL }
296};
297
298/** Maximum length in characters a HTTP server path can have (excluding termination). */
299#define RTHTTPSERVER_MAX_PATH RTPATH_MAX
300
301
302/*********************************************************************************************************************************
303* Internal functions *
304*********************************************************************************************************************************/
305
306/**
307 * Guesses the HTTP MIME type based on a given file extension.
308 *
309 * Note: Has to include the beginning dot, e.g. ".mp3" (see IPRT).
310 *
311 * @returns Guessed MIME type, or "application/octet-stream" if not found.
312 * @param pszFileExt File extension to guess MIME type for.
313 */
314static const char *rtHttpServerGuessMIMEType(const char *pszFileExt)
315{
316 if (pszFileExt)
317 {
318 size_t i = 0;
319 while (s_aFileExtMIMEType[i++].pszExt) /* Slow, but does the job for now. */
320 {
321 if (!RTStrICmp(pszFileExt, s_aFileExtMIMEType[i].pszExt))
322 return s_aFileExtMIMEType[i].pszMIMEType;
323 }
324 }
325
326 return "application/octet-stream";
327}
328
329/**
330 * Initializes a HTTP body.
331 *
332 * @param pBody Body to initialize.
333 * @param cbSize Size of body (in bytes) to allocate. Optional and can be 0.
334 */
335static int rtHttpServerBodyInit(PRTHTTPBODY pBody, size_t cbSize)
336{
337 if (cbSize)
338 {
339 pBody->pvBody = RTMemAlloc(cbSize);
340 AssertPtrReturn(pBody->pvBody, VERR_NO_MEMORY);
341 pBody->cbBodyAlloc = cbSize;
342 }
343 else
344 {
345 pBody->pvBody = NULL;
346 pBody->cbBodyAlloc = 0;
347 }
348
349 pBody->cbBodyUsed = 0;
350 pBody->offBody = 0;
351
352 return VINF_SUCCESS;
353}
354
355/**
356 * Destroys a HTTP body.
357 *
358 * @param pBody Body to destroy.
359 */
360static void rtHttpServerBodyDestroy(PRTHTTPBODY pBody)
361{
362 if (!pBody)
363 return;
364
365 if (pBody->pvBody)
366 {
367 Assert(pBody->cbBodyAlloc);
368
369 RTMemFree(pBody->pvBody);
370 pBody->pvBody = NULL;
371 }
372
373 pBody->cbBodyAlloc = 0;
374 pBody->cbBodyUsed = 0;
375 pBody->offBody = 0;
376}
377
378/**
379 * Allocates and initializes a new client request.
380 *
381 * @returns Pointer to the new client request, or NULL on OOM.
382 * Needs to be free'd with rtHttpServerReqFree().
383 */
384static PRTHTTPSERVERREQ rtHttpServerReqAlloc(void)
385{
386 PRTHTTPSERVERREQ pReq = (PRTHTTPSERVERREQ)RTMemAllocZ(sizeof(RTHTTPSERVERREQ));
387 AssertPtrReturn(pReq, NULL);
388
389 int rc2 = RTHttpHeaderListInit(&pReq->hHdrLst);
390 AssertRC(rc2);
391
392 rc2 = rtHttpServerBodyInit(&pReq->Body, 0 /* cbSize */);
393 AssertRC(rc2);
394
395 return pReq;
396}
397
398/**
399 * Frees a formerly allocated client request.
400 *
401 * @param pReq Pointer to client request to free.
402 * The pointer will be invalid on return.
403 */
404static void rtHttpServerReqFree(PRTHTTPSERVERREQ pReq)
405{
406 if (!pReq)
407 return;
408
409 RTStrFree(pReq->pszUrl);
410
411 RTHttpHeaderListDestroy(pReq->hHdrLst);
412
413 rtHttpServerBodyDestroy(&pReq->Body);
414
415 RTMemFree(pReq);
416 pReq = NULL;
417}
418
419/**
420 * Initializes a HTTP server response with an allocated body size.
421 *
422 * @returns VBox status code.
423 * @param pResp HTTP server response to initialize.
424 * @param cbBody Body size (in bytes) to allocate.
425 */
426RTR3DECL(int) RTHttpServerResponseInitEx(PRTHTTPSERVERRESP pResp, size_t cbBody)
427{
428 pResp->enmSts = RTHTTPSTATUS_INTERNAL_NOT_SET;
429
430 int rc = RTHttpHeaderListInit(&pResp->hHdrLst);
431 AssertRCReturn(rc, rc);
432
433 rc = rtHttpServerBodyInit(&pResp->Body, cbBody);
434
435 return rc;
436}
437
438/**
439 * Initializes a HTTP server response.
440 *
441 * @returns VBox status code.
442 * @param pResp HTTP server response to initialize.
443 */
444RTR3DECL(int) RTHttpServerResponseInit(PRTHTTPSERVERRESP pResp)
445{
446 return RTHttpServerResponseInitEx(pResp, 0 /* cbBody */);
447}
448
449/**
450 * Destroys a formerly initialized HTTP server response.
451 *
452 * @param pResp Pointer to HTTP server response to destroy.
453 */
454RTR3DECL(void) RTHttpServerResponseDestroy(PRTHTTPSERVERRESP pResp)
455{
456 if (!pResp)
457 return;
458
459 pResp->enmSts = RTHTTPSTATUS_INTERNAL_NOT_SET;
460
461 RTHttpHeaderListDestroy(pResp->hHdrLst);
462
463 rtHttpServerBodyDestroy(&pResp->Body);
464}
465
466
467/*********************************************************************************************************************************
468* Protocol Functions *
469*********************************************************************************************************************************/
470
471/**
472 * Logs the HTTP protocol communication to the debug logger (2).
473 *
474 * @param pClient Client to log communication for.
475 * @param fWrite Whether the server is writing (to client) or reading (from client).
476 * @param pszData Actual protocol communication data to log.
477 */
478static void rtHttpServerLogProto(PRTHTTPSERVERCLIENT pClient, bool fWrite, const char *pszData)
479{
480 RT_NOREF(pClient);
481
482#ifdef LOG_ENABLED
483 if (!pszData) /* Nothing to log? Bail out. */
484 return;
485
486 char **ppapszStrings;
487 size_t cStrings;
488 int rc2 = RTStrSplit(pszData, strlen(pszData), RTHTTPSERVER_HTTP11_EOL_STR, &ppapszStrings, &cStrings);
489 if (RT_SUCCESS(rc2))
490 {
491 for (size_t i = 0; i < cStrings; i++)
492 {
493 Log2(("%s %s\n", fWrite ? ">" : "<", ppapszStrings[i]));
494 RTStrFree(ppapszStrings[i]);
495 }
496
497 RTMemFree(ppapszStrings);
498 }
499#else
500 RT_NOREF(fWrite, pszData);
501#endif
502}
503
504/**
505 * Writes HTTP protocol communication data to a connected client.
506 *
507 * @returns VBox status code.
508 * @param pClient Client to write data to.
509 * @param pszData Data to write. Must be zero-terminated.
510 */
511static int rtHttpServerWriteProto(PRTHTTPSERVERCLIENT pClient, const char *pszData)
512{
513 rtHttpServerLogProto(pClient, true /* fWrite */, pszData);
514 return RTTcpWrite(pClient->hSocket, pszData, strlen(pszData));
515}
516
517/**
518 * Main function for sending a response back to the client.
519 *
520 * @returns VBox status code.
521 * @param pClient Client to reply to.
522 * @param enmSts Status code to send.
523 */
524static int rtHttpServerSendResponse(PRTHTTPSERVERCLIENT pClient, RTHTTPSTATUS enmSts)
525{
526 char *pszResp;
527 int rc = RTStrAPrintf(&pszResp,
528 "%s %RU32 %s\r\n", RTHTTPVER_1_1_STR, enmSts, RTHttpStatusToStr(enmSts));
529 AssertRCReturn(rc, rc);
530 rc = rtHttpServerWriteProto(pClient, pszResp);
531 RTStrFree(pszResp);
532
533 LogFlowFuncLeaveRC(rc);
534 return rc;
535}
536
537/**
538 * Main function for sending response headers back to the client.
539 *
540 * @returns VBox status code.
541 * @param pClient Client to reply to.
542 * @param pHdrLst Header list to send. Optional and can be NULL.
543 */
544static int rtHttpServerSendResponseHdrEx(PRTHTTPSERVERCLIENT pClient, PRTHTTPHEADERLIST pHdrLst)
545{
546 RTHTTPHEADERLIST HdrLst;
547 int rc = RTHttpHeaderListInit(&HdrLst);
548 AssertRCReturn(rc, rc);
549
550#ifdef DEBUG
551 /* Include a timestamp when running a debug build. */
552 RTTIMESPEC tsNow;
553 char szTS[64];
554 RTTimeSpecToString(RTTimeNow(&tsNow), szTS, sizeof(szTS));
555 rc = RTHttpHeaderListAdd(HdrLst, "Date", szTS, strlen(szTS), RTHTTPHEADERLISTADD_F_BACK);
556 AssertRCReturn(rc, rc);
557#endif
558
559 /* Note: Deliberately don't include the VBox version due to security reasons. */
560 rc = RTHttpHeaderListAdd(HdrLst, "Server", "Oracle VirtualBox", strlen("Oracle VirtualBox"), RTHTTPHEADERLISTADD_F_BACK);
561 AssertRCReturn(rc, rc);
562
563#ifdef IPRT_HTTP_WITH_WEBDAV
564 rc = RTHttpHeaderListAdd(HdrLst, "Allow", "GET, HEAD, PROPFIND", strlen("GET, HEAD, PROPFIND"), RTHTTPHEADERLISTADD_F_BACK);
565 AssertRCReturn(rc, rc);
566 rc = RTHttpHeaderListAdd(HdrLst, "DAV", "1", strlen("1"), RTHTTPHEADERLISTADD_F_BACK); /* Note: v1 is sufficient for read-only access. */
567 AssertRCReturn(rc, rc);
568#endif
569
570 char *pszHdr = NULL;
571
572 size_t i = 0;
573 const char *pszEntry;
574 while ((pszEntry = RTHttpHeaderListGetByOrdinal(HdrLst, i++)) != NULL)
575 {
576 rc = RTStrAAppend(&pszHdr, pszEntry);
577 AssertRCBreak(rc);
578 rc = RTStrAAppend(&pszHdr, RTHTTPSERVER_HTTP11_EOL_STR);
579 AssertRCBreak(rc);
580 }
581
582 /* Append optional headers, if any. */
583 if (pHdrLst)
584 {
585 i = 0;
586 while ((pszEntry = RTHttpHeaderListGetByOrdinal(*pHdrLst, i++)) != NULL)
587 {
588 rc = RTStrAAppend(&pszHdr, pszEntry);
589 AssertRCBreak(rc);
590 rc = RTStrAAppend(&pszHdr, RTHTTPSERVER_HTTP11_EOL_STR);
591 AssertRCBreak(rc);
592 }
593 }
594
595 if (RT_SUCCESS(rc))
596 {
597 /* Append trailing EOL. */
598 rc = RTStrAAppend(&pszHdr, RTHTTPSERVER_HTTP11_EOL_STR);
599 if (RT_SUCCESS(rc))
600 rc = rtHttpServerWriteProto(pClient, pszHdr);
601 }
602
603 RTStrFree(pszHdr);
604
605 RTHttpHeaderListDestroy(HdrLst);
606
607 LogFlowFunc(("rc=%Rrc\n", rc));
608 return rc;
609}
610
611/**
612 * Replies with (three digit) response status back to the client, extended version.
613 *
614 * @returns VBox status code.
615 * @param pClient Client to reply to.
616 * @param enmSts Status code to send.
617 * @param pHdrLst Header list to send. Optional and can be NULL.
618 */
619static int rtHttpServerSendResponseEx(PRTHTTPSERVERCLIENT pClient, RTHTTPSTATUS enmSts, PRTHTTPHEADERLIST pHdrLst)
620{
621 int rc = rtHttpServerSendResponse(pClient, enmSts);
622 if (RT_SUCCESS(rc))
623 rc = rtHttpServerSendResponseHdrEx(pClient, pHdrLst);
624
625 return rc;
626}
627
628/**
629 * Replies with (three digit) response status back to the client.
630 *
631 * @returns VBox status code.
632 * @param pClient Client to reply to.
633 * @param enmSts Status code to send.
634 */
635static int rtHttpServerSendResponseSimple(PRTHTTPSERVERCLIENT pClient, RTHTTPSTATUS enmSts)
636{
637 return rtHttpServerSendResponseEx(pClient, enmSts, NULL /* pHdrLst */);
638}
639
640/**
641 * Sends a chunk of the response body to the client.
642 *
643 * @returns VBox status code.
644 * @param pClient Client to send body to.
645 * @param pvBuf Data buffer to send.
646 * @param cbBuf Size (in bytes) of data buffer to send.
647 * @param pcbSent Where to store the sent bytes. Optional and can be NULL.
648 */
649static int rtHttpServerSendResponseBody(PRTHTTPSERVERCLIENT pClient, void *pvBuf, size_t cbBuf, size_t *pcbSent)
650{
651 int rc = RTTcpWrite(pClient->hSocket, pvBuf, cbBuf);
652 if ( RT_SUCCESS(rc)
653 && pcbSent)
654 *pcbSent = cbBuf;
655
656 return rc;
657}
658
659/**
660 * Resolves a VBox status code to a HTTP status code.
661 *
662 * @returns Resolved HTTP status code, or RTHTTPSTATUS_INTERNALSERVERERROR if not able to resolve.
663 * @param rc VBox status code to resolve.
664 */
665static RTHTTPSTATUS rtHttpServerRcToStatus(int rc)
666{
667 switch (rc)
668 {
669 case VINF_SUCCESS: return RTHTTPSTATUS_OK;
670 case VERR_INVALID_PARAMETER: return RTHTTPSTATUS_BADREQUEST;
671 case VERR_INVALID_POINTER: return RTHTTPSTATUS_BADREQUEST;
672 case VERR_NOT_IMPLEMENTED: return RTHTTPSTATUS_NOTIMPLEMENTED;
673 case VERR_NOT_SUPPORTED: return RTHTTPSTATUS_NOTIMPLEMENTED;
674 case VERR_PATH_NOT_FOUND: return RTHTTPSTATUS_NOTFOUND;
675 case VERR_FILE_NOT_FOUND: return RTHTTPSTATUS_NOTFOUND;
676 case VERR_IS_A_DIRECTORY: return RTHTTPSTATUS_FORBIDDEN;
677 case VERR_NOT_FOUND: return RTHTTPSTATUS_NOTFOUND;
678 default:
679 break;
680 }
681
682 AssertMsgFailed(("rc=%Rrc not handled for HTTP status\n", rc));
683 return RTHTTPSTATUS_INTERNALSERVERERROR;
684}
685
686
687/*********************************************************************************************************************************
688* Command Protocol Handlers *
689*********************************************************************************************************************************/
690
691/**
692 * Handler for the GET method.
693 *
694 * @returns VBox status code.
695 * @param pClient Client to handle GET method for.
696 * @param pReq Client request to handle.
697 */
698static DECLCALLBACK(int) rtHttpServerHandleGET(PRTHTTPSERVERCLIENT pClient, PRTHTTPSERVERREQ pReq)
699{
700 LogFlowFuncEnter();
701
702 int rc = VINF_SUCCESS;
703
704 /* If a low-level GET request handler is defined, call it and return. */
705 RTHTTPSERVER_HANDLE_CALLBACK_VA_RET(pfnOnGetRequest, pReq);
706
707 RTFSOBJINFO fsObj;
708 RT_ZERO(fsObj); /* Shut up MSVC. */
709
710 char *pszMIMEHint = NULL;
711
712 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnQueryInfo, pReq, &fsObj, &pszMIMEHint);
713 if (RT_FAILURE(rc))
714 return rc;
715
716 void *pvHandle = NULL;
717 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnOpen, pReq, &pvHandle);
718
719 if (RT_SUCCESS(rc))
720 {
721 size_t cbBuf = _64K;
722 void *pvBuf = RTMemAlloc(cbBuf);
723 AssertPtrReturn(pvBuf, VERR_NO_MEMORY);
724
725 for (;;)
726 {
727 RTHTTPHEADERLIST HdrLst;
728 rc = RTHttpHeaderListInit(&HdrLst);
729 AssertRCReturn(rc, rc);
730
731 char szVal[16];
732
733 /* Note: For directories fsObj.cbObject contains the actual size (in bytes)
734 * of the body data for the directory listing. */
735
736 ssize_t cch = RTStrPrintf2(szVal, sizeof(szVal), "%RU64", fsObj.cbObject);
737 AssertBreakStmt(cch, VERR_BUFFER_OVERFLOW);
738 rc = RTHttpHeaderListAdd(HdrLst, "Content-Length", szVal, strlen(szVal), RTHTTPHEADERLISTADD_F_BACK);
739 AssertRCBreak(rc);
740
741 cch = RTStrPrintf2(szVal, sizeof(szVal), "identity");
742 AssertBreakStmt(cch, VERR_BUFFER_OVERFLOW);
743 rc = RTHttpHeaderListAdd(HdrLst, "Content-Encoding", szVal, strlen(szVal), RTHTTPHEADERLISTADD_F_BACK);
744 AssertRCBreak(rc);
745
746 if (pszMIMEHint == NULL)
747 {
748 const char *pszMIME = rtHttpServerGuessMIMEType(RTPathSuffix(pReq->pszUrl));
749 rc = RTHttpHeaderListAdd(HdrLst, "Content-Type", pszMIME, strlen(pszMIME), RTHTTPHEADERLISTADD_F_BACK);
750 }
751 else
752 {
753 rc = RTHttpHeaderListAdd(HdrLst, "Content-Type", pszMIMEHint, strlen(pszMIMEHint), RTHTTPHEADERLISTADD_F_BACK);
754 RTStrFree(pszMIMEHint);
755 pszMIMEHint = NULL;
756 }
757 AssertRCReturn(rc, rc);
758
759 if (pClient->State.msKeepAlive)
760 {
761 /* If the client requested to keep alive the connection,
762 * always override this with 30s and report this back to the client. */
763 pClient->State.msKeepAlive = RT_MS_30SEC; /** @todo Make this configurable. */
764#ifdef DEBUG_andy
765 pClient->State.msKeepAlive = 5000;
766#endif
767 cch = RTStrPrintf2(szVal, sizeof(szVal), "timeout=%RU64", pClient->State.msKeepAlive / RT_MS_1SEC); /** @todo No pipelining support here yet. */
768 AssertBreakStmt(cch, VERR_BUFFER_OVERFLOW);
769 rc = RTHttpHeaderListAdd(HdrLst, "Keep-Alive", szVal, strlen(szVal), RTHTTPHEADERLISTADD_F_BACK);
770 AssertRCReturn(rc, rc);
771 }
772
773 rc = rtHttpServerSendResponseEx(pClient, RTHTTPSTATUS_OK, &HdrLst);
774
775 RTHttpHeaderListDestroy(HdrLst);
776
777 if (rc == VERR_BROKEN_PIPE) /* Could happen on fast reloads. */
778 break;
779 AssertRCReturn(rc, rc);
780
781 size_t cbToRead = fsObj.cbObject;
782 size_t cbRead = 0; /* Shut up GCC. */
783 size_t cbWritten = 0; /* Ditto. */
784 while (cbToRead)
785 {
786 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnRead, pvHandle, pvBuf, RT_MIN(cbBuf, cbToRead), &cbRead);
787 if (RT_FAILURE(rc))
788 break;
789 rc = rtHttpServerSendResponseBody(pClient, pvBuf, cbRead, &cbWritten);
790 AssertBreak(cbToRead >= cbWritten);
791 cbToRead -= cbWritten;
792 if (rc == VERR_NET_CONNECTION_RESET_BY_PEER) /* Clients often apruptly abort the connection when done. */
793 {
794 rc = VINF_SUCCESS;
795 break;
796 }
797 AssertRCBreak(rc);
798 }
799
800 break;
801 } /* for (;;) */
802
803 RTMemFree(pvBuf);
804
805 int rc2 = rc; /* Save rc. */
806
807 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnClose, pvHandle);
808
809 if (RT_FAILURE(rc2)) /* Restore original rc on failure. */
810 rc = rc2;
811 }
812
813 LogFlowFuncLeaveRC(rc);
814 return rc;
815}
816
817/**
818 * Handler for the HEAD method.
819 *
820 * @returns VBox status code.
821 * @param pClient Client to handle HEAD method for.
822 * @param pReq Client request to handle.
823 */
824static DECLCALLBACK(int) rtHttpServerHandleHEAD(PRTHTTPSERVERCLIENT pClient, PRTHTTPSERVERREQ pReq)
825{
826 LogFlowFuncEnter();
827
828 /* If a low-level HEAD request handler is defined, call it and return. */
829 RTHTTPSERVER_HANDLE_CALLBACK_VA_RET(pfnOnHeadRequest, pReq);
830
831 int rc = VINF_SUCCESS;
832
833 RTFSOBJINFO fsObj;
834 RT_ZERO(fsObj); /* Shut up MSVC. */
835
836 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnQueryInfo, pReq, &fsObj, NULL /* pszMIMEHint */);
837 if (RT_SUCCESS(rc))
838 {
839 RTHTTPHEADERLIST HdrLst;
840 rc = RTHttpHeaderListInit(&HdrLst);
841 AssertRCReturn(rc, rc);
842
843 /*
844 * Note: A response to a HEAD request does not have a body.
845 * All entity headers below are assumed to describe the the response a similar GET
846 * request would return (but then with a body).
847 */
848 char szVal[16];
849
850 ssize_t cch = RTStrPrintf2(szVal, sizeof(szVal), "%RU64", fsObj.cbObject);
851 AssertReturn(cch, VERR_BUFFER_OVERFLOW);
852 rc = RTHttpHeaderListAdd(HdrLst, "Content-Length", szVal, strlen(szVal), RTHTTPHEADERLISTADD_F_BACK);
853 AssertRCReturn(rc, rc);
854
855 cch = RTStrPrintf2(szVal, sizeof(szVal), "identity");
856 AssertReturn(cch, VERR_BUFFER_OVERFLOW);
857 rc = RTHttpHeaderListAdd(HdrLst, "Content-Encoding", szVal, strlen(szVal), RTHTTPHEADERLISTADD_F_BACK);
858 AssertRCReturn(rc, rc);
859
860 const char *pszMIME = rtHttpServerGuessMIMEType(RTPathSuffix(pReq->pszUrl));
861 rc = RTHttpHeaderListAdd(HdrLst, "Content-Type", pszMIME, strlen(pszMIME), RTHTTPHEADERLISTADD_F_BACK);
862 AssertRCReturn(rc, rc);
863
864 rc = rtHttpServerSendResponseEx(pClient, RTHTTPSTATUS_OK, &HdrLst);
865 AssertRCReturn(rc, rc);
866
867 RTHttpHeaderListDestroy(HdrLst);
868 }
869
870 LogFlowFuncLeaveRC(rc);
871 return rc;
872}
873
874#ifdef IPRT_HTTP_WITH_WEBDAV
875/**
876 * Handler for the OPTIONS method.
877 *
878 * @returns VBox status code.
879 * @param pClient Client to handle OPTIONS method for.
880 * @param pReq Client request to handle.
881 */
882static DECLCALLBACK(int) rtHttpServerHandleOPTIONS(PRTHTTPSERVERCLIENT pClient, PRTHTTPSERVERREQ pReq)
883{
884 LogFlowFuncEnter();
885
886 RT_NOREF(pReq);
887
888 int rc = rtHttpServerSendResponseEx(pClient, RTHTTPSTATUS_OK, NULL /* pHdrLst */);
889
890 LogFlowFuncLeaveRC(rc);
891 return rc;
892}
893
894/**
895 * Handler for the PROPFIND (WebDAV) method.
896 *
897 * @returns VBox status code.
898 * @param pClient Client to handle PROPFIND method for.
899 * @param pReq Client request to handle.
900 */
901static DECLCALLBACK(int) rtHttpServerHandlePROPFIND(PRTHTTPSERVERCLIENT pClient, PRTHTTPSERVERREQ pReq)
902{
903 LogFlowFuncEnter();
904
905 int rc = VINF_SUCCESS;
906
907 /* If a low-level GET request handler is defined, call it and return. */
908 RTHTTPSERVER_HANDLE_CALLBACK_VA_RET(pfnOnGetRequest, pReq);
909
910 RTFSOBJINFO fsObj;
911 RT_ZERO(fsObj); /* Shut up MSVC. */
912
913 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnQueryInfo, pReq, &fsObj, NULL /* pszMIMEHint */);
914 if (RT_FAILURE(rc))
915 return rc;
916
917 void *pvHandle = NULL;
918 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnOpen, pReq, &pvHandle);
919
920 if (RT_SUCCESS(rc))
921 {
922 size_t cbBuf = _64K;
923 void *pvBuf = RTMemAlloc(cbBuf);
924 AssertPtrReturn(pvBuf, VERR_NO_MEMORY);
925
926 for (;;)
927 {
928 RTHTTPHEADERLIST HdrLst;
929 rc = RTHttpHeaderListInit(&HdrLst);
930 AssertRCReturn(rc, rc);
931
932 char szVal[16];
933
934 rc = RTHttpHeaderListAdd(HdrLst, "Content-Type", "text/xml; charset=utf-8", strlen("text/xml; charset=utf-8"), RTHTTPHEADERLISTADD_F_BACK);
935 AssertRCBreak(rc);
936
937 /* Note: For directories fsObj.cbObject contains the actual size (in bytes)
938 * of the body data for the directory listing. */
939
940 ssize_t cch = RTStrPrintf2(szVal, sizeof(szVal), "%RU64", fsObj.cbObject);
941 AssertBreakStmt(cch, VERR_BUFFER_OVERFLOW);
942 rc = RTHttpHeaderListAdd(HdrLst, "Content-Length", szVal, strlen(szVal), RTHTTPHEADERLISTADD_F_BACK);
943 AssertRCBreak(rc);
944
945 rc = rtHttpServerSendResponseEx(pClient, RTHTTPSTATUS_MULTISTATUS, &HdrLst);
946 AssertRCReturn(rc, rc);
947
948 RTHttpHeaderListDestroy(HdrLst);
949
950 size_t cbToRead = fsObj.cbObject;
951 size_t cbRead = 0; /* Shut up GCC. */
952 size_t cbWritten = 0; /* Ditto. */
953 while (cbToRead)
954 {
955 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnRead, pvHandle, pvBuf, RT_MIN(cbBuf, cbToRead), &cbRead);
956 if (RT_FAILURE(rc))
957 break;
958 //rtHttpServerLogProto(pClient, true /* fWrite */, (const char *)pvBuf);
959 rc = rtHttpServerSendResponseBody(pClient, pvBuf, cbRead, &cbWritten);
960 AssertBreak(cbToRead >= cbWritten);
961 cbToRead -= cbWritten;
962 if (rc == VERR_NET_CONNECTION_RESET_BY_PEER) /* Clients often apruptly abort the connection when done. */
963 {
964 rc = VINF_SUCCESS;
965 break;
966 }
967 AssertRCBreak(rc);
968 }
969
970 break;
971 } /* for (;;) */
972
973 RTMemFree(pvBuf);
974
975 int rc2 = rc; /* Save rc. */
976
977 RTHTTPSERVER_HANDLE_CALLBACK_VA(pfnClose, pvHandle);
978
979 if (RT_FAILURE(rc2)) /* Restore original rc on failure. */
980 rc = rc2;
981 }
982
983 LogFlowFuncLeaveRC(rc);
984 return rc;
985}
986#endif /* IPRT_HTTP_WITH_WEBDAV */
987
988/**
989 * Validates if a given path is valid or not.
990 *
991 * @returns \c true if path is valid, or \c false if not.
992 * @param pszPath Path to check.
993 * @param fIsAbsolute Whether the path to check is an absolute path or not.
994 */
995static bool rtHttpServerPathIsValid(const char *pszPath, bool fIsAbsolute)
996{
997 if (!pszPath)
998 return false;
999
1000 bool fIsValid = strlen(pszPath)
1001 && RTStrIsValidEncoding(pszPath)
1002 && RTStrStr(pszPath, "..") == NULL; /** @todo Very crude for now -- improve this. */
1003 if ( fIsValid
1004 && fIsAbsolute)
1005 {
1006 RTFSOBJINFO objInfo;
1007 int rc2 = RTPathQueryInfo(pszPath, &objInfo, RTFSOBJATTRADD_NOTHING);
1008 if (RT_SUCCESS(rc2))
1009 {
1010 fIsValid = RTFS_IS_DIRECTORY(objInfo.Attr.fMode)
1011 || RTFS_IS_FILE(objInfo.Attr.fMode);
1012
1013 /* No symlinks and other stuff not allowed. */
1014 }
1015 else
1016 fIsValid = false;
1017 }
1018
1019 LogFlowFunc(("pszPath=%s -> %RTbool\n", pszPath, fIsValid));
1020 return fIsValid;
1021
1022}
1023
1024/**
1025 * Parses headers and sets (replaces) a given header list.
1026 *
1027 * @returns VBox status code.
1028 * @param hList Header list to fill parsed headers in.
1029 * @param cStrings Number of strings to parse for \a papszStrings.
1030 * @param papszStrings Array of strings to parse.
1031 */
1032static int rtHttpServerParseHeaders(RTHTTPHEADERLIST hList, size_t cStrings, char **papszStrings)
1033{
1034 /* Nothing to parse left? Bail out early. */
1035 if ( !cStrings
1036 || !papszStrings)
1037 return VINF_SUCCESS;
1038
1039#ifdef LOG_ENABLED
1040 for (size_t i = 0; i < cStrings; i++)
1041 LogFlowFunc(("Header: %s\n", papszStrings[i]));
1042#endif
1043
1044 int rc = RTHttpHeaderListSet(hList, cStrings, papszStrings);
1045
1046 LogFlowFunc(("rc=%Rrc, cHeaders=%zu\n", rc, cStrings));
1047 return rc;
1048}
1049
1050/**
1051 * Main function for parsing and allocating a client request.
1052 *
1053 * See: https://tools.ietf.org/html/rfc2616#section-2.2
1054 *
1055 * @returns VBox status code.
1056 * @param pClient Client to parse request from.
1057 * @param pszReq Request string with headers to parse.
1058 * @param cbReq Size (in bytes) of request string to parse.
1059 * @param ppReq Where to store the allocated client request on success.
1060 * Needs to be free'd via rtHttpServerReqFree().
1061 */
1062static int rtHttpServerParseRequest(PRTHTTPSERVERCLIENT pClient, const char *pszReq, size_t cbReq,
1063 PRTHTTPSERVERREQ *ppReq)
1064{
1065 RT_NOREF(pClient);
1066
1067 AssertPtrReturn(pszReq, VERR_INVALID_POINTER);
1068 AssertReturn(cbReq, VERR_INVALID_PARAMETER);
1069
1070 /* We only support UTF-8 charset for now. */
1071 AssertReturn(RTStrIsValidEncoding(pszReq), VERR_INVALID_PARAMETER);
1072
1073 char **ppapszReq = NULL;
1074 size_t cReq = 0;
1075 int rc = RTStrSplit(pszReq, cbReq, RTHTTPSERVER_HTTP11_EOL_STR, &ppapszReq, &cReq);
1076 if (RT_FAILURE(rc))
1077 return rc;
1078
1079 if (!cReq)
1080 return VERR_INVALID_PARAMETER;
1081
1082#ifdef LOG_ENABLED
1083 for (size_t i = 0; i < cReq; i++)
1084 LogFlowFunc(("%s\n", ppapszReq[i]));
1085#endif
1086
1087 PRTHTTPSERVERREQ pReq = NULL;
1088
1089 char **ppapszFirstLine = NULL;
1090 size_t cFirstLine = 0;
1091 rc = RTStrSplit(ppapszReq[0], strlen(ppapszReq[0]), " ", &ppapszFirstLine, &cFirstLine);
1092 if (RT_SUCCESS(rc))
1093 {
1094 if (cFirstLine < 3) /* At leat the method, path and version has to be present. */
1095 rc = VERR_INVALID_PARAMETER;
1096 }
1097
1098 while (RT_SUCCESS(rc)) /* To use break. */
1099 {
1100 pReq = rtHttpServerReqAlloc();
1101 AssertPtrBreakStmt(pReq, rc = VERR_NO_MEMORY);
1102
1103 /*
1104 * Parse method to use. Method names are case sensitive.
1105 */
1106 const char *pszMethod = ppapszFirstLine[0];
1107 if (!RTStrCmp(pszMethod, "GET")) pReq->enmMethod = RTHTTPMETHOD_GET;
1108 else if (!RTStrCmp(pszMethod, "HEAD")) pReq->enmMethod = RTHTTPMETHOD_HEAD;
1109#ifdef IPRT_HTTP_WITH_WEBDAV
1110 else if (!RTStrCmp(pszMethod, "OPTIONS")) pReq->enmMethod = RTHTTPMETHOD_OPTIONS;
1111 else if (!RTStrCmp(pszMethod, "PROPFIND")) pReq->enmMethod = RTHTTPMETHOD_PROPFIND;
1112#endif
1113 else
1114 {
1115 rc = VERR_NOT_SUPPORTED;
1116 break;
1117 }
1118
1119 /*
1120 * Parse requested path.
1121 */
1122 /** @todo Do URL unescaping here. */
1123 const char *pszPath = ppapszFirstLine[1];
1124 if (!rtHttpServerPathIsValid(pszPath, false /* fIsAbsolute */))
1125 {
1126 rc = VERR_PATH_NOT_FOUND;
1127 break;
1128 }
1129
1130 pReq->pszUrl = RTStrDup(pszPath);
1131
1132 /*
1133 * Parse HTTP version to use.
1134 * We're picky heree: Only HTTP 1.1 is supported by now.
1135 */
1136 const char *pszVer = ppapszFirstLine[2];
1137 if (RTStrCmp(pszVer, RTHTTPVER_1_1_STR)) /** @todo Use RTStrVersionCompare. Later. */
1138 {
1139 rc = VERR_NOT_SUPPORTED;
1140 break;
1141 }
1142
1143 /** @todo Anything else needed for the first line here? */
1144
1145 /*
1146 * Process headers, if any.
1147 */
1148 if (cReq > 1)
1149 {
1150 rc = rtHttpServerParseHeaders(pReq->hHdrLst, cReq - 1, &ppapszReq[1]);
1151 if (RT_SUCCESS(rc))
1152 {
1153 if (RTHttpHeaderListGet(pReq->hHdrLst, "Connection", RTSTR_MAX))
1154 pClient->State.msKeepAlive = RT_MS_30SEC; /** @todo Insert the real value here. */
1155 }
1156 }
1157 break;
1158 } /* for (;;) */
1159
1160 /*
1161 * Cleanup.
1162 */
1163
1164 for (size_t i = 0; i < cFirstLine; i++)
1165 RTStrFree(ppapszFirstLine[i]);
1166 RTMemFree(ppapszFirstLine);
1167
1168 for (size_t i = 0; i < cReq; i++)
1169 RTStrFree(ppapszReq[i]);
1170 RTMemFree(ppapszReq);
1171
1172 if (RT_FAILURE(rc))
1173 {
1174 rtHttpServerReqFree(pReq);
1175 pReq = NULL;
1176 }
1177 else
1178 *ppReq = pReq;
1179
1180 return rc;
1181}
1182
1183/**
1184 * Main function for processing client requests.
1185 *
1186 * @returns VBox status code.
1187 * @param pClient Client to process request for.
1188 * @param pszReq Request string to parse and handle.
1189 * @param cbReq Size (in bytes) of request string.
1190 */
1191static int rtHttpServerProcessRequest(PRTHTTPSERVERCLIENT pClient, char *pszReq, size_t cbReq)
1192{
1193 RTHTTPSTATUS enmSts = RTHTTPSTATUS_INTERNAL_NOT_SET;
1194
1195 PRTHTTPSERVERREQ pReq = NULL; /* Shut up GCC. */
1196 int rc = rtHttpServerParseRequest(pClient, pszReq, cbReq, &pReq);
1197 if (RT_SUCCESS(rc))
1198 {
1199 LogFlowFunc(("Request %s %s\n", RTHttpMethodToStr(pReq->enmMethod), pReq->pszUrl));
1200
1201 unsigned i = 0;
1202 for (; i < RT_ELEMENTS(g_aMethodMap); i++)
1203 {
1204 const RTHTTPSERVERMETHOD_ENTRY *pMethodEntry = &g_aMethodMap[i];
1205 if (pReq->enmMethod == pMethodEntry->enmMethod)
1206 {
1207 /* Hand in request to method handler. */
1208 int rcMethod = pMethodEntry->pfnMethod(pClient, pReq);
1209 if (RT_FAILURE(rcMethod))
1210 LogFunc(("Request %s %s failed with %Rrc\n", RTHttpMethodToStr(pReq->enmMethod), pReq->pszUrl, rcMethod));
1211
1212 enmSts = rtHttpServerRcToStatus(rcMethod);
1213 break;
1214 }
1215 }
1216
1217 if (i == RT_ELEMENTS(g_aMethodMap))
1218 enmSts = RTHTTPSTATUS_NOTIMPLEMENTED;
1219
1220 rtHttpServerReqFree(pReq);
1221 }
1222 else
1223 enmSts = RTHTTPSTATUS_BADREQUEST;
1224
1225 if (enmSts != RTHTTPSTATUS_INTERNAL_NOT_SET)
1226 {
1227 int rc2 = rtHttpServerSendResponseSimple(pClient, enmSts);
1228 if (RT_SUCCESS(rc))
1229 rc = rc2;
1230 }
1231
1232 LogFlowFuncLeaveRC(rc);
1233 return rc;
1234}
1235
1236/**
1237 * Main loop for processing client requests.
1238 *
1239 * @returns VBox status code.
1240 * @param pClient Client to process requests for.
1241 */
1242static int rtHttpServerClientMain(PRTHTTPSERVERCLIENT pClient)
1243{
1244 int rc;
1245
1246 char szReq[RTHTTPSERVER_MAX_REQ_LEN + 1];
1247
1248 LogFlowFunc(("Client connected\n"));
1249
1250 /* Initialize client state. */
1251 pClient->State.msKeepAlive = 0;
1252
1253 RTMSINTERVAL cWaitMs = RT_INDEFINITE_WAIT; /* The first wait always waits indefinitely. */
1254 uint64_t tsLastReadMs = 0;
1255
1256 for (;;)
1257 {
1258 rc = RTTcpSelectOne(pClient->hSocket, cWaitMs);
1259 if (RT_FAILURE(rc))
1260 {
1261 LogFlowFunc(("RTTcpSelectOne=%Rrc (cWaitMs=%RU64)\n", rc, cWaitMs));
1262 if (rc == VERR_TIMEOUT)
1263 {
1264 if (pClient->State.msKeepAlive) /* Keep alive handling needed? */
1265 {
1266 if (!tsLastReadMs)
1267 tsLastReadMs = RTTimeMilliTS();
1268 const uint64_t tsDeltaMs = pClient->State.msKeepAlive - (RTTimeMilliTS() - tsLastReadMs);
1269 LogFlowFunc(("tsLastReadMs=%RU64, tsDeltaMs=%RU64\n", tsLastReadMs, tsDeltaMs));
1270 Log3Func(("Keep alive active (%RU32ms): %RU64ms remaining\n", pClient->State.msKeepAlive, tsDeltaMs));
1271 if ( tsDeltaMs > cWaitMs
1272 && tsDeltaMs < pClient->State.msKeepAlive)
1273 continue;
1274
1275 LogFunc(("Keep alive active: Client did not respond within %RU32ms, closing\n", pClient->State.msKeepAlive));
1276 rc = VINF_SUCCESS;
1277 break;
1278 }
1279 }
1280
1281 break;
1282 }
1283
1284 LogFlowFunc(("Reading client request ...\n"));
1285
1286 tsLastReadMs = RTTimeMilliTS();
1287 cWaitMs = 200; /* All consequtive waits do busy waiting for now. */
1288
1289 char *pszReq = szReq;
1290 size_t cbRead;
1291 size_t cbToRead = sizeof(szReq);
1292 size_t cbReadTotal = 0;
1293
1294 do
1295 {
1296 rc = RTTcpReadNB(pClient->hSocket, pszReq, cbToRead, &cbRead);
1297 if (RT_FAILURE(rc))
1298 break;
1299
1300 if (!cbRead)
1301 break;
1302
1303 /* Make sure to terminate string read so far. */
1304 pszReq[cbRead] = '\0';
1305
1306 /* End of request reached? */
1307 /** @todo BUGBUG Improve this. */
1308 char *pszEOR = RTStrStr(&pszReq[cbReadTotal], "\r\n\r\n");
1309 if (pszEOR)
1310 {
1311 cbReadTotal = pszEOR - pszReq;
1312 *pszEOR = '\0';
1313 break;
1314 }
1315
1316 AssertBreak(cbToRead >= cbRead);
1317 cbToRead -= cbRead;
1318 cbReadTotal += cbRead;
1319 AssertBreak(cbReadTotal <= sizeof(szReq));
1320 pszReq += cbRead;
1321
1322 } while (cbToRead);
1323
1324 if ( RT_SUCCESS(rc)
1325 && cbReadTotal)
1326 {
1327 LogFlowFunc(("Received client request (%zu bytes)\n", cbReadTotal));
1328
1329 rtHttpServerLogProto(pClient, false /* fWrite */, szReq);
1330
1331 rc = rtHttpServerProcessRequest(pClient, szReq, cbReadTotal);
1332 }
1333 else
1334 break;
1335
1336 } /* for */
1337
1338 if (RT_FAILURE(rc))
1339 {
1340 switch (rc)
1341 {
1342 case VERR_NET_CONNECTION_RESET_BY_PEER:
1343 {
1344 LogFunc(("Client closed the connection\n"));
1345 rc = VINF_SUCCESS;
1346 break;
1347 }
1348
1349 default:
1350 LogFunc(("Client processing failed with %Rrc\n", rc));
1351 break;
1352 }
1353 }
1354
1355 LogFlowFuncLeaveRC(rc);
1356 return rc;
1357}
1358
1359/**
1360 * Per-client thread for serving the server's control connection.
1361 *
1362 * @returns VBox status code.
1363 * @param hSocket Socket handle to use for the control connection.
1364 * @param pvUser User-provided arguments. Of type PRTHTTPSERVERINTERNAL.
1365 */
1366static DECLCALLBACK(int) rtHttpServerClientThread(RTSOCKET hSocket, void *pvUser)
1367{
1368 PRTHTTPSERVERINTERNAL pThis = (PRTHTTPSERVERINTERNAL)pvUser;
1369 RTHTTPSERVER_VALID_RETURN(pThis);
1370
1371 LogFlowFuncEnter();
1372
1373 RTHTTPSERVERCLIENT Client;
1374 RT_ZERO(Client);
1375
1376 Client.pServer = pThis;
1377 Client.hSocket = hSocket;
1378
1379 return rtHttpServerClientMain(&Client);
1380}
1381
1382RTR3DECL(int) RTHttpServerCreate(PRTHTTPSERVER hHttpServer, const char *pszAddress, uint16_t uPort,
1383 PRTHTTPSERVERCALLBACKS pCallbacks, void *pvUser, size_t cbUser)
1384{
1385 AssertPtrReturn(hHttpServer, VERR_INVALID_POINTER);
1386 AssertPtrReturn(pszAddress, VERR_INVALID_POINTER);
1387 AssertReturn (uPort, VERR_INVALID_PARAMETER);
1388 AssertPtrReturn(pCallbacks, VERR_INVALID_POINTER);
1389 /* pvUser is optional. */
1390
1391 int rc;
1392
1393 PRTHTTPSERVERINTERNAL pThis = (PRTHTTPSERVERINTERNAL)RTMemAllocZ(sizeof(RTHTTPSERVERINTERNAL));
1394 if (pThis)
1395 {
1396 pThis->u32Magic = RTHTTPSERVER_MAGIC;
1397 pThis->Callbacks = *pCallbacks;
1398 pThis->pvUser = pvUser;
1399 pThis->cbUser = cbUser;
1400
1401 rc = RTTcpServerCreate(pszAddress, uPort, RTTHREADTYPE_DEFAULT, "httpsrv",
1402 rtHttpServerClientThread, pThis /* pvUser */, &pThis->pTCPServer);
1403 if (RT_SUCCESS(rc))
1404 {
1405 *hHttpServer = (RTHTTPSERVER)pThis;
1406 }
1407 }
1408 else
1409 rc = VERR_NO_MEMORY;
1410
1411 return rc;
1412}
1413
1414RTR3DECL(int) RTHttpServerDestroy(RTHTTPSERVER hHttpServer)
1415{
1416 if (hHttpServer == NIL_RTHTTPSERVER)
1417 return VINF_SUCCESS;
1418
1419 PRTHTTPSERVERINTERNAL pThis = hHttpServer;
1420 RTHTTPSERVER_VALID_RETURN(pThis);
1421
1422 AssertPtr(pThis->pTCPServer);
1423
1424 int rc = VINF_SUCCESS;
1425
1426 PRTHTTPSERVERCALLBACKS pCallbacks = &pThis->Callbacks;
1427 if (pCallbacks->pfnDestroy)
1428 {
1429 RTHTTPCALLBACKDATA Data = { NULL /* pClient */, pThis->pvUser, pThis->cbUser };
1430 rc = pCallbacks->pfnDestroy(&Data);
1431 }
1432
1433 if (RT_SUCCESS(rc))
1434 {
1435 rc = RTTcpServerDestroy(pThis->pTCPServer);
1436 if (RT_SUCCESS(rc))
1437 {
1438 pThis->u32Magic = RTHTTPSERVER_MAGIC_DEAD;
1439
1440 RTMemFree(pThis);
1441 }
1442 }
1443
1444 return rc;
1445}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette