2 * Wininet - Http Implementation
4 * Copyright 1999 Corel Corporation
14 #include "debugtools.h"
19 #include <sys/types.h>
20 #ifdef HAVE_SYS_SOCKET_H
21 # include <sys/socket.h>
31 DEFAULT_DEBUG_CHANNEL(wininet)
33 #define HTTPHEADER " HTTP/1.0"
34 #define HTTPHOSTHEADER "\r\nHost: "
35 #define MAXHOSTNAME 100
36 #define MAX_FIELD_VALUE_LEN 256
37 #define MAX_FIELD_LEN 256
40 #define HTTP_REFERER "Referer"
41 #define HTTP_ACCEPT "Accept"
43 #define HTTP_ADDHDR_FLAG_ADD 0x20000000
44 #define HTTP_ADDHDR_FLAG_ADD_IF_NEW 0x10000000
45 #define HTTP_ADDHDR_FLAG_COALESCE 0x40000000
46 #define HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA 0x40000000
47 #define HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON 0x01000000
48 #define HTTP_ADDHDR_FLAG_REPLACE 0x80000000
49 #define HTTP_ADDHDR_FLAG_REQ 0x02000000
52 BOOL HTTP_OpenConnection(LPWININETHTTPREQA lpwhr);
53 int HTTP_WriteDataToStream(LPWININETHTTPREQA lpwhr,
54 void *Buffer, int BytesToWrite);
55 int HTTP_ReadDataFromStream(LPWININETHTTPREQA lpwhr,
56 void *Buffer, int BytesToRead);
57 BOOL HTTP_GetResponseHeaders(LPWININETHTTPREQA lpwhr);
58 BOOL HTTP_ProcessHeader(LPWININETHTTPREQA lpwhr, LPCSTR field, LPCSTR value, DWORD dwModifier);
59 void HTTP_CloseConnection(LPWININETHTTPREQA lpwhr);
60 BOOL HTTP_InterpretHttpHeader(LPSTR buffer, LPSTR field, INT fieldlen, LPSTR value, INT valuelen);
61 INT HTTP_GetStdHeaderIndex(LPCSTR lpszField);
62 INT HTTP_InsertCustomHeader(LPWININETHTTPREQA lpwhr, LPHTTPHEADERA lpHdr);
63 INT HTTP_GetCustomHeaderIndex(LPWININETHTTPREQA lpwhr, LPCSTR lpszField);
65 /***********************************************************************
66 * HttpAddRequestHeadersA (WININET.68)
68 * Adds one or more HTTP header to the request handler
75 INTERNETAPI BOOL WINAPI HttpAddRequestHeadersA(HINTERNET hHttpRequest,
76 LPCSTR lpszHeader, DWORD dwHeaderLength, DWORD dwModifier)
81 CHAR value[MAX_FIELD_VALUE_LEN], field[MAX_FIELD_LEN];
82 BOOL bSuccess = FALSE;
83 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
85 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
87 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
91 buffer = HEAP_strdupA(GetProcessHeap(), 0, lpszHeader);
98 while (*lpszEnd != '\0')
100 if (*lpszEnd == '\r' && *(lpszEnd + 1) == '\n')
105 if (*lpszEnd == '\0')
110 if (HTTP_InterpretHttpHeader(lpszStart, field, MAX_FIELD_LEN, value, MAX_FIELD_VALUE_LEN))
111 bSuccess = HTTP_ProcessHeader(lpwhr, field, value, dwModifier | HTTP_ADDHDR_FLAG_REQ);
113 lpszStart = lpszEnd + 2; /* Jump over \0\n */
117 HeapFree(GetProcessHeap(), 0, buffer);
122 /***********************************************************************
123 * HttpOpenRequestA (WININET.72)
125 * Open a HTTP request handle
128 * HINTERNET a HTTP request handle on success
132 INTERNETAPI HINTERNET WINAPI HttpOpenRequestA(HINTERNET hHttpSession,
133 LPCSTR lpszVerb, LPCSTR lpszObjectName, LPCSTR lpszVersion,
134 LPCSTR lpszReferrer , LPCSTR *lpszAcceptTypes,
135 DWORD dwFlags, DWORD dwContext)
137 LPWININETHTTPSESSIONA lpwhs = (LPWININETHTTPSESSIONA) hHttpSession;
138 LPWININETAPPINFOA hIC = NULL;
142 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
144 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
148 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
150 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC)
152 WORKREQUEST workRequest;
154 workRequest.asyncall = HTTPOPENREQUESTA;
155 workRequest.HFTPSESSION = (DWORD)hHttpSession;
156 workRequest.LPSZVERB = (DWORD)HEAP_strdupA(GetProcessHeap(), 0, lpszVerb);
157 workRequest.LPSZOBJECTNAME = (DWORD)HEAP_strdupA(GetProcessHeap(), 0, lpszObjectName);
158 workRequest.LPSZVERSION = (DWORD)HEAP_strdupA(GetProcessHeap(), 0, lpszVersion);
159 workRequest.LPSZREFERRER = (DWORD)HEAP_strdupA(GetProcessHeap(), 0, lpszReferrer);
160 workRequest.LPSZACCEPTTYPES = (DWORD)lpszAcceptTypes;
161 workRequest.DWFLAGS = dwFlags;
162 workRequest.DWCONTEXT = dwContext;
164 return (HINTERNET)INTERNET_AsyncCall(&workRequest);
168 return HTTP_HttpOpenRequestA(hHttpSession, lpszVerb, lpszObjectName,
169 lpszVersion, lpszReferrer, lpszAcceptTypes, dwFlags, dwContext);
174 /***********************************************************************
175 * HTTP_HttpOpenRequestA (internal)
177 * Open a HTTP request handle
180 * HINTERNET a HTTP request handle on success
184 INTERNETAPI HINTERNET WINAPI HTTP_HttpOpenRequestA(HINTERNET hHttpSession,
185 LPCSTR lpszVerb, LPCSTR lpszObjectName, LPCSTR lpszVersion,
186 LPCSTR lpszReferrer , LPCSTR *lpszAcceptTypes,
187 DWORD dwFlags, DWORD dwContext)
189 LPWININETHTTPSESSIONA lpwhs = (LPWININETHTTPSESSIONA) hHttpSession;
190 LPWININETAPPINFOA hIC = NULL;
191 LPWININETHTTPREQA lpwhr;
195 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
197 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
201 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
203 lpwhr = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WININETHTTPREQA));
206 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
207 return (HINTERNET) NULL;
210 lpwhr->hdr.htype = WH_HHTTPREQ;
211 lpwhr->hdr.lpwhparent = hHttpSession;
212 lpwhr->hdr.dwFlags = dwFlags;
213 lpwhr->hdr.dwContext = dwContext;
214 lpwhr->nSocketFD = INVALID_SOCKET;
216 if (NULL != lpszObjectName && strlen(lpszObjectName))
217 lpwhr->lpszPath = HEAP_strdupA(GetProcessHeap(), 0, lpszObjectName);
219 if (NULL != lpszReferrer && strlen(lpszReferrer))
220 HTTP_ProcessHeader(lpwhr, HTTP_REFERER, lpszReferrer, HTTP_ADDHDR_FLAG_COALESCE);
223 if (NULL != lpszAcceptTypes && strlen(*lpszAcceptTypes))
224 HTTP_ProcessHeader(lpwhr, HTTP_ACCEPT, *lpszAcceptTypes, HTTP_ADDHDR_FLAG_COALESCE);
226 if (NULL == lpszVerb)
227 lpwhr->lpszVerb = HEAP_strdupA(GetProcessHeap(), 0, "GET");
228 else if (strlen(lpszVerb))
229 lpwhr->lpszVerb = HEAP_strdupA(GetProcessHeap(), 0, lpszVerb);
231 if (NULL != lpszReferrer)
233 char buf[MAXHOSTNAME];
234 URL_COMPONENTSA UrlComponents;
236 UrlComponents.lpszExtraInfo = NULL;
237 UrlComponents.lpszPassword = NULL;
238 UrlComponents.lpszScheme = NULL;
239 UrlComponents.lpszUrlPath = NULL;
240 UrlComponents.lpszUserName = NULL;
241 UrlComponents.lpszHostName = buf;
242 UrlComponents.dwHostNameLength = MAXHOSTNAME;
244 InternetCrackUrlA(lpszReferrer, 0, 0, &UrlComponents);
245 if (strlen(UrlComponents.lpszHostName))
246 lpwhr->lpszHostName = HEAP_strdupA(GetProcessHeap(), 0, UrlComponents.lpszHostName);
248 lpwhr->lpszHostName = HEAP_strdupA(GetProcessHeap(), 0,
249 lpwhs->lpszServerName);
252 if (hIC->lpfnStatusCB)
254 INTERNET_ASYNC_RESULT iar;
256 iar.dwResult = (DWORD)lpwhr;
257 iar.dwError = ERROR_SUCCESS;
259 hIC->lpfnStatusCB(hHttpSession, dwContext, INTERNET_STATUS_HANDLE_CREATED,
260 &iar, sizeof(INTERNET_ASYNC_RESULT));
263 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC && hIC->lpfnStatusCB)
265 INTERNET_ASYNC_RESULT iar;
267 iar.dwResult = (DWORD)lpwhr;
268 iar.dwError = lpwhr ? ERROR_SUCCESS : INTERNET_GetLastError();
269 hIC->lpfnStatusCB(hHttpSession, dwContext, INTERNET_STATUS_REQUEST_COMPLETE,
270 &iar, sizeof(INTERNET_ASYNC_RESULT));
273 return (HINTERNET) lpwhr;
277 /***********************************************************************
278 * HttpQueryInfoA (WININET.74)
280 * Queries for information about an HTTP request
287 BOOL WINAPI HttpQueryInfoA(HINTERNET hHttpRequest, DWORD dwInfoLevel,
288 LPVOID lpBuffer, LPDWORD lpdwBufferLength, LPDWORD lpdwIndex)
290 LPHTTPHEADERA lphttpHdr = NULL;
291 BOOL bSuccess = FALSE;
292 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
294 TRACE("(0x%08lx)--> %ld\n", dwInfoLevel, dwInfoLevel);
296 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
298 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
302 /* Find requested header structure */
303 if ((dwInfoLevel & ~HTTP_QUERY_MODIFIER_FLAGS_MASK) == HTTP_QUERY_CUSTOM)
305 INT index = HTTP_GetCustomHeaderIndex(lpwhr, (LPSTR)lpBuffer);
310 lphttpHdr = &lpwhr->pCustHeaders[index];
314 INT index = dwInfoLevel & ~HTTP_QUERY_MODIFIER_FLAGS_MASK;
316 if (index == HTTP_QUERY_RAW_HEADERS_CRLF || index == HTTP_QUERY_RAW_HEADERS)
318 INT i, delim, size = 0, cnt = 0;
320 delim = index == HTTP_QUERY_RAW_HEADERS_CRLF ? 2 : 1;
322 /* Calculate length of custom reuqest headers */
323 for (i = 0; i < lpwhr->nCustHeaders; i++)
325 if ((~lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST) && lpwhr->pCustHeaders[i].lpszField &&
326 lpwhr->pCustHeaders[i].lpszValue)
328 size += strlen(lpwhr->pCustHeaders[i].lpszField) +
329 strlen(lpwhr->pCustHeaders[i].lpszValue) + delim + 2;
333 /* Calculate the length of stadard request headers */
334 for (i = 0; i <= HTTP_QUERY_MAX; i++)
336 if ((~lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST) && lpwhr->StdHeaders[i].lpszField &&
337 lpwhr->StdHeaders[i].lpszValue)
339 size += strlen(lpwhr->StdHeaders[i].lpszField) +
340 strlen(lpwhr->StdHeaders[i].lpszValue) + delim + 2;
346 if (size + 1 > *lpdwBufferLength)
348 *lpdwBufferLength = size + 1;
349 INTERNET_SetLastError(ERROR_INSUFFICIENT_BUFFER);
353 /* Append standard request heades */
354 for (i = 0; i <= HTTP_QUERY_MAX; i++)
356 if ((~lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST) &&
357 lpwhr->StdHeaders[i].lpszField &&
358 lpwhr->StdHeaders[i].lpszValue)
360 cnt += sprintf(lpBuffer + cnt, "%s: %s%s", lpwhr->StdHeaders[i].lpszField, lpwhr->StdHeaders[i].lpszValue,
361 index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "\0");
365 /* Append custom request heades */
366 for (i = 0; i < lpwhr->nCustHeaders; i++)
368 if ((~lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST) &&
369 lpwhr->pCustHeaders[i].lpszField &&
370 lpwhr->pCustHeaders[i].lpszValue)
372 cnt += sprintf(lpBuffer + cnt, "%s: %s%s",
373 lpwhr->pCustHeaders[i].lpszField, lpwhr->pCustHeaders[i].lpszValue,
374 index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "\0");
378 strcpy(lpBuffer + cnt, index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "");
380 *lpdwBufferLength = cnt + delim;
384 else if (index >= 0 && index <= HTTP_QUERY_MAX && lpwhr->StdHeaders[index].lpszValue)
386 lphttpHdr = &lpwhr->StdHeaders[index];
392 /* Ensure header satisifies requested attributes */
393 if ((dwInfoLevel & HTTP_QUERY_FLAG_REQUEST_HEADERS) &&
394 (~lphttpHdr->wFlags & HDR_ISREQUEST))
397 /* coalesce value to reuqested type */
398 if (dwInfoLevel & HTTP_QUERY_FLAG_NUMBER)
400 *(int *)lpBuffer = atoi(lphttpHdr->lpszValue);
403 else if (dwInfoLevel & HTTP_QUERY_FLAG_SYSTEMTIME)
409 tmpTime = ConvertTimeString(lphttpHdr->lpszValue);
411 tmpTM = *gmtime(&tmpTime);
412 STHook = (SYSTEMTIME *) lpBuffer;
416 STHook->wDay = tmpTM.tm_mday;
417 STHook->wHour = tmpTM.tm_hour;
418 STHook->wMilliseconds = 0;
419 STHook->wMinute = tmpTM.tm_min;
420 STHook->wDayOfWeek = tmpTM.tm_wday;
421 STHook->wMonth = tmpTM.tm_mon + 1;
422 STHook->wSecond = tmpTM.tm_sec;
423 STHook->wYear = tmpTM.tm_year;
427 else if (dwInfoLevel & HTTP_QUERY_FLAG_COALESCE)
429 if (*lpdwIndex >= lphttpHdr->wCount)
431 INTERNET_SetLastError(ERROR_HTTP_HEADER_NOT_FOUND);
435 //! Copy strncpy(lpBuffer, lphttpHdr[*lpdwIndex], len);
441 INT len = strlen(lphttpHdr->lpszValue);
443 if (len + 1 > *lpdwBufferLength)
445 *lpdwBufferLength = len + 1;
446 INTERNET_SetLastError(ERROR_INSUFFICIENT_BUFFER);
450 strncpy(lpBuffer, lphttpHdr->lpszValue, len);
451 *lpdwBufferLength = len;
456 TRACE("%d <--\n", bSuccess);
461 /***********************************************************************
462 * HttpSendRequestExA (WININET)
464 * Sends the specified request to the HTTP server and allows chunked
467 BOOL WINAPI HttpSendRequestExA(HINTERNET hRequest,
468 LPINTERNET_BUFFERSA lpBuffersIn,
469 LPINTERNET_BUFFERSA lpBuffersOut,
470 DWORD dwFlags, DWORD dwContext)
472 FIXME("(%p, %p, %p, %08lx, %08lx): stub\n", hRequest, lpBuffersIn,
473 lpBuffersOut, dwFlags, dwContext);
477 /***********************************************************************
478 * HttpSendRequestA (WININET.76)
480 * Sends the specified request to the HTTP server
487 BOOL WINAPI HttpSendRequestA(HINTERNET hHttpRequest, LPCSTR lpszHeaders,
488 DWORD dwHeaderLength, LPVOID lpOptional ,DWORD dwOptionalLength)
490 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
491 LPWININETHTTPSESSIONA lpwhs = NULL;
492 LPWININETAPPINFOA hIC = NULL;
494 TRACE("0x%08lx\n", (unsigned long)hHttpRequest);
496 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
498 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
502 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
503 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
505 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
509 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
510 if (NULL == hIC || hIC->hdr.htype != WH_HINIT)
512 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
516 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC)
518 WORKREQUEST workRequest;
520 workRequest.asyncall = HTTPSENDREQUESTA;
521 workRequest.HFTPSESSION = (DWORD)hHttpRequest;
522 workRequest.LPSZHEADER = (DWORD)HEAP_strdupA(GetProcessHeap(), 0, lpszHeaders);
523 workRequest.DWHEADERLENGTH = dwHeaderLength;
524 workRequest.LPOPTIONAL = (DWORD)lpOptional;
525 workRequest.DWOPTIONALLENGTH = dwOptionalLength;
527 return INTERNET_AsyncCall(&workRequest);
531 return HTTP_HttpSendRequestA(hHttpRequest, lpszHeaders,
532 dwHeaderLength, lpOptional, dwOptionalLength);
537 /***********************************************************************
538 * HTTP_HttpSendRequestA (internal)
540 * Sends the specified request to the HTTP server
547 BOOL WINAPI HTTP_HttpSendRequestA(HINTERNET hHttpRequest, LPCSTR lpszHeaders,
548 DWORD dwHeaderLength, LPVOID lpOptional ,DWORD dwOptionalLength)
552 BOOL bSuccess = FALSE;
553 LPSTR requestString = NULL;
554 INT requestStringLen;
555 INT headerLength = 0;
556 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
557 LPWININETHTTPSESSIONA lpwhs = NULL;
558 LPWININETAPPINFOA hIC = NULL;
560 TRACE("0x%08lx\n", (ULONG)hHttpRequest);
562 /* Verify our tree of internet handles */
563 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
565 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
569 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
570 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
572 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
576 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
577 if (NULL == hIC || hIC->hdr.htype != WH_HINIT)
579 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
583 /* Clear any error information */
584 INTERNET_SetLastError(0);
586 /* We must have a verb */
587 if (NULL == lpwhr->lpszVerb)
592 /* If we don't have a path we set it to root */
593 if (NULL == lpwhr->lpszPath)
594 lpwhr->lpszPath = HEAP_strdupA(GetProcessHeap(), 0, "/");
596 /* Calculate length of request string */
598 strlen(lpwhr->lpszVerb) +
599 strlen(lpwhr->lpszPath) +
600 (lpwhr->lpszHostName ? (strlen(HTTPHOSTHEADER) + strlen(lpwhr->lpszHostName)) : 0) +
604 /* Add length of passed headers */
607 headerLength = -1 == dwHeaderLength ? strlen(lpszHeaders) : dwHeaderLength;
608 requestStringLen += headerLength + 2; /* \r\n */
611 /* Calculate length of custom request headers */
612 for (i = 0; i < lpwhr->nCustHeaders; i++)
614 if (lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST)
616 requestStringLen += strlen(lpwhr->pCustHeaders[i].lpszField) +
617 strlen(lpwhr->pCustHeaders[i].lpszValue) + 4; /*: \r\n */
621 /* Calculate the length of standard request headers */
622 for (i = 0; i <= HTTP_QUERY_MAX; i++)
624 if (lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST)
626 requestStringLen += strlen(lpwhr->StdHeaders[i].lpszField) +
627 strlen(lpwhr->StdHeaders[i].lpszValue) + 4; /*: \r\n */
631 /* Allocate string to hold entire request */
632 requestString = HeapAlloc(GetProcessHeap(), 0, requestStringLen + 1);
633 if (NULL == requestString)
635 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
639 /* Build request string */
640 cnt = sprintf(requestString, "%s %s%s%s",
643 lpwhr->lpszHostName ? (HTTPHEADER HTTPHOSTHEADER) : HTTPHEADER,
644 lpwhr->lpszHostName ? lpwhr->lpszHostName : "");
646 /* Append standard request headers */
647 for (i = 0; i <= HTTP_QUERY_MAX; i++)
649 if (lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST)
651 cnt += sprintf(requestString + cnt, "\r\n%s: %s",
652 lpwhr->StdHeaders[i].lpszField, lpwhr->StdHeaders[i].lpszValue);
656 /* Append custom request heades */
657 for (i = 0; i < lpwhr->nCustHeaders; i++)
659 if (lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST)
661 cnt += sprintf(requestString + cnt, "\r\n%s: %s",
662 lpwhr->pCustHeaders[i].lpszField, lpwhr->pCustHeaders[i].lpszValue);
666 /* Append passed request headers */
669 strcpy(requestString + cnt, "\r\n");
671 strcpy(requestString + cnt, lpszHeaders);
675 /* Set termination string for request */
676 strcpy(requestString + cnt, "\r\n\r\n");
678 if (hIC->lpfnStatusCB)
679 hIC->lpfnStatusCB(hHttpRequest, lpwhr->hdr.dwContext, INTERNET_STATUS_SENDING_REQUEST, NULL, 0);
681 TRACE("(%s) len(%d)\n", requestString, requestStringLen);
682 /* Send the request and store the results */
683 if (!HTTP_OpenConnection(lpwhr))
686 cnt = INTERNET_WriteDataToStream(lpwhr->nSocketFD, requestString, requestStringLen);
691 if (HTTP_GetResponseHeaders(lpwhr))
697 HeapFree(GetProcessHeap(), 0, requestString);
699 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC && hIC->lpfnStatusCB)
701 INTERNET_ASYNC_RESULT iar;
703 iar.dwResult = (DWORD)bSuccess;
704 iar.dwError = bSuccess ? ERROR_SUCCESS : INTERNET_GetLastError();
705 hIC->lpfnStatusCB(hHttpRequest, lpwhr->hdr.dwContext, INTERNET_STATUS_REQUEST_COMPLETE,
706 &iar, sizeof(INTERNET_ASYNC_RESULT));
714 /***********************************************************************
715 * HTTP_Connect (internal)
717 * Create http session handle
720 * HINTERNET a session handle on success
724 HINTERNET HTTP_Connect(HINTERNET hInternet, LPCSTR lpszServerName,
725 INTERNET_PORT nServerPort, LPCSTR lpszUserName,
726 LPCSTR lpszPassword, DWORD dwFlags, DWORD dwContext)
728 BOOL bSuccess = FALSE;
729 LPWININETAPPINFOA hIC = NULL;
730 LPWININETHTTPSESSIONA lpwhs = NULL;
734 if (((LPWININETHANDLEHEADER)hInternet)->htype != WH_HINIT)
737 hIC = (LPWININETAPPINFOA) hInternet;
739 lpwhs = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WININETHTTPSESSIONA));
742 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
746 if (hIC->lpfnStatusCB)
747 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_RESOLVING_NAME,
748 (LPVOID)lpszServerName, strlen(lpszServerName));
750 if (nServerPort == INTERNET_INVALID_PORT_NUMBER)
751 nServerPort = INTERNET_DEFAULT_HTTP_PORT;
753 if (!GetAddress(lpszServerName, nServerPort, &lpwhs->phostent, &lpwhs->socketAddress))
755 INTERNET_SetLastError(ERROR_INTERNET_NAME_NOT_RESOLVED);
759 if (hIC->lpfnStatusCB)
760 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_NAME_RESOLVED,
761 (LPVOID)lpszServerName, strlen(lpszServerName));
763 lpwhs->hdr.htype = WH_HHTTPSESSION;
764 lpwhs->hdr.lpwhparent = (LPWININETHANDLEHEADER)hInternet;
765 lpwhs->hdr.dwFlags = dwFlags;
766 lpwhs->hdr.dwContext = dwContext;
767 if (NULL != lpszServerName)
768 lpwhs->lpszServerName = HEAP_strdupA(GetProcessHeap(), 0, lpszServerName);
769 if (NULL != lpszUserName)
770 lpwhs->lpszUserName = HEAP_strdupA(GetProcessHeap(), 0, lpszUserName);
771 lpwhs->nServerPort = nServerPort;
773 if (hIC->lpfnStatusCB)
775 INTERNET_ASYNC_RESULT iar;
777 iar.dwResult = (DWORD)lpwhs;
778 iar.dwError = ERROR_SUCCESS;
780 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_HANDLE_CREATED,
781 &iar, sizeof(INTERNET_ASYNC_RESULT));
787 if (!bSuccess && lpwhs)
789 HeapFree(GetProcessHeap(), 0, lpwhs);
793 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC && hIC->lpfnStatusCB)
795 INTERNET_ASYNC_RESULT iar;
797 iar.dwResult = (DWORD)lpwhs;
798 iar.dwError = bSuccess ? ERROR_SUCCESS : INTERNET_GetLastError();
799 hIC->lpfnStatusCB(hInternet, dwContext, INTERNET_STATUS_REQUEST_COMPLETE,
800 &iar, sizeof(INTERNET_ASYNC_RESULT));
803 return (HINTERNET)lpwhs;
807 /***********************************************************************
808 * HTTP_OpenConnection (internal)
810 * Connect to a web server
817 BOOL HTTP_OpenConnection(LPWININETHTTPREQA lpwhr)
819 BOOL bSuccess = FALSE;
821 LPWININETHTTPSESSIONA lpwhs;
825 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
827 INTERNET_SetLastError(ERROR_INVALID_PARAMETER);
831 lpwhs = (LPWININETHTTPSESSIONA)lpwhr->hdr.lpwhparent;
833 lpwhr->nSocketFD = socket(lpwhs->phostent->h_addrtype,SOCK_STREAM,0);
834 if (INVALID_SOCKET == lpwhr->nSocketFD)
836 WARN("Socket creation failed\n");
840 result = connect(lpwhr->nSocketFD, (struct sockaddr *)&lpwhs->socketAddress,
841 sizeof(lpwhs->socketAddress));
843 if (SOCKET_ERROR == result)
845 WARN("Unable to connect to host (%s)\n", strerror(errno));
852 TRACE(": %d\n", bSuccess);
857 /***********************************************************************
858 * HTTP_GetResponseHeaders (internal)
860 * Read server response
867 BOOL HTTP_GetResponseHeaders(LPWININETHTTPREQA lpwhr)
870 CHAR buffer[MAX_REPLY_LEN];
871 DWORD buflen = MAX_REPLY_LEN;
872 BOOL bSuccess = FALSE;
873 CHAR value[MAX_FIELD_VALUE_LEN], field[MAX_FIELD_LEN];
877 if (INVALID_SOCKET == lpwhr->nSocketFD)
881 * We should first receive 'HTTP/1.x nnn' where nnn is the status code.
883 if (!INTERNET_GetNextLine(lpwhr->nSocketFD, buffer, &buflen))
886 if (strncmp(buffer, "HTTP", 4) != 0)
890 HTTP_ProcessHeader(lpwhr, "Status", buffer+9, (HTTP_ADDREQ_FLAG_ADD | HTTP_ADDREQ_FLAG_REPLACE));
892 /* Parse each response line */
895 buflen = MAX_REPLY_LEN;
896 if (INTERNET_GetNextLine(lpwhr->nSocketFD, buffer, &buflen))
898 if (!HTTP_InterpretHttpHeader(buffer, field, MAX_FIELD_LEN, value, MAX_FIELD_VALUE_LEN))
901 HTTP_ProcessHeader(lpwhr, field, value, (HTTP_ADDREQ_FLAG_ADD | HTTP_ADDREQ_FLAG_REPLACE));
919 /***********************************************************************
920 * HTTP_InterpretHttpHeader (internal)
922 * Parse server response
929 INT stripSpaces(LPCSTR lpszSrc, LPSTR lpszStart, INT *len)
936 while (*lpszSrc == ' ' && *lpszSrc != '\0')
940 while(*lpsztmp != '\0')
943 srclen = lpsztmp - lpszSrc + 1;
948 *len = min(*len, srclen);
949 strncpy(lpszStart, lpszSrc, *len);
950 lpszStart[*len] = '\0';
956 BOOL HTTP_InterpretHttpHeader(LPSTR buffer, LPSTR field, INT fieldlen, LPSTR value, INT valuelen)
959 BOOL bSuccess = FALSE;
966 pd = strchr(buffer, ':');
970 if (stripSpaces(buffer, field, &fieldlen) > 0)
972 if (stripSpaces(pd+1, value, &valuelen) > 0)
977 TRACE("%d: field(%s) Value(%s)\n", bSuccess, field, value);
982 /***********************************************************************
983 * HTTP_GetStdHeaderIndex (internal)
985 * Lookup field index in standard http header array
987 * FIXME: This should be stuffed into a hash table
989 INT HTTP_GetStdHeaderIndex(LPCSTR lpszField)
993 if (!strcasecmp(lpszField, "Content-Length"))
994 index = HTTP_QUERY_CONTENT_LENGTH;
995 else if (!strcasecmp(lpszField,"Status"))
996 index = HTTP_QUERY_STATUS_CODE;
997 else if (!strcasecmp(lpszField,"Content-Type"))
998 index = HTTP_QUERY_CONTENT_TYPE;
999 else if (!strcasecmp(lpszField,"Last-Modified"))
1000 index = HTTP_QUERY_LAST_MODIFIED;
1001 else if (!strcasecmp(lpszField,"Location"))
1002 index = HTTP_QUERY_LOCATION;
1003 else if (!strcasecmp(lpszField,"Accept"))
1004 index = HTTP_QUERY_ACCEPT;
1005 else if (!strcasecmp(lpszField,"Referer"))
1006 index = HTTP_QUERY_REFERER;
1007 else if (!strcasecmp(lpszField,"Content-Transfer-Encoding"))
1008 index = HTTP_QUERY_CONTENT_TRANSFER_ENCODING;
1009 else if (!strcasecmp(lpszField,"Date"))
1010 index = HTTP_QUERY_DATE;
1011 else if (!strcasecmp(lpszField,"Server"))
1012 index = HTTP_QUERY_SERVER;
1013 else if (!strcasecmp(lpszField,"Connection"))
1014 index = HTTP_QUERY_CONNECTION;
1015 else if (!strcasecmp(lpszField,"ETag"))
1016 index = HTTP_QUERY_ETAG;
1017 else if (!strcasecmp(lpszField,"Accept-Ranges"))
1018 index = HTTP_QUERY_ACCEPT_RANGES;
1019 else if (!strcasecmp(lpszField,"Expires"))
1020 index = HTTP_QUERY_EXPIRES;
1021 else if (!strcasecmp(lpszField,"Mime-Version"))
1022 index = HTTP_QUERY_MIME_VERSION;
1025 FIXME("Couldn't find %s in standard header table\n", lpszField);
1032 /***********************************************************************
1033 * HTTP_ProcessHeader (internal)
1035 * Stuff header into header tables according to <dwModifier>
1039 #define COALESCEFLASG (HTTP_ADDHDR_FLAG_COALESCE|HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA|HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON)
1041 BOOL HTTP_ProcessHeader(LPWININETHTTPREQA lpwhr, LPCSTR field, LPCSTR value, DWORD dwModifier)
1043 LPHTTPHEADERA lphttpHdr = NULL;
1044 BOOL bSuccess = FALSE;
1047 TRACE("%s:%s - 0x%08x\n", field, value, (unsigned int)dwModifier);
1049 /* Adjust modifier flags */
1050 if (dwModifier & COALESCEFLASG)
1051 dwModifier |= HTTP_ADDHDR_FLAG_ADD;
1053 /* Try to get index into standard header array */
1054 index = HTTP_GetStdHeaderIndex(field);
1057 lphttpHdr = &lpwhr->StdHeaders[index];
1059 else /* Find or create new custom header */
1061 index = HTTP_GetCustomHeaderIndex(lpwhr, field);
1064 if (dwModifier & HTTP_ADDHDR_FLAG_ADD_IF_NEW)
1068 lphttpHdr = &lpwhr->pCustHeaders[index];
1074 hdr.lpszField = (LPSTR)field;
1075 hdr.lpszValue = (LPSTR)value;
1076 hdr.wFlags = hdr.wCount = 0;
1078 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1079 hdr.wFlags |= HDR_ISREQUEST;
1081 index = HTTP_InsertCustomHeader(lpwhr, &hdr);
1086 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1087 lphttpHdr->wFlags |= HDR_ISREQUEST;
1089 lphttpHdr->wFlags &= ~HDR_ISREQUEST;
1091 if (!lphttpHdr->lpszValue && (dwModifier & (HTTP_ADDHDR_FLAG_ADD|HTTP_ADDHDR_FLAG_ADD_IF_NEW)))
1095 if (!lpwhr->StdHeaders[index].lpszField)
1097 lphttpHdr->lpszField = HEAP_strdupA(GetProcessHeap(), 0, field);
1099 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1100 lphttpHdr->wFlags |= HDR_ISREQUEST;
1103 slen = strlen(value) + 1;
1104 lphttpHdr->lpszValue = HeapAlloc(GetProcessHeap(), 0, slen);
1105 if (lphttpHdr->lpszValue)
1107 memcpy(lphttpHdr->lpszValue, value, slen);
1112 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1115 else if (lphttpHdr->lpszValue)
1117 if (dwModifier & HTTP_ADDHDR_FLAG_REPLACE)
1122 len = strlen(value);
1126 //! if custom header delete from array
1127 HeapFree(GetProcessHeap(), 0, lphttpHdr->lpszValue);
1128 lphttpHdr->lpszValue = NULL;
1133 lpsztmp = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lphttpHdr->lpszValue, len+1);
1136 lphttpHdr->lpszValue = lpsztmp;
1137 strcpy(lpsztmp, value);
1142 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1146 else if (dwModifier & COALESCEFLASG)
1151 INT origlen = strlen(lphttpHdr->lpszValue);
1152 INT valuelen = strlen(value);
1154 if (dwModifier & HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA)
1157 lphttpHdr->wFlags |= HDR_COMMADELIMITED;
1159 else if (dwModifier & HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON)
1162 lphttpHdr->wFlags |= HDR_COMMADELIMITED;
1165 len = origlen + valuelen + (ch > 0) ? 1 : 0;
1167 lpsztmp = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lphttpHdr->lpszValue, len+1);
1170 /* FIXME: Increment lphttpHdr->wCount. Perhaps lpszValue should be an array */
1173 lphttpHdr->lpszValue[origlen] = ch;
1177 memcpy(&lphttpHdr->lpszValue[origlen], value, valuelen);
1178 lphttpHdr->lpszValue[len] = '\0';
1183 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1192 /***********************************************************************
1193 * HTTP_CloseConnection (internal)
1195 * Close socket connection
1198 VOID HTTP_CloseConnection(LPWININETHTTPREQA lpwhr)
1200 if (lpwhr->nSocketFD != INVALID_SOCKET)
1202 close(lpwhr->nSocketFD);
1203 lpwhr->nSocketFD = INVALID_SOCKET;
1208 /***********************************************************************
1209 * HTTP_CloseHTTPRequestHandle (internal)
1211 * Deallocate request handle
1214 void HTTP_CloseHTTPRequestHandle(LPWININETHTTPREQA lpwhr)
1220 if (lpwhr->nSocketFD != INVALID_SOCKET)
1221 HTTP_CloseConnection(lpwhr);
1223 if (lpwhr->lpszPath)
1224 HeapFree(GetProcessHeap(), 0, lpwhr->lpszPath);
1225 if (lpwhr->lpszVerb)
1226 HeapFree(GetProcessHeap(), 0, lpwhr->lpszVerb);
1227 if (lpwhr->lpszHostName)
1228 HeapFree(GetProcessHeap(), 0, lpwhr->lpszHostName);
1230 for (i = 0; i <= HTTP_QUERY_MAX; i++)
1232 if (lpwhr->StdHeaders[i].lpszField)
1233 HeapFree(GetProcessHeap(), 0, lpwhr->StdHeaders[i].lpszField);
1234 if (lpwhr->StdHeaders[i].lpszValue)
1235 HeapFree(GetProcessHeap(), 0, lpwhr->StdHeaders[i].lpszValue);
1238 for (i = 0; i < lpwhr->nCustHeaders; i++)
1240 if (lpwhr->pCustHeaders[i].lpszField)
1241 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders[i].lpszField);
1242 if (lpwhr->pCustHeaders[i].lpszValue)
1243 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders[i].lpszValue);
1246 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders);
1247 HeapFree(GetProcessHeap(), 0, lpwhr);
1251 /***********************************************************************
1252 * HTTP_CloseHTTPSessionHandle (internal)
1254 * Deallocate session handle
1257 void HTTP_CloseHTTPSessionHandle(LPWININETHTTPSESSIONA lpwhs)
1261 if (lpwhs->lpszServerName)
1262 HeapFree(GetProcessHeap(), 0, lpwhs->lpszServerName);
1263 if (lpwhs->lpszUserName)
1264 HeapFree(GetProcessHeap(), 0, lpwhs->lpszUserName);
1265 HeapFree(GetProcessHeap(), 0, lpwhs);
1269 /***********************************************************************
1270 * HTTP_GetCustomHeaderIndex (internal)
1272 * Return index of custom header from header array
1275 INT HTTP_GetCustomHeaderIndex(LPWININETHTTPREQA lpwhr, LPCSTR lpszField)
1279 TRACE("%s\n", lpszField);
1281 for (index = 0; index < lpwhr->nCustHeaders; index++)
1283 if (!strcasecmp(lpwhr->pCustHeaders[index].lpszField, lpszField))
1288 if (index >= lpwhr->nCustHeaders)
1291 TRACE("Return: %d\n", index);
1296 /***********************************************************************
1297 * HTTP_InsertCustomHeader (internal)
1299 * Insert header into array
1302 INT HTTP_InsertCustomHeader(LPWININETHTTPREQA lpwhr, LPHTTPHEADERA lpHdr)
1305 LPHTTPHEADERA lph = NULL;
1307 TRACE("%s: %s\n", lpHdr->lpszField, lpHdr->lpszValue);
1308 count = lpwhr->nCustHeaders + 1;
1310 lph = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lpwhr->pCustHeaders, sizeof(HTTPHEADERA) * count);
1312 lph = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(HTTPHEADERA) * count);
1316 lpwhr->pCustHeaders = lph;
1317 lpwhr->pCustHeaders[count-1].lpszField = HEAP_strdupA(GetProcessHeap(), 0, lpHdr->lpszField);
1318 lpwhr->pCustHeaders[count-1].lpszValue = HEAP_strdupA(GetProcessHeap(), 0, lpHdr->lpszValue);
1319 lpwhr->pCustHeaders[count-1].wFlags = lpHdr->wFlags;
1320 lpwhr->pCustHeaders[count-1].wCount= lpHdr->wCount;
1321 lpwhr->nCustHeaders++;
1325 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1329 TRACE("%d <--\n", count-1);
1334 /***********************************************************************
1335 * HTTP_DeleteCustomHeader (internal)
1337 * Delete header from array
1340 BOOL HTTP_DeleteCustomHeader(INT index)