|
|
//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1996 - 1999
//
// File: httptran.cxx
//
//--------------------------------------------------------------------------
#include <windows.h>
#include <tchar.h>
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include "GTTran.h"
#include "IHGTTran.h"
#include "httptran.hxx"
DWORD __stdcall SendExp(HUTTRAN hTran, DWORD dwEncodeType, DWORD cbSendBuff, const BYTE * pbSendBuff) { return( ((CHttpTran *) hTran)->Send(dwEncodeType, cbSendBuff, pbSendBuff)); }
DWORD __stdcall ReceiveExp(HUTTRAN hTran, DWORD * pdwEncodeType, DWORD * pcbReceiveBuff, BYTE ** ppbReceiveBuff) { return( ((CHttpTran *) hTran)->Receive(pdwEncodeType, pcbReceiveBuff, ppbReceiveBuff)); }
DWORD __stdcall CloseExp(HUTTRAN hTran) { return(((CHttpTran *) hTran)->Close()); }
DWORD __stdcall FreeExp(HUTTRAN hTran, BYTE * pb) { return(((CHttpTran *) hTran)->Free(pb)); }
DWORD __stdcall OpenExp(HUTTRAN * phTran, const TCHAR * tszBinding, DWORD fOpen) {
DWORD err;
assert(phTran != NULL);
if( (*phTran = (HUTTRAN) new CHttpTran(tszBinding)) == NULL ) return(ERROR_NOT_ENOUGH_MEMORY);
err = ((CHttpTran *) *phTran)->Open(tszBinding, fOpen);
if(err != ERROR_SUCCESS) delete (CHttpTran *) (*phTran);
return(err); }
DWORD CHttpTran::Open(const TCHAR * tszURL, DWORD fOpenT) {
TCHAR tszDomanName[_MAX_PATH]; TCHAR tszPort[12]; TCHAR * ptch; TCHAR * ptchT; DWORD err;
// did we get a flag?
if( (fOpenT & (GTREAD | GTWRITE)) == 0 ) return(ERROR_INVALID_PARAMETER);
// is it a readonly flag, then do gets
fOpen = fOpenT;
assert(tszURL != NULL);
// we must have http://
assert(_tcslen(tszURL) > 7); assert(_tcsnicmp(tszURL, TEXT("http://"), 7) == 0);
// copy the Doman Name
ptch = (TCHAR *) &tszURL[7]; ptchT = tszDomanName; while(*ptch != _T('/') && *ptch != _T(':') && *ptch != 0) *ptchT++ = *ptch++; *ptchT = 0;
// parse out the port number
tszPort[0] = 0; if(*ptch == _T(':')) { ptchT = tszPort; while(*ptch != _T('/') && *ptch != 0) *ptchT++ = *ptch++; *ptchT = 0; }
// Note, we don't support port numbers
assert(tszPort[0] == 0);
// save away what to look up.
tszPartURL = new TCHAR[(_tcslen(ptch) + 1)]; if (NULL == tszPartURL) { delete this; return(ERROR_NOT_ENOUGH_MEMORY); }
_tcscpy(tszPartURL, ptch);
// INTERNET_OPEN_TYPE_DIRECT,
if( (hIOpen = InternetOpen( TEXT("Transport"), INTERNET_OPEN_TYPE_PRECONFIG, NULL, NULL, 0)) == NULL ||
(hIConnect = InternetConnect(hIOpen, tszDomanName, INTERNET_INVALID_PORT_NUMBER, NULL, NULL, INTERNET_SERVICE_HTTP, 0, 0)) == NULL ) { err = GetLastError(); delete this; return(err); }
// If this is a GET, do a dummy send
if( fOpen == GTREAD && ((hIHttp = HttpOpenRequest(hIConnect, TEXT("GET"), tszPartURL, HTTP_VERSION, NULL, NULL, INTERNET_FLAG_DONT_CACHE, 0)) == NULL || HttpSendRequest(hIHttp, TEXT("Accept: */*\r\n"), (DWORD) -1, NULL, 0) == FALSE) ) { err = GetLastError(); delete this; return(err); }
return(ERROR_SUCCESS); }
DWORD CHttpTran::Send(DWORD dwEncodeType, DWORD cbSendBuff, const BYTE * pbSendBuff) {
TCHAR tszBuff[1024]; DWORD err; TCHAR * tszContentType;
if( pbRecBuf != NULL || (fOpen & GTWRITE) != GTWRITE) return(ERROR_INVALID_PARAMETER);
switch( dwEncodeType ) { case ASN_ENCODING: tszContentType = TEXT("application/x-octet-stream-asn"); break; case TLV_ENCODING: tszContentType = TEXT("application/x-octet-stream-tlv"); break; case IDL_ENCODING: tszContentType = TEXT("application/x-octet-stream-idl"); break; case OCTET_ENCODING: tszContentType = TEXT("application/octet-stream"); break; default: tszContentType = TEXT("text/*"); break; }
// say how long the buffer is
_stprintf(tszBuff, TEXT("Content-Type: %s\r\nContent-Length: %d\r\nAccept: %s\r\n"), tszContentType, cbSendBuff,tszContentType);
if( (hIHttp = HttpOpenRequest(hIConnect, TEXT("POST"), tszPartURL, HTTP_VERSION, NULL, NULL, INTERNET_FLAG_DONT_CACHE, 0)) == NULL ) { return(GetLastError()); }
// send of the request, this will wait for a response
if( HttpSendRequest(hIHttp, tszBuff, (DWORD) -1, (LPVOID) pbSendBuff, cbSendBuff) == FALSE ) {
err = GetLastError(); // close out the handle
assert(hIHttp != NULL); InternetCloseHandle(hIHttp); hIHttp = NULL;
return(err); }
return(ERROR_SUCCESS); }
DWORD CHttpTran::Receive(DWORD * pdwEncodeType, DWORD * pcbReceiveBuff, BYTE ** ppbReceiveBuff) {
TCHAR tszBuff[1024]; DWORD cbBuff, cbBuffRead, cbBuffT; DWORD err;
assert(pcbReceiveBuff != NULL && ppbReceiveBuff != NULL); *ppbReceiveBuff = NULL; *pcbReceiveBuff = 0;
if( pbRecBuf != NULL || (fOpen & GTREAD) != GTREAD || hIHttp == NULL) return(ERROR_INVALID_PARAMETER);
// get the content type
if( pdwEncodeType != NULL) {
cbBuff = sizeof(tszBuff); if(HttpQueryInfo( hIHttp, HTTP_QUERY_CONTENT_TYPE, tszBuff, &cbBuff, NULL) == FALSE) return(GetLastError());
assert(cbBuff > 0);
if(!_tcscmp(TEXT("application/x-octet-stream-asn"), tszBuff)) *pdwEncodeType = ASN_ENCODING; else if(!_tcscmp(TEXT("application/x-octet-stream-idl"), tszBuff)) *pdwEncodeType = IDL_ENCODING; else if(!_tcscmp(TEXT("application/x-octet-stream-tlv"), tszBuff)) *pdwEncodeType = TLV_ENCODING; else if(!_tcscmp(TEXT("application/octet-stream"), tszBuff)) *pdwEncodeType = OCTET_ENCODING; else *pdwEncodeType = ASCII_ENCODING;
}
// now get the length of the buffer returned
cbBuff = sizeof(tszBuff); if(HttpQueryInfo( hIHttp, HTTP_QUERY_CONTENT_LENGTH, tszBuff, &cbBuff, NULL) == FALSE) return(GetLastError());
assert(cbBuff > 0); cbBuff = _ttol(tszBuff);
// allocate a buffer
if( (pbRecBuf = new BYTE[cbBuff]) == NULL ) return(ERROR_NOT_ENOUGH_MEMORY);
// read the data
cbBuffRead = 0; while(cbBuffRead < cbBuff) { cbBuffT = 0; if(InternetReadFile(hIHttp, &pbRecBuf[cbBuffRead], (cbBuff - cbBuffRead), &cbBuffT) == FALSE ) { err = GetLastError(); delete [] pbRecBuf; pbRecBuf = NULL; return(err); } cbBuffRead += cbBuffT; }
// close out the handle
InternetCloseHandle(hIHttp); hIHttp = NULL;
// pass back the info
*ppbReceiveBuff = pbRecBuf; *pcbReceiveBuff = cbBuff; return(ERROR_SUCCESS); }
DWORD CHttpTran::Free(BYTE * pb) { assert(pb == pbRecBuf); delete [] pbRecBuf; pbRecBuf = NULL; return(ERROR_SUCCESS); }
DWORD CHttpTran::Close(void) { delete this; return(ERROR_SUCCESS); }
|