blob: aacbee37e28ef6d71aa84b12586604d026ad29e9 [file] [log] [blame]
Vincent Béron9a624912002-05-31 23:06:46 +00001/*
Ulrich Czekallac2757242000-06-11 20:04:44 +00002 * Wininet - Http Implementation
3 *
4 * Copyright 1999 Corel Corporation
Aric Stewartff9b9d42002-06-21 23:59:49 +00005 * Copyright 2002 CodeWeavers Inc.
David Hammerton852c7ae2003-06-20 23:26:56 +00006 * Copyright 2002 TransGaming Technologies Inc.
Ulrich Czekallac2757242000-06-11 20:04:44 +00007 *
8 * Ulrich Czekalla
Aric Stewartff9b9d42002-06-21 23:59:49 +00009 * Aric Stewart
David Hammerton852c7ae2003-06-20 23:26:56 +000010 * David Hammerton
Ulrich Czekallac2757242000-06-11 20:04:44 +000011 *
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000012 * This library is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU Lesser General Public
14 * License as published by the Free Software Foundation; either
15 * version 2.1 of the License, or (at your option) any later version.
16 *
17 * This library is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * Lesser General Public License for more details.
21 *
22 * You should have received a copy of the GNU Lesser General Public
23 * License along with this library; if not, write to the Free Software
24 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
Ulrich Czekallac2757242000-06-11 20:04:44 +000025 */
26
Patrik Stridvall4710be22000-06-23 15:47:14 +000027#include "config.h"
28
Ulrich Czekallac2757242000-06-11 20:04:44 +000029#include <sys/types.h>
Patrik Stridvall4710be22000-06-23 15:47:14 +000030#ifdef HAVE_SYS_SOCKET_H
31# include <sys/socket.h>
32#endif
Ulrich Czekallac2757242000-06-11 20:04:44 +000033#include <stdio.h>
34#include <stdlib.h>
Patrik Stridvalld016f812002-08-17 00:43:16 +000035#ifdef HAVE_UNISTD_H
36# include <unistd.h>
37#endif
Ulrich Czekallac2757242000-06-11 20:04:44 +000038#include <errno.h>
Jon Griffiths4ab15582001-01-22 02:17:29 +000039#include <string.h>
Chris Morganb9807b42001-02-15 21:24:07 +000040#include <time.h>
Ulrich Czekallac2757242000-06-11 20:04:44 +000041
Guy Albertelliaafec982001-11-06 22:31:19 +000042#include "windef.h"
43#include "winbase.h"
44#include "wininet.h"
45#include "winreg.h"
46#include "winerror.h"
Jon Griffiths603f20f2001-12-11 00:30:17 +000047#define NO_SHLWAPI_STREAM
Guy Albertelliaafec982001-11-06 22:31:19 +000048#include "shlwapi.h"
49
Ulrich Czekallac2757242000-06-11 20:04:44 +000050#include "internet.h"
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000051#include "wine/debug.h"
Alberto Massarid476a5a2002-11-12 02:13:04 +000052#include "wine/unicode.h"
Ulrich Czekallac2757242000-06-11 20:04:44 +000053
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000054WINE_DEFAULT_DEBUG_CHANNEL(wininet);
Ulrich Czekallac2757242000-06-11 20:04:44 +000055
56#define HTTPHEADER " HTTP/1.0"
57#define HTTPHOSTHEADER "\r\nHost: "
58#define MAXHOSTNAME 100
59#define MAX_FIELD_VALUE_LEN 256
60#define MAX_FIELD_LEN 256
61
62
Alberto Massaribc8bd722002-12-06 23:20:31 +000063#define HTTP_REFERER "Referer"
64#define HTTP_ACCEPT "Accept"
65#define HTTP_USERAGENT "User-Agent"
Ulrich Czekallac2757242000-06-11 20:04:44 +000066
67#define HTTP_ADDHDR_FLAG_ADD 0x20000000
68#define HTTP_ADDHDR_FLAG_ADD_IF_NEW 0x10000000
69#define HTTP_ADDHDR_FLAG_COALESCE 0x40000000
70#define HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA 0x40000000
71#define HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON 0x01000000
72#define HTTP_ADDHDR_FLAG_REPLACE 0x80000000
73#define HTTP_ADDHDR_FLAG_REQ 0x02000000
74
75
76BOOL HTTP_OpenConnection(LPWININETHTTPREQA lpwhr);
Vincent Béron9a624912002-05-31 23:06:46 +000077int HTTP_WriteDataToStream(LPWININETHTTPREQA lpwhr,
Ulrich Czekallac2757242000-06-11 20:04:44 +000078 void *Buffer, int BytesToWrite);
Vincent Béron9a624912002-05-31 23:06:46 +000079int HTTP_ReadDataFromStream(LPWININETHTTPREQA lpwhr,
Ulrich Czekallac2757242000-06-11 20:04:44 +000080 void *Buffer, int BytesToRead);
81BOOL HTTP_GetResponseHeaders(LPWININETHTTPREQA lpwhr);
82BOOL HTTP_ProcessHeader(LPWININETHTTPREQA lpwhr, LPCSTR field, LPCSTR value, DWORD dwModifier);
83void HTTP_CloseConnection(LPWININETHTTPREQA lpwhr);
84BOOL HTTP_InterpretHttpHeader(LPSTR buffer, LPSTR field, INT fieldlen, LPSTR value, INT valuelen);
85INT HTTP_GetStdHeaderIndex(LPCSTR lpszField);
Mike McCormacka1c16d22003-07-22 03:17:52 +000086BOOL HTTP_InsertCustomHeader(LPWININETHTTPREQA lpwhr, LPHTTPHEADERA lpHdr);
Ulrich Czekallac2757242000-06-11 20:04:44 +000087INT HTTP_GetCustomHeaderIndex(LPWININETHTTPREQA lpwhr, LPCSTR lpszField);
Mike McCormacka1c16d22003-07-22 03:17:52 +000088BOOL HTTP_DeleteCustomHeader(LPWININETHTTPREQA lpwhr, INT index);
Ulrich Czekallac2757242000-06-11 20:04:44 +000089
Alexandre Julliard0e44f632000-11-16 00:28:52 +000090inline static LPSTR HTTP_strdup( LPCSTR str )
91{
92 LPSTR ret = HeapAlloc( GetProcessHeap(), 0, strlen(str) + 1 );
93 if (ret) strcpy( ret, str );
94 return ret;
95}
96
Ulrich Czekallac2757242000-06-11 20:04:44 +000097/***********************************************************************
Patrik Stridvall8b216b32001-06-19 18:20:47 +000098 * HttpAddRequestHeadersA (WININET.@)
Ulrich Czekallac2757242000-06-11 20:04:44 +000099 *
100 * Adds one or more HTTP header to the request handler
101 *
102 * RETURNS
103 * TRUE on success
104 * FALSE on failure
105 *
106 */
Vincent Béron9a624912002-05-31 23:06:46 +0000107BOOL WINAPI HttpAddRequestHeadersA(HINTERNET hHttpRequest,
Ulrich Czekallac2757242000-06-11 20:04:44 +0000108 LPCSTR lpszHeader, DWORD dwHeaderLength, DWORD dwModifier)
109{
110 LPSTR lpszStart;
111 LPSTR lpszEnd;
112 LPSTR buffer;
113 CHAR value[MAX_FIELD_VALUE_LEN], field[MAX_FIELD_LEN];
114 BOOL bSuccess = FALSE;
115 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
116
David Hammerton852c7ae2003-06-20 23:26:56 +0000117 TRACE("%p, %s, %li, %li\n", hHttpRequest, lpszHeader, dwHeaderLength,
118 dwModifier);
119
Aric Stewartff9b9d42002-06-21 23:59:49 +0000120
Ulrich Czekallac2757242000-06-11 20:04:44 +0000121 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
122 {
123 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
124 return FALSE;
125 }
126
Uwe Bonnes1cd00da2003-02-25 03:57:59 +0000127 if (!lpszHeader)
128 return TRUE;
David Hammerton852c7ae2003-06-20 23:26:56 +0000129
130 TRACE("copying header: %s\n", lpszHeader);
Alexandre Julliard0e44f632000-11-16 00:28:52 +0000131 buffer = HTTP_strdup(lpszHeader);
Ulrich Czekallac2757242000-06-11 20:04:44 +0000132 lpszStart = buffer;
133
134 do
135 {
136 lpszEnd = lpszStart;
137
138 while (*lpszEnd != '\0')
139 {
140 if (*lpszEnd == '\r' && *(lpszEnd + 1) == '\n')
141 break;
142 lpszEnd++;
143 }
144
145 if (*lpszEnd == '\0')
146 break;
147
148 *lpszEnd = '\0';
149
David Hammerton852c7ae2003-06-20 23:26:56 +0000150 TRACE("interpreting header %s\n", debugstr_a(lpszStart));
Ulrich Czekallac2757242000-06-11 20:04:44 +0000151 if (HTTP_InterpretHttpHeader(lpszStart, field, MAX_FIELD_LEN, value, MAX_FIELD_VALUE_LEN))
152 bSuccess = HTTP_ProcessHeader(lpwhr, field, value, dwModifier | HTTP_ADDHDR_FLAG_REQ);
153
154 lpszStart = lpszEnd + 2; /* Jump over \0\n */
155
156 } while (bSuccess);
157
158 HeapFree(GetProcessHeap(), 0, buffer);
159 return bSuccess;
160}
161
Chris Morgana8b32162002-09-27 22:05:23 +0000162/***********************************************************************
163 * HttpEndRequestA (WININET.@)
164 *
165 * Ends an HTTP request that was started by HttpSendRequestEx
166 *
167 * RETURNS
168 * TRUE if successful
169 * FALSE on failure
170 *
171 */
172BOOL WINAPI HttpEndRequestA(HINTERNET hRequest, LPINTERNET_BUFFERSA lpBuffersOut,
173 DWORD dwFlags, DWORD dwContext)
174{
175 FIXME("stub\n");
176 return FALSE;
177}
178
179/***********************************************************************
180 * HttpEndRequestW (WININET.@)
181 *
182 * Ends an HTTP request that was started by HttpSendRequestEx
183 *
184 * RETURNS
185 * TRUE if successful
186 * FALSE on failure
187 *
188 */
189BOOL WINAPI HttpEndRequestW(HINTERNET hRequest, LPINTERNET_BUFFERSW lpBuffersOut,
190 DWORD dwFlags, DWORD dwContext)
191{
192 FIXME("stub\n");
193 return FALSE;
194}
Ulrich Czekallac2757242000-06-11 20:04:44 +0000195
196/***********************************************************************
Patrik Stridvall8b216b32001-06-19 18:20:47 +0000197 * HttpOpenRequestA (WININET.@)
Ulrich Czekallac2757242000-06-11 20:04:44 +0000198 *
199 * Open a HTTP request handle
200 *
201 * RETURNS
202 * HINTERNET a HTTP request handle on success
203 * NULL on failure
204 *
205 */
Patrik Stridvall3c0211f2001-09-11 00:32:32 +0000206HINTERNET WINAPI HttpOpenRequestA(HINTERNET hHttpSession,
Ulrich Czekallac2757242000-06-11 20:04:44 +0000207 LPCSTR lpszVerb, LPCSTR lpszObjectName, LPCSTR lpszVersion,
Vincent Béron9a624912002-05-31 23:06:46 +0000208 LPCSTR lpszReferrer , LPCSTR *lpszAcceptTypes,
Ulrich Czekallac2757242000-06-11 20:04:44 +0000209 DWORD dwFlags, DWORD dwContext)
210{
211 LPWININETHTTPSESSIONA lpwhs = (LPWININETHTTPSESSIONA) hHttpSession;
212 LPWININETAPPINFOA hIC = NULL;
213
David Hammerton852c7ae2003-06-20 23:26:56 +0000214 TRACE("(%p, %s, %s, %s, %s, %p, %08lx, %08lx)\n", hHttpSession,
215 debugstr_a(lpszVerb), lpszObjectName,
216 debugstr_a(lpszVersion), debugstr_a(lpszReferrer), lpszAcceptTypes,
217 dwFlags, dwContext);
Alberto Massaribc8bd722002-12-06 23:20:31 +0000218 if(lpszAcceptTypes!=NULL)
219 {
220 int i;
221 for(i=0;lpszAcceptTypes[i]!=NULL;i++)
222 TRACE("\taccept type: %s\n",lpszAcceptTypes[i]);
223 }
Ulrich Czekallac2757242000-06-11 20:04:44 +0000224
225 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
226 {
227 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
228 return FALSE;
229 }
Ulrich Czekallac2757242000-06-11 20:04:44 +0000230 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
231
Aric Stewartff9b9d42002-06-21 23:59:49 +0000232 /*
233 * My tests seem to show that the windows version does not
234 * become asynchronous until after this point. And anyhow
235 * if this call was asynchronous then how would you get the
236 * necessary HINTERNET pointer returned by this function.
237 *
238 * I am leaving this here just in case I am wrong
239 *
240 * if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC)
241 */
242 if (0)
Ulrich Czekallac2757242000-06-11 20:04:44 +0000243 {
244 WORKREQUEST workRequest;
245
246 workRequest.asyncall = HTTPOPENREQUESTA;
247 workRequest.HFTPSESSION = (DWORD)hHttpSession;
Alexandre Julliard0e44f632000-11-16 00:28:52 +0000248 workRequest.LPSZVERB = (DWORD)HTTP_strdup(lpszVerb);
249 workRequest.LPSZOBJECTNAME = (DWORD)HTTP_strdup(lpszObjectName);
Aric Stewartff9b9d42002-06-21 23:59:49 +0000250 if (lpszVersion)
251 workRequest.LPSZVERSION = (DWORD)HTTP_strdup(lpszVersion);
252 else
253 workRequest.LPSZVERSION = 0;
254 if (lpszReferrer)
255 workRequest.LPSZREFERRER = (DWORD)HTTP_strdup(lpszReferrer);
256 else
257 workRequest.LPSZREFERRER = 0;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000258 workRequest.LPSZACCEPTTYPES = (DWORD)lpszAcceptTypes;
259 workRequest.DWFLAGS = dwFlags;
260 workRequest.DWCONTEXT = dwContext;
261
Aric Stewartff9b9d42002-06-21 23:59:49 +0000262 INTERNET_AsyncCall(&workRequest);
David Hammerton852c7ae2003-06-20 23:26:56 +0000263 TRACE ("returning NULL\n");
Aric Stewartff9b9d42002-06-21 23:59:49 +0000264 return NULL;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000265 }
266 else
267 {
David Hammerton852c7ae2003-06-20 23:26:56 +0000268 HINTERNET rec = HTTP_HttpOpenRequestA(hHttpSession, lpszVerb, lpszObjectName,
269 lpszVersion, lpszReferrer, lpszAcceptTypes,
270 dwFlags, dwContext);
271 TRACE("returning %p\n", rec);
272 return rec;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000273 }
274}
275
David Hammerton852c7ae2003-06-20 23:26:56 +0000276
Alberto Massarid476a5a2002-11-12 02:13:04 +0000277/***********************************************************************
278 * HttpOpenRequestW (WININET.@)
279 *
280 * Open a HTTP request handle
281 *
282 * RETURNS
283 * HINTERNET a HTTP request handle on success
284 * NULL on failure
285 *
David Hammerton852c7ae2003-06-20 23:26:56 +0000286 * FIXME: This should be the other way around (A should call W)
Alberto Massarid476a5a2002-11-12 02:13:04 +0000287 */
288HINTERNET WINAPI HttpOpenRequestW(HINTERNET hHttpSession,
289 LPCWSTR lpszVerb, LPCWSTR lpszObjectName, LPCWSTR lpszVersion,
290 LPCWSTR lpszReferrer , LPCWSTR *lpszAcceptTypes,
291 DWORD dwFlags, DWORD dwContext)
292{
David Hammerton852c7ae2003-06-20 23:26:56 +0000293 CHAR *szVerb = NULL, *szObjectName = NULL;
294 CHAR *szVersion = NULL, *szReferrer = NULL, **szAcceptTypes = NULL;
295 INT len;
296 INT acceptTypesCount;
297 HINTERNET rc = FALSE;
298 TRACE("(%p, %s, %s, %s, %s, %p, %08lx, %08lx)\n", hHttpSession,
299 debugstr_w(lpszVerb), debugstr_w(lpszObjectName),
300 debugstr_w(lpszVersion), debugstr_w(lpszReferrer), lpszAcceptTypes,
301 dwFlags, dwContext);
Alberto Massaribc8bd722002-12-06 23:20:31 +0000302
David Hammerton852c7ae2003-06-20 23:26:56 +0000303 if (lpszVerb)
304 {
305 len = lstrlenW(lpszVerb)+1;
306 if (!(szVerb = (CHAR *) HeapAlloc(GetProcessHeap(), 0, len * sizeof(CHAR))))
307 goto end;
308 WideCharToMultiByte(CP_ACP, -1, lpszVerb, -1, szVerb, len, NULL, NULL);
309 }
310
311 if (lpszObjectName)
312 {
313 len = lstrlenW(lpszObjectName)+1;
314 if (!(szObjectName = (CHAR *) HeapAlloc(GetProcessHeap(), 0, len * sizeof(CHAR))))
315 goto end;
316 WideCharToMultiByte(CP_ACP, -1, lpszObjectName, -1, szObjectName, len, NULL, NULL);
317 }
318
319 if (lpszVersion)
320 {
321 len = lstrlenW(lpszVersion)+1;
322 if (!(szVersion = (CHAR *) HeapAlloc(GetProcessHeap(), 0, len * sizeof(CHAR))))
323 goto end;
324 WideCharToMultiByte(CP_ACP, -1, lpszVersion, -1, szVersion, len, NULL, NULL);
325 }
326
327 if (lpszReferrer)
328 {
329 len = lstrlenW(lpszReferrer)+1;
330 if (!(szReferrer = (CHAR *) HeapAlloc(GetProcessHeap(), 0, len * sizeof(CHAR))))
331 goto end;
332 WideCharToMultiByte(CP_ACP, -1, lpszReferrer, -1, szReferrer, len, NULL, NULL);
333 }
334
335 acceptTypesCount = 0;
336 if (lpszAcceptTypes)
337 {
338 while (lpszAcceptTypes[acceptTypesCount]) { acceptTypesCount++; } /* find out how many there are */
339 szAcceptTypes = HeapAlloc(GetProcessHeap(), 0, sizeof(CHAR *) * acceptTypesCount);
340 acceptTypesCount = 0;
341 while (lpszAcceptTypes[acceptTypesCount])
342 {
343 len = lstrlenW(lpszAcceptTypes[acceptTypesCount])+1;
344 if (!(szAcceptTypes[acceptTypesCount] = (CHAR *) HeapAlloc(GetProcessHeap(),
345 0, len * sizeof(CHAR))))
346 goto end;
347 WideCharToMultiByte(CP_ACP, -1, lpszAcceptTypes[acceptTypesCount],
348 -1, szAcceptTypes[acceptTypesCount], len, NULL, NULL);
349 acceptTypesCount++;
350 }
351 }
352 else szAcceptTypes = 0;
353
354 rc = HttpOpenRequestA(hHttpSession, (LPCSTR)szVerb, (LPCSTR)szObjectName,
355 (LPCSTR)szVersion, (LPCSTR)szReferrer,
356 (LPCSTR *)szAcceptTypes, dwFlags, dwContext);
357
358end:
359 if (szAcceptTypes)
360 {
361 acceptTypesCount = 0;
362 while (szAcceptTypes[acceptTypesCount])
363 {
364 HeapFree(GetProcessHeap(), 0, szAcceptTypes[acceptTypesCount]);
365 acceptTypesCount++;
366 }
367 HeapFree(GetProcessHeap(), 0, szAcceptTypes);
368 }
369 if (szReferrer) HeapFree(GetProcessHeap(), 0, szReferrer);
370 if (szVersion) HeapFree(GetProcessHeap(), 0, szVersion);
371 if (szObjectName) HeapFree(GetProcessHeap(), 0, szObjectName);
372 if (szVerb) HeapFree(GetProcessHeap(), 0, szVerb);
373
374 return rc;
Alberto Massarid476a5a2002-11-12 02:13:04 +0000375}
Ulrich Czekallac2757242000-06-11 20:04:44 +0000376
377/***********************************************************************
Mike McCormacka1c16d22003-07-22 03:17:52 +0000378 * HTTP_Base64
379 */
380static UINT HTTP_Base64( LPCSTR bin, LPSTR base64 )
381{
382 UINT n = 0, x;
383 static LPSTR HTTP_Base64Enc =
384 "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
385
386 while( bin[0] )
387 {
388 /* first 6 bits, all from bin[0] */
389 base64[n++] = HTTP_Base64Enc[(bin[0] & 0xfc) >> 2];
390 x = (bin[0] & 3) << 4;
391
392 /* next 6 bits, 2 from bin[0] and 4 from bin[1] */
393 if( !bin[1] )
394 {
395 base64[n++] = HTTP_Base64Enc[x];
396 base64[n++] = '=';
397 base64[n++] = '=';
398 break;
399 }
400 base64[n++] = HTTP_Base64Enc[ x | ( (bin[1]&0xf0) >> 4 ) ];
401 x = ( bin[1] & 0x0f ) << 2;
402
403 /* next 6 bits 4 from bin[1] and 2 from bin[2] */
404 if( !bin[2] )
405 {
406 base64[n++] = HTTP_Base64Enc[x];
407 base64[n++] = '=';
408 break;
409 }
410 base64[n++] = HTTP_Base64Enc[ x | ( (bin[2]&0xc0 ) >> 6 ) ];
411
412 /* last 6 bits, all from bin [2] */
413 base64[n++] = HTTP_Base64Enc[ bin[2] & 0x3f ];
414 bin += 3;
415 }
416 base64[n] = 0;
417 return n;
418}
419
420/***********************************************************************
421 * HTTP_EncodeBasicAuth
422 *
423 * Encode the basic authentication string for HTTP 1.1
424 */
425static LPSTR HTTP_EncodeBasicAuth( LPCSTR username, LPCSTR password)
426{
427 UINT len;
428 LPSTR in, out, szBasic = "Basic ";
429
430 len = strlen( username ) + 1 + strlen ( password ) + 1;
431 in = HeapAlloc( GetProcessHeap(), 0, len );
432 if( !in )
433 return NULL;
434
435 len = strlen(szBasic) +
436 (strlen( username ) + 1 + strlen ( password ))*2 + 1 + 1;
437 out = HeapAlloc( GetProcessHeap(), 0, len );
438 if( out )
439 {
440 strcpy( in, username );
441 strcat( in, ":" );
442 strcat( in, password );
443 strcpy( out, szBasic );
444 HTTP_Base64( in, &out[strlen(out)] );
445 }
446 HeapFree( GetProcessHeap(), 0, in );
447
448 return out;
449}
450
451/***********************************************************************
452 * HTTP_InsertProxyAuthorization
453 *
454 * Insert the basic authorization field in the request header
455 */
456BOOL HTTP_InsertProxyAuthorization( LPWININETHTTPREQA lpwhr,
457 LPCSTR username, LPCSTR password )
458{
459 HTTPHEADERA hdr;
460 INT index;
461
462 hdr.lpszField = "Proxy-Authorization";
463 hdr.lpszValue = HTTP_EncodeBasicAuth( username, password );
464 hdr.wFlags = HDR_ISREQUEST;
465 hdr.wCount = 0;
466 if( !hdr.lpszValue )
467 return FALSE;
468
469 TRACE("Inserting %s = %s\n",
470 debugstr_a( hdr.lpszField ), debugstr_a( hdr.lpszValue ) );
471
472 /* remove the old proxy authorization header */
473 index = HTTP_GetCustomHeaderIndex( lpwhr, hdr.lpszField );
474 if( index >=0 )
475 HTTP_DeleteCustomHeader( lpwhr, index );
476
477 HTTP_InsertCustomHeader(lpwhr, &hdr);
478 HeapFree( GetProcessHeap(), 0, hdr.lpszValue );
479
480 return TRUE;
481}
482
483/***********************************************************************
484 * HTTP_DealWithProxy
485 */
486static BOOL HTTP_DealWithProxy( LPWININETAPPINFOA hIC,
487 LPWININETHTTPSESSIONA lpwhs, LPWININETHTTPREQA lpwhr)
488{
489 char buf[MAXHOSTNAME];
490 char proxy[MAXHOSTNAME + 15]; /* 15 == "http://" + sizeof(port#) + ":/\0" */
491 char* url, *szNul = "";
492 URL_COMPONENTSA UrlComponents;
493
494 memset( &UrlComponents, 0, sizeof UrlComponents );
495 UrlComponents.dwStructSize = sizeof UrlComponents;
496 UrlComponents.lpszHostName = buf;
497 UrlComponents.dwHostNameLength = MAXHOSTNAME;
498
499 sprintf(proxy, "http://%s/", hIC->lpszProxy);
500 if( !InternetCrackUrlA(proxy, 0, 0, &UrlComponents) )
501 return FALSE;
502 if( UrlComponents.dwHostNameLength == 0 )
503 return FALSE;
504
505 if( !lpwhr->lpszPath )
506 lpwhr->lpszPath = szNul;
507 TRACE("server='%s' path='%s'\n",
508 lpwhs->lpszServerName, lpwhr->lpszPath);
509 /* for constant 15 see above */
510 url = HeapAlloc(GetProcessHeap(), 0,
511 strlen(lpwhs->lpszServerName) + strlen(lpwhr->lpszPath) + 15);
512
513 if(UrlComponents.nPort == INTERNET_INVALID_PORT_NUMBER)
514 UrlComponents.nPort = INTERNET_DEFAULT_HTTP_PORT;
515
516 sprintf(url, "http://%s:%d", lpwhs->lpszServerName,
517 lpwhs->nServerPort);
518 if( lpwhr->lpszPath[0] != '/' )
519 strcat( url, "/" );
520 strcat(url, lpwhr->lpszPath);
521 if(lpwhr->lpszPath != szNul)
522 HeapFree(GetProcessHeap(), 0, lpwhr->lpszPath);
523 lpwhr->lpszPath = url;
524 /* FIXME: Do I have to free lpwhs->lpszServerName here ? */
525 lpwhs->lpszServerName = HTTP_strdup(UrlComponents.lpszHostName);
526 lpwhs->nServerPort = UrlComponents.nPort;
527
528 return TRUE;
529}
530
531/***********************************************************************
Ulrich Czekallac2757242000-06-11 20:04:44 +0000532 * HTTP_HttpOpenRequestA (internal)
533 *
534 * Open a HTTP request handle
535 *
536 * RETURNS
537 * HINTERNET a HTTP request handle on success
538 * NULL on failure
539 *
540 */
Patrik Stridvall3c0211f2001-09-11 00:32:32 +0000541HINTERNET WINAPI HTTP_HttpOpenRequestA(HINTERNET hHttpSession,
Ulrich Czekallac2757242000-06-11 20:04:44 +0000542 LPCSTR lpszVerb, LPCSTR lpszObjectName, LPCSTR lpszVersion,
Vincent Béron9a624912002-05-31 23:06:46 +0000543 LPCSTR lpszReferrer , LPCSTR *lpszAcceptTypes,
Ulrich Czekallac2757242000-06-11 20:04:44 +0000544 DWORD dwFlags, DWORD dwContext)
545{
546 LPWININETHTTPSESSIONA lpwhs = (LPWININETHTTPSESSIONA) hHttpSession;
547 LPWININETAPPINFOA hIC = NULL;
548 LPWININETHTTPREQA lpwhr;
David Hammerton852c7ae2003-06-20 23:26:56 +0000549 LPSTR lpszCookies;
550 LPSTR lpszUrl = NULL;
551 DWORD nCookieSize;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000552
Aric Stewartff9b9d42002-06-21 23:59:49 +0000553 TRACE("--> \n");
Ulrich Czekallac2757242000-06-11 20:04:44 +0000554
555 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
556 {
557 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
558 return FALSE;
559 }
560
561 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
562
563 lpwhr = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WININETHTTPREQA));
564 if (NULL == lpwhr)
565 {
566 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
567 return (HINTERNET) NULL;
568 }
569
570 lpwhr->hdr.htype = WH_HHTTPREQ;
571 lpwhr->hdr.lpwhparent = hHttpSession;
572 lpwhr->hdr.dwFlags = dwFlags;
573 lpwhr->hdr.dwContext = dwContext;
David Hammerton852c7ae2003-06-20 23:26:56 +0000574 NETCON_init(&lpwhr->netConnection, dwFlags & INTERNET_FLAG_SECURE);
Ulrich Czekallac2757242000-06-11 20:04:44 +0000575
Huw D M Davies0aebee92001-01-21 21:09:00 +0000576 if (NULL != lpszObjectName && strlen(lpszObjectName)) {
577 DWORD needed = 0;
Aric Stewartff9b9d42002-06-21 23:59:49 +0000578 HRESULT rc;
579 rc = UrlEscapeA(lpszObjectName, NULL, &needed, URL_ESCAPE_SPACES_ONLY);
580 if (rc != E_POINTER)
581 needed = strlen(lpszObjectName)+1;
Huw D M Davies0aebee92001-01-21 21:09:00 +0000582 lpwhr->lpszPath = HeapAlloc(GetProcessHeap(), 0, needed);
Aric Stewartff9b9d42002-06-21 23:59:49 +0000583 rc = UrlEscapeA(lpszObjectName, lpwhr->lpszPath, &needed,
Huw D M Davies0aebee92001-01-21 21:09:00 +0000584 URL_ESCAPE_SPACES_ONLY);
Aric Stewartff9b9d42002-06-21 23:59:49 +0000585 if (rc)
586 {
587 ERR("Unable to escape string!(%s) (%ld)\n",lpszObjectName,rc);
588 strcpy(lpwhr->lpszPath,lpszObjectName);
589 }
Huw D M Davies0aebee92001-01-21 21:09:00 +0000590 }
Ulrich Czekallac2757242000-06-11 20:04:44 +0000591
Alberto Massaribc8bd722002-12-06 23:20:31 +0000592 if (NULL != lpszReferrer && strlen(lpszReferrer))
David Hammerton852c7ae2003-06-20 23:26:56 +0000593 HTTP_ProcessHeader(lpwhr, HTTP_REFERER, lpszReferrer, HTTP_ADDHDR_FLAG_COALESCE);
Alberto Massaribc8bd722002-12-06 23:20:31 +0000594
595 if(lpszAcceptTypes!=NULL)
596 {
597 int i;
598 for(i=0;lpszAcceptTypes[i]!=NULL;i++)
599 HTTP_ProcessHeader(lpwhr, HTTP_ACCEPT, lpszAcceptTypes[i], HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA|HTTP_ADDHDR_FLAG_REQ|HTTP_ADDHDR_FLAG_ADD_IF_NEW);
600 }
Ulrich Czekallac2757242000-06-11 20:04:44 +0000601
602 if (NULL == lpszVerb)
Alexandre Julliard0e44f632000-11-16 00:28:52 +0000603 lpwhr->lpszVerb = HTTP_strdup("GET");
Ulrich Czekallac2757242000-06-11 20:04:44 +0000604 else if (strlen(lpszVerb))
Alexandre Julliard0e44f632000-11-16 00:28:52 +0000605 lpwhr->lpszVerb = HTTP_strdup(lpszVerb);
Ulrich Czekallac2757242000-06-11 20:04:44 +0000606
David Hammerton852c7ae2003-06-20 23:26:56 +0000607 if (NULL != lpszReferrer && strlen(lpszReferrer))
Ulrich Czekallac2757242000-06-11 20:04:44 +0000608 {
609 char buf[MAXHOSTNAME];
610 URL_COMPONENTSA UrlComponents;
611
Mike McCormacka1c16d22003-07-22 03:17:52 +0000612 memset( &UrlComponents, 0, sizeof UrlComponents );
613 UrlComponents.dwStructSize = sizeof UrlComponents;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000614 UrlComponents.lpszHostName = buf;
615 UrlComponents.dwHostNameLength = MAXHOSTNAME;
616
617 InternetCrackUrlA(lpszReferrer, 0, 0, &UrlComponents);
618 if (strlen(UrlComponents.lpszHostName))
Alexandre Julliard0e44f632000-11-16 00:28:52 +0000619 lpwhr->lpszHostName = HTTP_strdup(UrlComponents.lpszHostName);
Andreas Mohr34965562000-08-26 20:31:48 +0000620 } else {
Alexandre Julliard0e44f632000-11-16 00:28:52 +0000621 lpwhr->lpszHostName = HTTP_strdup(lpwhs->lpszServerName);
Ulrich Czekallac2757242000-06-11 20:04:44 +0000622 }
Mike McCormacka1c16d22003-07-22 03:17:52 +0000623 if (NULL != hIC->lpszProxy && hIC->lpszProxy[0] != 0)
624 HTTP_DealWithProxy( hIC, lpwhs, lpwhr );
Ulrich Czekallac2757242000-06-11 20:04:44 +0000625
David Hammerton852c7ae2003-06-20 23:26:56 +0000626 if (hIC->lpszAgent)
627 {
628 char *agent_header = HeapAlloc(GetProcessHeap(), 0, strlen(hIC->lpszAgent) + 1 + 14);
629 sprintf(agent_header, "User-Agent: %s\r\n", hIC->lpszAgent);
630 HttpAddRequestHeadersA((HINTERNET)lpwhr, agent_header, strlen(agent_header),
631 HTTP_ADDREQ_FLAG_ADD);
632 HeapFree(GetProcessHeap(), 0, agent_header);
633 }
634
635 lpszUrl = HeapAlloc(GetProcessHeap(), 0, strlen(lpwhr->lpszHostName) + 1 + 7);
636 sprintf(lpszUrl, "http://%s", lpwhr->lpszHostName);
637 if (InternetGetCookieA(lpszUrl, NULL, NULL, &nCookieSize))
638 {
639 int cnt = 0;
640
641 lpszCookies = HeapAlloc(GetProcessHeap(), 0, nCookieSize + 1 + 8);
642
643 cnt += sprintf(lpszCookies, "Cookie: ");
644 InternetGetCookieA(lpszUrl, NULL, lpszCookies + cnt, &nCookieSize);
645 cnt += nCookieSize - 1;
646 sprintf(lpszCookies + cnt, "\r\n");
647
648 HttpAddRequestHeadersA((HINTERNET)lpwhr, lpszCookies, strlen(lpszCookies),
649 HTTP_ADDREQ_FLAG_ADD);
650 HeapFree(GetProcessHeap(), 0, lpszCookies);
651 }
652 HeapFree(GetProcessHeap(), 0, lpszUrl);
653
654
655
Ulrich Czekallac2757242000-06-11 20:04:44 +0000656 if (hIC->lpfnStatusCB)
657 {
658 INTERNET_ASYNC_RESULT iar;
659
660 iar.dwResult = (DWORD)lpwhr;
661 iar.dwError = ERROR_SUCCESS;
662
Aric Stewartff9b9d42002-06-21 23:59:49 +0000663 SendAsyncCallback(hIC, hHttpSession, dwContext,
664 INTERNET_STATUS_HANDLE_CREATED, &iar,
665 sizeof(INTERNET_ASYNC_RESULT));
Ulrich Czekallac2757242000-06-11 20:04:44 +0000666 }
667
Aric Stewartff9b9d42002-06-21 23:59:49 +0000668 /*
669 * A STATUS_REQUEST_COMPLETE is NOT sent here as per my tests on windows
670 */
671
672 /*
673 * According to my tests. The name is not resolved until a request is Opened
674 */
675 SendAsyncCallback(hIC, hHttpSession, dwContext,
676 INTERNET_STATUS_RESOLVING_NAME,
677 lpwhs->lpszServerName,
678 strlen(lpwhs->lpszServerName)+1);
Aric Stewartff9b9d42002-06-21 23:59:49 +0000679 if (!GetAddress(lpwhs->lpszServerName, lpwhs->nServerPort,
680 &lpwhs->phostent, &lpwhs->socketAddress))
Ulrich Czekallac2757242000-06-11 20:04:44 +0000681 {
Aric Stewartff9b9d42002-06-21 23:59:49 +0000682 INTERNET_SetLastError(ERROR_INTERNET_NAME_NOT_RESOLVED);
683 return FALSE;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000684 }
685
Aric Stewartff9b9d42002-06-21 23:59:49 +0000686 SendAsyncCallback(hIC, hHttpSession, lpwhr->hdr.dwContext,
687 INTERNET_STATUS_NAME_RESOLVED,
688 &(lpwhs->socketAddress),
689 sizeof(struct sockaddr_in));
690
691 TRACE("<--\n");
Ulrich Czekallac2757242000-06-11 20:04:44 +0000692 return (HINTERNET) lpwhr;
693}
694
695
696/***********************************************************************
Patrik Stridvall8b216b32001-06-19 18:20:47 +0000697 * HttpQueryInfoA (WININET.@)
Ulrich Czekallac2757242000-06-11 20:04:44 +0000698 *
699 * Queries for information about an HTTP request
700 *
701 * RETURNS
702 * TRUE on success
703 * FALSE on failure
704 *
705 */
706BOOL WINAPI HttpQueryInfoA(HINTERNET hHttpRequest, DWORD dwInfoLevel,
707 LPVOID lpBuffer, LPDWORD lpdwBufferLength, LPDWORD lpdwIndex)
708{
Vincent Béron9a624912002-05-31 23:06:46 +0000709 LPHTTPHEADERA lphttpHdr = NULL;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000710 BOOL bSuccess = FALSE;
711 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
Vincent Béron9a624912002-05-31 23:06:46 +0000712
Ulrich Czekallac2757242000-06-11 20:04:44 +0000713 TRACE("(0x%08lx)--> %ld\n", dwInfoLevel, dwInfoLevel);
714
715 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
716 {
717 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
718 return FALSE;
719 }
720
721 /* Find requested header structure */
722 if ((dwInfoLevel & ~HTTP_QUERY_MODIFIER_FLAGS_MASK) == HTTP_QUERY_CUSTOM)
723 {
724 INT index = HTTP_GetCustomHeaderIndex(lpwhr, (LPSTR)lpBuffer);
725
726 if (index < 0)
727 goto lend;
728
729 lphttpHdr = &lpwhr->pCustHeaders[index];
730 }
731 else
732 {
733 INT index = dwInfoLevel & ~HTTP_QUERY_MODIFIER_FLAGS_MASK;
734
735 if (index == HTTP_QUERY_RAW_HEADERS_CRLF || index == HTTP_QUERY_RAW_HEADERS)
736 {
737 INT i, delim, size = 0, cnt = 0;
738
739 delim = index == HTTP_QUERY_RAW_HEADERS_CRLF ? 2 : 1;
740
741 /* Calculate length of custom reuqest headers */
742 for (i = 0; i < lpwhr->nCustHeaders; i++)
743 {
744 if ((~lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST) && lpwhr->pCustHeaders[i].lpszField &&
745 lpwhr->pCustHeaders[i].lpszValue)
746 {
Vincent Béron9a624912002-05-31 23:06:46 +0000747 size += strlen(lpwhr->pCustHeaders[i].lpszField) +
Ulrich Czekallac2757242000-06-11 20:04:44 +0000748 strlen(lpwhr->pCustHeaders[i].lpszValue) + delim + 2;
749 }
750 }
751
752 /* Calculate the length of stadard request headers */
753 for (i = 0; i <= HTTP_QUERY_MAX; i++)
754 {
Vincent Béron9a624912002-05-31 23:06:46 +0000755 if ((~lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST) && lpwhr->StdHeaders[i].lpszField &&
Ulrich Czekallac2757242000-06-11 20:04:44 +0000756 lpwhr->StdHeaders[i].lpszValue)
757 {
Vincent Béron9a624912002-05-31 23:06:46 +0000758 size += strlen(lpwhr->StdHeaders[i].lpszField) +
Ulrich Czekallac2757242000-06-11 20:04:44 +0000759 strlen(lpwhr->StdHeaders[i].lpszValue) + delim + 2;
760 }
761 }
Ulrich Czekallac2757242000-06-11 20:04:44 +0000762 size += delim;
763
764 if (size + 1 > *lpdwBufferLength)
765 {
766 *lpdwBufferLength = size + 1;
767 INTERNET_SetLastError(ERROR_INSUFFICIENT_BUFFER);
768 goto lend;
769 }
770
771 /* Append standard request heades */
772 for (i = 0; i <= HTTP_QUERY_MAX; i++)
773 {
774 if ((~lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST) &&
775 lpwhr->StdHeaders[i].lpszField &&
776 lpwhr->StdHeaders[i].lpszValue)
777 {
Gregg Mattinson7c4cb512002-07-03 21:10:43 +0000778 cnt += sprintf((char*)lpBuffer + cnt, "%s: %s%s", lpwhr->StdHeaders[i].lpszField, lpwhr->StdHeaders[i].lpszValue,
Ulrich Czekallac2757242000-06-11 20:04:44 +0000779 index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "\0");
780 }
781 }
782
783 /* Append custom request heades */
784 for (i = 0; i < lpwhr->nCustHeaders; i++)
785 {
786 if ((~lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST) &&
787 lpwhr->pCustHeaders[i].lpszField &&
788 lpwhr->pCustHeaders[i].lpszValue)
789 {
Gregg Mattinson7c4cb512002-07-03 21:10:43 +0000790 cnt += sprintf((char*)lpBuffer + cnt, "%s: %s%s",
Ulrich Czekallac2757242000-06-11 20:04:44 +0000791 lpwhr->pCustHeaders[i].lpszField, lpwhr->pCustHeaders[i].lpszValue,
792 index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "\0");
793 }
794 }
795
Gregg Mattinson7c4cb512002-07-03 21:10:43 +0000796 strcpy((char*)lpBuffer + cnt, index == HTTP_QUERY_RAW_HEADERS_CRLF ? "\r\n" : "");
Ulrich Czekallac2757242000-06-11 20:04:44 +0000797
798 *lpdwBufferLength = cnt + delim;
799 bSuccess = TRUE;
800 goto lend;
801 }
802 else if (index >= 0 && index <= HTTP_QUERY_MAX && lpwhr->StdHeaders[index].lpszValue)
Vincent Béron9a624912002-05-31 23:06:46 +0000803 {
Ulrich Czekallac2757242000-06-11 20:04:44 +0000804 lphttpHdr = &lpwhr->StdHeaders[index];
805 }
806 else
807 goto lend;
808 }
809
810 /* Ensure header satisifies requested attributes */
Vincent Béron9a624912002-05-31 23:06:46 +0000811 if ((dwInfoLevel & HTTP_QUERY_FLAG_REQUEST_HEADERS) &&
Ulrich Czekallac2757242000-06-11 20:04:44 +0000812 (~lphttpHdr->wFlags & HDR_ISREQUEST))
813 goto lend;
Vincent Béron9a624912002-05-31 23:06:46 +0000814
Ulrich Czekallac2757242000-06-11 20:04:44 +0000815 /* coalesce value to reuqested type */
816 if (dwInfoLevel & HTTP_QUERY_FLAG_NUMBER)
817 {
818 *(int *)lpBuffer = atoi(lphttpHdr->lpszValue);
819 bSuccess = TRUE;
820 }
821 else if (dwInfoLevel & HTTP_QUERY_FLAG_SYSTEMTIME)
822 {
823 time_t tmpTime;
824 struct tm tmpTM;
825 SYSTEMTIME *STHook;
Vincent Béron9a624912002-05-31 23:06:46 +0000826
Ulrich Czekallac2757242000-06-11 20:04:44 +0000827 tmpTime = ConvertTimeString(lphttpHdr->lpszValue);
828
829 tmpTM = *gmtime(&tmpTime);
830 STHook = (SYSTEMTIME *) lpBuffer;
831 if(STHook==NULL)
832 goto lend;
833
834 STHook->wDay = tmpTM.tm_mday;
835 STHook->wHour = tmpTM.tm_hour;
836 STHook->wMilliseconds = 0;
837 STHook->wMinute = tmpTM.tm_min;
838 STHook->wDayOfWeek = tmpTM.tm_wday;
839 STHook->wMonth = tmpTM.tm_mon + 1;
840 STHook->wSecond = tmpTM.tm_sec;
841 STHook->wYear = tmpTM.tm_year;
842
843 bSuccess = TRUE;
844 }
845 else if (dwInfoLevel & HTTP_QUERY_FLAG_COALESCE)
846 {
847 if (*lpdwIndex >= lphttpHdr->wCount)
848 {
849 INTERNET_SetLastError(ERROR_HTTP_HEADER_NOT_FOUND);
850 }
851 else
852 {
Patrik Stridvallfc2be7e2002-04-29 18:48:56 +0000853 /* Copy strncpy(lpBuffer, lphttpHdr[*lpdwIndex], len); */
Ulrich Czekallac2757242000-06-11 20:04:44 +0000854 (*lpdwIndex)++;
855 }
856 }
857 else
858 {
859 INT len = strlen(lphttpHdr->lpszValue);
860
861 if (len + 1 > *lpdwBufferLength)
862 {
863 *lpdwBufferLength = len + 1;
864 INTERNET_SetLastError(ERROR_INSUFFICIENT_BUFFER);
865 goto lend;
866 }
867
868 strncpy(lpBuffer, lphttpHdr->lpszValue, len);
Aric Stewartff9b9d42002-06-21 23:59:49 +0000869 ((char*)lpBuffer)[len]=0;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000870 *lpdwBufferLength = len;
Vincent Béron9a624912002-05-31 23:06:46 +0000871 bSuccess = TRUE;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000872 }
873
874lend:
875 TRACE("%d <--\n", bSuccess);
876 return bSuccess;
877}
878
Alberto Massarid476a5a2002-11-12 02:13:04 +0000879/***********************************************************************
880 * HttpQueryInfoW (WININET.@)
881 *
882 * Queries for information about an HTTP request
883 *
884 * RETURNS
885 * TRUE on success
886 * FALSE on failure
887 *
888 */
889BOOL WINAPI HttpQueryInfoW(HINTERNET hHttpRequest, DWORD dwInfoLevel,
890 LPVOID lpBuffer, LPDWORD lpdwBufferLength, LPDWORD lpdwIndex)
891{
892 BOOL result;
893 DWORD charLen=*lpdwBufferLength;
894 char* tempBuffer=HeapAlloc(GetProcessHeap(), 0, charLen);
895 result=HttpQueryInfoA(hHttpRequest, dwInfoLevel, tempBuffer, &charLen, lpdwIndex);
896 if((dwInfoLevel & HTTP_QUERY_FLAG_NUMBER) ||
897 (dwInfoLevel & HTTP_QUERY_FLAG_SYSTEMTIME))
898 {
899 memcpy(lpBuffer,tempBuffer,charLen);
900 }
901 else
902 {
903 int nChars=MultiByteToWideChar(CP_ACP,0, tempBuffer,charLen,lpBuffer,*lpdwBufferLength);
904 *lpdwBufferLength=nChars;
905 }
906 HeapFree(GetProcessHeap(), 0, tempBuffer);
907 return result;
908}
Huw D M Daviesf9b6d7b2000-10-28 00:30:23 +0000909
910/***********************************************************************
Patrik Stridvall3ca98232001-06-20 23:03:14 +0000911 * HttpSendRequestExA (WININET.@)
Huw D M Daviesf9b6d7b2000-10-28 00:30:23 +0000912 *
913 * Sends the specified request to the HTTP server and allows chunked
914 * transfers
915 */
916BOOL WINAPI HttpSendRequestExA(HINTERNET hRequest,
917 LPINTERNET_BUFFERSA lpBuffersIn,
918 LPINTERNET_BUFFERSA lpBuffersOut,
919 DWORD dwFlags, DWORD dwContext)
920{
921 FIXME("(%p, %p, %p, %08lx, %08lx): stub\n", hRequest, lpBuffersIn,
922 lpBuffersOut, dwFlags, dwContext);
923 return FALSE;
924}
925
Ulrich Czekallac2757242000-06-11 20:04:44 +0000926/***********************************************************************
Patrik Stridvall8b216b32001-06-19 18:20:47 +0000927 * HttpSendRequestA (WININET.@)
Ulrich Czekallac2757242000-06-11 20:04:44 +0000928 *
929 * Sends the specified request to the HTTP server
930 *
931 * RETURNS
932 * TRUE on success
933 * FALSE on failure
934 *
935 */
936BOOL WINAPI HttpSendRequestA(HINTERNET hHttpRequest, LPCSTR lpszHeaders,
937 DWORD dwHeaderLength, LPVOID lpOptional ,DWORD dwOptionalLength)
Vincent Béron9a624912002-05-31 23:06:46 +0000938{
Ulrich Czekallac2757242000-06-11 20:04:44 +0000939 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
940 LPWININETHTTPSESSIONA lpwhs = NULL;
941 LPWININETAPPINFOA hIC = NULL;
942
David Hammerton6226f3f2003-08-05 18:31:02 +0000943 TRACE("(0x%08lx, %p (%s), %li, %p, %li)\n", (unsigned long)hHttpRequest,
944 lpszHeaders, debugstr_a(lpszHeaders), dwHeaderLength, lpOptional, dwOptionalLength);
Ulrich Czekallac2757242000-06-11 20:04:44 +0000945
946 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
947 {
948 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
949 return FALSE;
950 }
951
952 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
953 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
954 {
955 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
956 return FALSE;
957 }
958
959 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
960 if (NULL == hIC || hIC->hdr.htype != WH_HINIT)
961 {
962 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
963 return FALSE;
964 }
965
966 if (hIC->hdr.dwFlags & INTERNET_FLAG_ASYNC)
967 {
968 WORKREQUEST workRequest;
969
Aric Stewartff9b9d42002-06-21 23:59:49 +0000970 workRequest.asyncall = HTTPSENDREQUESTA;
971 workRequest.HFTPSESSION = (DWORD)hHttpRequest;
972 if (lpszHeaders)
973 workRequest.LPSZHEADER = (DWORD)HTTP_strdup(lpszHeaders);
974 else
975 workRequest.LPSZHEADER = 0;
976 workRequest.DWHEADERLENGTH = dwHeaderLength;
977 workRequest.LPOPTIONAL = (DWORD)lpOptional;
978 workRequest.DWOPTIONALLENGTH = dwOptionalLength;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000979
Aric Stewartff9b9d42002-06-21 23:59:49 +0000980 INTERNET_AsyncCall(&workRequest);
981 /*
Alberto Massaribc8bd722002-12-06 23:20:31 +0000982 * This is from windows.
Aric Stewartff9b9d42002-06-21 23:59:49 +0000983 */
Alberto Massaribc8bd722002-12-06 23:20:31 +0000984 SetLastError(ERROR_IO_PENDING);
Aric Stewartff9b9d42002-06-21 23:59:49 +0000985 return 0;
Ulrich Czekallac2757242000-06-11 20:04:44 +0000986 }
987 else
988 {
Vincent Béron9a624912002-05-31 23:06:46 +0000989 return HTTP_HttpSendRequestA(hHttpRequest, lpszHeaders,
Ulrich Czekallac2757242000-06-11 20:04:44 +0000990 dwHeaderLength, lpOptional, dwOptionalLength);
991 }
992}
993
Alberto Massarid476a5a2002-11-12 02:13:04 +0000994/***********************************************************************
995 * HttpSendRequestW (WININET.@)
996 *
997 * Sends the specified request to the HTTP server
998 *
999 * RETURNS
1000 * TRUE on success
1001 * FALSE on failure
1002 *
1003 */
1004BOOL WINAPI HttpSendRequestW(HINTERNET hHttpRequest, LPCWSTR lpszHeaders,
1005 DWORD dwHeaderLength, LPVOID lpOptional ,DWORD dwOptionalLength)
1006{
1007 BOOL result;
1008 char* szHeaders=NULL;
1009 DWORD nLen=dwHeaderLength;
1010 if(lpszHeaders!=NULL)
1011 {
1012 if(nLen==-1)
1013 nLen=strlenW(lpszHeaders);
1014 szHeaders=(char*)malloc(nLen+1);
1015 WideCharToMultiByte(CP_ACP,0,lpszHeaders,nLen,szHeaders,nLen,NULL,NULL);
1016 }
1017 result=HttpSendRequestA(hHttpRequest, szHeaders, dwHeaderLength, lpOptional, dwOptionalLength);
1018 if(szHeaders!=NULL)
1019 free(szHeaders);
1020 return result;
1021}
Ulrich Czekallac2757242000-06-11 20:04:44 +00001022
1023/***********************************************************************
Alberto Massaribc8bd722002-12-06 23:20:31 +00001024 * HTTP_HandleRedirect (internal)
1025 */
1026static BOOL HTTP_HandleRedirect(LPWININETHTTPREQA lpwhr, LPCSTR lpszUrl, LPCSTR lpszHeaders,
1027 DWORD dwHeaderLength, LPVOID lpOptional, DWORD dwOptionalLength)
1028{
1029 LPWININETHTTPSESSIONA lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
1030 LPWININETAPPINFOA hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
1031 char path[2048];
1032 if(lpszUrl[0]=='/')
1033 {
1034 /* if it's an absolute path, keep the same session info */
1035 strcpy(path,lpszUrl);
1036 }
Mike McCormacka1c16d22003-07-22 03:17:52 +00001037 else if (NULL != hIC->lpszProxy && hIC->lpszProxy[0] != 0)
1038 {
1039 TRACE("Redirect through proxy\n");
1040 strcpy(path,lpszUrl);
1041 }
Alberto Massaribc8bd722002-12-06 23:20:31 +00001042 else
1043 {
1044 URL_COMPONENTSA urlComponents;
1045 char protocol[32], hostName[MAXHOSTNAME], userName[1024];
1046 char password[1024], extra[1024];
1047 urlComponents.dwStructSize = sizeof(URL_COMPONENTSA);
1048 urlComponents.lpszScheme = protocol;
1049 urlComponents.dwSchemeLength = 32;
1050 urlComponents.lpszHostName = hostName;
1051 urlComponents.dwHostNameLength = MAXHOSTNAME;
1052 urlComponents.lpszUserName = userName;
1053 urlComponents.dwUserNameLength = 1024;
1054 urlComponents.lpszPassword = password;
1055 urlComponents.dwPasswordLength = 1024;
1056 urlComponents.lpszUrlPath = path;
1057 urlComponents.dwUrlPathLength = 2048;
1058 urlComponents.lpszExtraInfo = extra;
1059 urlComponents.dwExtraInfoLength = 1024;
1060 if(!InternetCrackUrlA(lpszUrl, strlen(lpszUrl), 0, &urlComponents))
1061 return FALSE;
1062
1063 if (urlComponents.nPort == INTERNET_INVALID_PORT_NUMBER)
1064 urlComponents.nPort = INTERNET_DEFAULT_HTTP_PORT;
1065
Mike McCormacka1c16d22003-07-22 03:17:52 +00001066#if 0
1067 /*
1068 * This upsets redirects to binary files on sourceforge.net
1069 * and gives an html page instead of the target file
1070 * Examination of the HTTP request sent by native wininet.dll
1071 * reveals that it doesn't send a referrer in that case.
1072 * Maybe there's a flag that enables this, or maybe a referrer
1073 * shouldn't be added in case of a redirect.
1074 */
1075
1076 /* consider the current host as the referrer */
Alberto Massaribc8bd722002-12-06 23:20:31 +00001077 if (NULL != lpwhs->lpszServerName && strlen(lpwhs->lpszServerName))
Mike McCormacka1c16d22003-07-22 03:17:52 +00001078 HTTP_ProcessHeader(lpwhr, HTTP_REFERER, lpwhs->lpszServerName,
1079 HTTP_ADDHDR_FLAG_REQ|HTTP_ADDREQ_FLAG_REPLACE|
1080 HTTP_ADDHDR_FLAG_ADD_IF_NEW);
1081#endif
Alberto Massaribc8bd722002-12-06 23:20:31 +00001082
1083 if (NULL != lpwhs->lpszServerName)
1084 HeapFree(GetProcessHeap(), 0, lpwhs->lpszServerName);
1085 lpwhs->lpszServerName = HTTP_strdup(hostName);
1086 if (NULL != lpwhs->lpszUserName)
1087 HeapFree(GetProcessHeap(), 0, lpwhs->lpszUserName);
1088 lpwhs->lpszUserName = HTTP_strdup(userName);
1089 lpwhs->nServerPort = urlComponents.nPort;
1090
1091 if (NULL != lpwhr->lpszHostName)
1092 HeapFree(GetProcessHeap(), 0, lpwhr->lpszHostName);
1093 lpwhr->lpszHostName=HTTP_strdup(hostName);
1094
1095 SendAsyncCallback(hIC, lpwhs, lpwhr->hdr.dwContext,
1096 INTERNET_STATUS_RESOLVING_NAME,
1097 lpwhs->lpszServerName,
1098 strlen(lpwhs->lpszServerName)+1);
1099
1100 if (!GetAddress(lpwhs->lpszServerName, lpwhs->nServerPort,
1101 &lpwhs->phostent, &lpwhs->socketAddress))
1102 {
1103 INTERNET_SetLastError(ERROR_INTERNET_NAME_NOT_RESOLVED);
1104 return FALSE;
1105 }
1106
1107 SendAsyncCallback(hIC, lpwhs, lpwhr->hdr.dwContext,
1108 INTERNET_STATUS_NAME_RESOLVED,
1109 &(lpwhs->socketAddress),
1110 sizeof(struct sockaddr_in));
1111
1112 }
1113
1114 if(lpwhr->lpszPath)
1115 HeapFree(GetProcessHeap(), 0, lpwhr->lpszPath);
1116 lpwhr->lpszPath=NULL;
1117 if (strlen(path))
1118 {
1119 DWORD needed = 0;
1120 HRESULT rc;
1121 rc = UrlEscapeA(path, NULL, &needed, URL_ESCAPE_SPACES_ONLY);
1122 if (rc != E_POINTER)
1123 needed = strlen(path)+1;
1124 lpwhr->lpszPath = HeapAlloc(GetProcessHeap(), 0, needed);
1125 rc = UrlEscapeA(path, lpwhr->lpszPath, &needed,
1126 URL_ESCAPE_SPACES_ONLY);
1127 if (rc)
1128 {
1129 ERR("Unable to escape string!(%s) (%ld)\n",path,rc);
1130 strcpy(lpwhr->lpszPath,path);
1131 }
1132 }
1133
1134 return HttpSendRequestA((HINTERNET)lpwhr, lpszHeaders, dwHeaderLength, lpOptional, dwOptionalLength);
1135}
1136
1137/***********************************************************************
Ulrich Czekallac2757242000-06-11 20:04:44 +00001138 * HTTP_HttpSendRequestA (internal)
1139 *
1140 * Sends the specified request to the HTTP server
1141 *
1142 * RETURNS
1143 * TRUE on success
1144 * FALSE on failure
1145 *
1146 */
1147BOOL WINAPI HTTP_HttpSendRequestA(HINTERNET hHttpRequest, LPCSTR lpszHeaders,
1148 DWORD dwHeaderLength, LPVOID lpOptional ,DWORD dwOptionalLength)
1149{
1150 INT cnt;
1151 INT i;
1152 BOOL bSuccess = FALSE;
1153 LPSTR requestString = NULL;
1154 INT requestStringLen;
Aric Stewartff9b9d42002-06-21 23:59:49 +00001155 INT responseLen;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001156 INT headerLength = 0;
1157 LPWININETHTTPREQA lpwhr = (LPWININETHTTPREQA) hHttpRequest;
1158 LPWININETHTTPSESSIONA lpwhs = NULL;
1159 LPWININETAPPINFOA hIC = NULL;
David Hammerton852c7ae2003-06-20 23:26:56 +00001160 BOOL loop_next = FALSE;
1161 int CustHeaderIndex;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001162
Aric Stewartff9b9d42002-06-21 23:59:49 +00001163 TRACE("--> 0x%08lx\n", (ULONG)hHttpRequest);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001164
1165 /* Verify our tree of internet handles */
1166 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
1167 {
1168 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
1169 return FALSE;
1170 }
1171
1172 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
1173 if (NULL == lpwhs || lpwhs->hdr.htype != WH_HHTTPSESSION)
1174 {
1175 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
1176 return FALSE;
1177 }
1178
1179 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
1180 if (NULL == hIC || hIC->hdr.htype != WH_HINIT)
1181 {
1182 INTERNET_SetLastError(ERROR_INTERNET_INCORRECT_HANDLE_TYPE);
1183 return FALSE;
1184 }
1185
1186 /* Clear any error information */
1187 INTERNET_SetLastError(0);
1188
Aric Stewartff9b9d42002-06-21 23:59:49 +00001189
Ulrich Czekallac2757242000-06-11 20:04:44 +00001190 /* We must have a verb */
1191 if (NULL == lpwhr->lpszVerb)
1192 {
1193 goto lend;
1194 }
1195
David Hammerton6226f3f2003-08-05 18:31:02 +00001196 /* if we are using optional stuff, we must add the fixed header of that option length */
1197 if (lpOptional && dwOptionalLength)
1198 {
1199 char contentLengthStr[sizeof("Content-Length: ") + 20 /* int */ + 2 /* \n\r */];
1200 sprintf(contentLengthStr, "Content-Length: %li\r\n", dwOptionalLength);
1201 HttpAddRequestHeadersA(hHttpRequest, contentLengthStr, -1L, HTTP_ADDREQ_FLAG_ADD);
1202 }
1203
David Hammerton852c7ae2003-06-20 23:26:56 +00001204 do
Nikolas Zimmermann76598822001-10-04 18:12:41 +00001205 {
David Hammerton852c7ae2003-06-20 23:26:56 +00001206 TRACE("Going to url %s %s\n", debugstr_a(lpwhr->lpszHostName), debugstr_a(lpwhr->lpszPath));
1207 loop_next = FALSE;
Nikolas Zimmermann76598822001-10-04 18:12:41 +00001208
David Hammerton852c7ae2003-06-20 23:26:56 +00001209 /* If we don't have a path we set it to root */
1210 if (NULL == lpwhr->lpszPath)
1211 lpwhr->lpszPath = HTTP_strdup("/");
Ulrich Czekallac2757242000-06-11 20:04:44 +00001212
David Hammerton852c7ae2003-06-20 23:26:56 +00001213 if(strncmp(lpwhr->lpszPath, "http://", sizeof("http://") -1) != 0
1214 && lpwhr->lpszPath[0] != '/') /* not an absolute path ?? --> fix it !! */
1215 {
1216 char *fixurl = HeapAlloc(GetProcessHeap(), 0, strlen(lpwhr->lpszPath) + 2);
1217 *fixurl = '/';
1218 strcpy(fixurl + 1, lpwhr->lpszPath);
1219 HeapFree( GetProcessHeap(), 0, lpwhr->lpszPath );
1220 lpwhr->lpszPath = fixurl;
1221 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001222
David Hammerton852c7ae2003-06-20 23:26:56 +00001223 /* Calculate length of request string */
1224 requestStringLen =
1225 strlen(lpwhr->lpszVerb) +
1226 strlen(lpwhr->lpszPath) +
1227 strlen(HTTPHEADER) +
1228 5; /* " \r\n\r\n" */
1229
1230 /* Add length of passed headers */
1231 if (lpszHeaders)
1232 {
1233 headerLength = -1 == dwHeaderLength ? strlen(lpszHeaders) : dwHeaderLength;
1234 requestStringLen += headerLength + 2; /* \r\n */
1235 }
1236
Mike McCormacka1c16d22003-07-22 03:17:52 +00001237
1238 /* if there isa proxy username and password, add it to the headers */
1239 if( hIC && (hIC->lpszProxyUsername || hIC->lpszProxyPassword ) )
1240 {
1241 HTTP_InsertProxyAuthorization( lpwhr, hIC->lpszProxyUsername, hIC->lpszProxyPassword );
1242 }
1243
David Hammerton852c7ae2003-06-20 23:26:56 +00001244 /* Calculate length of custom request headers */
1245 for (i = 0; i < lpwhr->nCustHeaders; i++)
1246 {
Ulrich Czekallac2757242000-06-11 20:04:44 +00001247 if (lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST)
1248 {
David Hammerton852c7ae2003-06-20 23:26:56 +00001249 requestStringLen += strlen(lpwhr->pCustHeaders[i].lpszField) +
1250 strlen(lpwhr->pCustHeaders[i].lpszValue) + 4; /*: \r\n */
Ulrich Czekallac2757242000-06-11 20:04:44 +00001251 }
David Hammerton852c7ae2003-06-20 23:26:56 +00001252 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001253
David Hammerton852c7ae2003-06-20 23:26:56 +00001254 /* Calculate the length of standard request headers */
1255 for (i = 0; i <= HTTP_QUERY_MAX; i++)
1256 {
1257 if (lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST)
1258 {
1259 requestStringLen += strlen(lpwhr->StdHeaders[i].lpszField) +
1260 strlen(lpwhr->StdHeaders[i].lpszValue) + 4; /*: \r\n */
1261 }
1262 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001263
David Hammerton852c7ae2003-06-20 23:26:56 +00001264 if (lpwhr->lpszHostName)
1265 requestStringLen += (strlen(HTTPHOSTHEADER) + strlen(lpwhr->lpszHostName));
Ulrich Czekallac2757242000-06-11 20:04:44 +00001266
David Hammerton6226f3f2003-08-05 18:31:02 +00001267 /* if there is optional data to send, add the length */
1268 if (lpOptional)
1269 {
1270 requestStringLen += dwOptionalLength;
1271 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001272
David Hammerton852c7ae2003-06-20 23:26:56 +00001273 /* Allocate string to hold entire request */
1274 requestString = HeapAlloc(GetProcessHeap(), 0, requestStringLen + 1);
1275 if (NULL == requestString)
1276 {
1277 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1278 goto lend;
1279 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001280
David Hammerton852c7ae2003-06-20 23:26:56 +00001281 /* Build request string */
1282 cnt = sprintf(requestString, "%s %s%s",
1283 lpwhr->lpszVerb,
1284 lpwhr->lpszPath,
1285 HTTPHEADER);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001286
David Hammerton852c7ae2003-06-20 23:26:56 +00001287 /* Append standard request headers */
1288 for (i = 0; i <= HTTP_QUERY_MAX; i++)
1289 {
1290 if (lpwhr->StdHeaders[i].wFlags & HDR_ISREQUEST)
1291 {
1292 cnt += sprintf(requestString + cnt, "\r\n%s: %s",
1293 lpwhr->StdHeaders[i].lpszField, lpwhr->StdHeaders[i].lpszValue);
1294 TRACE("Adding header %s (%s)\n",lpwhr->StdHeaders[i].lpszField,lpwhr->StdHeaders[i].lpszValue);
1295 }
1296 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001297
David Hammerton852c7ae2003-06-20 23:26:56 +00001298 /* Append custom request heades */
1299 for (i = 0; i < lpwhr->nCustHeaders; i++)
1300 {
1301 if (lpwhr->pCustHeaders[i].wFlags & HDR_ISREQUEST)
1302 {
1303 cnt += sprintf(requestString + cnt, "\r\n%s: %s",
1304 lpwhr->pCustHeaders[i].lpszField, lpwhr->pCustHeaders[i].lpszValue);
1305 TRACE("Adding custom header %s (%s)\n",lpwhr->pCustHeaders[i].lpszField,lpwhr->pCustHeaders[i].lpszValue);
1306 }
1307 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001308
David Hammerton852c7ae2003-06-20 23:26:56 +00001309 if (lpwhr->lpszHostName)
1310 cnt += sprintf(requestString + cnt, "%s%s", HTTPHOSTHEADER, lpwhr->lpszHostName);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001311
David Hammerton852c7ae2003-06-20 23:26:56 +00001312 /* Append passed request headers */
1313 if (lpszHeaders)
1314 {
1315 strcpy(requestString + cnt, "\r\n");
1316 cnt += 2;
1317 strcpy(requestString + cnt, lpszHeaders);
1318 cnt += headerLength;
1319 }
Aric Stewartff9b9d42002-06-21 23:59:49 +00001320
David Hammerton6226f3f2003-08-05 18:31:02 +00001321 /* Set (header) termination string for request */
1322 if (memcmp((requestString + cnt) - 4, "\r\n\r\n", 4) != 0)
1323 { /* only add it if the request string doesn't already
1324 have the thing.. (could happen if the custom header
1325 added it */
1326 strcpy(requestString + cnt, "\r\n");
1327 cnt += 2;
1328 }
1329 else
1330 requestStringLen -= 2;
1331
1332 /* if optional data, append it */
1333 if (lpOptional)
1334 {
1335 memcpy(requestString + cnt, lpOptional, dwOptionalLength);
1336 cnt += dwOptionalLength;
1337 /* we also have to decrease the expected string length by two,
1338 * since we won't be adding on those following \r\n's */
1339 requestStringLen -= 2;
1340 }
1341 else
1342 { /* if there is no optional data, add on another \r\n just to be safe */
1343 /* termination for request */
1344 strcpy(requestString + cnt, "\r\n");
1345 cnt += 2;
1346 }
Aric Stewartff9b9d42002-06-21 23:59:49 +00001347
David Hammerton852c7ae2003-06-20 23:26:56 +00001348 TRACE("(%s) len(%d)\n", requestString, requestStringLen);
1349 /* Send the request and store the results */
1350 if (!HTTP_OpenConnection(lpwhr))
1351 goto lend;
Aric Stewartff9b9d42002-06-21 23:59:49 +00001352
David Hammerton852c7ae2003-06-20 23:26:56 +00001353 SendAsyncCallback(hIC, hHttpRequest, lpwhr->hdr.dwContext,
1354 INTERNET_STATUS_SENDING_REQUEST, NULL, 0);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001355
David Hammerton852c7ae2003-06-20 23:26:56 +00001356 NETCON_send(&lpwhr->netConnection, requestString, requestStringLen,
1357 0, &cnt);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001358
David Hammerton852c7ae2003-06-20 23:26:56 +00001359
1360 SendAsyncCallback(hIC, hHttpRequest, lpwhr->hdr.dwContext,
1361 INTERNET_STATUS_REQUEST_SENT,
1362 &requestStringLen,sizeof(DWORD));
1363
1364 SendAsyncCallback(hIC, hHttpRequest, lpwhr->hdr.dwContext,
1365 INTERNET_STATUS_RECEIVING_RESPONSE, NULL, 0);
1366
1367 if (cnt < 0)
1368 goto lend;
1369
1370 responseLen = HTTP_GetResponseHeaders(lpwhr);
1371 if (responseLen)
Ulrich Czekallac2757242000-06-11 20:04:44 +00001372 bSuccess = TRUE;
1373
David Hammerton852c7ae2003-06-20 23:26:56 +00001374 SendAsyncCallback(hIC, hHttpRequest, lpwhr->hdr.dwContext,
1375 INTERNET_STATUS_RESPONSE_RECEIVED, &responseLen,
1376 sizeof(DWORD));
1377
1378 /* process headers here. Is this right? */
1379 CustHeaderIndex = HTTP_GetCustomHeaderIndex(lpwhr, "Set-Cookie");
1380 if (CustHeaderIndex >= 0)
1381 {
1382 LPHTTPHEADERA setCookieHeader;
1383 int nPosStart = 0, nPosEnd = 0;
1384
1385 setCookieHeader = &lpwhr->pCustHeaders[CustHeaderIndex];
1386
1387 while (setCookieHeader->lpszValue[nPosEnd] != '\0')
1388 {
1389 LPSTR buf_cookie, cookie_name, cookie_data;
1390 LPSTR buf_url;
1391 LPSTR domain = NULL;
1392 int nEqualPos = 0;
1393 while (setCookieHeader->lpszValue[nPosEnd] != ';' && setCookieHeader->lpszValue[nPosEnd] != ',' &&
1394 setCookieHeader->lpszValue[nPosEnd] != '\0')
1395 {
1396 nPosEnd++;
1397 }
1398 if (setCookieHeader->lpszValue[nPosEnd] == ';')
1399 {
1400 /* fixme: not case sensitive, strcasestr is gnu only */
1401 int nDomainPosEnd = 0;
1402 int nDomainPosStart = 0, nDomainLength = 0;
1403 LPSTR lpszDomain = strstr(&setCookieHeader->lpszValue[nPosEnd], "domain=");
1404 if (lpszDomain)
1405 { /* they have specified their own domain, lets use it */
1406 while (lpszDomain[nDomainPosEnd] != ';' && lpszDomain[nDomainPosEnd] != ',' &&
1407 lpszDomain[nDomainPosEnd] != '\0')
1408 {
1409 nDomainPosEnd++;
1410 }
1411 nDomainPosStart = strlen("domain=");
1412 nDomainLength = (nDomainPosEnd - nDomainPosStart) + 1;
1413 domain = HeapAlloc(GetProcessHeap(), 0, nDomainLength + 1);
1414 strncpy(domain, &lpszDomain[nDomainPosStart], nDomainLength);
1415 domain[nDomainLength] = '\0';
1416 }
1417 }
1418 if (setCookieHeader->lpszValue[nPosEnd] == '\0') break;
1419 buf_cookie = HeapAlloc(GetProcessHeap(), 0, (nPosEnd - nPosStart) + 1);
1420 strncpy(buf_cookie, &setCookieHeader->lpszValue[nPosStart], (nPosEnd - nPosStart));
1421 buf_cookie[(nPosEnd - nPosStart)] = '\0';
1422 TRACE("%s\n", buf_cookie);
1423 while (buf_cookie[nEqualPos] != '=' && buf_cookie[nEqualPos] != '\0')
1424 {
1425 nEqualPos++;
1426 }
1427 if (buf_cookie[nEqualPos] == '\0' || buf_cookie[nEqualPos + 1] == '\0')
1428 {
1429 HeapFree(GetProcessHeap(), 0, buf_cookie);
1430 break;
1431 }
1432
1433 cookie_name = HeapAlloc(GetProcessHeap(), 0, nEqualPos + 1);
1434 strncpy(cookie_name, buf_cookie, nEqualPos);
1435 cookie_name[nEqualPos] = '\0';
1436 cookie_data = &buf_cookie[nEqualPos + 1];
1437
1438
1439 buf_url = HeapAlloc(GetProcessHeap(), 0, strlen((domain ? domain : lpwhr->lpszHostName)) + strlen(lpwhr->lpszPath) + 9);
1440 sprintf(buf_url, "http://%s/", (domain ? domain : lpwhr->lpszHostName)); /* FIXME PATH!!! */
1441 InternetSetCookieA(buf_url, cookie_name, cookie_data);
1442
1443 HeapFree(GetProcessHeap(), 0, buf_url);
1444 HeapFree(GetProcessHeap(), 0, buf_cookie);
1445 HeapFree(GetProcessHeap(), 0, cookie_name);
1446 if (domain) HeapFree(GetProcessHeap(), 0, domain);
1447 nPosStart = nPosEnd;
1448 }
1449 }
David Hammerton852c7ae2003-06-20 23:26:56 +00001450 }
1451 while (loop_next);
Aric Stewartff9b9d42002-06-21 23:59:49 +00001452
Ulrich Czekallac2757242000-06-11 20:04:44 +00001453lend:
1454
1455 if (requestString)
1456 HeapFree(GetProcessHeap(), 0, requestString);
1457
Alberto Massaribc8bd722002-12-06 23:20:31 +00001458 /* TODO: send notification for P3P header */
1459
1460 if(!(hIC->hdr.dwFlags & INTERNET_FLAG_NO_AUTO_REDIRECT) && bSuccess)
1461 {
1462 DWORD dwCode,dwCodeLength=sizeof(DWORD),dwIndex=0;
1463 if(HttpQueryInfoA(hHttpRequest,HTTP_QUERY_FLAG_NUMBER|HTTP_QUERY_STATUS_CODE,&dwCode,&dwCodeLength,&dwIndex) &&
1464 (dwCode==302 || dwCode==301))
1465 {
1466 char szNewLocation[2048];
1467 DWORD dwBufferSize=2048;
1468 dwIndex=0;
1469 if(HttpQueryInfoA(hHttpRequest,HTTP_QUERY_LOCATION,szNewLocation,&dwBufferSize,&dwIndex))
1470 {
1471 SendAsyncCallback(hIC, hHttpRequest, lpwhr->hdr.dwContext,
1472 INTERNET_STATUS_REDIRECT, szNewLocation,
1473 dwBufferSize);
1474 return HTTP_HandleRedirect(lpwhr, szNewLocation, lpszHeaders,
1475 dwHeaderLength, lpOptional, dwOptionalLength);
1476 }
1477 }
1478 }
Aric Stewartff9b9d42002-06-21 23:59:49 +00001479
1480 if (hIC->lpfnStatusCB)
Ulrich Czekallac2757242000-06-11 20:04:44 +00001481 {
1482 INTERNET_ASYNC_RESULT iar;
Vincent Béron9a624912002-05-31 23:06:46 +00001483
Ulrich Czekallac2757242000-06-11 20:04:44 +00001484 iar.dwResult = (DWORD)bSuccess;
1485 iar.dwError = bSuccess ? ERROR_SUCCESS : INTERNET_GetLastError();
Aric Stewartff9b9d42002-06-21 23:59:49 +00001486
1487 SendAsyncCallback(hIC, hHttpRequest, lpwhr->hdr.dwContext,
1488 INTERNET_STATUS_REQUEST_COMPLETE, &iar,
1489 sizeof(INTERNET_ASYNC_RESULT));
Ulrich Czekallac2757242000-06-11 20:04:44 +00001490 }
1491
1492 TRACE("<--\n");
1493 return bSuccess;
1494}
1495
1496
1497/***********************************************************************
1498 * HTTP_Connect (internal)
1499 *
1500 * Create http session handle
1501 *
1502 * RETURNS
1503 * HINTERNET a session handle on success
1504 * NULL on failure
1505 *
1506 */
Vincent Béron9a624912002-05-31 23:06:46 +00001507HINTERNET HTTP_Connect(HINTERNET hInternet, LPCSTR lpszServerName,
Ulrich Czekallac2757242000-06-11 20:04:44 +00001508 INTERNET_PORT nServerPort, LPCSTR lpszUserName,
1509 LPCSTR lpszPassword, DWORD dwFlags, DWORD dwContext)
1510{
1511 BOOL bSuccess = FALSE;
1512 LPWININETAPPINFOA hIC = NULL;
1513 LPWININETHTTPSESSIONA lpwhs = NULL;
1514
Aric Stewartff9b9d42002-06-21 23:59:49 +00001515 TRACE("-->\n");
Ulrich Czekallac2757242000-06-11 20:04:44 +00001516
1517 if (((LPWININETHANDLEHEADER)hInternet)->htype != WH_HINIT)
1518 goto lerror;
1519
1520 hIC = (LPWININETAPPINFOA) hInternet;
Aric Stewartff9b9d42002-06-21 23:59:49 +00001521 hIC->hdr.dwContext = dwContext;
Dominik Strasser94c02fe2003-04-14 21:32:36 +00001522
Ulrich Czekallac2757242000-06-11 20:04:44 +00001523 lpwhs = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WININETHTTPSESSIONA));
1524 if (NULL == lpwhs)
1525 {
1526 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1527 goto lerror;
1528 }
1529
Aric Stewartff9b9d42002-06-21 23:59:49 +00001530 /*
1531 * According to my tests. The name is not resolved until a request is sent
1532 */
Ulrich Czekallac2757242000-06-11 20:04:44 +00001533
1534 if (nServerPort == INTERNET_INVALID_PORT_NUMBER)
1535 nServerPort = INTERNET_DEFAULT_HTTP_PORT;
1536
Ulrich Czekallac2757242000-06-11 20:04:44 +00001537 lpwhs->hdr.htype = WH_HHTTPSESSION;
1538 lpwhs->hdr.lpwhparent = (LPWININETHANDLEHEADER)hInternet;
1539 lpwhs->hdr.dwFlags = dwFlags;
1540 lpwhs->hdr.dwContext = dwContext;
Dominik Strasser94c02fe2003-04-14 21:32:36 +00001541 if(hIC->lpszProxy && hIC->dwAccessType == INTERNET_OPEN_TYPE_PROXY) {
1542 if(strchr(hIC->lpszProxy, ' '))
1543 FIXME("Several proxies not implemented.\n");
1544 if(hIC->lpszProxyBypass)
1545 FIXME("Proxy bypass is ignored.\n");
1546 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001547 if (NULL != lpszServerName)
Alexandre Julliard0e44f632000-11-16 00:28:52 +00001548 lpwhs->lpszServerName = HTTP_strdup(lpszServerName);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001549 if (NULL != lpszUserName)
Alexandre Julliard0e44f632000-11-16 00:28:52 +00001550 lpwhs->lpszUserName = HTTP_strdup(lpszUserName);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001551 lpwhs->nServerPort = nServerPort;
1552
1553 if (hIC->lpfnStatusCB)
1554 {
1555 INTERNET_ASYNC_RESULT iar;
1556
1557 iar.dwResult = (DWORD)lpwhs;
1558 iar.dwError = ERROR_SUCCESS;
1559
Aric Stewartff9b9d42002-06-21 23:59:49 +00001560 SendAsyncCallback(hIC, hInternet, dwContext,
1561 INTERNET_STATUS_HANDLE_CREATED, &iar,
1562 sizeof(INTERNET_ASYNC_RESULT));
Ulrich Czekallac2757242000-06-11 20:04:44 +00001563 }
1564
1565 bSuccess = TRUE;
1566
1567lerror:
1568 if (!bSuccess && lpwhs)
1569 {
1570 HeapFree(GetProcessHeap(), 0, lpwhs);
1571 lpwhs = NULL;
1572 }
1573
Aric Stewartff9b9d42002-06-21 23:59:49 +00001574/*
1575 * a INTERNET_STATUS_REQUEST_COMPLETE is NOT sent here as per my tests on
1576 * windows
1577 */
Vincent Béron9a624912002-05-31 23:06:46 +00001578
David Hammerton852c7ae2003-06-20 23:26:56 +00001579 TRACE("%p -->\n", hInternet);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001580 return (HINTERNET)lpwhs;
1581}
1582
1583
1584/***********************************************************************
1585 * HTTP_OpenConnection (internal)
1586 *
1587 * Connect to a web server
1588 *
1589 * RETURNS
1590 *
1591 * TRUE on success
1592 * FALSE on failure
1593 */
1594BOOL HTTP_OpenConnection(LPWININETHTTPREQA lpwhr)
1595{
1596 BOOL bSuccess = FALSE;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001597 LPWININETHTTPSESSIONA lpwhs;
Aric Stewartff9b9d42002-06-21 23:59:49 +00001598 LPWININETAPPINFOA hIC = NULL;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001599
Aric Stewartff9b9d42002-06-21 23:59:49 +00001600 TRACE("-->\n");
1601
Ulrich Czekallac2757242000-06-11 20:04:44 +00001602
1603 if (NULL == lpwhr || lpwhr->hdr.htype != WH_HHTTPREQ)
1604 {
1605 INTERNET_SetLastError(ERROR_INVALID_PARAMETER);
1606 goto lend;
1607 }
1608
1609 lpwhs = (LPWININETHTTPSESSIONA)lpwhr->hdr.lpwhparent;
1610
Aric Stewartff9b9d42002-06-21 23:59:49 +00001611 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
1612 SendAsyncCallback(hIC, lpwhr, lpwhr->hdr.dwContext,
1613 INTERNET_STATUS_CONNECTING_TO_SERVER,
1614 &(lpwhs->socketAddress),
1615 sizeof(struct sockaddr_in));
1616
David Hammerton852c7ae2003-06-20 23:26:56 +00001617 if (!NETCON_create(&lpwhr->netConnection, lpwhs->phostent->h_addrtype,
1618 SOCK_STREAM, 0))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001619 {
1620 WARN("Socket creation failed\n");
1621 goto lend;
1622 }
1623
David Hammerton852c7ae2003-06-20 23:26:56 +00001624 if (!NETCON_connect(&lpwhr->netConnection, (struct sockaddr *)&lpwhs->socketAddress,
1625 sizeof(lpwhs->socketAddress)))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001626 {
Andreas Mohr34965562000-08-26 20:31:48 +00001627 WARN("Unable to connect to host (%s)\n", strerror(errno));
Ulrich Czekallac2757242000-06-11 20:04:44 +00001628 goto lend;
1629 }
1630
Aric Stewartff9b9d42002-06-21 23:59:49 +00001631 SendAsyncCallback(hIC, lpwhr, lpwhr->hdr.dwContext,
1632 INTERNET_STATUS_CONNECTED_TO_SERVER,
1633 &(lpwhs->socketAddress),
1634 sizeof(struct sockaddr_in));
1635
Ulrich Czekallac2757242000-06-11 20:04:44 +00001636 bSuccess = TRUE;
1637
1638lend:
Aric Stewartff9b9d42002-06-21 23:59:49 +00001639 TRACE("%d <--\n", bSuccess);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001640 return bSuccess;
1641}
1642
1643
1644/***********************************************************************
1645 * HTTP_GetResponseHeaders (internal)
1646 *
1647 * Read server response
1648 *
1649 * RETURNS
1650 *
1651 * TRUE on success
1652 * FALSE on error
1653 */
1654BOOL HTTP_GetResponseHeaders(LPWININETHTTPREQA lpwhr)
1655{
1656 INT cbreaks = 0;
1657 CHAR buffer[MAX_REPLY_LEN];
1658 DWORD buflen = MAX_REPLY_LEN;
1659 BOOL bSuccess = FALSE;
Aric Stewartff9b9d42002-06-21 23:59:49 +00001660 INT rc = 0;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001661 CHAR value[MAX_FIELD_VALUE_LEN], field[MAX_FIELD_LEN];
1662
Aric Stewartff9b9d42002-06-21 23:59:49 +00001663 TRACE("-->\n");
Ulrich Czekallac2757242000-06-11 20:04:44 +00001664
David Hammerton852c7ae2003-06-20 23:26:56 +00001665 if (!NETCON_connected(&lpwhr->netConnection))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001666 goto lend;
1667
Vincent Béron9a624912002-05-31 23:06:46 +00001668 /*
Aric Stewartff9b9d42002-06-21 23:59:49 +00001669 * HACK peek at the buffer
1670 */
David Hammerton852c7ae2003-06-20 23:26:56 +00001671 NETCON_recv(&lpwhr->netConnection, buffer, buflen, MSG_PEEK, &rc);
Aric Stewartff9b9d42002-06-21 23:59:49 +00001672
1673 /*
Ulrich Czekallac2757242000-06-11 20:04:44 +00001674 * We should first receive 'HTTP/1.x nnn' where nnn is the status code.
1675 */
Aric Stewartff9b9d42002-06-21 23:59:49 +00001676 buflen = MAX_REPLY_LEN;
David Hammerton852c7ae2003-06-20 23:26:56 +00001677 memset(buffer, 0, MAX_REPLY_LEN);
1678 if (!NETCON_getNextLine(&lpwhr->netConnection, buffer, &buflen))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001679 goto lend;
1680
1681 if (strncmp(buffer, "HTTP", 4) != 0)
1682 goto lend;
Vincent Béron9a624912002-05-31 23:06:46 +00001683
Ulrich Czekallac2757242000-06-11 20:04:44 +00001684 buffer[12]='\0';
1685 HTTP_ProcessHeader(lpwhr, "Status", buffer+9, (HTTP_ADDREQ_FLAG_ADD | HTTP_ADDREQ_FLAG_REPLACE));
1686
1687 /* Parse each response line */
1688 do
1689 {
1690 buflen = MAX_REPLY_LEN;
David Hammerton852c7ae2003-06-20 23:26:56 +00001691 if (NETCON_getNextLine(&lpwhr->netConnection, buffer, &buflen))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001692 {
David Hammerton852c7ae2003-06-20 23:26:56 +00001693 TRACE("got line %s, now interpretting\n", debugstr_a(buffer));
Ulrich Czekallac2757242000-06-11 20:04:44 +00001694 if (!HTTP_InterpretHttpHeader(buffer, field, MAX_FIELD_LEN, value, MAX_FIELD_VALUE_LEN))
1695 break;
1696
1697 HTTP_ProcessHeader(lpwhr, field, value, (HTTP_ADDREQ_FLAG_ADD | HTTP_ADDREQ_FLAG_REPLACE));
1698 }
1699 else
1700 {
1701 cbreaks++;
1702 if (cbreaks >= 2)
1703 break;
1704 }
1705 }while(1);
1706
1707 bSuccess = TRUE;
1708
1709lend:
1710
Aric Stewartff9b9d42002-06-21 23:59:49 +00001711 TRACE("<--\n");
1712 if (bSuccess)
1713 return rc;
1714 else
1715 return FALSE;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001716}
1717
1718
1719/***********************************************************************
1720 * HTTP_InterpretHttpHeader (internal)
1721 *
1722 * Parse server response
1723 *
1724 * RETURNS
1725 *
1726 * TRUE on success
1727 * FALSE on error
1728 */
1729INT stripSpaces(LPCSTR lpszSrc, LPSTR lpszStart, INT *len)
1730{
1731 LPCSTR lpsztmp;
1732 INT srclen;
1733
1734 srclen = 0;
1735
1736 while (*lpszSrc == ' ' && *lpszSrc != '\0')
1737 lpszSrc++;
Vincent Béron9a624912002-05-31 23:06:46 +00001738
Ulrich Czekallac2757242000-06-11 20:04:44 +00001739 lpsztmp = lpszSrc;
1740 while(*lpsztmp != '\0')
1741 {
1742 if (*lpsztmp != ' ')
1743 srclen = lpsztmp - lpszSrc + 1;
1744
1745 lpsztmp++;
1746 }
1747
1748 *len = min(*len, srclen);
1749 strncpy(lpszStart, lpszSrc, *len);
1750 lpszStart[*len] = '\0';
1751
1752 return *len;
1753}
1754
1755
1756BOOL HTTP_InterpretHttpHeader(LPSTR buffer, LPSTR field, INT fieldlen, LPSTR value, INT valuelen)
1757{
1758 CHAR *pd;
1759 BOOL bSuccess = FALSE;
1760
1761 TRACE("\n");
1762
1763 *field = '\0';
1764 *value = '\0';
1765
1766 pd = strchr(buffer, ':');
1767 if (pd)
1768 {
1769 *pd = '\0';
1770 if (stripSpaces(buffer, field, &fieldlen) > 0)
1771 {
1772 if (stripSpaces(pd+1, value, &valuelen) > 0)
1773 bSuccess = TRUE;
1774 }
1775 }
1776
1777 TRACE("%d: field(%s) Value(%s)\n", bSuccess, field, value);
1778 return bSuccess;
1779}
1780
1781
1782/***********************************************************************
1783 * HTTP_GetStdHeaderIndex (internal)
1784 *
John R. Sheets646d2a22000-07-23 13:34:43 +00001785 * Lookup field index in standard http header array
Ulrich Czekallac2757242000-06-11 20:04:44 +00001786 *
1787 * FIXME: This should be stuffed into a hash table
1788 */
1789INT HTTP_GetStdHeaderIndex(LPCSTR lpszField)
1790{
1791 INT index = -1;
1792
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001793 if (!strcasecmp(lpszField, "Content-Length"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001794 index = HTTP_QUERY_CONTENT_LENGTH;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001795 else if (!strcasecmp(lpszField,"Status"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001796 index = HTTP_QUERY_STATUS_CODE;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001797 else if (!strcasecmp(lpszField,"Content-Type"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001798 index = HTTP_QUERY_CONTENT_TYPE;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001799 else if (!strcasecmp(lpszField,"Last-Modified"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001800 index = HTTP_QUERY_LAST_MODIFIED;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001801 else if (!strcasecmp(lpszField,"Location"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001802 index = HTTP_QUERY_LOCATION;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001803 else if (!strcasecmp(lpszField,"Accept"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001804 index = HTTP_QUERY_ACCEPT;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001805 else if (!strcasecmp(lpszField,"Referer"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001806 index = HTTP_QUERY_REFERER;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001807 else if (!strcasecmp(lpszField,"Content-Transfer-Encoding"))
Ulrich Czekallac2757242000-06-11 20:04:44 +00001808 index = HTTP_QUERY_CONTENT_TRANSFER_ENCODING;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001809 else if (!strcasecmp(lpszField,"Date"))
John R. Sheets646d2a22000-07-23 13:34:43 +00001810 index = HTTP_QUERY_DATE;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001811 else if (!strcasecmp(lpszField,"Server"))
John R. Sheets646d2a22000-07-23 13:34:43 +00001812 index = HTTP_QUERY_SERVER;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001813 else if (!strcasecmp(lpszField,"Connection"))
John R. Sheets646d2a22000-07-23 13:34:43 +00001814 index = HTTP_QUERY_CONNECTION;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001815 else if (!strcasecmp(lpszField,"ETag"))
John R. Sheets646d2a22000-07-23 13:34:43 +00001816 index = HTTP_QUERY_ETAG;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001817 else if (!strcasecmp(lpszField,"Accept-Ranges"))
John R. Sheets646d2a22000-07-23 13:34:43 +00001818 index = HTTP_QUERY_ACCEPT_RANGES;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001819 else if (!strcasecmp(lpszField,"Expires"))
John R. Sheets646d2a22000-07-23 13:34:43 +00001820 index = HTTP_QUERY_EXPIRES;
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00001821 else if (!strcasecmp(lpszField,"Mime-Version"))
John R. Sheets66b4dd22000-08-03 22:16:39 +00001822 index = HTTP_QUERY_MIME_VERSION;
Johan Dahlin664b9bb2001-12-17 20:50:53 +00001823 else if (!strcasecmp(lpszField,"Pragma"))
1824 index = HTTP_QUERY_PRAGMA;
1825 else if (!strcasecmp(lpszField,"Cache-Control"))
1826 index = HTTP_QUERY_CACHE_CONTROL;
1827 else if (!strcasecmp(lpszField,"Content-Length"))
1828 index = HTTP_QUERY_CONTENT_LENGTH;
1829 else if (!strcasecmp(lpszField,"User-Agent"))
1830 index = HTTP_QUERY_USER_AGENT;
Mike McCormacka1c16d22003-07-22 03:17:52 +00001831 else if (!strcasecmp(lpszField,"Proxy-Authenticate"))
1832 index = HTTP_QUERY_PROXY_AUTHENTICATE;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001833 else
1834 {
Aric Stewartff9b9d42002-06-21 23:59:49 +00001835 TRACE("Couldn't find %s in standard header table\n", lpszField);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001836 }
1837
1838 return index;
1839}
1840
1841
1842/***********************************************************************
1843 * HTTP_ProcessHeader (internal)
1844 *
1845 * Stuff header into header tables according to <dwModifier>
1846 *
1847 */
1848
1849#define COALESCEFLASG (HTTP_ADDHDR_FLAG_COALESCE|HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA|HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON)
1850
1851BOOL HTTP_ProcessHeader(LPWININETHTTPREQA lpwhr, LPCSTR field, LPCSTR value, DWORD dwModifier)
1852{
1853 LPHTTPHEADERA lphttpHdr = NULL;
1854 BOOL bSuccess = FALSE;
1855 INT index;
1856
Alberto Massaribc8bd722002-12-06 23:20:31 +00001857 TRACE("--> %s: %s - 0x%08x\n", field, value, (unsigned int)dwModifier);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001858
1859 /* Adjust modifier flags */
1860 if (dwModifier & COALESCEFLASG)
1861 dwModifier |= HTTP_ADDHDR_FLAG_ADD;
1862
1863 /* Try to get index into standard header array */
1864 index = HTTP_GetStdHeaderIndex(field);
1865 if (index >= 0)
1866 {
1867 lphttpHdr = &lpwhr->StdHeaders[index];
1868 }
1869 else /* Find or create new custom header */
1870 {
Aric Stewartff9b9d42002-06-21 23:59:49 +00001871 index = HTTP_GetCustomHeaderIndex(lpwhr, field);
1872 if (index >= 0)
1873 {
1874 if (dwModifier & HTTP_ADDHDR_FLAG_ADD_IF_NEW)
1875 {
1876 return FALSE;
1877 }
1878 lphttpHdr = &lpwhr->pCustHeaders[index];
1879 }
1880 else
1881 {
1882 HTTPHEADERA hdr;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001883
Aric Stewartff9b9d42002-06-21 23:59:49 +00001884 hdr.lpszField = (LPSTR)field;
1885 hdr.lpszValue = (LPSTR)value;
1886 hdr.wFlags = hdr.wCount = 0;
Ulrich Czekallac2757242000-06-11 20:04:44 +00001887
1888 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1889 hdr.wFlags |= HDR_ISREQUEST;
1890
Mike McCormacka1c16d22003-07-22 03:17:52 +00001891 return HTTP_InsertCustomHeader(lpwhr, &hdr);
Aric Stewartff9b9d42002-06-21 23:59:49 +00001892 }
Ulrich Czekallac2757242000-06-11 20:04:44 +00001893 }
Vincent Béron9a624912002-05-31 23:06:46 +00001894
Ulrich Czekallac2757242000-06-11 20:04:44 +00001895 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1896 lphttpHdr->wFlags |= HDR_ISREQUEST;
1897 else
1898 lphttpHdr->wFlags &= ~HDR_ISREQUEST;
Vincent Béron9a624912002-05-31 23:06:46 +00001899
Ulrich Czekallac2757242000-06-11 20:04:44 +00001900 if (!lphttpHdr->lpszValue && (dwModifier & (HTTP_ADDHDR_FLAG_ADD|HTTP_ADDHDR_FLAG_ADD_IF_NEW)))
1901 {
1902 INT slen;
1903
1904 if (!lpwhr->StdHeaders[index].lpszField)
1905 {
Alexandre Julliard0e44f632000-11-16 00:28:52 +00001906 lphttpHdr->lpszField = HTTP_strdup(field);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001907
1908 if (dwModifier & HTTP_ADDHDR_FLAG_REQ)
1909 lphttpHdr->wFlags |= HDR_ISREQUEST;
1910 }
1911
1912 slen = strlen(value) + 1;
1913 lphttpHdr->lpszValue = HeapAlloc(GetProcessHeap(), 0, slen);
1914 if (lphttpHdr->lpszValue)
1915 {
1916 memcpy(lphttpHdr->lpszValue, value, slen);
1917 bSuccess = TRUE;
1918 }
1919 else
1920 {
1921 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1922 }
1923 }
Vincent Béron9a624912002-05-31 23:06:46 +00001924 else if (lphttpHdr->lpszValue)
Ulrich Czekallac2757242000-06-11 20:04:44 +00001925 {
Vincent Béron9a624912002-05-31 23:06:46 +00001926 if (dwModifier & HTTP_ADDHDR_FLAG_REPLACE)
Ulrich Czekallac2757242000-06-11 20:04:44 +00001927 {
1928 LPSTR lpsztmp;
1929 INT len;
1930
1931 len = strlen(value);
1932
1933 if (len <= 0)
1934 {
Patrik Stridvallfc2be7e2002-04-29 18:48:56 +00001935 /* if custom header delete from array */
Ulrich Czekallac2757242000-06-11 20:04:44 +00001936 HeapFree(GetProcessHeap(), 0, lphttpHdr->lpszValue);
1937 lphttpHdr->lpszValue = NULL;
1938 bSuccess = TRUE;
1939 }
1940 else
1941 {
1942 lpsztmp = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lphttpHdr->lpszValue, len+1);
1943 if (lpsztmp)
1944 {
1945 lphttpHdr->lpszValue = lpsztmp;
1946 strcpy(lpsztmp, value);
1947 bSuccess = TRUE;
1948 }
1949 else
1950 {
Alberto Massaribc8bd722002-12-06 23:20:31 +00001951 WARN("HeapReAlloc (%d bytes) failed\n",len+1);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001952 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1953 }
1954 }
1955 }
1956 else if (dwModifier & COALESCEFLASG)
1957 {
1958 LPSTR lpsztmp;
1959 CHAR ch = 0;
1960 INT len = 0;
1961 INT origlen = strlen(lphttpHdr->lpszValue);
1962 INT valuelen = strlen(value);
1963
1964 if (dwModifier & HTTP_ADDHDR_FLAG_COALESCE_WITH_COMMA)
1965 {
1966 ch = ',';
1967 lphttpHdr->wFlags |= HDR_COMMADELIMITED;
1968 }
1969 else if (dwModifier & HTTP_ADDHDR_FLAG_COALESCE_WITH_SEMICOLON)
1970 {
1971 ch = ';';
1972 lphttpHdr->wFlags |= HDR_COMMADELIMITED;
1973 }
1974
Alberto Massaribc8bd722002-12-06 23:20:31 +00001975 len = origlen + valuelen + ((ch > 0) ? 1 : 0);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001976
1977 lpsztmp = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lphttpHdr->lpszValue, len+1);
1978 if (lpsztmp)
1979 {
Vincent Béron9a624912002-05-31 23:06:46 +00001980 /* FIXME: Increment lphttpHdr->wCount. Perhaps lpszValue should be an array */
Ulrich Czekallac2757242000-06-11 20:04:44 +00001981 if (ch > 0)
1982 {
1983 lphttpHdr->lpszValue[origlen] = ch;
1984 origlen++;
1985 }
1986
1987 memcpy(&lphttpHdr->lpszValue[origlen], value, valuelen);
1988 lphttpHdr->lpszValue[len] = '\0';
1989 bSuccess = TRUE;
1990 }
1991 else
1992 {
Alberto Massaribc8bd722002-12-06 23:20:31 +00001993 WARN("HeapReAlloc (%d bytes) failed\n",len+1);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001994 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
1995 }
1996 }
1997 }
Alberto Massaribc8bd722002-12-06 23:20:31 +00001998 TRACE("<-- %d\n",bSuccess);
Ulrich Czekallac2757242000-06-11 20:04:44 +00001999 return bSuccess;
2000}
2001
2002
2003/***********************************************************************
2004 * HTTP_CloseConnection (internal)
2005 *
2006 * Close socket connection
2007 *
2008 */
2009VOID HTTP_CloseConnection(LPWININETHTTPREQA lpwhr)
2010{
Aric Stewartff9b9d42002-06-21 23:59:49 +00002011
2012
2013 LPWININETHTTPSESSIONA lpwhs = NULL;
2014 LPWININETAPPINFOA hIC = NULL;
2015
2016 TRACE("%p\n",lpwhr);
2017
2018 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
2019 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
2020
2021 SendAsyncCallback(hIC, lpwhr, lpwhr->hdr.dwContext,
2022 INTERNET_STATUS_CLOSING_CONNECTION, 0, 0);
2023
David Hammerton852c7ae2003-06-20 23:26:56 +00002024 if (NETCON_connected(&lpwhr->netConnection))
2025 {
2026 NETCON_close(&lpwhr->netConnection);
2027 }
Aric Stewartff9b9d42002-06-21 23:59:49 +00002028
2029 SendAsyncCallback(hIC, lpwhr, lpwhr->hdr.dwContext,
2030 INTERNET_STATUS_CONNECTION_CLOSED, 0, 0);
Ulrich Czekallac2757242000-06-11 20:04:44 +00002031}
2032
2033
2034/***********************************************************************
2035 * HTTP_CloseHTTPRequestHandle (internal)
2036 *
2037 * Deallocate request handle
2038 *
2039 */
2040void HTTP_CloseHTTPRequestHandle(LPWININETHTTPREQA lpwhr)
2041{
2042 int i;
Aric Stewartff9b9d42002-06-21 23:59:49 +00002043 LPWININETHTTPSESSIONA lpwhs = NULL;
2044 LPWININETAPPINFOA hIC = NULL;
Ulrich Czekallac2757242000-06-11 20:04:44 +00002045
2046 TRACE("\n");
2047
David Hammerton852c7ae2003-06-20 23:26:56 +00002048 if (NETCON_connected(&lpwhr->netConnection))
Ulrich Czekallac2757242000-06-11 20:04:44 +00002049 HTTP_CloseConnection(lpwhr);
2050
Aric Stewartff9b9d42002-06-21 23:59:49 +00002051 lpwhs = (LPWININETHTTPSESSIONA) lpwhr->hdr.lpwhparent;
2052 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
2053
2054 SendAsyncCallback(hIC, lpwhr, lpwhr->hdr.dwContext,
2055 INTERNET_STATUS_HANDLE_CLOSING, lpwhr,
2056 sizeof(HINTERNET));
2057
Ulrich Czekallac2757242000-06-11 20:04:44 +00002058 if (lpwhr->lpszPath)
2059 HeapFree(GetProcessHeap(), 0, lpwhr->lpszPath);
2060 if (lpwhr->lpszVerb)
2061 HeapFree(GetProcessHeap(), 0, lpwhr->lpszVerb);
2062 if (lpwhr->lpszHostName)
2063 HeapFree(GetProcessHeap(), 0, lpwhr->lpszHostName);
2064
2065 for (i = 0; i <= HTTP_QUERY_MAX; i++)
2066 {
2067 if (lpwhr->StdHeaders[i].lpszField)
2068 HeapFree(GetProcessHeap(), 0, lpwhr->StdHeaders[i].lpszField);
2069 if (lpwhr->StdHeaders[i].lpszValue)
2070 HeapFree(GetProcessHeap(), 0, lpwhr->StdHeaders[i].lpszValue);
2071 }
2072
2073 for (i = 0; i < lpwhr->nCustHeaders; i++)
2074 {
2075 if (lpwhr->pCustHeaders[i].lpszField)
2076 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders[i].lpszField);
2077 if (lpwhr->pCustHeaders[i].lpszValue)
2078 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders[i].lpszValue);
2079 }
2080
2081 HeapFree(GetProcessHeap(), 0, lpwhr->pCustHeaders);
2082 HeapFree(GetProcessHeap(), 0, lpwhr);
2083}
2084
2085
2086/***********************************************************************
2087 * HTTP_CloseHTTPSessionHandle (internal)
2088 *
2089 * Deallocate session handle
2090 *
2091 */
2092void HTTP_CloseHTTPSessionHandle(LPWININETHTTPSESSIONA lpwhs)
2093{
Aric Stewartff9b9d42002-06-21 23:59:49 +00002094 LPWININETAPPINFOA hIC = NULL;
David Hammerton852c7ae2003-06-20 23:26:56 +00002095 TRACE("%p\n", lpwhs);
Ulrich Czekallac2757242000-06-11 20:04:44 +00002096
Aric Stewartff9b9d42002-06-21 23:59:49 +00002097 hIC = (LPWININETAPPINFOA) lpwhs->hdr.lpwhparent;
2098
2099 SendAsyncCallback(hIC, lpwhs, lpwhs->hdr.dwContext,
2100 INTERNET_STATUS_HANDLE_CLOSING, lpwhs,
2101 sizeof(HINTERNET));
2102
Ulrich Czekallac2757242000-06-11 20:04:44 +00002103 if (lpwhs->lpszServerName)
2104 HeapFree(GetProcessHeap(), 0, lpwhs->lpszServerName);
2105 if (lpwhs->lpszUserName)
2106 HeapFree(GetProcessHeap(), 0, lpwhs->lpszUserName);
2107 HeapFree(GetProcessHeap(), 0, lpwhs);
2108}
2109
2110
2111/***********************************************************************
2112 * HTTP_GetCustomHeaderIndex (internal)
2113 *
2114 * Return index of custom header from header array
2115 *
2116 */
2117INT HTTP_GetCustomHeaderIndex(LPWININETHTTPREQA lpwhr, LPCSTR lpszField)
2118{
2119 INT index;
2120
2121 TRACE("%s\n", lpszField);
2122
2123 for (index = 0; index < lpwhr->nCustHeaders; index++)
2124 {
Alexandre Julliard5ab9d862000-08-09 22:35:05 +00002125 if (!strcasecmp(lpwhr->pCustHeaders[index].lpszField, lpszField))
Ulrich Czekallac2757242000-06-11 20:04:44 +00002126 break;
2127
2128 }
2129
2130 if (index >= lpwhr->nCustHeaders)
2131 index = -1;
2132
2133 TRACE("Return: %d\n", index);
2134 return index;
2135}
2136
2137
2138/***********************************************************************
2139 * HTTP_InsertCustomHeader (internal)
2140 *
2141 * Insert header into array
2142 *
2143 */
Mike McCormacka1c16d22003-07-22 03:17:52 +00002144BOOL HTTP_InsertCustomHeader(LPWININETHTTPREQA lpwhr, LPHTTPHEADERA lpHdr)
Ulrich Czekallac2757242000-06-11 20:04:44 +00002145{
2146 INT count;
2147 LPHTTPHEADERA lph = NULL;
Mike McCormacka1c16d22003-07-22 03:17:52 +00002148 BOOL r = FALSE;
Ulrich Czekallac2757242000-06-11 20:04:44 +00002149
Aric Stewartff9b9d42002-06-21 23:59:49 +00002150 TRACE("--> %s: %s\n", lpHdr->lpszField, lpHdr->lpszValue);
Ulrich Czekallac2757242000-06-11 20:04:44 +00002151 count = lpwhr->nCustHeaders + 1;
2152 if (count > 1)
2153 lph = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, lpwhr->pCustHeaders, sizeof(HTTPHEADERA) * count);
2154 else
2155 lph = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(HTTPHEADERA) * count);
2156
2157 if (NULL != lph)
2158 {
2159 lpwhr->pCustHeaders = lph;
Alexandre Julliard0e44f632000-11-16 00:28:52 +00002160 lpwhr->pCustHeaders[count-1].lpszField = HTTP_strdup(lpHdr->lpszField);
2161 lpwhr->pCustHeaders[count-1].lpszValue = HTTP_strdup(lpHdr->lpszValue);
Ulrich Czekallac2757242000-06-11 20:04:44 +00002162 lpwhr->pCustHeaders[count-1].wFlags = lpHdr->wFlags;
2163 lpwhr->pCustHeaders[count-1].wCount= lpHdr->wCount;
2164 lpwhr->nCustHeaders++;
Mike McCormacka1c16d22003-07-22 03:17:52 +00002165 r = TRUE;
Ulrich Czekallac2757242000-06-11 20:04:44 +00002166 }
2167 else
2168 {
2169 INTERNET_SetLastError(ERROR_OUTOFMEMORY);
Ulrich Czekallac2757242000-06-11 20:04:44 +00002170 }
2171
Mike McCormacka1c16d22003-07-22 03:17:52 +00002172 return r;
Ulrich Czekallac2757242000-06-11 20:04:44 +00002173}
2174
2175
2176/***********************************************************************
2177 * HTTP_DeleteCustomHeader (internal)
2178 *
2179 * Delete header from array
Mike McCormacka1c16d22003-07-22 03:17:52 +00002180 * If this function is called, the indexs may change.
Ulrich Czekallac2757242000-06-11 20:04:44 +00002181 */
Mike McCormacka1c16d22003-07-22 03:17:52 +00002182BOOL HTTP_DeleteCustomHeader(LPWININETHTTPREQA lpwhr, INT index)
Ulrich Czekallac2757242000-06-11 20:04:44 +00002183{
Mike McCormacka1c16d22003-07-22 03:17:52 +00002184 if( lpwhr->nCustHeaders <= 0 )
2185 return FALSE;
2186 if( lpwhr->nCustHeaders >= index )
2187 return FALSE;
2188 lpwhr->nCustHeaders--;
2189
2190 memmove( &lpwhr->pCustHeaders[index], &lpwhr->pCustHeaders[index+1],
2191 (lpwhr->nCustHeaders - index)* sizeof(HTTPHEADERA) );
2192 memset( &lpwhr->pCustHeaders[lpwhr->nCustHeaders], 0, sizeof(HTTPHEADERA) );
2193
2194 return TRUE;
Ulrich Czekallac2757242000-06-11 20:04:44 +00002195}
Alberto Massarib09eef22002-11-13 04:08:26 +00002196
2197/***********************************************************************
2198 * IsHostInProxyBypassList (@)
2199 *
2200 * Undocumented
2201 *
2202 */
2203BOOL WINAPI IsHostInProxyBypassList(DWORD flags, LPCSTR szHost, DWORD length)
2204{
2205 FIXME("STUB: flags=%ld host=%s length=%ld\n",flags,szHost,length);
2206 return FALSE;
2207}