2 * Wininet - Http Implementation
4 * Copyright 1999 Corel Corporation
15 #include "debugtools.h"
19 #include <sys/types.h>
20 #ifdef HAVE_SYS_SOCKET_H
21 # include <sys/socket.h>
32 DEFAULT_DEBUG_CHANNEL(wininet);
34 #define HTTPHEADER " HTTP/1.0"
35 #define HTTPHOSTHEADER "\r\nHost: "
36 #define MAXHOSTNAME 100
37 #define MAX_FIELD_VALUE_LEN 256
38 #define MAX_FIELD_LEN 256
41 #define HTTP_REFERER "Referer"
42 #define HTTP_ACCEPT "Accept"
44 #define HTTP_ADDHDR_FLAG_ADD 0x20000000
45 #define HTTP_ADDHDR_FLAG_ADD_IF_NEW 0x10000000
46 #define HTTP_ADDHDR_FLAG_COALESCE 0x40000000
47 #define HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA 0x40000000
48 #define HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON 0x01000000
49 #define HTTP_ADDHDR_FLAG_REPLACE 0x80000000
50 #define HTTP_ADDHDR_FLAG_REQ 0x02000000
53 BOOL HTTP_OpenConnection(LPWININETHTTPREQA lpwhr);
54 int HTTP_WriteDataToStream(LPWININETHTTPREQA lpwhr,
55 void *Buffer, int BytesToWrite);
56 int HTTP_ReadDataFromStream(LPWININETHTTPREQA lpwhr,
57 void *Buffer, int BytesToRead);
58 BOOL HTTP_GetResponseHeaders(LPWININETHTTPREQA lpwhr);
59 BOOL HTTP_ProcessHeader(LPWININETHTTPREQA lpwhr, LPCSTR field, LPCSTR value, DWORD dwModifier);
60 void HTTP_CloseConnection(LPWININETHTTPREQA lpwhr);
61 BOOL HTTP_InterpretHttpHeader(LPSTR buffer, LPSTR field, INT fieldlen, LPSTR value, INT valuelen);
62 INT HTTP_GetStdHeaderIndex(LPCSTR lpszField);
63 INT HTTP_InsertCustomHeader(LPWININETHTTPREQA lpwhr, LPHTTPHEADERA lpHdr);
64 INT HTTP_GetCustomHeaderIndex(LPWININETHTTPREQA lpwhr, LPCSTR lpszField);
66 inline static LPSTR HTTP_strdup( LPCSTR str )
68 LPSTR ret = HeapAlloc( GetProcessHeap(), 0, strlen(str) + 1 );
69 if (ret) strcpy( ret, str );
73 /***********************************************************************
74 * HttpAddRequestHeadersA (WININET.@)
76 * Adds one or more HTTP header to the request handler
83 BOOL WINAPI HttpAddRequestHeadersA(HINTERNET hHttpRequest,
84 LPCSTR lpszHeader, DWORD dwHeaderLength, DWORD dwModifier)
89 CHAR value[MAX_FIELD_VALUE_LEN], field[MAX_FIELD_LEN];
90 BOOL bSuccess = FALSE;
91 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
93 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
95 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
99 buffer = HTTP_strdup(lpszHeader);
106 while (*lpszEnd != '\0')
108 if (*lpszEnd == '\r' && *(lpszEnd + 1) == '\n')
113 if (*lpszEnd == '\0')
118 if (HTTP_InterpretHttpHeader(lpszStart, field, MAX_FIELD_LEN, value, MAX_FIELD_VALUE_LEN))
119 bSuccess = HTTP_ProcessHeader(lpwhr, field, value, dwModifier | HTTP_ADDHDR_FLAG_REQ);
121 lpszStart = lpszEnd + 2; /* Jump over \0\n */
125 HeapFree(GetProcessHeap(), 0, buffer);
130 /***********************************************************************
131 * HttpOpenRequestA (WININET.@)
133 * Open a HTTP request handle
136 * HINTERNET a HTTP request handle on success
140 HINTERNET WINAPI HttpOpenRequestA(HINTERNET hHttpSession,
141 LPCSTR lpszVerb, LPCSTR lpszObjectName, LPCSTR lpszVersion,
142 LPCSTR lpszReferrer , LPCSTR *lpszAcceptTypes,
143 DWORD dwFlags, DWORD dwContext)
145 LPWININETHTTPSESSIONA lpwhs = (LPWININETHTTPSESSIONA) hHttpSession;
146 LPWININETAPPINFOA hIC = NULL;
150 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
152 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
156 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
158 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC)
160 WORKREQUEST workRequest;
162 workRequest.asyncall = HTTPOPENREQUESTA;
163 workRequest.HFTPSESSION = (DWORD)hHttpSession;
164 workRequest.LPSZVERB = (DWORD)HTTP_strdup(lpszVerb);
165 workRequest.LPSZOBJECTNAME = (DWORD)HTTP_strdup(lpszObjectName);
166 workRequest.LPSZVERSION = (DWORD)HTTP_strdup(lpszVersion);
167 workRequest.LPSZREFERRER = (DWORD)HTTP_strdup(lpszReferrer);
168 workRequest.LPSZACCEPTTYPES = (DWORD)lpszAcceptTypes;
169 workRequest.DWFLAGS = dwFlags;
170 workRequest.DWCONTEXT = dwContext;
172 return (HINTERNET)INTERNET_AsyncCall(&workRequest);
176 return HTTP_HttpOpenRequestA(hHttpSession, lpszVerb, lpszObjectName,
177 lpszVersion, lpszReferrer, lpszAcceptTypes, dwFlags, dwContext);
182 /***********************************************************************
183 * HTTP_HttpOpenRequestA (internal)
185 * Open a HTTP request handle
188 * HINTERNET a HTTP request handle on success
192 HINTERNET WINAPI HTTP_HttpOpenRequestA(HINTERNET hHttpSession,
193 LPCSTR lpszVerb, LPCSTR lpszObjectName, LPCSTR lpszVersion,
194 LPCSTR lpszReferrer , LPCSTR *lpszAcceptTypes,
195 DWORD dwFlags, DWORD dwContext)
197 LPWININETHTTPSESSIONA lpwhs = (LPWININETHTTPSESSIONA) hHttpSession;
198 LPWININETAPPINFOA hIC = NULL;
199 LPWININETHTTPREQA lpwhr;
203 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
205 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
209 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
211 lpwhr = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WININETHTTPREQA));
214 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
215 return (HINTERNET) NULL;
218 lpwhr->hdr.htype = WH_HHTTPREQ;
219 lpwhr->hdr.lpwhparent = hHttpSession;
220 lpwhr->hdr.dwFlags = dwFlags;
221 lpwhr->hdr.dwContext = dwContext;
222 lpwhr->nSocketFD = -1;
224 if (NULL != lpszObjectName && strlen(lpszObjectName)) {
226 UrlEscapeA(lpszObjectName, NULL, &needed, URL_ESCAPE_SPACES_ONLY);
227 lpwhr->lpszPath = HeapAlloc(GetProcessHeap(), 0, needed);
228 UrlEscapeA(lpszObjectName, lpwhr->lpszPath, &needed,
229 URL_ESCAPE_SPACES_ONLY);
232 if (NULL != lpszReferrer && strlen(lpszReferrer))
233 HTTP_ProcessHeader(lpwhr, HTTP_REFERER, lpszReferrer, HTTP_ADDHDR_FLAG_COALESCE);
236 if (NULL != lpszAcceptTypes && strlen(*lpszAcceptTypes))
237 HTTP_ProcessHeader(lpwhr, HTTP_ACCEPT, *lpszAcceptTypes, HTTP_ADDHDR_FLAG_COALESCE);
239 if (NULL == lpszVerb)
240 lpwhr->lpszVerb = HTTP_strdup("GET");
241 else if (strlen(lpszVerb))
242 lpwhr->lpszVerb = HTTP_strdup(lpszVerb);
244 if (NULL != lpszReferrer)
246 char buf[MAXHOSTNAME];
247 URL_COMPONENTSA UrlComponents;
249 UrlComponents.lpszExtraInfo = NULL;
250 UrlComponents.lpszPassword = NULL;
251 UrlComponents.lpszScheme = NULL;
252 UrlComponents.lpszUrlPath = NULL;
253 UrlComponents.lpszUserName = NULL;
254 UrlComponents.lpszHostName = buf;
255 UrlComponents.dwHostNameLength = MAXHOSTNAME;
257 InternetCrackUrlA(lpszReferrer, 0, 0, &UrlComponents);
258 if (strlen(UrlComponents.lpszHostName))
259 lpwhr->lpszHostName = HTTP_strdup(UrlComponents.lpszHostName);
261 lpwhr->lpszHostName = HTTP_strdup(lpwhs->lpszServerName);
264 if (hIC->lpfnStatusCB)
266 INTERNET_ASYNC_RESULT iar;
268 iar.dwResult = (DWORD)lpwhr;
269 iar.dwError = ERROR_SUCCESS;
271 hIC->lpfnStatusCB(hHttpSession, dwContext, INTERNET_STATUS_HANDLE_CREATED,
272 &iar, sizeof(INTERNET_ASYNC_RESULT));
275 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC && hIC->lpfnStatusCB)
277 INTERNET_ASYNC_RESULT iar;
279 iar.dwResult = (DWORD)lpwhr;
280 iar.dwError = lpwhr ? ERROR_SUCCESS : INTERNET_GetLastError();
281 hIC->lpfnStatusCB(hHttpSession, dwContext, INTERNET_STATUS_REQUEST_COMPLETE,
282 &iar, sizeof(INTERNET_ASYNC_RESULT));
285 return (HINTERNET) lpwhr;
289 /***********************************************************************
290 * HttpQueryInfoA (WININET.@)
292 * Queries for information about an HTTP request
299 BOOL WINAPI HttpQueryInfoA(HINTERNET hHttpRequest, DWORD dwInfoLevel,
300 LPVOID lpBuffer, LPDWORD lpdwBufferLength, LPDWORD lpdwIndex)
302 LPHTTPHEADERA lphttpHdr = NULL;
303 BOOL bSuccess = FALSE;
304 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
306 TRACE("(0x%08lx)--> %ld\n", dwInfoLevel, dwInfoLevel);
308 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
310 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
314 /* Find requested header structure */
315 if ((dwInfoLevel & ~HTTP_QUERY_MODIFIER_FLAGS_MASK) == HTTP_QUERY_CUSTOM)
317 INT index = HTTP_GetCustomHeaderIndex(lpwhr, (LPSTR)lpBuffer);
322 lphttpHdr = &lpwhr->pCustHeaders[index];
326 INT index = dwInfoLevel & ~HTTP_QUERY_MODIFIER_FLAGS_MASK;
328 if (index == HTTP_QUERY_RAW_HEADERS_CRLF || index == HTTP_QUERY_RAW_HEADERS)
330 INT i, delim, size = 0, cnt = 0;
332 delim = index == HTTP_QUERY_RAW_HEADERS_CRLF ? 2 : 1;
334 /* Calculate length of custom reuqest headers */
335 for (i = 0; i < lpwhr->nCustHeaders; i++)
337 if ((~lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST) && lpwhr->pCustHeaders[i].lpszField &&
338 lpwhr->pCustHeaders[i].lpszValue)
340 size += strlen(lpwhr->pCustHeaders[i].lpszField) +
341 strlen(lpwhr->pCustHeaders[i].lpszValue) + delim + 2;
345 /* Calculate the length of stadard request headers */
346 for (i = 0; i <= HTTP_QUERY_MAX; i++)
348 if ((~lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST) && lpwhr->StdHeaders[i].lpszField &&
349 lpwhr->StdHeaders[i].lpszValue)
351 size += strlen(lpwhr->StdHeaders[i].lpszField) +
352 strlen(lpwhr->StdHeaders[i].lpszValue) + delim + 2;
358 if (size + 1 > *lpdwBufferLength)
360 *lpdwBufferLength = size + 1;
361 INTERNET_SetLastError(ERROR_INSUFFICIENT_BUFFER);
365 /* Append standard request heades */
366 for (i = 0; i <= HTTP_QUERY_MAX; i++)
368 if ((~lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST) &&
369 lpwhr->StdHeaders[i].lpszField &&
370 lpwhr->StdHeaders[i].lpszValue)
372 cnt += sprintf(lpBuffer + cnt, "%s: %s%s", lpwhr->StdHeaders[i].lpszField, lpwhr->StdHeaders[i].lpszValue,
373 index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "\0");
377 /* Append custom request heades */
378 for (i = 0; i < lpwhr->nCustHeaders; i++)
380 if ((~lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST) &&
381 lpwhr->pCustHeaders[i].lpszField &&
382 lpwhr->pCustHeaders[i].lpszValue)
384 cnt += sprintf(lpBuffer + cnt, "%s: %s%s",
385 lpwhr->pCustHeaders[i].lpszField, lpwhr->pCustHeaders[i].lpszValue,
386 index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "\0");
390 strcpy(lpBuffer + cnt, index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "");
392 *lpdwBufferLength = cnt + delim;
396 else if (index >= 0 && index <= HTTP_QUERY_MAX && lpwhr->StdHeaders[index].lpszValue)
398 lphttpHdr = &lpwhr->StdHeaders[index];
404 /* Ensure header satisifies requested attributes */
405 if ((dwInfoLevel & HTTP_QUERY_FLAG_REQUEST_HEADERS) &&
406 (~lphttpHdr->wFlags & HDR_ISREQUEST))
409 /* coalesce value to reuqested type */
410 if (dwInfoLevel & HTTP_QUERY_FLAG_NUMBER)
412 *(int *)lpBuffer = atoi(lphttpHdr->lpszValue);
415 else if (dwInfoLevel & HTTP_QUERY_FLAG_SYSTEMTIME)
421 tmpTime = ConvertTimeString(lphttpHdr->lpszValue);
423 tmpTM = *gmtime(&tmpTime);
424 STHook = (SYSTEMTIME *) lpBuffer;
428 STHook->wDay = tmpTM.tm_mday;
429 STHook->wHour = tmpTM.tm_hour;
430 STHook->wMilliseconds = 0;
431 STHook->wMinute = tmpTM.tm_min;
432 STHook->wDayOfWeek = tmpTM.tm_wday;
433 STHook->wMonth = tmpTM.tm_mon + 1;
434 STHook->wSecond = tmpTM.tm_sec;
435 STHook->wYear = tmpTM.tm_year;
439 else if (dwInfoLevel & HTTP_QUERY_FLAG_COALESCE)
441 if (*lpdwIndex >= lphttpHdr->wCount)
443 INTERNET_SetLastError(ERROR_HTTP_HEADER_NOT_FOUND);
447 //! Copy strncpy(lpBuffer, lphttpHdr[*lpdwIndex], len);
453 INT len = strlen(lphttpHdr->lpszValue);
455 if (len + 1 > *lpdwBufferLength)
457 *lpdwBufferLength = len + 1;
458 INTERNET_SetLastError(ERROR_INSUFFICIENT_BUFFER);
462 strncpy(lpBuffer, lphttpHdr->lpszValue, len);
463 *lpdwBufferLength = len;
468 TRACE("%d <--\n", bSuccess);
473 /***********************************************************************
474 * HttpSendRequestExA (WININET.@)
476 * Sends the specified request to the HTTP server and allows chunked
479 BOOL WINAPI HttpSendRequestExA(HINTERNET hRequest,
480 LPINTERNET_BUFFERSA lpBuffersIn,
481 LPINTERNET_BUFFERSA lpBuffersOut,
482 DWORD dwFlags, DWORD dwContext)
484 FIXME("(%p, %p, %p, %08lx, %08lx): stub\n", hRequest, lpBuffersIn,
485 lpBuffersOut, dwFlags, dwContext);
489 /***********************************************************************
490 * HttpSendRequestA (WININET.@)
492 * Sends the specified request to the HTTP server
499 BOOL WINAPI HttpSendRequestA(HINTERNET hHttpRequest, LPCSTR lpszHeaders,
500 DWORD dwHeaderLength, LPVOID lpOptional ,DWORD dwOptionalLength)
502 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
503 LPWININETHTTPSESSIONA lpwhs = NULL;
504 LPWININETAPPINFOA hIC = NULL;
506 TRACE("0x%08lx\n", (unsigned long)hHttpRequest);
508 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
510 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
514 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
515 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
517 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
521 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
522 if (NULL == hIC || hIC->hdr.htype != WH_HINIT)
524 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
528 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC)
530 WORKREQUEST workRequest;
532 workRequest.asyncall = HTTPSENDREQUESTA;
533 workRequest.HFTPSESSION = (DWORD)hHttpRequest;
534 workRequest.LPSZHEADER = (DWORD)HTTP_strdup(lpszHeaders);
535 workRequest.DWHEADERLENGTH = dwHeaderLength;
536 workRequest.LPOPTIONAL = (DWORD)lpOptional;
537 workRequest.DWOPTIONALLENGTH = dwOptionalLength;
539 return INTERNET_AsyncCall(&workRequest);
543 return HTTP_HttpSendRequestA(hHttpRequest, lpszHeaders,
544 dwHeaderLength, lpOptional, dwOptionalLength);
549 /***********************************************************************
550 * HTTP_HttpSendRequestA (internal)
552 * Sends the specified request to the HTTP server
559 BOOL WINAPI HTTP_HttpSendRequestA(HINTERNET hHttpRequest, LPCSTR lpszHeaders,
560 DWORD dwHeaderLength, LPVOID lpOptional ,DWORD dwOptionalLength)
564 BOOL bSuccess = FALSE;
565 LPSTR requestString = NULL;
566 INT requestStringLen;
567 INT headerLength = 0;
568 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
569 LPWININETHTTPSESSIONA lpwhs = NULL;
570 LPWININETAPPINFOA hIC = NULL;
572 TRACE("0x%08lx\n", (ULONG)hHttpRequest);
574 /* Verify our tree of internet handles */
575 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
577 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
581 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
582 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
584 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
588 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
589 if (NULL == hIC || hIC->hdr.htype != WH_HINIT)
591 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
595 /* Clear any error information */
596 INTERNET_SetLastError(0);
598 /* We must have a verb */
599 if (NULL == lpwhr->lpszVerb)
604 /* If we don't have a path we set it to root */
605 if (NULL == lpwhr->lpszPath)
606 lpwhr->lpszPath = HTTP_strdup("/");
608 if(lpwhr->lpszPath[0] != '/') /* not an absolute path ?? --> fix it !! */
610 char *fixurl = HeapAlloc(GetProcessHeap(), 0, strlen(lpwhr->lpszPath) + 2);
612 strcpy(fixurl + 1, lpwhr->lpszPath);
613 HeapFree( GetProcessHeap(), 0, lpwhr->lpszPath );
614 lpwhr->lpszPath = fixurl;
617 /* Calculate length of request string */
619 strlen(lpwhr->lpszVerb) +
620 strlen(lpwhr->lpszPath) +
621 (lpwhr->lpszHostName ? (strlen(HTTPHOSTHEADER) + strlen(lpwhr->lpszHostName)) : 0) +
625 /* Add length of passed headers */
628 headerLength = -1 == dwHeaderLength ? strlen(lpszHeaders) : dwHeaderLength;
629 requestStringLen += headerLength + 2; /* \r\n */
632 /* Calculate length of custom request headers */
633 for (i = 0; i < lpwhr->nCustHeaders; i++)
635 if (lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST)
637 requestStringLen += strlen(lpwhr->pCustHeaders[i].lpszField) +
638 strlen(lpwhr->pCustHeaders[i].lpszValue) + 4; /*: \r\n */
642 /* Calculate the length of standard request headers */
643 for (i = 0; i <= HTTP_QUERY_MAX; i++)
645 if (lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST)
647 requestStringLen += strlen(lpwhr->StdHeaders[i].lpszField) +
648 strlen(lpwhr->StdHeaders[i].lpszValue) + 4; /*: \r\n */
652 /* Allocate string to hold entire request */
653 requestString = HeapAlloc(GetProcessHeap(), 0, requestStringLen + 1);
654 if (NULL == requestString)
656 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
660 /* Build request string */
661 cnt = sprintf(requestString, "%s %s%s%s",
664 lpwhr->lpszHostName ? (HTTPHEADER HTTPHOSTHEADER) : HTTPHEADER,
665 lpwhr->lpszHostName ? lpwhr->lpszHostName : "");
667 /* Append standard request headers */
668 for (i = 0; i <= HTTP_QUERY_MAX; i++)
670 if (lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST)
672 cnt += sprintf(requestString + cnt, "\r\n%s: %s",
673 lpwhr->StdHeaders[i].lpszField, lpwhr->StdHeaders[i].lpszValue);
677 /* Append custom request heades */
678 for (i = 0; i < lpwhr->nCustHeaders; i++)
680 if (lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST)
682 cnt += sprintf(requestString + cnt, "\r\n%s: %s",
683 lpwhr->pCustHeaders[i].lpszField, lpwhr->pCustHeaders[i].lpszValue);
687 /* Append passed request headers */
690 strcpy(requestString + cnt, "\r\n");
692 strcpy(requestString + cnt, lpszHeaders);
696 /* Set termination string for request */
697 strcpy(requestString + cnt, "\r\n\r\n");
699 if (hIC->lpfnStatusCB)
700 hIC->lpfnStatusCB(hHttpRequest, lpwhr->hdr.dwContext, INTERNET_STATUS_SENDING_REQUEST, NULL, 0);
702 TRACE("(%s) len(%d)\n", requestString, requestStringLen);
703 /* Send the request and store the results */
704 if (!HTTP_OpenConnection(lpwhr))
707 cnt = INTERNET_WriteDataToStream(lpwhr->nSocketFD, requestString, requestStringLen);
712 if (HTTP_GetResponseHeaders(lpwhr))
718 HeapFree(GetProcessHeap(), 0, requestString);
720 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC && hIC->lpfnStatusCB)
722 INTERNET_ASYNC_RESULT iar;
724 iar.dwResult = (DWORD)bSuccess;
725 iar.dwError = bSuccess ? ERROR_SUCCESS : INTERNET_GetLastError();
726 hIC->lpfnStatusCB(hHttpRequest, lpwhr->hdr.dwContext, INTERNET_STATUS_REQUEST_COMPLETE,
727 &iar, sizeof(INTERNET_ASYNC_RESULT));
735 /***********************************************************************
736 * HTTP_Connect (internal)
738 * Create http session handle
741 * HINTERNET a session handle on success
745 HINTERNET HTTP_Connect(HINTERNET hInternet, LPCSTR lpszServerName,
746 INTERNET_PORT nServerPort, LPCSTR lpszUserName,
747 LPCSTR lpszPassword, DWORD dwFlags, DWORD dwContext)
749 BOOL bSuccess = FALSE;
750 LPWININETAPPINFOA hIC = NULL;
751 LPWININETHTTPSESSIONA lpwhs = NULL;
755 if (((LPWININETHANDLEHEADER)hInternet)->htype != WH_HINIT)
758 hIC = (LPWININETAPPINFOA) hInternet;
760 lpwhs = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WININETHTTPSESSIONA));
763 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
767 if (hIC->lpfnStatusCB)
768 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_RESOLVING_NAME,
769 (LPVOID)lpszServerName, strlen(lpszServerName));
771 if (nServerPort == INTERNET_INVALID_PORT_NUMBER)
772 nServerPort = INTERNET_DEFAULT_HTTP_PORT;
774 if (!GetAddress(lpszServerName, nServerPort, &lpwhs->phostent, &lpwhs->socketAddress))
776 INTERNET_SetLastError(ERROR_INTERNET_NAME_NOT_RESOLVED);
780 if (hIC->lpfnStatusCB)
781 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_NAME_RESOLVED,
782 (LPVOID)lpszServerName, strlen(lpszServerName));
784 lpwhs->hdr.htype = WH_HHTTPSESSION;
785 lpwhs->hdr.lpwhparent = (LPWININETHANDLEHEADER)hInternet;
786 lpwhs->hdr.dwFlags = dwFlags;
787 lpwhs->hdr.dwContext = dwContext;
788 if (NULL != lpszServerName)
789 lpwhs->lpszServerName = HTTP_strdup(lpszServerName);
790 if (NULL != lpszUserName)
791 lpwhs->lpszUserName = HTTP_strdup(lpszUserName);
792 lpwhs->nServerPort = nServerPort;
794 if (hIC->lpfnStatusCB)
796 INTERNET_ASYNC_RESULT iar;
798 iar.dwResult = (DWORD)lpwhs;
799 iar.dwError = ERROR_SUCCESS;
801 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_HANDLE_CREATED,
802 &iar, sizeof(INTERNET_ASYNC_RESULT));
808 if (!bSuccess && lpwhs)
810 HeapFree(GetProcessHeap(), 0, lpwhs);
814 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC && hIC->lpfnStatusCB)
816 INTERNET_ASYNC_RESULT iar;
818 iar.dwResult = (DWORD)lpwhs;
819 iar.dwError = bSuccess ? ERROR_SUCCESS : INTERNET_GetLastError();
820 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_REQUEST_COMPLETE,
821 &iar, sizeof(INTERNET_ASYNC_RESULT));
824 return (HINTERNET)lpwhs;
828 /***********************************************************************
829 * HTTP_OpenConnection (internal)
831 * Connect to a web server
838 BOOL HTTP_OpenConnection(LPWININETHTTPREQA lpwhr)
840 BOOL bSuccess = FALSE;
842 LPWININETHTTPSESSIONA lpwhs;
846 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
848 INTERNET_SetLastError(ERROR_INVALID_PARAMETER);
852 lpwhs = (LPWININETHTTPSESSIONA)lpwhr->hdr.lpwhparent;
854 lpwhr->nSocketFD = socket(lpwhs->phostent->h_addrtype,SOCK_STREAM,0);
855 if (lpwhr->nSocketFD == -1)
857 WARN("Socket creation failed\n");
861 result = connect(lpwhr->nSocketFD, (struct sockaddr *)&lpwhs->socketAddress,
862 sizeof(lpwhs->socketAddress));
866 WARN("Unable to connect to host (%s)\n", strerror(errno));
873 TRACE(": %d\n", bSuccess);
878 /***********************************************************************
879 * HTTP_GetResponseHeaders (internal)
881 * Read server response
888 BOOL HTTP_GetResponseHeaders(LPWININETHTTPREQA lpwhr)
891 CHAR buffer[MAX_REPLY_LEN];
892 DWORD buflen = MAX_REPLY_LEN;
893 BOOL bSuccess = FALSE;
894 CHAR value[MAX_FIELD_VALUE_LEN], field[MAX_FIELD_LEN];
898 if (lpwhr->nSocketFD == -1)
902 * We should first receive 'HTTP/1.x nnn' where nnn is the status code.
904 if (!INTERNET_GetNextLine(lpwhr->nSocketFD, buffer, &buflen))
907 if (strncmp(buffer, "HTTP", 4) != 0)
911 HTTP_ProcessHeader(lpwhr, "Status", buffer+9, (HTTP_ADDREQ_FLAG_ADD | HTTP_ADDREQ_FLAG_REPLACE));
913 /* Parse each response line */
916 buflen = MAX_REPLY_LEN;
917 if (INTERNET_GetNextLine(lpwhr->nSocketFD, buffer, &buflen))
919 if (!HTTP_InterpretHttpHeader(buffer, field, MAX_FIELD_LEN, value, MAX_FIELD_VALUE_LEN))
922 HTTP_ProcessHeader(lpwhr, field, value, (HTTP_ADDREQ_FLAG_ADD | HTTP_ADDREQ_FLAG_REPLACE));
940 /***********************************************************************
941 * HTTP_InterpretHttpHeader (internal)
943 * Parse server response
950 INT stripSpaces(LPCSTR lpszSrc, LPSTR lpszStart, INT *len)
957 while (*lpszSrc == ' ' && *lpszSrc != '\0')
961 while(*lpsztmp != '\0')
964 srclen = lpsztmp - lpszSrc + 1;
969 *len = min(*len, srclen);
970 strncpy(lpszStart, lpszSrc, *len);
971 lpszStart[*len] = '\0';
977 BOOL HTTP_InterpretHttpHeader(LPSTR buffer, LPSTR field, INT fieldlen, LPSTR value, INT valuelen)
980 BOOL bSuccess = FALSE;
987 pd = strchr(buffer, ':');
991 if (stripSpaces(buffer, field, &fieldlen) > 0)
993 if (stripSpaces(pd+1, value, &valuelen) > 0)
998 TRACE("%d: field(%s) Value(%s)\n", bSuccess, field, value);
1003 /***********************************************************************
1004 * HTTP_GetStdHeaderIndex (internal)
1006 * Lookup field index in standard http header array
1008 * FIXME: This should be stuffed into a hash table
1010 INT HTTP_GetStdHeaderIndex(LPCSTR lpszField)
1014 if (!strcasecmp(lpszField, "Content-Length"))
1015 index = HTTP_QUERY_CONTENT_LENGTH;
1016 else if (!strcasecmp(lpszField,"Status"))
1017 index = HTTP_QUERY_STATUS_CODE;
1018 else if (!strcasecmp(lpszField,"Content-Type"))
1019 index = HTTP_QUERY_CONTENT_TYPE;
1020 else if (!strcasecmp(lpszField,"Last-Modified"))
1021 index = HTTP_QUERY_LAST_MODIFIED;
1022 else if (!strcasecmp(lpszField,"Location"))
1023 index = HTTP_QUERY_LOCATION;
1024 else if (!strcasecmp(lpszField,"Accept"))
1025 index = HTTP_QUERY_ACCEPT;
1026 else if (!strcasecmp(lpszField,"Referer"))
1027 index = HTTP_QUERY_REFERER;
1028 else if (!strcasecmp(lpszField,"Content-Transfer-Encoding"))
1029 index = HTTP_QUERY_CONTENT_TRANSFER_ENCODING;
1030 else if (!strcasecmp(lpszField,"Date"))
1031 index = HTTP_QUERY_DATE;
1032 else if (!strcasecmp(lpszField,"Server"))
1033 index = HTTP_QUERY_SERVER;
1034 else if (!strcasecmp(lpszField,"Connection"))
1035 index = HTTP_QUERY_CONNECTION;
1036 else if (!strcasecmp(lpszField,"ETag"))
1037 index = HTTP_QUERY_ETAG;
1038 else if (!strcasecmp(lpszField,"Accept-Ranges"))
1039 index = HTTP_QUERY_ACCEPT_RANGES;
1040 else if (!strcasecmp(lpszField,"Expires"))
1041 index = HTTP_QUERY_EXPIRES;
1042 else if (!strcasecmp(lpszField,"Mime-Version"))
1043 index = HTTP_QUERY_MIME_VERSION;
1046 FIXME("Couldn't find %s in standard header table\n", lpszField);
1053 /***********************************************************************
1054 * HTTP_ProcessHeader (internal)
1056 * Stuff header into header tables according to <dwModifier>
1060 #define COALESCEFLASG (HTTP_ADDHDR_FLAG_COALESCE|HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA|HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON)
1062 BOOL HTTP_ProcessHeader(LPWININETHTTPREQA lpwhr, LPCSTR field, LPCSTR value, DWORD dwModifier)
1064 LPHTTPHEADERA lphttpHdr = NULL;
1065 BOOL bSuccess = FALSE;
1068 TRACE("%s:%s - 0x%08x\n", field, value, (unsigned int)dwModifier);
1070 /* Adjust modifier flags */
1071 if (dwModifier & COALESCEFLASG)
1072 dwModifier |= HTTP_ADDHDR_FLAG_ADD;
1074 /* Try to get index into standard header array */
1075 index = HTTP_GetStdHeaderIndex(field);
1078 lphttpHdr = &lpwhr->StdHeaders[index];
1080 else /* Find or create new custom header */
1082 index = HTTP_GetCustomHeaderIndex(lpwhr, field);
1085 if (dwModifier & HTTP_ADDHDR_FLAG_ADD_IF_NEW)
1089 lphttpHdr = &lpwhr->pCustHeaders[index];
1095 hdr.lpszField = (LPSTR)field;
1096 hdr.lpszValue = (LPSTR)value;
1097 hdr.wFlags = hdr.wCount = 0;
1099 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1100 hdr.wFlags |= HDR_ISREQUEST;
1102 index = HTTP_InsertCustomHeader(lpwhr, &hdr);
1107 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1108 lphttpHdr->wFlags |= HDR_ISREQUEST;
1110 lphttpHdr->wFlags &= ~HDR_ISREQUEST;
1112 if (!lphttpHdr->lpszValue && (dwModifier & (HTTP_ADDHDR_FLAG_ADD|HTTP_ADDHDR_FLAG_ADD_IF_NEW)))
1116 if (!lpwhr->StdHeaders[index].lpszField)
1118 lphttpHdr->lpszField = HTTP_strdup(field);
1120 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1121 lphttpHdr->wFlags |= HDR_ISREQUEST;
1124 slen = strlen(value) + 1;
1125 lphttpHdr->lpszValue = HeapAlloc(GetProcessHeap(), 0, slen);
1126 if (lphttpHdr->lpszValue)
1128 memcpy(lphttpHdr->lpszValue, value, slen);
1133 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1136 else if (lphttpHdr->lpszValue)
1138 if (dwModifier & HTTP_ADDHDR_FLAG_REPLACE)
1143 len = strlen(value);
1147 //! if custom header delete from array
1148 HeapFree(GetProcessHeap(), 0, lphttpHdr->lpszValue);
1149 lphttpHdr->lpszValue = NULL;
1154 lpsztmp = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lphttpHdr->lpszValue, len+1);
1157 lphttpHdr->lpszValue = lpsztmp;
1158 strcpy(lpsztmp, value);
1163 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1167 else if (dwModifier & COALESCEFLASG)
1172 INT origlen = strlen(lphttpHdr->lpszValue);
1173 INT valuelen = strlen(value);
1175 if (dwModifier & HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA)
1178 lphttpHdr->wFlags |= HDR_COMMADELIMITED;
1180 else if (dwModifier & HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON)
1183 lphttpHdr->wFlags |= HDR_COMMADELIMITED;
1186 len = origlen + valuelen + (ch > 0) ? 1 : 0;
1188 lpsztmp = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lphttpHdr->lpszValue, len+1);
1191 /* FIXME: Increment lphttpHdr->wCount. Perhaps lpszValue should be an array */
1194 lphttpHdr->lpszValue[origlen] = ch;
1198 memcpy(&lphttpHdr->lpszValue[origlen], value, valuelen);
1199 lphttpHdr->lpszValue[len] = '\0';
1204 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1213 /***********************************************************************
1214 * HTTP_CloseConnection (internal)
1216 * Close socket connection
1219 VOID HTTP_CloseConnection(LPWININETHTTPREQA lpwhr)
1221 if (lpwhr->nSocketFD != -1)
1223 close(lpwhr->nSocketFD);
1224 lpwhr->nSocketFD = -1;
1229 /***********************************************************************
1230 * HTTP_CloseHTTPRequestHandle (internal)
1232 * Deallocate request handle
1235 void HTTP_CloseHTTPRequestHandle(LPWININETHTTPREQA lpwhr)
1241 if (lpwhr->nSocketFD != -1)
1242 HTTP_CloseConnection(lpwhr);
1244 if (lpwhr->lpszPath)
1245 HeapFree(GetProcessHeap(), 0, lpwhr->lpszPath);
1246 if (lpwhr->lpszVerb)
1247 HeapFree(GetProcessHeap(), 0, lpwhr->lpszVerb);
1248 if (lpwhr->lpszHostName)
1249 HeapFree(GetProcessHeap(), 0, lpwhr->lpszHostName);
1251 for (i = 0; i <= HTTP_QUERY_MAX; i++)
1253 if (lpwhr->StdHeaders[i].lpszField)
1254 HeapFree(GetProcessHeap(), 0, lpwhr->StdHeaders[i].lpszField);
1255 if (lpwhr->StdHeaders[i].lpszValue)
1256 HeapFree(GetProcessHeap(), 0, lpwhr->StdHeaders[i].lpszValue);
1259 for (i = 0; i < lpwhr->nCustHeaders; i++)
1261 if (lpwhr->pCustHeaders[i].lpszField)
1262 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders[i].lpszField);
1263 if (lpwhr->pCustHeaders[i].lpszValue)
1264 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders[i].lpszValue);
1267 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders);
1268 HeapFree(GetProcessHeap(), 0, lpwhr);
1272 /***********************************************************************
1273 * HTTP_CloseHTTPSessionHandle (internal)
1275 * Deallocate session handle
1278 void HTTP_CloseHTTPSessionHandle(LPWININETHTTPSESSIONA lpwhs)
1282 if (lpwhs->lpszServerName)
1283 HeapFree(GetProcessHeap(), 0, lpwhs->lpszServerName);
1284 if (lpwhs->lpszUserName)
1285 HeapFree(GetProcessHeap(), 0, lpwhs->lpszUserName);
1286 HeapFree(GetProcessHeap(), 0, lpwhs);
1290 /***********************************************************************
1291 * HTTP_GetCustomHeaderIndex (internal)
1293 * Return index of custom header from header array
1296 INT HTTP_GetCustomHeaderIndex(LPWININETHTTPREQA lpwhr, LPCSTR lpszField)
1300 TRACE("%s\n", lpszField);
1302 for (index = 0; index < lpwhr->nCustHeaders; index++)
1304 if (!strcasecmp(lpwhr->pCustHeaders[index].lpszField, lpszField))
1309 if (index >= lpwhr->nCustHeaders)
1312 TRACE("Return: %d\n", index);
1317 /***********************************************************************
1318 * HTTP_InsertCustomHeader (internal)
1320 * Insert header into array
1323 INT HTTP_InsertCustomHeader(LPWININETHTTPREQA lpwhr, LPHTTPHEADERA lpHdr)
1326 LPHTTPHEADERA lph = NULL;
1328 TRACE("%s: %s\n", lpHdr->lpszField, lpHdr->lpszValue);
1329 count = lpwhr->nCustHeaders + 1;
1331 lph = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lpwhr->pCustHeaders, sizeof(HTTPHEADERA) * count);
1333 lph = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(HTTPHEADERA) * count);
1337 lpwhr->pCustHeaders = lph;
1338 lpwhr->pCustHeaders[count-1].lpszField = HTTP_strdup(lpHdr->lpszField);
1339 lpwhr->pCustHeaders[count-1].lpszValue = HTTP_strdup(lpHdr->lpszValue);
1340 lpwhr->pCustHeaders[count-1].wFlags = lpHdr->wFlags;
1341 lpwhr->pCustHeaders[count-1].wCount= lpHdr->wCount;
1342 lpwhr->nCustHeaders++;
1346 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1350 TRACE("%d <--\n", count-1);
1355 /***********************************************************************
1356 * HTTP_DeleteCustomHeader (internal)
1358 * Delete header from array
1361 BOOL HTTP_DeleteCustomHeader(INT index)