2020-09-30 16:53:55 +02:00

640 lines
18 KiB
C++
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1994 Microsoft Corporation
Module Name:
open.cxx
Abstract:
This file contains the implementation of the HttpOpenRequestA API.
The following functions are exported by this module:
HttpOpenRequestA
WinHttpOpenRequest
ParseHttpUrl
ParseHttpUrl_Fsm
Author:
Keith Moore (keithmo) 16-Nov-1994
Revision History:
Modified to make HttpOpenRequestA remotable. madana (2/8/95)
--*/
#include <wininetp.h>
#include "httpp.h"
//
// functions
//
INTERNETAPI
HINTERNET
WINAPI
HttpOpenRequestA(
IN HINTERNET hConnect,
IN LPCSTR lpszVerb OPTIONAL,
IN LPCSTR lpszObjectName OPTIONAL,
IN LPCSTR lpszVersion OPTIONAL,
IN LPCSTR lpszReferrer OPTIONAL,
IN LPCSTR FAR * lplpszAcceptTypes OPTIONAL,
IN DWORD dwFlags,
IN DWORD_PTR dwContext
)
/*++
Routine Description:
Creates a new HTTP request handle and stores the specified parameters
in that context.
Arguments:
hConnect - An open Internet handle returned by InternetConnect()
lpszVerb - The verb to use in the request. May be NULL in which
case "GET" will be used
lpszObjectName - The target object for the specified verb. This is
typically a file name, an executable module, or a
search specifier. May be NULL in which case the empty
string will be used
lpszVersion - The version string for the request. May be NULL in
which case "HTTP/1.0" will be used
lpszReferrer - Specifies the address (URI) of the document from
which the URI in the request (lpszObjectName) was
obtained. May be NULL in which case no referer is
specified
lplpszAcceptTypes - Points to a NULL-terminated array of LPCTSTR pointers
to content-types accepted by the client. This value
may be NULL in which case the default content-type
(text/html) is used
dwFlags - open options
dwContext - app-supplied context value for call-backs
BUGBUG: WHAT IS THE DEFAULT CONTENT-TRANSFER-ENCODING?
Return Value:
HINTERNET
Success - non-NULL (open) handle to an HTTP request
Failure - NULL. Error status is available by calling GetLastError()
--*/
{
DEBUG_ENTER_API((DBG_API,
Handle,
"HttpOpenRequestA",
"%#x, %.80q, %.80q, %.80q, %.80q, %#x, %#08x, %#08x",
hConnect,
lpszVerb,
lpszObjectName,
lpszVersion,
lpszReferrer,
lplpszAcceptTypes,
dwFlags,
dwContext
));
DWORD error;
HINTERNET hConnectMapped = NULL;
BOOL fRequestUsingProxy;
HINTERNET hRequest = NULL;
if (!GlobalDataInitialized) {
error = ERROR_WINHTTP_NOT_INITIALIZED;
goto done;
}
//
// get the per-thread info
//
LPINTERNET_THREAD_INFO lpThreadInfo;
lpThreadInfo = InternetGetThreadInfo();
if (lpThreadInfo == NULL) {
error = ERROR_WINHTTP_INTERNAL_ERROR;
goto done;
}
_InternetIncNestingCount();
//
// map the handle
//
error = MapHandleToAddress(hConnect, (LPVOID *)&hConnectMapped, FALSE);
if (error != ERROR_SUCCESS) {
goto quit;
}
//
// find path from internet handle and validate handle
//
BOOL isLocal;
BOOL isAsync;
error = RIsHandleLocal(hConnectMapped,
&isLocal,
&isAsync,
TypeHttpConnectHandle
);
if (error != ERROR_SUCCESS) {
goto quit;
}
//
// validate parameters. Allow lpszVerb to default to "GET" if a NULL pointer
// is supplied
//
if (!ARGUMENT_PRESENT(lpszVerb) || (*lpszVerb == '\0')) {
lpszVerb = DEFAULT_HTTP_REQUEST_VERB;
}
//
// if a NULL pointer or empty string is supplied for the object name, then
// convert to the default object name (root object)
//
if (!ARGUMENT_PRESENT(lpszObjectName) || (*lpszObjectName == '\0')) {
lpszObjectName = "/";
}
// check the rest of the parameters
if (dwFlags & ~WINHTTP_OPEN_REQUEST_FLAGS_MASK)
{
error = ERROR_INVALID_PARAMETER;
goto quit;
}
// default to the current supported version
char versionBuffer[sizeof("HTTP/4294967295.4294967295")];
DWORD verMajor;
DWORD verMinor;
if (!ARGUMENT_PRESENT(lpszVersion) || (*lpszVersion == '\0')) {
wsprintf(versionBuffer,
"HTTP/%d.%d",
HttpVersionInfo.dwMajorVersion,
HttpVersionInfo.dwMinorVersion
);
lpszVersion = versionBuffer;
verMajor = HttpVersionInfo.dwMajorVersion;
verMinor = HttpVersionInfo.dwMinorVersion;
} else if (strnicmp(lpszVersion, "HTTP/", sizeof("HTTP/") - 1) == 0) {
LPSTR p = (LPSTR)lpszVersion + sizeof("HTTP/") - 1;
ExtractInt(&p, 0, (LPINT)&verMajor);
while (!isdigit(*p) && (*p != '\0')) {
++p;
}
ExtractInt(&p, 0, (LPINT)&verMinor);
} else {
verMajor = 1;
verMinor = 0;
}
//
// if we have HTTP 1.1 enabled in the registry and the version is < 1.1
// then convert
//
if (GlobalEnableHttp1_1
&& (((verMajor == 1) && (verMinor == 0)) || (verMajor < 1))) {
lpszVersion = "HTTP/1.1";
}
//
// allow empty strings to be equivalent to NULL pointer
//
if (ARGUMENT_PRESENT(lpszReferrer) && (*lpszReferrer == '\0')) {
lpszReferrer = NULL;
}
// get the target port
INTERNET_CONNECT_HANDLE_OBJECT * pConnect;
pConnect = (INTERNET_CONNECT_HANDLE_OBJECT *)hConnectMapped;
INTERNET_PORT hostPort;
hostPort = pConnect->GetHostPort();
//
// set the per-thread info: parent handle object
//
_InternetSetObjectHandle(lpThreadInfo, hConnect, hConnectMapped);
//
// make local HTTP request handle object before we can add headers to it
//
error = RMakeHttpReqObjectHandle(hConnectMapped,
&hRequest,
NULL, // (CLOSE_HANDLE_FUNC)wHttpCloseRequest
dwFlags,
dwContext
);
if (error != ERROR_SUCCESS) {
goto quit;
}
HTTP_REQUEST_HANDLE_OBJECT * pRequest;
pRequest = (HTTP_REQUEST_HANDLE_OBJECT *)hRequest;
//
// add the request line
//
INET_ASSERT((lpszVerb != NULL) && (*lpszVerb != '\0'));
INET_ASSERT((lpszObjectName != NULL) && (*lpszObjectName != '\0'));
INET_ASSERT((lpszVersion != NULL) && (*lpszVersion != '\0'));
if (!pRequest->LockHeaders())
{
error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
//
// encode the URL-path
//
error = pRequest->AddRequest((LPSTR)lpszVerb,
(LPSTR)lpszObjectName,
(LPSTR)lpszVersion
);
if (error != ERROR_SUCCESS) {
pRequest->UnlockHeaders();
goto quit;
}
//
// set the method type from the verb
//
pRequest->SetMethodType(lpszVerb);
//
// add the headers
//
if (lpszReferrer != NULL) {
error = pRequest->AddRequestHeader(HTTP_QUERY_REFERER,
(LPSTR)lpszReferrer,
lstrlen(lpszReferrer),
0,
CLEAN_HEADER
);
if (error != ERROR_SUCCESS) {
pRequest->UnlockHeaders();
goto quit;
}
}
if (lplpszAcceptTypes != NULL) {
while (*lplpszAcceptTypes) {
error = pRequest->AddRequestHeader(HTTP_QUERY_ACCEPT,
(LPSTR)*lplpszAcceptTypes,
lstrlen(*(LPSTR*)lplpszAcceptTypes),
0,
CLEAN_HEADER | COALESCE_HEADER_WITH_COMMA
);
if (error != ERROR_SUCCESS) {
pRequest->UnlockHeaders();
goto quit;
}
++lplpszAcceptTypes;
}
}
INET_ASSERT(error == ERROR_SUCCESS);
pRequest->UnlockHeaders();
//
// change the object state to opened
//
pRequest->SetState(HttpRequestStateOpen);
((HTTP_REQUEST_HANDLE_OBJECT *)hRequest)->SetRequestUsingProxy(
FALSE
);
if (hostPort == INTERNET_INVALID_PORT_NUMBER)
{
if (dwFlags & WINHTTP_FLAG_SECURE)
{
pRequest->SetHostPort(INTERNET_DEFAULT_HTTPS_PORT);
}
else
{
pRequest->SetHostPort(INTERNET_DEFAULT_HTTP_PORT);
}
}
else
{
pRequest->SetHostPort(hostPort);
}
//
// if the object name is not set then all cache methods fail
//
URLGEN_FUNC fn;
fn = (URLGEN_FUNC)pHttpGetUrlString;
//
// BUGBUG - change prototype to take LPCSTR
//
error = pRequest->SetObjectName((LPSTR)lpszObjectName,
NULL,
&fn
);
quit:
_InternetDecNestingCount(1);
done:
if (error != ERROR_SUCCESS) {
if (hRequest != NULL) {
WinHttpCloseHandle(((HANDLE_OBJECT *)hRequest)->GetPseudoHandle());
}
DEBUG_ERROR(HTTP, error);
SetLastError(error);
hRequest = NULL;
} else {
//
// success - don't return the object address, return the pseudo-handle
// value we generated
//
hRequest = ((HANDLE_OBJECT *)hRequest)->GetPseudoHandle();
}
if (hConnectMapped != NULL) {
DereferenceObject((LPVOID)hConnectMapped);
}
DEBUG_LEAVE_API(hRequest);
return hRequest;
}
INTERNETAPI
HINTERNET
WINAPI
WinHttpOpenRequest(
IN HINTERNET hConnect,
IN LPCWSTR lpszVerb,
IN LPCWSTR lpszObjectName,
IN LPCWSTR lpszVersion,
IN LPCWSTR lpszReferrer OPTIONAL,
IN LPCWSTR FAR * lplpszAcceptTypes OPTIONAL,
IN DWORD dwFlags
)
/*++
Routine Description:
Creates a new HTTP request handle and stores the specified parameters
in that context.
Arguments:
hHttpSession - An open Internet handle returned by InternetConnect()
lpszVerb - The verb to use in the request
lpszObjectName - The target object for the specified verb. This is
typically a file name, an executable module, or a
search specifier
lpszVersion - The version string for the request
lpszReferrer - Specifies the address (URI) of the document from
which the URI in the request (lpszObjectName) was
obtained. May be NULL in which case no referer is
specified
lplpszAcceptTypes - Points to a NULL-terminated array of LPCTSTR pointers
to content-types accepted by the client. This value
may be NULL in which case the default content-type
(text/html) is used
dwFlags - open options
BUGBUG: WHAT IS THE DEFAULT CONTENT-TRANSFER-ENCODING?
Return Value:
!NULL - An open handle to an HTTP request.
NULL - The operation failed. Error status is available by calling
GetLastError().
Comments:
--*/
{
DEBUG_ENTER_API((DBG_API,
Handle,
"WinHttpOpenRequest",
"%#x, %.80wq, %.80wq, %.80wq, %.80wq, %#x, %#08x",
hConnect,
lpszVerb,
lpszObjectName,
lpszVersion,
lpszReferrer,
lplpszAcceptTypes,
dwFlags
));
HINTERNET hConnectMapped = NULL;
INTERNET_CONNECT_HANDLE_OBJECT * pConnect;
DWORD dwErr = ERROR_SUCCESS;
HINTERNET hInternet = NULL;
MEMORYPACKET mpVerb, mpObjectName, mpVersion, mpReferrer;
MEMORYPACKETTABLE mptAcceptTypes;
BOOL isLocal;
BOOL isAsync;
if (dwFlags &~ (WINHTTP_OPEN_REQUEST_FLAGS_MASK))
{
dwErr = ERROR_INVALID_PARAMETER;
goto cleanup;
}
// map the handle
dwErr = MapHandleToAddress(hConnect, (LPVOID *)&hConnectMapped, FALSE);
if (dwErr != ERROR_SUCCESS)
{
goto cleanup;
}
// find path from internet handle and validate handle
dwErr = RIsHandleLocal(hConnectMapped,
&isLocal,
&isAsync,
TypeHttpConnectHandle
);
if (dwErr != ERROR_SUCCESS)
{
goto cleanup;
}
if (lpszVerb)
{
if (IsBadStringPtrW(lpszVerb, -1))
{
dwErr = ERROR_INVALID_PARAMETER;
goto cleanup;
}
ALLOC_MB(lpszVerb,0,mpVerb);
if (!mpVerb.psStr)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
UNICODE_TO_ANSI(lpszVerb,mpVerb);
}
if (lpszObjectName)
{
if (IsBadStringPtrW(lpszObjectName, -1))
{
dwErr = ERROR_INVALID_PARAMETER;
goto cleanup;
}
pConnect = (INTERNET_CONNECT_HANDLE_OBJECT*)hConnectMapped;
DWORD dwCodePage = pConnect->GetCodePage();
dwErr = ConvertUnicodeToMultiByte(lpszObjectName, dwCodePage, &mpObjectName,
(dwFlags&(WINHTTP_FLAG_ESCAPE_PERCENT|WINHTTP_FLAG_NULL_CODEPAGE))|WINHTTP_FLAG_DEFAULT_ESCAPE );
if (dwErr != ERROR_SUCCESS)
{
goto cleanup;
}
}
if (lpszVersion)
{
if (IsBadStringPtrW(lpszVersion, -1))
{
dwErr = ERROR_INVALID_PARAMETER;
goto cleanup;
}
ALLOC_MB(lpszVersion,0,mpVersion);
if (!mpVersion.psStr)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
UNICODE_TO_ANSI(lpszVersion,mpVersion);
}
if (lpszReferrer)
{
if (IsBadStringPtrW(lpszReferrer, -1))
{
dwErr = ERROR_INVALID_PARAMETER;
goto cleanup;
}
ALLOC_MB(lpszReferrer,0,mpReferrer);
if (!mpReferrer.psStr)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
UNICODE_TO_ANSI(lpszReferrer,mpReferrer);
}
// Create a table of ansi strings
if (lplpszAcceptTypes)
{
WORD csTmp=0;
do
{
if (IsBadReadPtr(lplpszAcceptTypes+csTmp*sizeof(LPCWSTR), sizeof(LPCWSTR)))
{
dwErr = ERROR_INVALID_PARAMETER;
goto cleanup;
}
if (lplpszAcceptTypes[csTmp])
{
if (IsBadStringPtrW(lplpszAcceptTypes[csTmp++], -1))
{
dwErr = ERROR_INVALID_PARAMETER;
goto cleanup;
}
}
else
break;
}
while (TRUE);
mptAcceptTypes.SetUpFor(csTmp);
for (WORD ce=0; ce < csTmp; ce++)
{
mptAcceptTypes.pdwAlloc[ce] = (lstrlenW(lplpszAcceptTypes[ce]) + 1)*sizeof(WCHAR);
mptAcceptTypes.ppsStr[ce] = (LPSTR)ALLOC_BYTES(mptAcceptTypes.pdwAlloc[ce]*sizeof(CHAR));
if (!mptAcceptTypes.ppsStr[ce])
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
goto cleanup;
}
mptAcceptTypes.pdwSize[ce] = WideCharToMultiByte(CP_ACP,
0,
lplpszAcceptTypes[ce],
mptAcceptTypes.pdwAlloc[ce]/sizeof(WCHAR),
mptAcceptTypes.ppsStr[ce],
mptAcceptTypes.pdwAlloc[ce],NULL,NULL);
}
}
hInternet = HttpOpenRequestA(hConnect, mpVerb.psStr, mpObjectName.psStr, mpVersion.psStr,
mpReferrer.psStr, (LPCSTR*)mptAcceptTypes.ppsStr,
dwFlags, NULL);
cleanup:
if (hConnectMapped != NULL)
{
DereferenceObject((LPVOID)hConnectMapped);
}
if (dwErr!=ERROR_SUCCESS)
{
SetLastError(dwErr);
DEBUG_ERROR(HTTP, dwErr);
}
DEBUG_LEAVE_API(hInternet);
return hInternet;
}