mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3527 lines
81 KiB
3527 lines
81 KiB
/*++
|
|
|
|
Copyright (c) 2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
ssinc.cxx
|
|
|
|
Abstract:
|
|
|
|
This module contains the server side include processing code. We
|
|
aim for support as specified by iis\spec\ssi.doc. The code is based
|
|
on existing SSI support done in iis\svcs\w3\server\ssinc.cxx.
|
|
|
|
Author:
|
|
|
|
Ming Lu (MingLu) 10-Apr-2000
|
|
|
|
--*/
|
|
|
|
#include "precomp.hxx"
|
|
|
|
//
|
|
// Globals
|
|
//
|
|
|
|
UINT g_MonthToDayCount[] = {
|
|
0,
|
|
31,
|
|
31 + 28,
|
|
31 + 28 + 31,
|
|
31 + 28 + 31 + 30,
|
|
31 + 28 + 31 + 30 + 31,
|
|
31 + 28 + 31 + 30 + 31 + 30,
|
|
31 + 28 + 31 + 30 + 31 + 30 + 31,
|
|
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31,
|
|
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30,
|
|
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31,
|
|
31 + 28 + 31 + 30 + 31 + 30 + 31 + 31 + 30 + 31 + 30,
|
|
} ;
|
|
|
|
//
|
|
// Prototypes
|
|
//
|
|
|
|
extern "C" {
|
|
|
|
BOOL
|
|
WINAPI
|
|
DLLEntry(
|
|
HINSTANCE hDll,
|
|
DWORD dwReason,
|
|
LPVOID lpvReserved
|
|
);
|
|
}
|
|
|
|
class SSI_ELEMENT_LIST;
|
|
|
|
|
|
VOID
|
|
InitializeSSIGlobals( VOID );
|
|
|
|
//
|
|
// Global Data
|
|
//
|
|
|
|
DECLARE_DEBUG_PRINTS_OBJECT()
|
|
DECLARE_DEBUG_VARIABLE();
|
|
|
|
//
|
|
// This is the list of supported commands
|
|
//
|
|
|
|
struct _SSI_CMD_MAP
|
|
{
|
|
CHAR * pszCommand;
|
|
DWORD cchCommand;
|
|
SSI_COMMANDS ssiCmd;
|
|
}
|
|
SSICmdMap[] =
|
|
{
|
|
"#include ", 9, SSI_CMD_INCLUDE,
|
|
"#echo ", 6, SSI_CMD_ECHO,
|
|
"#fsize ", 7, SSI_CMD_FSIZE,
|
|
"#flastmod ",10, SSI_CMD_FLASTMOD,
|
|
"#config ", 8, SSI_CMD_CONFIG,
|
|
"#exec ", 6, SSI_CMD_EXEC,
|
|
NULL, 0, SSI_CMD_UNKNOWN
|
|
};
|
|
|
|
//
|
|
// This is the list of supported tags
|
|
//
|
|
|
|
struct _SSI_TAG_MAP
|
|
{
|
|
CHAR * pszTag;
|
|
DWORD cchTag;
|
|
SSI_TAGS ssiTag;
|
|
}
|
|
SSITagMap[] =
|
|
{
|
|
"var", 3, SSI_TAG_VAR,
|
|
"file", 4, SSI_TAG_FILE,
|
|
"virtual", 7, SSI_TAG_VIRTUAL,
|
|
"errmsg", 6, SSI_TAG_ERRMSG,
|
|
"timefmt", 7, SSI_TAG_TIMEFMT,
|
|
"sizefmt", 7, SSI_TAG_SIZEFMT,
|
|
"cmd", 3, SSI_TAG_CMD,
|
|
"cgi", 3, SSI_TAG_CGI,
|
|
"isa", 3, SSI_TAG_ISA,
|
|
NULL, 0, SSI_TAG_UNKNOWN
|
|
};
|
|
|
|
//
|
|
// This is a list of #ECHO variables not supported by ISAPI
|
|
//
|
|
|
|
struct _SSI_VAR_MAP
|
|
{
|
|
CHAR * pszMap;
|
|
DWORD cchMap;
|
|
SSI_VARS ssiMap;
|
|
}
|
|
SSIVarMap[] =
|
|
{
|
|
"DOCUMENT_NAME", 13, SSI_VAR_DOCUMENT_NAME,
|
|
"DOCUMENT_URI", 12, SSI_VAR_DOCUMENT_URI,
|
|
"QUERY_STRING_UNESCAPED", 22, SSI_VAR_QUERY_STRING_UNESCAPED,
|
|
"DATE_LOCAL", 10, SSI_VAR_DATE_LOCAL,
|
|
"DATE_GMT", 8, SSI_VAR_DATE_GMT,
|
|
"LAST_MODIFIED", 13, SSI_VAR_LAST_MODIFIED,
|
|
NULL, 0, SSI_VAR_UNKNOWN
|
|
};
|
|
|
|
//
|
|
// #EXEC CMD is BAD. Disable it by default
|
|
//
|
|
|
|
BOOL fEnableCmdDirective = FALSE;
|
|
|
|
//
|
|
// Pointer to file cache
|
|
//
|
|
|
|
W3_FILE_INFO_CACHE * g_pFileCache = NULL;
|
|
|
|
//
|
|
// Class Definitions
|
|
//
|
|
|
|
// class SSI_FILE
|
|
//
|
|
// File structure. All high level functions should use this
|
|
// structure instead of dealing with handle specifics themselves.
|
|
|
|
class SSI_FILE
|
|
{
|
|
private:
|
|
STRU _strFilename;
|
|
W3_FILE_INFO * _hHandle;
|
|
HANDLE _hMapHandle;
|
|
PVOID _pvMappedBase;
|
|
BOOL _fValid;
|
|
BOOL _fCloseOnDestroy;
|
|
|
|
//
|
|
// Track the current number of open handles for this file.
|
|
//
|
|
|
|
DWORD _cRefCount;
|
|
CRITICAL_SECTION _csRef;
|
|
|
|
public:
|
|
|
|
SSI_FILE(
|
|
IN STRU * pstrFilename,
|
|
IN W3_FILE_INFO * pOpenFile
|
|
) : _hHandle ( pOpenFile ),
|
|
_hMapHandle ( NULL ),
|
|
_pvMappedBase ( NULL ),
|
|
_fValid ( FALSE ),
|
|
_cRefCount ( 0),
|
|
_fCloseOnDestroy( FALSE )
|
|
{
|
|
InitializeCriticalSection( &_csRef );
|
|
|
|
if ( FAILED( _strFilename.Copy( pstrFilename->QueryStr() ) ) )
|
|
{
|
|
return;
|
|
}
|
|
_fValid = TRUE;
|
|
}
|
|
|
|
SSI_FILE(
|
|
IN STRU * pstrFilename,
|
|
IN HANDLE hUser
|
|
) : _hHandle ( NULL ),
|
|
_hMapHandle ( NULL ),
|
|
_pvMappedBase ( NULL ),
|
|
_fValid ( FALSE ),
|
|
_cRefCount ( 0),
|
|
_fCloseOnDestroy( TRUE )
|
|
{
|
|
FILE_CACHE_USER fileUser;
|
|
HRESULT hr;
|
|
|
|
InitializeCriticalSection( &_csRef );
|
|
|
|
if ( FAILED( _strFilename.Copy( pstrFilename->QueryStr() ) ) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
fileUser._hToken = hUser;
|
|
|
|
hr = g_pFileCache->GetFileInfo( _strFilename,
|
|
NULL,
|
|
&fileUser,
|
|
TRUE,
|
|
&_hHandle );
|
|
if ( FAILED( hr ) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
_fValid = TRUE;
|
|
}
|
|
|
|
~SSI_FILE()
|
|
{
|
|
if ( _fCloseOnDestroy )
|
|
{
|
|
if ( _hHandle )
|
|
{
|
|
_hHandle->DereferenceCacheEntry();
|
|
}
|
|
}
|
|
DeleteCriticalSection( &_csRef );
|
|
}
|
|
|
|
VOID
|
|
Lock(
|
|
VOID
|
|
)
|
|
{
|
|
EnterCriticalSection( &_csRef );
|
|
}
|
|
|
|
VOID
|
|
UnLock(
|
|
VOID
|
|
)
|
|
{
|
|
LeaveCriticalSection( &_csRef );
|
|
}
|
|
|
|
BOOL
|
|
IsValid(
|
|
VOID
|
|
) const
|
|
{
|
|
return _fValid;
|
|
}
|
|
|
|
PSECURITY_DESCRIPTOR
|
|
GetSecDesc(
|
|
VOID
|
|
) const
|
|
{
|
|
return _hHandle->QuerySecDesc();
|
|
}
|
|
|
|
BOOL
|
|
SSICreateFileMapping(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Creates a mapping to a file
|
|
|
|
--*/
|
|
{
|
|
HANDLE hHandle;
|
|
|
|
if ( _hHandle->QueryFileBuffer() )
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
hHandle = _hHandle->QueryFileHandle();
|
|
if ( _hMapHandle != NULL )
|
|
{
|
|
if ( !SSICloseMapHandle() )
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
_hMapHandle = ::CreateFileMapping( hHandle,
|
|
NULL,
|
|
PAGE_READONLY,
|
|
0,
|
|
0,
|
|
NULL );
|
|
|
|
if ( _hMapHandle == NULL )
|
|
{
|
|
DBGPRINTF(( DBG_CONTEXT,
|
|
"CreateFileMapping failed with %d\n",
|
|
GetLastError() ));
|
|
}
|
|
|
|
return _hMapHandle != NULL;
|
|
}
|
|
|
|
BOOL
|
|
SSICloseMapHandle(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Closes mapping to a file
|
|
|
|
--*/
|
|
{
|
|
if ( _hMapHandle != NULL )
|
|
{
|
|
::CloseHandle( _hMapHandle );
|
|
_hMapHandle = NULL;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL
|
|
SSIMapViewOfFile(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Maps file to address
|
|
|
|
--*/
|
|
{
|
|
if ( _hHandle->QueryFileBuffer() )
|
|
{
|
|
_pvMappedBase = _hHandle->QueryFileBuffer();
|
|
return TRUE;
|
|
}
|
|
|
|
if ( _pvMappedBase != NULL )
|
|
{
|
|
if ( !SSIUnmapViewOfFile() )
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
_pvMappedBase = ::MapViewOfFile( _hMapHandle,
|
|
FILE_MAP_READ,
|
|
0,
|
|
0,
|
|
0 );
|
|
if ( _pvMappedBase == NULL )
|
|
{
|
|
DBGPRINTF(( DBG_CONTEXT,
|
|
"MapViewOfFile() failed with %d\n",
|
|
GetLastError() ));
|
|
}
|
|
return _pvMappedBase != NULL;
|
|
}
|
|
|
|
BOOL
|
|
SSIUnmapViewOfFile(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Unmaps file
|
|
|
|
--*/
|
|
{
|
|
if ( !_hHandle->QueryFileBuffer() && _pvMappedBase != NULL )
|
|
{
|
|
::UnmapViewOfFile( _pvMappedBase );
|
|
_pvMappedBase = NULL;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
DWORD
|
|
SSIGetFileAttributes(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Gets the attributes of a file
|
|
|
|
--*/
|
|
{
|
|
return _hHandle->QueryAttributes();
|
|
}
|
|
|
|
BOOL
|
|
SSIGetFileSize(
|
|
OUT DWORD * pdwLowWord,
|
|
OUT DWORD * pdwHighWord
|
|
)
|
|
/*++
|
|
|
|
Gets the size of the file.
|
|
|
|
--*/
|
|
{
|
|
LARGE_INTEGER liSize;
|
|
|
|
_hHandle->QuerySize( &liSize );
|
|
|
|
*pdwLowWord = liSize.LowPart;
|
|
*pdwHighWord = liSize.HighPart;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL
|
|
SSIGetLastModTime(
|
|
OUT FILETIME * ftTime
|
|
)
|
|
/*++
|
|
|
|
Gets the Last modification time of a file.
|
|
|
|
--*/
|
|
{
|
|
_hHandle->QueryLastWriteTime( ftTime );
|
|
return TRUE;
|
|
}
|
|
|
|
PVOID
|
|
GetMappedBase(
|
|
VOID
|
|
)
|
|
{
|
|
return _pvMappedBase;
|
|
}
|
|
|
|
STRU &
|
|
GetFilename(
|
|
VOID
|
|
)
|
|
{
|
|
return _strFilename;
|
|
}
|
|
};
|
|
|
|
// Class SSI_ELEMENT_ITEM
|
|
//
|
|
// Represents a SSI command or block of static text in the document
|
|
|
|
class SSI_ELEMENT_ITEM
|
|
{
|
|
private:
|
|
DWORD _Signature;
|
|
SSI_COMMANDS _ssiCmd;
|
|
SSI_TAGS _ssiTag;
|
|
STRA * _pstrTagValue;
|
|
|
|
DWORD _cbBegin; // Only used for Byte range command
|
|
DWORD _cbLength; // Only used for Byte range command
|
|
public:
|
|
|
|
LIST_ENTRY _ListEntry;
|
|
|
|
SSI_ELEMENT_ITEM( VOID )
|
|
: _ssiCmd ( SSI_CMD_UNKNOWN ),
|
|
_ssiTag ( SSI_TAG_UNKNOWN ),
|
|
_Signature( SIGNATURE_SEI ),
|
|
_pstrTagValue( NULL )
|
|
{
|
|
_ListEntry.Flink = NULL;
|
|
}
|
|
|
|
~SSI_ELEMENT_ITEM()
|
|
{
|
|
if ( _pstrTagValue != NULL )
|
|
{
|
|
delete _pstrTagValue;
|
|
}
|
|
|
|
DBG_ASSERT( _ListEntry.Flink == NULL );
|
|
|
|
_Signature = SIGNATURE_SEI_FREE;
|
|
}
|
|
|
|
VOID
|
|
SetByteRange(
|
|
IN DWORD cbBegin,
|
|
IN DWORD cbLength
|
|
)
|
|
{
|
|
_ssiCmd = SSI_CMD_BYTERANGE;
|
|
_cbBegin = cbBegin;
|
|
_cbLength = cbLength;
|
|
}
|
|
|
|
BOOL
|
|
SetCommand(
|
|
IN SSI_COMMANDS ssiCmd,
|
|
IN SSI_TAGS ssiTag,
|
|
IN CHAR * achTag
|
|
)
|
|
{
|
|
_ssiCmd = ssiCmd;
|
|
_ssiTag = ssiTag;
|
|
|
|
_pstrTagValue = new STRA();
|
|
if( _pstrTagValue == NULL )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if( FAILED( _pstrTagValue->Copy( achTag ) ) )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
SSI_COMMANDS
|
|
QueryCommand(
|
|
VOID
|
|
) const
|
|
{
|
|
return _ssiCmd;
|
|
}
|
|
|
|
SSI_TAGS
|
|
QueryTag(
|
|
VOID
|
|
) const
|
|
{
|
|
return _ssiTag;
|
|
}
|
|
|
|
STRA *
|
|
QueryTagValue(
|
|
VOID
|
|
) const
|
|
{
|
|
return _pstrTagValue;
|
|
}
|
|
|
|
BOOL
|
|
CheckSignature(
|
|
VOID
|
|
) const
|
|
{
|
|
return _Signature == SIGNATURE_SEI;
|
|
}
|
|
|
|
DWORD
|
|
QueryBegin(
|
|
VOID
|
|
) const
|
|
{
|
|
return _cbBegin;
|
|
}
|
|
|
|
DWORD
|
|
QueryLength(
|
|
VOID
|
|
) const
|
|
{
|
|
return _cbLength;
|
|
}
|
|
};
|
|
|
|
// Class SSI_ELEMENT_LIST
|
|
//
|
|
// This object sits as a cache blob under a file to be processed as a
|
|
// server side include. It represents an interpreted list of data
|
|
// elements that make up the file itself.
|
|
//
|
|
|
|
class SSI_ELEMENT_LIST : public ASSOCIATED_FILE_OBJECT
|
|
{
|
|
|
|
private:
|
|
DWORD _Signature;
|
|
LIST_ENTRY _ListHead;
|
|
|
|
//
|
|
// These are for tracking the memory mapped file
|
|
//
|
|
|
|
DWORD _cRefCount;
|
|
CRITICAL_SECTION _csRef;
|
|
|
|
//
|
|
// Provides the utilities needed to open/manipulate files
|
|
//
|
|
|
|
SSI_FILE * _pssiFile;
|
|
|
|
//
|
|
// Name of URL. Used to resolve FILE="xxx" filenames
|
|
//
|
|
|
|
STRU _strURL;
|
|
|
|
|
|
public:
|
|
SSI_ELEMENT_LIST()
|
|
: _Signature ( SIGNATURE_SEL ),
|
|
_pssiFile( NULL ),
|
|
_cRefCount( 0 )
|
|
{
|
|
InitializeListHead( &_ListHead );
|
|
INITIALIZE_CRITICAL_SECTION( &_csRef );
|
|
}
|
|
|
|
~SSI_ELEMENT_LIST()
|
|
{
|
|
SSI_ELEMENT_ITEM * pSEI;
|
|
|
|
while ( !IsListEmpty( &_ListHead ))
|
|
{
|
|
pSEI = CONTAINING_RECORD( _ListHead.Flink,
|
|
SSI_ELEMENT_ITEM,
|
|
_ListEntry );
|
|
|
|
RemoveEntryList( &pSEI->_ListEntry );
|
|
pSEI->_ListEntry.Flink = NULL;
|
|
delete pSEI;
|
|
}
|
|
UnMap();
|
|
|
|
if( _pssiFile != NULL )
|
|
{
|
|
delete _pssiFile;
|
|
}
|
|
|
|
DeleteCriticalSection( &_csRef );
|
|
_Signature = SIGNATURE_SEL_FREE;
|
|
}
|
|
|
|
VOID
|
|
Cleanup(
|
|
VOID
|
|
)
|
|
{
|
|
delete this;
|
|
}
|
|
|
|
LIST_ENTRY *
|
|
QueryListHead(
|
|
VOID
|
|
)
|
|
{
|
|
return &_ListHead;
|
|
};
|
|
|
|
BOOL
|
|
AppendByteRange(
|
|
IN DWORD cbStart,
|
|
IN DWORD cbLength
|
|
)
|
|
{
|
|
SSI_ELEMENT_ITEM * pSEI;
|
|
|
|
pSEI = new SSI_ELEMENT_ITEM;
|
|
|
|
if ( !pSEI )
|
|
{
|
|
SetLastError( ERROR_NOT_ENOUGH_MEMORY );
|
|
return FALSE;
|
|
}
|
|
|
|
pSEI->SetByteRange( cbStart,
|
|
cbLength );
|
|
AppendItem( pSEI );
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL
|
|
AppendCommand(
|
|
IN SSI_COMMANDS ssiCmd,
|
|
IN SSI_TAGS ssiTag,
|
|
IN CHAR * pszTag
|
|
)
|
|
{
|
|
SSI_ELEMENT_ITEM * pSEI;
|
|
|
|
pSEI = new SSI_ELEMENT_ITEM;
|
|
|
|
if ( !pSEI )
|
|
{
|
|
SetLastError( ERROR_NOT_ENOUGH_MEMORY );
|
|
return FALSE;
|
|
}
|
|
|
|
if ( !pSEI->SetCommand( ssiCmd,
|
|
ssiTag,
|
|
pszTag ))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
AppendItem( pSEI );
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
VOID
|
|
AppendItem(
|
|
IN SSI_ELEMENT_ITEM * pSEI
|
|
)
|
|
{
|
|
InsertTailList( &_ListHead,
|
|
&pSEI->_ListEntry );
|
|
}
|
|
|
|
CHAR * QueryData(
|
|
VOID
|
|
) const
|
|
{
|
|
return ( CHAR * ) _pssiFile->GetMappedBase();
|
|
}
|
|
|
|
PSECURITY_DESCRIPTOR
|
|
QuerySecDesc(
|
|
VOID
|
|
)
|
|
{
|
|
return _pssiFile->GetSecDesc();
|
|
}
|
|
|
|
BOOL
|
|
CheckSignature(
|
|
VOID
|
|
) const
|
|
{
|
|
return _Signature == SIGNATURE_SEL;
|
|
}
|
|
|
|
VOID
|
|
Lock(
|
|
VOID
|
|
)
|
|
{
|
|
EnterCriticalSection( &_csRef );
|
|
}
|
|
|
|
VOID
|
|
UnLock(
|
|
VOID
|
|
)
|
|
{
|
|
LeaveCriticalSection( &_csRef );
|
|
}
|
|
|
|
BOOL
|
|
UnMap(
|
|
VOID
|
|
)
|
|
{
|
|
Lock();
|
|
if ( _cRefCount && !--_cRefCount )
|
|
{
|
|
DBG_REQUIRE( _pssiFile->SSIUnmapViewOfFile() );
|
|
DBG_REQUIRE( _pssiFile->SSICloseMapHandle() );
|
|
}
|
|
UnLock();
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL
|
|
Map(
|
|
VOID
|
|
)
|
|
{
|
|
Lock();
|
|
if ( _cRefCount++ == 0 )
|
|
{
|
|
if ( !_pssiFile->SSICreateFileMapping() )
|
|
{
|
|
UnLock();
|
|
return FALSE;
|
|
}
|
|
if ( !_pssiFile->SSIMapViewOfFile() )
|
|
{
|
|
UnMap();
|
|
UnLock();
|
|
return FALSE;
|
|
}
|
|
}
|
|
UnLock();
|
|
return TRUE;
|
|
}
|
|
|
|
VOID
|
|
SetFile(
|
|
IN SSI_FILE * pssiFile
|
|
)
|
|
{
|
|
_pssiFile = pssiFile;
|
|
}
|
|
|
|
SSI_FILE *
|
|
GetFile(
|
|
VOID
|
|
)
|
|
{
|
|
return _pssiFile;
|
|
}
|
|
|
|
HRESULT
|
|
SetURL(
|
|
IN STRU * pstrURL
|
|
)
|
|
{
|
|
return _strURL.Copy( pstrURL->QueryStr() );
|
|
}
|
|
|
|
|
|
static
|
|
BOOL
|
|
SSIFreeContextRoutine(
|
|
VOID * pvContext
|
|
)
|
|
{
|
|
DBG_ASSERT( ( ( SSI_ELEMENT_LIST * ) pvContext) -> CheckSignature() );
|
|
|
|
delete ( SSI_ELEMENT_LIST * ) pvContext;
|
|
return TRUE;
|
|
}
|
|
};
|
|
|
|
//
|
|
// SSI_REQUEST methods implementation
|
|
//
|
|
|
|
//static
|
|
ALLOC_CACHE_HANDLER * SSI_REQUEST::sm_pachSSIRequests = NULL;
|
|
|
|
SSI_REQUEST::SSI_REQUEST( EXTENSION_CONTROL_BLOCK * pECB )
|
|
: _pECB( pECB ),
|
|
_fBaseFile( TRUE ),
|
|
_fValid( FALSE ),
|
|
_pchErrorBuff ( NULL )
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Constructor
|
|
|
|
--*/
|
|
|
|
{
|
|
DBG_ASSERT( _pECB != NULL );
|
|
STACK_BUFFER (buffTemp, 512);
|
|
DWORD cbSize = buffTemp.QuerySize();
|
|
|
|
if (!_pECB->GetServerVariable(_pECB->ConnID,
|
|
"UNICODE_URL",
|
|
buffTemp.QueryPtr(),
|
|
&cbSize))
|
|
{
|
|
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER ||
|
|
!buffTemp.Resize(cbSize))
|
|
{
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Now, we should have enough buffer, try again
|
|
//
|
|
if (!_pECB->GetServerVariable(_pECB->ConnID,
|
|
"UNICODE_URL",
|
|
buffTemp.QueryPtr(),
|
|
&cbSize))
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
if (FAILED(_strURL.Copy( (LPWSTR)buffTemp.QueryPtr() )))
|
|
{
|
|
return;
|
|
}
|
|
|
|
cbSize = buffTemp.QuerySize();
|
|
if (!_pECB->GetServerVariable(_pECB->ConnID,
|
|
"UNICODE_SCRIPT_TRANSLATED",
|
|
buffTemp.QueryPtr(),
|
|
&cbSize))
|
|
{
|
|
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER ||
|
|
!buffTemp.Resize(cbSize))
|
|
{
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Now, we should have enough buffer, try again
|
|
//
|
|
if (!_pECB->GetServerVariable(_pECB->ConnID,
|
|
"UNICODE_SCRIPT_TRANSLATED",
|
|
buffTemp.QueryPtr(),
|
|
&cbSize))
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
if (FAILED(_strFilename.Copy( (LPWSTR)buffTemp.QueryPtr() )))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ( !_pECB->ServerSupportFunction(
|
|
_pECB->ConnID,
|
|
HSE_REQ_GET_IMPERSONATION_TOKEN,
|
|
&_hUser,
|
|
NULL,
|
|
NULL ) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
_pSIF = new SSI_INCLUDE_FILE ( this, _strFilename, _strURL, NULL );
|
|
|
|
|
|
if ( _pSIF == NULL || !_pSIF->IsValid() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if( !_IOBuffer.Resize(MAX_PATH) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if( FAILED( _IOString.Resize(MAX_PATH) ) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
|
|
_fValid = TRUE;
|
|
}
|
|
|
|
SSI_REQUEST::~SSI_REQUEST()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Destructor
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
if ( _pSIF != NULL )
|
|
{
|
|
//
|
|
// There should be no nested stm files
|
|
// State Machine was supposed to complete and cleanup child SSI_INCLUDE_FILES
|
|
// (also in the case of error)
|
|
//
|
|
|
|
DBG_ASSERT( _pSIF->GetParent() == NULL );
|
|
delete _pSIF;
|
|
}
|
|
|
|
if ( _pchErrorBuff != NULL )
|
|
{
|
|
::LocalFree( ( VOID * )_pchErrorBuff );
|
|
}
|
|
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::SSISendError(
|
|
IN DWORD dwMessageID,
|
|
IN LPSTR apszParms[]
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Send an SSI error
|
|
|
|
Arguments:
|
|
|
|
dwMessageId - Message ID
|
|
apszParms - Array of parameters
|
|
|
|
Return Value:
|
|
|
|
HRESULT (if couldn't find a custom error, this will fail)
|
|
|
|
--*/
|
|
{
|
|
DWORD cbSent;
|
|
|
|
if ( *( ( CHAR * )_strUserMessage.QueryStr() ) != '\0' )
|
|
{
|
|
//
|
|
// user specified message with #CONFIG ERRMSG=
|
|
//
|
|
|
|
return WriteToClient( _strUserMessage.QueryStr(),
|
|
_strUserMessage.QueryCCH(),
|
|
&cbSent );
|
|
}
|
|
else
|
|
{
|
|
DWORD cch;
|
|
HRESULT hr = E_FAIL;
|
|
CHAR chTemp1 = '\0';
|
|
CHAR chTemp2 = '\0';
|
|
|
|
if( _pchErrorBuff )
|
|
{
|
|
::LocalFree( ( VOID * )_pchErrorBuff );
|
|
_pchErrorBuff = NULL;
|
|
}
|
|
|
|
//
|
|
// Lame. I need to validate the parameters for being <1024
|
|
// otherwise FormatMessage uses SEH to determine when to resize and
|
|
// this will cause debuggers to break on the 1st change exception
|
|
//
|
|
|
|
if ( apszParms[ 0 ] != NULL &&
|
|
strlen( apszParms[ 0 ] ) > SSI_MAX_FORMAT_LEN )
|
|
{
|
|
chTemp1 = apszParms[ 0 ][ SSI_MAX_FORMAT_LEN ];
|
|
apszParms[ 0 ][ SSI_MAX_FORMAT_LEN ] = '\0';
|
|
}
|
|
|
|
if ( apszParms[ 1 ] != NULL &&
|
|
strlen( apszParms[ 1 ] ) > SSI_MAX_FORMAT_LEN )
|
|
{
|
|
chTemp2 = apszParms[ 1 ][ SSI_MAX_FORMAT_LEN ];
|
|
apszParms[ 1 ][ SSI_MAX_FORMAT_LEN ] = '\0';
|
|
}
|
|
|
|
cch = ::FormatMessageA( FORMAT_MESSAGE_ARGUMENT_ARRAY |
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_FROM_HMODULE,
|
|
GetModuleHandle( SSI_DLL_NAME ),
|
|
dwMessageID,
|
|
0,
|
|
( LPSTR ) &_pchErrorBuff,
|
|
0,
|
|
( va_list *) apszParms );
|
|
|
|
if ( chTemp1 != '\0' )
|
|
{
|
|
apszParms[ 0 ][ SSI_MAX_FORMAT_LEN ] = chTemp1;
|
|
}
|
|
|
|
if ( chTemp2 != '\0' )
|
|
{
|
|
apszParms[ 1 ][ SSI_MAX_FORMAT_LEN ] = chTemp2;
|
|
}
|
|
|
|
if( cch != 0 )
|
|
{
|
|
//
|
|
// WriteToClient will execute asynchronously so do not
|
|
// free _pchErrorBuffer before I/O completion
|
|
//
|
|
|
|
hr = WriteToClient( _pchErrorBuff,
|
|
cch,
|
|
&cbSent );
|
|
return hr;
|
|
}
|
|
else
|
|
{
|
|
return HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
}
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::SendCustomError(
|
|
HSE_CUSTOM_ERROR_INFO * pCustomErrorInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Try to have IIS send custom error on our behalf
|
|
|
|
Arguments:
|
|
|
|
pCustomErrorInfo - Describes custom error
|
|
|
|
Return Value:
|
|
|
|
HRESULT (if couldn't find a custom error, this will fail)
|
|
|
|
--*/
|
|
{
|
|
BOOL fRet;
|
|
|
|
fRet = _pECB->ServerSupportFunction( _pECB->ConnID,
|
|
HSE_REQ_SEND_CUSTOM_ERROR,
|
|
pCustomErrorInfo,
|
|
NULL,
|
|
NULL );
|
|
if ( !fRet )
|
|
{
|
|
return HRESULT_FROM_WIN32( GetLastError() );
|
|
}
|
|
else
|
|
{
|
|
return NO_ERROR;
|
|
}
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoFLastMod(
|
|
IN STRU * pstrFilename,
|
|
IN STRA * pstrTimeFmt,
|
|
IN SSI_ELEMENT_LIST * pList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Send the LastModTime of file to HTML stream
|
|
|
|
Arguments:
|
|
|
|
pstrFilename - Filename
|
|
pstrTimeFmt - Format of time -> follows strftime() convention
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
FILETIME ftTime;
|
|
FILETIME ftLocalTime;
|
|
SYSTEMTIME sysLocal;
|
|
|
|
if ( ( NULL == pList ) ||
|
|
wcscmp( pstrFilename->QueryStr(),
|
|
( pList->GetFile()->GetFilename().QueryStr() ) ) )
|
|
{
|
|
SSI_FILE ssiFile( pstrFilename, GetUserToken() );
|
|
|
|
if ( !ssiFile.IsValid() ||
|
|
( !ssiFile.SSIGetLastModTime( &ftTime ))
|
|
)
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pList->GetFile()->SSIGetLastModTime( &ftTime );
|
|
}
|
|
|
|
if ( ( !FileTimeToLocalFileTime( &ftTime, &ftLocalTime ) ) ||
|
|
( !FileTimeToSystemTime( &ftLocalTime, &sysLocal ) ) )
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
return SendDate( &sysLocal,
|
|
pstrTimeFmt );
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::SendDate(
|
|
IN SYSTEMTIME * psysTime,
|
|
IN STRA * pstrTimeFmt
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends a SYSTEMTIME in appropriate format to HTML stream
|
|
|
|
Arguments:
|
|
|
|
psysTime - SYSTEMTIME containing time to send
|
|
pstrTimeFmt - Format of time (follows strftime() convention)
|
|
fCalcDays - TRUE if days since the beginning of the year should be
|
|
calculated
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
struct tm tm;
|
|
|
|
// Convert SYSTEMTIME to 'struct tm'
|
|
|
|
tm.tm_sec = psysTime->wSecond;
|
|
tm.tm_min = psysTime->wMinute;
|
|
tm.tm_hour = psysTime->wHour;
|
|
tm.tm_mday = psysTime->wDay;
|
|
tm.tm_mon = psysTime->wMonth - 1;
|
|
tm.tm_year = psysTime->wYear - 1900;
|
|
tm.tm_wday = psysTime->wDayOfWeek;
|
|
tm.tm_yday = g_MonthToDayCount[ tm.tm_mon ] + tm.tm_mday - 1;
|
|
|
|
//
|
|
// Adjust for leap year - note that we do not handle 2100
|
|
//
|
|
|
|
if ( ( tm.tm_mon ) > 1 && !( psysTime->wYear & 3 ) )
|
|
{
|
|
++tm.tm_yday;
|
|
}
|
|
|
|
tm.tm_isdst = -1; // Daylight savings time flag - have crt compute
|
|
|
|
_cbTimeBufferLen = strftime( _achTimeBuffer,
|
|
SSI_MAX_TIME_SIZE + 1,
|
|
( CHAR * )pstrTimeFmt->QueryStr(),
|
|
&tm );
|
|
|
|
if ( _cbTimeBufferLen == 0 )
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
return WriteToClient( _achTimeBuffer,
|
|
_cbTimeBufferLen,
|
|
&_cbTimeBufferLen );
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::LookupVirtualRoot( IN WCHAR * pszVirtual,
|
|
OUT STRU * pstrPhysical,
|
|
IN DWORD dwAccess )
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Lookup the given virtual path. Optionally ensure that its access
|
|
flags are valid for the require request.
|
|
|
|
Arguments:
|
|
|
|
pszVirtual = Virtual path to lookup
|
|
pstrPhysical = Contains the physical path
|
|
dwAccess = Access flags required for a valid request
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
HSE_URL_MAPEX_INFO URLMap;
|
|
DWORD dwMask;
|
|
STACK_STRA (strURL, 128);
|
|
HRESULT hr = E_FAIL;
|
|
|
|
//
|
|
// ServerSupportFunction doesn't accept unicode strings. Convert
|
|
//
|
|
|
|
if ( FAILED(hr = strURL.CopyW(pszVirtual)))
|
|
{
|
|
return hr;
|
|
|
|
}
|
|
|
|
if ( !_pECB->ServerSupportFunction( _pECB->ConnID,
|
|
HSE_REQ_MAP_URL_TO_PATH_EX,
|
|
strURL.QueryStr(),
|
|
NULL,
|
|
(PDWORD) &URLMap ) )
|
|
{
|
|
return HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
|
|
dwMask = URLMap.dwFlags;
|
|
|
|
if ( dwAccess & HSE_URL_FLAGS_READ )
|
|
{
|
|
//
|
|
// BUGBUG-MING: Add IsSecurePort()
|
|
//
|
|
/*
|
|
if ( !( dwMask & HSE_URL_FLAGS_READ ) ||
|
|
( ( dwMask & HSE_URL_FLAGS_SSL ) &&
|
|
!_pReq->IsSecurePort() ) )
|
|
*/
|
|
|
|
if ( !( dwMask & HSE_URL_FLAGS_READ ) ||
|
|
( dwMask & HSE_URL_FLAGS_SSL ) )
|
|
{
|
|
return HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
|
|
}
|
|
}
|
|
|
|
if( FAILED( pstrPhysical->CopyA( URLMap.lpszPath ) ) )
|
|
{
|
|
return HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoEchoISAPIVariable(
|
|
IN STRA * pstrVariable
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get ISAPI variable and if successful, send it to HTML stream
|
|
|
|
Arguments:
|
|
|
|
pstrVariable - Variable
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if ( FAILED( hr = GetVariable( ( CHAR * )pstrVariable->QueryStr(),
|
|
&_IOBuffer ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return WriteToClient( _IOBuffer.QueryPtr(),
|
|
strlen( ( CHAR * )_IOBuffer.QueryPtr() ),
|
|
&_cbIOLen );
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoEchoDateLocal(
|
|
IN STRA * pstrTimeFmt
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends local time (#ECHO VAR="DATE_LOCAL")
|
|
|
|
Arguments:
|
|
|
|
pstrTimefmt - Format of time (follows strftime() convention)
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
SYSTEMTIME sysTime;
|
|
|
|
::GetLocalTime( &sysTime );
|
|
return SendDate( &sysTime,
|
|
pstrTimeFmt );
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoEchoDateGMT(
|
|
IN STRA * pstrTimeFmt
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends GMT time (#ECHO VAR="DATE_GMT")
|
|
|
|
Arguments:
|
|
|
|
pstrTimefmt - Format of time (follows strftime() convention)
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
SYSTEMTIME sysTime;
|
|
|
|
::GetSystemTime( &sysTime );
|
|
return SendDate( &sysTime,
|
|
pstrTimeFmt );
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoEchoDocumentName(
|
|
IN STRU * pstrFilename
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends filename of current SSI document (#ECHO VAR="DOCUMENT_NAME")
|
|
|
|
Arguments:
|
|
|
|
pstrFilename - filename to print
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if ( FAILED( hr = (_IOString.CopyW(pstrFilename->QueryStr() ) ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return WriteToClient( _IOString.QueryStr(),
|
|
_IOString.QueryCCH(),
|
|
&_cbIOLen);
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoEchoDocumentURI(
|
|
IN STRU * pstrURL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends URL of current SSI document (#ECHO VAR="DOCUMENT_URI")
|
|
|
|
Arguments:
|
|
|
|
pstrURL - URL to print
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
--*/
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if ( FAILED( hr = _IOString.CopyW(pstrURL->QueryStr() ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return WriteToClient( _IOString.QueryStr(),
|
|
_IOString.QueryCCH(),
|
|
&_cbIOLen );
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoEchoQueryStringUnescaped(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends unescaped querystring to HTML stream (#ECHO VAR="QUERY_STRING_UNESCAPED")
|
|
|
|
Arguments:
|
|
|
|
none
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if ( FAILED( hr = _IOString.Copy( _pECB->lpszQueryString ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if ( FAILED( hr = _IOString.Unescape()) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return WriteToClient( _IOString.QueryStr(),
|
|
_IOString.QueryCCH(),
|
|
&_cbIOLen );
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoEchoLastModified(
|
|
IN STRU * pstrFilename,
|
|
IN STRA * pstrTimeFmt,
|
|
IN SSI_ELEMENT_LIST * pList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends LastModTime of current document (#ECHO VAR="LAST_MODIFIED")
|
|
|
|
Arguments:
|
|
|
|
pstrFilename - Filename of current SSI document
|
|
pstrTimeFmt - Time format (follows strftime() convention)
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
return DoFLastMod( pstrFilename,
|
|
pstrTimeFmt,
|
|
pList);
|
|
}
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoFSize(
|
|
IN STRU * pstrFilename,
|
|
IN BOOL bSizeFmtBytes,
|
|
IN SSI_ELEMENT_LIST * pList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sends file size of file to HTML stream
|
|
|
|
Arguments:
|
|
|
|
pstrfilename - Filename
|
|
bSizeFmtBytes - TRUE if count is in Bytes, FALSE if in KBytes
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
BOOL bRet;
|
|
DWORD cbSizeLow;
|
|
DWORD cbSizeHigh;
|
|
WCHAR achInputNumber[ SSI_MAX_NUMBER_STRING + 1 ];
|
|
WCHAR achOutputNumber[ SSI_MAX_NUMBER_STRING + 1 ];
|
|
NUMBERFMT nfNumberFormat;
|
|
int iOutputSize;
|
|
DWORD dwActualLen;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if ( ( NULL == pList ) ||
|
|
wcscmp( pstrFilename->QueryStr(),
|
|
( pList->GetFile()->GetFilename().QueryStr() ) ) )
|
|
{
|
|
SSI_FILE ssiFile( pstrFilename, GetUserToken() );
|
|
|
|
if ( !ssiFile.IsValid() ||
|
|
( !ssiFile.SSIGetFileSize( &cbSizeLow,
|
|
&cbSizeHigh ) )
|
|
)
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!pList->GetFile()->SSIGetFileSize( &cbSizeLow,
|
|
&cbSizeHigh ) )
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
|
|
if ( cbSizeHigh )
|
|
{
|
|
//BUGBUG-jaro: do we ignore extra large files intentionaly?
|
|
return E_FAIL;
|
|
}
|
|
|
|
if ( !bSizeFmtBytes )
|
|
{
|
|
// express in terms of KB
|
|
cbSizeLow /= 1000;
|
|
}
|
|
|
|
nfNumberFormat.NumDigits = 0;
|
|
nfNumberFormat.LeadingZero = 0;
|
|
nfNumberFormat.Grouping = 3;
|
|
nfNumberFormat.lpThousandSep = L",";
|
|
nfNumberFormat.lpDecimalSep = L".";
|
|
nfNumberFormat.NegativeOrder = 2;
|
|
|
|
_snwprintf( achInputNumber,
|
|
SSI_MAX_NUMBER_STRING + 1,
|
|
L"%ld",
|
|
cbSizeLow );
|
|
|
|
iOutputSize = GetNumberFormat( LOCALE_SYSTEM_DEFAULT,
|
|
0,
|
|
achInputNumber,
|
|
&nfNumberFormat,
|
|
achOutputNumber,
|
|
SSI_MAX_NUMBER_STRING + 1 );
|
|
if ( !iOutputSize )
|
|
{
|
|
return HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
|
|
//
|
|
// Do not count trailing '\0'
|
|
//
|
|
|
|
iOutputSize--;
|
|
|
|
//
|
|
// Convert from Unicode before sending to client
|
|
//
|
|
|
|
if ( FAILED( hr = _IOString.CopyW(achOutputNumber) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
|
|
return WriteToClient( _IOString.QueryStr(),
|
|
iOutputSize,
|
|
&_cbIOLen );
|
|
}
|
|
|
|
|
|
HRESULT
|
|
SSI_REQUEST::PrepareSSI(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Prepare esential data structures
|
|
|
|
Arguments:
|
|
|
|
none
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
return _pSIF->Prepare();
|
|
}
|
|
|
|
|
|
HRESULT
|
|
SSI_REQUEST::DoWork(
|
|
DWORD dwError
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is the top level routine for retrieving a server side include
|
|
file.
|
|
|
|
Arguments:
|
|
|
|
dwError - error of the last asynchronous operation (the one received by completion routine)
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
|
|
DBG_ASSERT ( _pSIF != NULL );
|
|
|
|
while( _pSIF != NULL )
|
|
{
|
|
//
|
|
// In the case that dwError != NO_ERROR
|
|
// _pSIF->DoWork() may be called multiple times to unwind state machine
|
|
// We will pass the same dwError in the case of multiple calls
|
|
// since that error is the primary reason why processing of this request
|
|
// must finish
|
|
//
|
|
hr = _pSIF->DoWork(dwError);
|
|
if ( hr == HRESULT_FROM_WIN32(ERROR_IO_PENDING) )
|
|
{
|
|
//
|
|
// If there is pending IO return to caller
|
|
//
|
|
return hr;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Either SSI_INCLUDE_FILE processing completed
|
|
// or there is nested include
|
|
//
|
|
// In the case of error this block is used to unwind state machine
|
|
//
|
|
if ( _pSIF->IsCompleted() )
|
|
{
|
|
//
|
|
// SSI_INCLUDE_FILE processing completed
|
|
// Cleanup and if there is parent SSI_INCLUDE_FILE continue with that one
|
|
//
|
|
|
|
SSI_INCLUDE_FILE * pParent = _pSIF->GetParent();
|
|
delete _pSIF;
|
|
_pSIF = pParent;
|
|
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Current SSI_INCLUDE_FILE _pSIF hasn't been completed yet. Continue
|
|
//
|
|
}
|
|
}
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//static
|
|
VOID WINAPI
|
|
SSI_REQUEST::HseIoCompletion(
|
|
IN EXTENSION_CONTROL_BLOCK * pECB,
|
|
IN PVOID pContext,
|
|
IN DWORD cbIO,
|
|
IN DWORD dwError
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is the callback function for handling completions of asynchronous IO.
|
|
This function performs necessary cleanup and resubmits additional IO
|
|
(if required).
|
|
|
|
Arguments:
|
|
|
|
pecb pointer to ECB containing parameters related to the request.
|
|
pContext context information supplied with the asynchronous IO call.
|
|
cbIO count of bytes of IO in the last call.
|
|
dwError Error if any, for the last IO operation.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
--*/
|
|
{
|
|
SSI_REQUEST * pRequest = (SSI_REQUEST *) pContext;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
//
|
|
// Continue processing SSI file
|
|
//
|
|
|
|
hr = pRequest->DoWork(dwError);
|
|
if ( hr == HRESULT_FROM_WIN32(ERROR_IO_PENDING) )
|
|
{
|
|
//
|
|
// pending IO operation
|
|
//
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Processing of current SSI request completed
|
|
// Do Cleanup
|
|
//
|
|
|
|
delete pRequest;
|
|
|
|
//
|
|
// Notify IIS that we are done with processing
|
|
//
|
|
|
|
pECB->ServerSupportFunction( pECB->ConnID,
|
|
HSE_REQ_DONE_WITH_SESSION,
|
|
NULL,
|
|
NULL,
|
|
NULL);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
//
|
|
// SSI_INCLUDE_FILE methods implementation
|
|
//
|
|
|
|
//static
|
|
ALLOC_CACHE_HANDLER * SSI_INCLUDE_FILE::sm_pachSSI_IncludeFiles = NULL;
|
|
|
|
|
|
SSI_INCLUDE_FILE::~SSI_INCLUDE_FILE( VOID )
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Destructor
|
|
|
|
--*/
|
|
|
|
{
|
|
if ( !_fSELCached && _pSEL )
|
|
{
|
|
delete _pSEL;
|
|
_pSEL = NULL;
|
|
}
|
|
|
|
if ( _pOpenFile )
|
|
{
|
|
_pOpenFile->DereferenceCacheEntry();
|
|
_pOpenFile = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
HRESULT
|
|
SSI_INCLUDE_FILE::Prepare( VOID )
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method builds the Server Side Include Element List the first
|
|
time a .stm file is sent. Subsequently, the element list is
|
|
checked out from the associated cache blob.
|
|
|
|
Note: The HTTP headers have already been sent at this point so for
|
|
any subsequent non-catastrophic errors, we have to insert them into
|
|
the output stream.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
DWORD dwError;
|
|
LPSTR apszParms[ 2 ] = { NULL, NULL };
|
|
CHAR pszNumBuf[ SSI_MAX_NUMBER_STRING ];
|
|
FILE_CACHE_USER fileUser;
|
|
HSE_CUSTOM_ERROR_INFO customErrorInfo;
|
|
|
|
DBG_ASSERT( _State == SIF_STATE_INITIALIZED );
|
|
|
|
DBG_ASSERT( _pRequest != NULL );
|
|
|
|
fileUser._hToken = _pRequest->GetUserToken();
|
|
|
|
hr = g_pFileCache->GetFileInfo( _strFilename,
|
|
NULL,
|
|
&fileUser,
|
|
TRUE,
|
|
&_pOpenFile );
|
|
if ( FAILED( hr ) )
|
|
{
|
|
goto failed;
|
|
}
|
|
|
|
//
|
|
// The source file is in the cache. Check whether we have
|
|
// associated a SSI_ELEMENT_LIST with it.
|
|
//
|
|
|
|
_fFileCached = _pOpenFile->QueryFileBuffer() != NULL;
|
|
|
|
_pSEL = ( SSI_ELEMENT_LIST * )_pOpenFile->QueryAssociatedObject();
|
|
if ( _pSEL )
|
|
{
|
|
_fSELCached = TRUE;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// build SSI_ELEMENT_LIST _pSEL
|
|
//
|
|
hr = BuildSEL();
|
|
if ( FAILED( hr ) )
|
|
{
|
|
goto failed;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Only bother to cache SEL if the file is cached
|
|
//
|
|
|
|
if ( !_fSELCached && _fFileCached )
|
|
{
|
|
if ( !_pOpenFile->SetAssociatedObject( _pSEL ) )
|
|
{
|
|
delete _pSEL;
|
|
_pSEL = (SSI_ELEMENT_LIST*) _pOpenFile->QueryAssociatedObject();
|
|
if ( !_pSEL )
|
|
{
|
|
DBG_ASSERT( FALSE );
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
else
|
|
{
|
|
_fSELCached = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_fSELCached = TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// adjust State
|
|
//
|
|
|
|
SetState( SIF_STATE_READY );
|
|
|
|
//
|
|
// If we got this far and this is the base file, we can send the
|
|
// 200 OK
|
|
//
|
|
|
|
if ( IsBaseFile() )
|
|
{
|
|
return _pRequest->SendResponseHeader( NULL,
|
|
SSI_HEADER );
|
|
|
|
}
|
|
else
|
|
{
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
failed:
|
|
dwError = WIN32_FROM_HRESULT(hr);
|
|
|
|
if ( IsBaseFile() )
|
|
{
|
|
//
|
|
// First try to have IIS send custom error
|
|
//
|
|
|
|
switch( dwError )
|
|
{
|
|
case ERROR_ACCESS_DENIED:
|
|
|
|
customErrorInfo.pszStatus = "401 Access Denied";
|
|
customErrorInfo.uHttpSubError = MD_ERROR_SUB401_LOGON_ACL;
|
|
customErrorInfo.fAsync = FALSE;
|
|
|
|
hr = _pRequest->SendCustomError( &customErrorInfo );
|
|
|
|
break;
|
|
case ERROR_FILE_NOT_FOUND:
|
|
case ERROR_PATH_NOT_FOUND:
|
|
|
|
customErrorInfo.pszStatus = "404 Object Not Found";
|
|
customErrorInfo.uHttpSubError = 0;
|
|
customErrorInfo.fAsync = FALSE;
|
|
|
|
hr = _pRequest->SendCustomError( &customErrorInfo );
|
|
|
|
break;
|
|
|
|
default:
|
|
hr = HRESULT_FROM_WIN32( ERROR_FILE_NOT_FOUND );
|
|
}
|
|
|
|
//
|
|
// If IIS could not send custom error, then send our own legacy
|
|
// error response
|
|
//
|
|
|
|
if ( hr == HRESULT_FROM_WIN32( ERROR_FILE_NOT_FOUND ) )
|
|
{
|
|
switch( dwError )
|
|
{
|
|
case ERROR_ACCESS_DENIED:
|
|
_pRequest->SendResponseHeader( SSI_ACCESS_DENIED,
|
|
SSI_HEADER
|
|
"<body><h1>"
|
|
SSI_ACCESS_DENIED
|
|
"</h1></body>" );
|
|
break;
|
|
case ERROR_FILE_NOT_FOUND:
|
|
case ERROR_PATH_NOT_FOUND:
|
|
_pRequest->SendResponseHeader( SSI_OBJECT_NOT_FOUND,
|
|
SSI_HEADER
|
|
"<body><h1>"
|
|
SSI_OBJECT_NOT_FOUND
|
|
"</h1></body>" );
|
|
break;
|
|
default:
|
|
|
|
STACK_STRA (strURL, 128);
|
|
if (FAILED( hr = strURL.CopyW(_strURL.QueryStr())))
|
|
{
|
|
break;
|
|
}
|
|
|
|
_pRequest->SendResponseHeader( NULL,
|
|
SSI_HEADER );
|
|
|
|
_ultoa( dwError, pszNumBuf, 10 );
|
|
apszParms[ 0 ] = strURL.QueryStr();
|
|
apszParms[ 1 ] = pszNumBuf;
|
|
|
|
hr = _pRequest->SSISendError( SSINCMSG_ERROR_HANDLING_FILE,
|
|
apszParms );
|
|
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
STACK_STRA (strURL, 128);
|
|
if (SUCCEEDED( hr = strURL.CopyW(_strURL.QueryStr())))
|
|
{
|
|
_ultoa( dwError, pszNumBuf, 10 );
|
|
apszParms[ 0 ] = strURL.QueryStr();
|
|
apszParms[ 1 ] = pszNumBuf;
|
|
|
|
hr = _pRequest->SSISendError( SSINCMSG_ERROR_HANDLING_FILE,
|
|
apszParms );
|
|
}
|
|
}
|
|
|
|
//
|
|
// In the case of error set state to Completed
|
|
//
|
|
SetState( SIF_STATE_COMPLETED );
|
|
|
|
return ( ( FAILED(hr) ) ? hr : E_FAIL);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
SSI_INCLUDE_FILE::BuildSEL( VOID )
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method opens and parses the specified server side include file and
|
|
builds SSI_ELEMENT_LIST (SEL)
|
|
|
|
Note: The HTTP headers have already been sent at this point so for any
|
|
subsequent non-catastrophic errors, we have to insert them into the output
|
|
stream.
|
|
|
|
We keep the file open but that's ok because if a change dir notification
|
|
occurs, the cache blob will get decached at which point we will close
|
|
all of our open handles.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
SSI_FILE * pssiFile = NULL;
|
|
SSI_ELEMENT_LIST * pSEL = NULL;
|
|
CHAR * pchBeginRange = NULL;
|
|
CHAR * pchFilePos = NULL;
|
|
CHAR * pchBeginFile = NULL;
|
|
CHAR * pchEOF = NULL;
|
|
DWORD cbSizeLow, cbSizeHigh;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
//
|
|
// Create the element list
|
|
//
|
|
|
|
_pSEL = new SSI_ELEMENT_LIST;
|
|
|
|
if ( _pSEL == NULL )
|
|
{
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
|
|
//
|
|
// Set the URL (to be used in calculating FILE="xxx" paths
|
|
//
|
|
|
|
_pSEL->SetURL( &_strURL );
|
|
|
|
//
|
|
// Open the file
|
|
//
|
|
|
|
pssiFile = new SSI_FILE( &_strFilename, _pOpenFile );
|
|
if ( pssiFile == NULL || !pssiFile->IsValid() )
|
|
{
|
|
if (pssiFile)
|
|
{
|
|
delete pssiFile;
|
|
pssiFile = NULL;
|
|
}
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
|
|
_pSEL->SetFile( pssiFile );
|
|
|
|
//
|
|
// Make sure a parent doesn't try and include a directory
|
|
//
|
|
|
|
if ( pssiFile->SSIGetFileAttributes() & FILE_ATTRIBUTE_DIRECTORY )
|
|
{
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
|
|
if ( !pssiFile->SSIGetFileSize( &cbSizeLow, &cbSizeHigh ) )
|
|
{
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
|
|
if ( cbSizeHigh )
|
|
{
|
|
hr = HRESULT_FROM_WIN32( ERROR_NOT_SUPPORTED );
|
|
goto failed;
|
|
}
|
|
|
|
//
|
|
// Create a file mapping, we shouldn't need to impersonate as we already
|
|
// have the file open
|
|
//
|
|
|
|
if ( !_pSEL->Map() )
|
|
{
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
|
|
pchFilePos = pchBeginFile = pchBeginRange = _pSEL->QueryData();
|
|
pchEOF = pchFilePos + cbSizeLow;
|
|
|
|
//
|
|
// Scan for "<!--" or "<%"
|
|
//
|
|
|
|
while ( TRUE )
|
|
{
|
|
while ( pchFilePos < pchEOF && *pchFilePos != '<' )
|
|
{
|
|
pchFilePos++;
|
|
}
|
|
|
|
if ( pchFilePos + 4 >= pchEOF )
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Is this one of our tags?
|
|
//
|
|
|
|
if ( pchFilePos[1] == '%' ||
|
|
!strncmp( pchFilePos, "<!--", 4 ))
|
|
{
|
|
CHAR * pchBeginTag = pchFilePos;
|
|
SSI_COMMANDS CommandType;
|
|
SSI_TAGS TagType;
|
|
CHAR achTagString[ SSI_MAX_PATH + 1 ];
|
|
BOOL fValidTag;
|
|
|
|
//
|
|
// Get the tag info. The file position will be advanced to the
|
|
// first character after the tag
|
|
//
|
|
|
|
if ( !ParseSSITag( &pchFilePos,
|
|
pchEOF,
|
|
&fValidTag,
|
|
&CommandType,
|
|
&TagType,
|
|
achTagString ) )
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// If it's a tag we don't recognize then ignore it
|
|
//
|
|
|
|
if ( !fValidTag )
|
|
{
|
|
pchFilePos++;
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// Add the data up to the tag as a byte range
|
|
//
|
|
|
|
if ( pchBeginRange != pchBeginTag )
|
|
{
|
|
if ( !_pSEL->AppendByteRange(
|
|
DIFF(pchBeginRange - pchBeginFile),
|
|
DIFF(pchBeginTag - pchBeginRange) ) )
|
|
{
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
}
|
|
|
|
pchBeginRange = pchFilePos;
|
|
|
|
//
|
|
// Add the tag
|
|
//
|
|
|
|
if ( !_pSEL->AppendCommand( CommandType,
|
|
TagType,
|
|
achTagString ))
|
|
{
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Not one of our tags, skip the openning angle bracket
|
|
//
|
|
|
|
pchFilePos++;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Tack on the last byte range
|
|
//
|
|
|
|
if ( pchFilePos > pchBeginRange )
|
|
{
|
|
if ( !_pSEL->AppendByteRange( DIFF(pchBeginRange - pchBeginFile),
|
|
DIFF(pchFilePos - pchBeginRange) ))
|
|
{
|
|
hr = E_FAIL;
|
|
goto failed;
|
|
}
|
|
}
|
|
|
|
|
|
return NO_ERROR;
|
|
|
|
failed:
|
|
|
|
if ( _pSEL != NULL )
|
|
{
|
|
//
|
|
// also deletes pssiFile if SetFile has been called.
|
|
//
|
|
delete _pSEL;
|
|
_pSEL = NULL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//static
|
|
BOOL
|
|
SSI_INCLUDE_FILE::ParseSSITag(
|
|
IN OUT CHAR * * ppchFilePos,
|
|
IN CHAR * pchEOF,
|
|
OUT BOOL * pfValidTag,
|
|
OUT SSI_COMMANDS * pCommandType,
|
|
OUT SSI_TAGS * pTagType,
|
|
OUT CHAR * pszTagString
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function picks apart an NCSA style server side include
|
|
expression
|
|
|
|
The general form of a server side include directive is:
|
|
|
|
<[!-- or %]#[command] [tag]="[value]"[-- or %]>
|
|
|
|
For example:
|
|
|
|
<!--#include file="myfile.txt"-->
|
|
<%#echo var="HTTP_USER_AGENT"%>
|
|
<!--#fsize virtual="/dir/bar.htm"-->
|
|
|
|
For valid commands and tags see \iis\specs\ssi.doc
|
|
|
|
Arguments:
|
|
|
|
ppchFilePos - Pointer to first character of tag on way in, pointer
|
|
to first character after tag on way out if the tag is valid
|
|
pchEOF - Points to first byte beyond the end of the file
|
|
pfValidTag - Set to TRUE if this is a tag we support and all of the
|
|
parameters have been supplied
|
|
pCommandType - Receives SSI command
|
|
pTagType - Receives SSI tag
|
|
pszTagString - Receives value of pTagType. Must be > SSI_MAX_PATH.
|
|
|
|
Return Value:
|
|
|
|
TRUE if no errors occurred.
|
|
|
|
--*/
|
|
{
|
|
CHAR * pchFilePos = *ppchFilePos;
|
|
CHAR * pchEOT;
|
|
CHAR * pchEndQuote;
|
|
DWORD i;
|
|
DWORD cbToCopy;
|
|
DWORD cbJumpLen = 0;
|
|
BOOL fNewStyle; // <% format
|
|
|
|
DBG_ASSERT( *pchFilePos == '<' );
|
|
|
|
//
|
|
// Assume this is bad tag
|
|
//
|
|
|
|
*pfValidTag = FALSE;
|
|
|
|
if ( !strncmp( pchFilePos, "<!--", 4 ) )
|
|
{
|
|
fNewStyle = FALSE;
|
|
}
|
|
else if ( !strncmp( pchFilePos, "<%", 2 ) )
|
|
{
|
|
fNewStyle = TRUE;
|
|
}
|
|
else
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
//
|
|
// Find the closing comment token (either --> or %>). The reason
|
|
// why we shouldn't simply look for a > is because we want to allow
|
|
// the user to embed HTML <tags> in the directive
|
|
// (ex. <!--#CONFIG ERRMSG="<B>ERROR!!!</B>-->)
|
|
//
|
|
|
|
pchEOT = strstr( pchFilePos, fNewStyle ? "%>" : "-->" );
|
|
if ( !pchEOT )
|
|
{
|
|
return FALSE;
|
|
}
|
|
cbJumpLen = fNewStyle ? 2 : 3;
|
|
|
|
//
|
|
// Find the '#' that prefixes the command
|
|
//
|
|
|
|
pchFilePos = SSISkipTo( pchFilePos, '#', pchEOT );
|
|
|
|
if ( !pchFilePos )
|
|
{
|
|
//
|
|
// No command, bail for this tag
|
|
//
|
|
// CODEWORK - Check for if expression here
|
|
//
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//
|
|
// Lookup the command
|
|
//
|
|
|
|
i = 0;
|
|
while ( SSICmdMap[i].pszCommand )
|
|
{
|
|
if ( *SSICmdMap[i].pszCommand == towlower( *pchFilePos ) &&
|
|
!_strnicmp( SSICmdMap[i].pszCommand,
|
|
pchFilePos,
|
|
SSICmdMap[i].cchCommand ))
|
|
{
|
|
*pCommandType = SSICmdMap[i].ssiCmd;
|
|
|
|
//
|
|
// Note the space after the command is included in
|
|
// cchCommand
|
|
//
|
|
|
|
pchFilePos += SSICmdMap[i].cchCommand;
|
|
goto FoundCommand;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
//
|
|
// Unrecognized command, bail
|
|
//
|
|
|
|
return TRUE;
|
|
|
|
FoundCommand:
|
|
|
|
//
|
|
// Next, find the tag name
|
|
//
|
|
|
|
pchFilePos = SSISkipWhite( pchFilePos, pchEOT );
|
|
|
|
if ( !pchFilePos )
|
|
return TRUE;
|
|
|
|
i = 0;
|
|
while ( SSITagMap[i].pszTag )
|
|
{
|
|
if ( *SSITagMap[i].pszTag == tolower( *pchFilePos ) &&
|
|
!_strnicmp( SSITagMap[i].pszTag,
|
|
pchFilePos,
|
|
SSITagMap[i].cchTag ))
|
|
{
|
|
*pTagType = SSITagMap[i].ssiTag;
|
|
pchFilePos += SSITagMap[i].cchTag;
|
|
goto FoundTag;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
//
|
|
// Tag not found, bail
|
|
//
|
|
|
|
return TRUE;
|
|
|
|
FoundTag:
|
|
|
|
//
|
|
// Skip to the quoted tag value, then find the close quote
|
|
//
|
|
|
|
pchFilePos = SSISkipTo( pchFilePos, '"', pchEOT );
|
|
|
|
if ( !pchFilePos )
|
|
return TRUE;
|
|
|
|
pchEndQuote = SSISkipTo( ++pchFilePos, '"', pchEOT );
|
|
|
|
if ( !pchEndQuote )
|
|
return TRUE;
|
|
|
|
cbToCopy = min( DIFF( pchEndQuote - pchFilePos ), SSI_MAX_PATH );
|
|
|
|
memcpy( pszTagString,
|
|
pchFilePos,
|
|
cbToCopy );
|
|
|
|
pszTagString[ cbToCopy ] = '\0';
|
|
|
|
*pfValidTag = TRUE;
|
|
|
|
*ppchFilePos = pchEOT + cbJumpLen;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//static
|
|
CHAR *
|
|
SSI_INCLUDE_FILE::SSISkipTo(
|
|
IN CHAR * pchFilePos,
|
|
IN CHAR ch,
|
|
IN CHAR * pchEOF
|
|
)
|
|
{
|
|
return ( CHAR * ) memchr( pchFilePos,
|
|
ch,
|
|
DIFF( pchEOF - pchFilePos ) );
|
|
}
|
|
|
|
//static
|
|
CHAR *
|
|
SSI_INCLUDE_FILE::SSISkipWhite(
|
|
IN CHAR * pchFilePos,
|
|
IN CHAR * pchEOF
|
|
)
|
|
{
|
|
while ( pchFilePos < pchEOF )
|
|
{
|
|
if ( !SAFEIsSpace( *pchFilePos ) )
|
|
return pchFilePos;
|
|
|
|
pchFilePos++;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
HRESULT
|
|
SSI_INCLUDE_FILE::DoWork(
|
|
DWORD dwError
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method walks the element list sending the appropriate chunks of
|
|
data
|
|
|
|
Arguments:
|
|
|
|
dwError - error of the last asynchronous operation
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
--*/
|
|
{
|
|
HSE_EXEC_URL_STATUS ExecUrlStatus;
|
|
SSI_ELEMENT_ITEM * pSEI;
|
|
DWORD dwID;
|
|
LPSTR apszParms[ 2 ] = { NULL, NULL };
|
|
CHAR achNumberBuffer[ SSI_MAX_NUMBER_STRING ];
|
|
HRESULT hr = E_FAIL;
|
|
|
|
DBG_ASSERT( _pRequest != NULL );
|
|
|
|
if ( dwError != NO_ERROR )
|
|
{
|
|
//
|
|
// Last asynchronous operation failed
|
|
// only in the state: SIF_STATE_EXEC_CHILD_PENDING we will proceed if error occured
|
|
//
|
|
|
|
if ( _State != SIF_STATE_EXEC_CHILD_PENDING )
|
|
{
|
|
//
|
|
// Completion error means that we are forced to finish processing
|
|
//
|
|
SetState( SIF_STATE_COMPLETED );
|
|
}
|
|
}
|
|
|
|
while( _State != SIF_STATE_COMPLETED )
|
|
{
|
|
switch( _State )
|
|
{
|
|
case SIF_STATE_INITIALIZED:
|
|
//
|
|
// Prepare data structures when called for the first time
|
|
// and send initial headers or errors
|
|
//
|
|
|
|
hr = Prepare();
|
|
if( FAILED( hr ) )
|
|
{
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
case SIF_STATE_READY:
|
|
//
|
|
// Auxiliary state to flag that all the necessary date structures
|
|
// were prepared but actual processing hasn't started yet.
|
|
// Switch to PROCESSING state
|
|
//
|
|
|
|
SetState( SIF_STATE_PROCESSING );
|
|
break;
|
|
|
|
case SIF_STATE_PROCESSING:
|
|
//
|
|
// There are few cases when Process returns
|
|
// a) request completed
|
|
// b) pending operation
|
|
// c) child include file to be processed
|
|
//
|
|
// any any case return back to caller
|
|
//
|
|
|
|
hr = Process();
|
|
return hr;
|
|
|
|
case SIF_STATE_INCLUDE_CHILD_PENDING:
|
|
//
|
|
// Child include completed. Restore processing of current include file
|
|
//
|
|
SetState( SIF_STATE_PROCESSING );
|
|
break;
|
|
|
|
case SIF_STATE_EXEC_CHILD_PENDING:
|
|
|
|
//
|
|
// We were able to spawn child request. Get the status
|
|
//
|
|
pSEI = CONTAINING_RECORD( _pCurrentEntry, SSI_ELEMENT_ITEM, _ListEntry );
|
|
if( _pRequest->GetECB()->ServerSupportFunction(
|
|
_pRequest->GetECB()->ConnID,
|
|
HSE_REQ_GET_EXEC_URL_STATUS,
|
|
&ExecUrlStatus,
|
|
NULL,
|
|
NULL
|
|
) )
|
|
{
|
|
if ( ExecUrlStatus.uHttpStatusCode >= 400 )
|
|
{
|
|
switch( ExecUrlStatus.uHttpStatusCode )
|
|
{
|
|
case 403:
|
|
dwID = SSINCMSG_NO_EXECUTE_PERMISSION;
|
|
break;
|
|
default:
|
|
dwID = SSINCMSG_CANT_EXEC_CGI_REPORT_HTTP_STATUS;
|
|
break;
|
|
}
|
|
|
|
_itoa( ExecUrlStatus.uHttpStatusCode, achNumberBuffer, 10 );
|
|
apszParms[ 0 ] = ( CHAR* )pSEI->QueryTagValue()->QueryStr();
|
|
apszParms[ 1 ] = achNumberBuffer;
|
|
}
|
|
else
|
|
{
|
|
dwID = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_ultoa( GetLastError(), achNumberBuffer, 10 );
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->QueryStr();
|
|
apszParms[ 1 ] = achNumberBuffer;
|
|
dwID = SSINCMSG_CANT_EXEC_CGI;
|
|
}
|
|
|
|
//
|
|
// EXEC_URL completed. Adjust State back to PROCESSING
|
|
//
|
|
|
|
SetState( SIF_STATE_PROCESSING );
|
|
|
|
if ( dwID != 0 )
|
|
{
|
|
hr = _pRequest->SSISendError( dwID, apszParms );
|
|
if ( FAILED (hr) )
|
|
{
|
|
//
|
|
// This doesn't mean necessarily fatal error
|
|
// ERROR_IO_PENDING was likely returned
|
|
// so we have to simply return and wait for completion
|
|
//
|
|
return hr;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
//
|
|
// Unexpected State
|
|
//
|
|
DBG_ASSERT( _State > SIF_STATE_UNINITIALIZED && _State < SIF_STATE_UNKNOWN );
|
|
return E_FAIL;
|
|
} // switch( _State )
|
|
}
|
|
return NO_ERROR;
|
|
}
|
|
|
|
HRESULT
|
|
SSI_INCLUDE_FILE::Process(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method walks the element list sending the appropriate chunks of
|
|
data
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
--*/
|
|
{
|
|
DWORD cbSent;
|
|
STACK_STRU( strPath, MAX_PATH);
|
|
SSI_ELEMENT_ITEM * pSEI;
|
|
HSE_EXEC_URL_STATUS ExecUrlStatus;
|
|
DWORD dwID;
|
|
LPSTR apszParms[ 2 ];
|
|
CHAR achNumberBuffer[ SSI_MAX_NUMBER_STRING ];
|
|
HRESULT hr = E_FAIL;
|
|
|
|
DBG_ASSERT( _pRequest != NULL );
|
|
|
|
|
|
if( _pCurrentEntry == NULL )
|
|
{
|
|
_pCurrentEntry = _pSEL->QueryListHead();
|
|
}
|
|
|
|
//
|
|
// Move CurrentEntry pointer to next element
|
|
//
|
|
_pCurrentEntry = _pCurrentEntry->Flink;
|
|
|
|
//
|
|
// Loop through each element and take the appropriate action
|
|
//
|
|
|
|
while( _pCurrentEntry != _pSEL->QueryListHead() )
|
|
{
|
|
|
|
DBG_ASSERT( _State == SIF_STATE_PROCESSING );
|
|
|
|
pSEI = CONTAINING_RECORD( _pCurrentEntry, SSI_ELEMENT_ITEM, _ListEntry );
|
|
|
|
DBG_ASSERT( pSEI->CheckSignature() );
|
|
|
|
dwID = 0;
|
|
|
|
switch ( pSEI->QueryCommand() )
|
|
{
|
|
case SSI_CMD_BYTERANGE:
|
|
if ( FAILED( hr = _pRequest->WriteToClient(
|
|
_pSEL->QueryData() + pSEI->QueryBegin(),
|
|
pSEI->QueryLength(),
|
|
&cbSent ) ) )
|
|
{
|
|
//
|
|
// This doesn't mean necessarily fatal error
|
|
// ERROR_IO_PENDING was likely returned
|
|
// so we have to simply return and wait for completion
|
|
//
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
case SSI_CMD_INCLUDE:
|
|
switch ( pSEI->QueryTag() )
|
|
{
|
|
case SSI_TAG_FILE:
|
|
case SSI_TAG_VIRTUAL:
|
|
{
|
|
STACK_STRU( strFullURL, MAX_PATH );
|
|
|
|
if ( FAILED ( hr = GetFullPath( pSEI,
|
|
&strPath,
|
|
HSE_URL_FLAGS_READ,
|
|
&_strURL,
|
|
&strFullURL ) ) )
|
|
{
|
|
_ultoa( WIN32_FROM_HRESULT(hr), achNumberBuffer, 10 );
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->
|
|
QueryStr();
|
|
apszParms[ 1 ] = achNumberBuffer;
|
|
dwID = SSINCMSG_ERROR_HANDLING_FILE;
|
|
break;
|
|
}
|
|
|
|
if ( IsRecursiveInclude( strPath ) )
|
|
{
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->
|
|
QueryStr();
|
|
apszParms[ 1 ] = NULL;
|
|
dwID = SSINCMSG_ERROR_RECURSIVE_INCLUDE;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Nested STM include
|
|
//
|
|
|
|
SSI_INCLUDE_FILE * pChild = new SSI_INCLUDE_FILE( _pRequest, strPath, strFullURL, this );
|
|
|
|
if ( pChild == NULL || !pChild->IsValid() )
|
|
{
|
|
_ultoa( WIN32_FROM_HRESULT(ERROR_OUTOFMEMORY), achNumberBuffer, 10 );
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->
|
|
QueryStr();
|
|
apszParms[ 1 ] = achNumberBuffer;
|
|
dwID = SSINCMSG_ERROR_HANDLING_FILE;
|
|
break;
|
|
|
|
}
|
|
|
|
_pRequest->SetCurrentIncludeFile( pChild );
|
|
|
|
SetState( SIF_STATE_INCLUDE_CHILD_PENDING );
|
|
|
|
//
|
|
// Return back to SSI_REQUEST (_pRequest)
|
|
// SSI_REQUEST will start executing it just added SSI_INCLUDE_FILE (pChild)
|
|
//
|
|
// This way recursive function calls that were used in the previous
|
|
// implementation can be avoided (to makes possible to implement
|
|
// asynchronous processing)
|
|
//
|
|
|
|
return NO_ERROR;
|
|
}
|
|
default:
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
break;
|
|
|
|
case SSI_CMD_FLASTMOD:
|
|
switch( pSEI->QueryTag() )
|
|
{
|
|
case SSI_TAG_FILE:
|
|
case SSI_TAG_VIRTUAL:
|
|
if ( FAILED( hr = GetFullPath( pSEI,
|
|
&strPath,
|
|
0,
|
|
&_strURL ) ) ||
|
|
FAILED( hr = _pRequest->DoFLastMod( &strPath,
|
|
&_strTimeFmt,
|
|
_pSEL ) ) )
|
|
{
|
|
if ( hr == HRESULT_FROM_WIN32(ERROR_IO_PENDING ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
_ultoa( WIN32_FROM_HRESULT(hr), achNumberBuffer, 10 );
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->
|
|
QueryStr();
|
|
apszParms[ 1 ] = achNumberBuffer;
|
|
dwID = SSINCMSG_CANT_DO_FLASTMOD;
|
|
}
|
|
break;
|
|
default:
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
break;
|
|
|
|
case SSI_CMD_CONFIG:
|
|
switch( pSEI->QueryTag() )
|
|
{
|
|
case SSI_TAG_ERRMSG:
|
|
if ( !_pRequest->SetUserErrorMessage(
|
|
pSEI->QueryTagValue() ) )
|
|
{
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
break;
|
|
case SSI_TAG_TIMEFMT:
|
|
if ( FAILED( _strTimeFmt.Resize(
|
|
pSEI->QueryTagValue()->QueryCCH() ) ) )
|
|
{
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
|
|
if ( FAILED( hr = _strTimeFmt.Copy( pSEI->QueryTagValue()->QueryStr() ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
break;
|
|
case SSI_TAG_SIZEFMT:
|
|
if ( _strnicmp( SSI_DEF_BYTES,
|
|
( CHAR * )pSEI->QueryTagValue()->QueryStr(),
|
|
SSI_DEF_BYTES_LEN ) == 0 )
|
|
{
|
|
_fSizeFmtBytes = TRUE;
|
|
}
|
|
else if ( _strnicmp( SSI_DEF_ABBREV,
|
|
( CHAR * )pSEI->QueryTagValue()->QueryStr(),
|
|
SSI_DEF_ABBREV_LEN ) == 0 )
|
|
{
|
|
_fSizeFmtBytes = FALSE;
|
|
}
|
|
else
|
|
{
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
break;
|
|
default:
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
break;
|
|
|
|
case SSI_CMD_FSIZE:
|
|
switch( pSEI->QueryTag() )
|
|
{
|
|
case SSI_TAG_FILE:
|
|
case SSI_TAG_VIRTUAL:
|
|
if ( FAILED ( hr = GetFullPath( pSEI,
|
|
&strPath,
|
|
0,
|
|
&_strURL ) ) ||
|
|
FAILED ( hr = _pRequest->DoFSize( &strPath,
|
|
_fSizeFmtBytes,
|
|
_pSEL) ) )
|
|
{
|
|
if ( hr == HRESULT_FROM_WIN32(ERROR_IO_PENDING ) )
|
|
{
|
|
return hr;
|
|
}
|
|
_ultoa( WIN32_FROM_HRESULT(hr), achNumberBuffer, 10 );
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->
|
|
QueryStr();
|
|
apszParms[ 1 ] = achNumberBuffer;
|
|
dwID = SSINCMSG_CANT_DO_FSIZE;
|
|
|
|
}
|
|
break;
|
|
default:
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
|
|
break;
|
|
|
|
case SSI_CMD_ECHO:
|
|
if ( pSEI->QueryTag() == SSI_TAG_VAR )
|
|
{
|
|
// First let ISAPI try to evaluate variable.
|
|
hr = _pRequest->DoEchoISAPIVariable(
|
|
pSEI->QueryTagValue() );
|
|
if ( hr == HRESULT_FROM_WIN32(ERROR_IO_PENDING ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if ( SUCCEEDED (hr ) )
|
|
{
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
DWORD dwVar;
|
|
HRESULT hrEcho = E_FAIL;
|
|
|
|
// if ISAPI couldn't resolve var, try internal list
|
|
if ( !FindInternalVariable( pSEI->QueryTagValue(),
|
|
&dwVar ) )
|
|
{
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->
|
|
QueryStr();
|
|
apszParms[ 1 ] = NULL;
|
|
dwID = SSINCMSG_CANT_FIND_VARIABLE;
|
|
}
|
|
else
|
|
{
|
|
switch( dwVar )
|
|
{
|
|
case SSI_VAR_DOCUMENT_NAME:
|
|
hrEcho = _pRequest->DoEchoDocumentName(
|
|
&_pSEL->GetFile()->GetFilename() );
|
|
break;
|
|
case SSI_VAR_DOCUMENT_URI:
|
|
hrEcho = _pRequest->DoEchoDocumentURI(
|
|
&_strURL );
|
|
break;
|
|
case SSI_VAR_QUERY_STRING_UNESCAPED:
|
|
hrEcho = _pRequest->DoEchoQueryStringUnescaped();
|
|
break;
|
|
case SSI_VAR_DATE_LOCAL:
|
|
hrEcho = _pRequest->DoEchoDateLocal(
|
|
&_strTimeFmt );
|
|
break;
|
|
case SSI_VAR_DATE_GMT:
|
|
hrEcho = _pRequest->DoEchoDateGMT(
|
|
&_strTimeFmt );
|
|
break;
|
|
case SSI_VAR_LAST_MODIFIED:
|
|
hrEcho = _pRequest->DoEchoLastModified(
|
|
&_pSEL->GetFile()->GetFilename(),
|
|
&_strTimeFmt,
|
|
_pSEL );
|
|
break;
|
|
default:
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->
|
|
QueryTagValue()->QueryStr();
|
|
apszParms[ 1 ] = NULL;
|
|
dwID = SSINCMSG_CANT_FIND_VARIABLE;
|
|
}
|
|
if ( hrEcho == HRESULT_FROM_WIN32(ERROR_IO_PENDING ) )
|
|
{
|
|
return hrEcho;
|
|
}
|
|
|
|
if ( FAILED ( hrEcho ) )
|
|
{
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->
|
|
QueryTagValue()->QueryStr();
|
|
apszParms[ 1 ] = NULL;
|
|
dwID = SSINCMSG_CANT_EVALUATE_VARIABLE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
|
|
break;
|
|
|
|
case SSI_CMD_EXEC:
|
|
{
|
|
SSI_EXEC_TYPE ssiExecType = SSI_EXEC_UNKNOWN;
|
|
|
|
if ( _pRequest->IsExecDisabled() )
|
|
{
|
|
dwID = SSINCMSG_EXEC_DISABLED;
|
|
}
|
|
else if ( pSEI->QueryTag() == SSI_TAG_CMD )
|
|
{
|
|
if ( !fEnableCmdDirective )
|
|
{
|
|
dwID = SSINCMSG_CMD_NOT_ENABLED;
|
|
}
|
|
else
|
|
{
|
|
ssiExecType = SSI_EXEC_CMD;
|
|
}
|
|
}
|
|
else if ( pSEI->QueryTag() == SSI_TAG_CGI )
|
|
{
|
|
ssiExecType = SSI_EXEC_CGI;
|
|
}
|
|
else if ( pSEI->QueryTag() == SSI_TAG_ISA )
|
|
{
|
|
ssiExecType = SSI_EXEC_ISA;
|
|
}
|
|
else
|
|
{
|
|
dwID = SSINCMSG_INVALID_TAG;
|
|
}
|
|
|
|
if ( ssiExecType != SSI_EXEC_UNKNOWN )
|
|
{
|
|
BOOL fOk = FALSE;
|
|
|
|
ZeroMemory( &_ExecUrlInfo, sizeof( _ExecUrlInfo ) );
|
|
|
|
//
|
|
// Make asynchronous Child Execute
|
|
//
|
|
_ExecUrlInfo.dwExecUrlFlags = HSE_EXEC_URL_NO_HEADERS |
|
|
HSE_EXEC_URL_ASYNC |
|
|
HSE_EXEC_URL_IGNORE_APPPOOL |
|
|
HSE_EXEC_URL_DISABLE_CUSTOM_ERROR |
|
|
HSE_EXEC_URL_IGNORE_VALIDATION_AND_RANGE;
|
|
|
|
if ( ssiExecType == SSI_EXEC_CMD )
|
|
{
|
|
_ExecUrlInfo.dwExecUrlFlags |= HSE_EXEC_URL_SSI_CMD;
|
|
}
|
|
|
|
_ExecUrlInfo.pszUrl = (LPSTR) pSEI->QueryTagValue()->QueryStr();
|
|
|
|
DBG_ASSERT( _ExecUrlInfo.pszUrl != NULL );
|
|
|
|
//
|
|
// Avoid execution of empty URL
|
|
//
|
|
|
|
SetState( SIF_STATE_EXEC_CHILD_PENDING );
|
|
|
|
if ( _ExecUrlInfo.pszUrl[0] != '\0' )
|
|
{
|
|
fOk = _pRequest->GetECB()->ServerSupportFunction(
|
|
_pRequest->GetECB()->ConnID,
|
|
HSE_REQ_EXEC_URL,
|
|
&_ExecUrlInfo,
|
|
NULL,
|
|
NULL
|
|
);
|
|
}
|
|
if ( !fOk )
|
|
{
|
|
SetState( SIF_STATE_PROCESSING );
|
|
_ultoa( GetLastError(), achNumberBuffer, 10 );
|
|
apszParms[ 0 ] = ( CHAR * )pSEI->QueryTagValue()->QueryStr();
|
|
apszParms[ 1 ] = achNumberBuffer;
|
|
dwID = SSINCMSG_CANT_EXEC_CGI;
|
|
}
|
|
else
|
|
{
|
|
return HRESULT_FROM_WIN32( ERROR_IO_PENDING );
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
dwID = SSINCMSG_NOT_SUPPORTED;
|
|
break;
|
|
}
|
|
if ( dwID )
|
|
{
|
|
hr = _pRequest->SSISendError( dwID, apszParms );
|
|
if ( FAILED (hr) )
|
|
{
|
|
//
|
|
// This doesn't mean necessarily fatal error
|
|
// ERROR_IO_PENDING was likely returned
|
|
// so we have to simply return and wait for completion
|
|
//
|
|
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Move to next element of SSI_ELEMENT_LIST
|
|
//
|
|
|
|
_pCurrentEntry = _pCurrentEntry->Flink;
|
|
}
|
|
//
|
|
// End of the list has been reached
|
|
// It means that processing of the current SSI_INCLUDE_FILE has completed
|
|
//
|
|
|
|
SetState( SIF_STATE_COMPLETED );
|
|
return NO_ERROR;
|
|
}
|
|
|
|
HRESULT
|
|
SSI_INCLUDE_FILE::GetFullPath(
|
|
IN SSI_ELEMENT_ITEM * pSEI,
|
|
OUT STRU * pstrPath,
|
|
IN DWORD dwPermission,
|
|
IN STRU * pstrCurrentURL,
|
|
OUT STRU * pstrURL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Used to resolve FILE= and VIRTUAL= references. Fills in the physical
|
|
path of such file references and optionally checks the permissions
|
|
of the virtual directory.
|
|
|
|
Arguments:
|
|
|
|
pSEI - Element item ( either FILE or VIRTUAL )
|
|
pstrPath - Filled in with physical path of file
|
|
dwPermission - Contains permissions that the virtual
|
|
path must satisfy. For example HSE_URL_FLAGS_READ.
|
|
If 0, then no permissions are checked
|
|
pstrCurrentURL - Current .STM URL being parsed
|
|
pstrURL - Full URL filled in here (may be NULL if only pstrPath is to be retrieved)
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
--*/
|
|
{
|
|
WCHAR * pszValue;
|
|
STRU strValue;
|
|
DWORD dwMask;
|
|
DWORD cbBufLen;
|
|
WCHAR achPath[ SSI_MAX_PATH + 1 ];
|
|
HRESULT hr = E_FAIL;
|
|
|
|
//
|
|
// We recalc the virtual root each time in case the root
|
|
// to directory mapping has changed
|
|
//
|
|
|
|
if ( FAILED( hr = strValue.CopyA( pSEI->QueryTagValue()->QueryStr() ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
pszValue = strValue.QueryStr();
|
|
|
|
if ( *pszValue == L'/' )
|
|
{
|
|
wcscpy( achPath, pszValue );
|
|
}
|
|
else if ( ( int )pSEI->QueryTag() == ( int )SSI_TAG_FILE )
|
|
{
|
|
wcscpy( achPath, pstrCurrentURL->QueryStr() );
|
|
LPWSTR pL = achPath + wcslen( achPath );
|
|
while ( pL > achPath && pL[ -1 ] != L'/' )
|
|
{
|
|
--pL;
|
|
}
|
|
|
|
if ( pL == achPath )
|
|
{
|
|
*pL++ = L'/';
|
|
}
|
|
wcscpy( pL, pszValue );
|
|
}
|
|
else
|
|
{
|
|
achPath[ 0 ] = L'/';
|
|
wcscpy( achPath + 1, pszValue );
|
|
}
|
|
|
|
//
|
|
// First canonicalize the URL to be #included
|
|
//
|
|
// BUGBUG-MING
|
|
//CanonURL( achPath, g_fIsDBCS );
|
|
|
|
//
|
|
// Map to a physical directory
|
|
//
|
|
|
|
if ( FAILED( hr =_pRequest->LookupVirtualRoot( achPath,
|
|
pstrPath,
|
|
dwPermission ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if( pstrURL == NULL )
|
|
{
|
|
return NO_ERROR;
|
|
}
|
|
|
|
if( FAILED( hr = pstrURL->Copy( achPath ) ) )
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//static
|
|
BOOL
|
|
SSI_INCLUDE_FILE::FindInternalVariable(
|
|
IN STRA * pstrVariable,
|
|
OUT PDWORD pdwID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Lookup internal list of SSI variables that aren't supported by ISAPI.
|
|
These include "DOCUMENT_NAME", "DATE_LOCAL", etc.
|
|
|
|
Arguments:
|
|
|
|
pstrVariable - Variable to check
|
|
pdwID - Variable ID (or SSI_VAR_UNKNOWN if not found)
|
|
|
|
Return Value:
|
|
|
|
TRUE on success, FALSE on failure
|
|
|
|
--*/
|
|
{
|
|
DWORD dwCounter = 0;
|
|
|
|
while ( ( SSIVarMap[ dwCounter ].pszMap != NULL ) &&
|
|
_strnicmp( SSIVarMap[ dwCounter ].pszMap,
|
|
( CHAR * )pstrVariable->QueryStr(),
|
|
SSIVarMap[ dwCounter ].cchMap ) )
|
|
{
|
|
dwCounter++;
|
|
}
|
|
if ( SSIVarMap[ dwCounter ].pszMap != NULL )
|
|
{
|
|
*pdwID = SSIVarMap[ dwCounter ].ssiMap;
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
*pdwID = SSI_VAR_UNKNOWN;
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
InitializeSSIGlobals( VOID )
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initialize global variables
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HKEY hKeyParam;
|
|
DWORD dwType;
|
|
DWORD nBytes;
|
|
DWORD dwValue;
|
|
DWORD err;
|
|
|
|
if ( RegOpenKeyEx( HKEY_LOCAL_MACHINE,
|
|
W3_PARAMETERS_KEY,
|
|
0,
|
|
KEY_READ,
|
|
&hKeyParam ) == NO_ERROR )
|
|
{
|
|
nBytes = sizeof( dwValue );
|
|
err = RegQueryValueExA( hKeyParam,
|
|
"SSIEnableCmdDirective",
|
|
NULL,
|
|
&dwType,
|
|
( LPBYTE )&dwValue,
|
|
&nBytes
|
|
);
|
|
|
|
if ( ( err == ERROR_SUCCESS ) && ( dwType == REG_DWORD ) )
|
|
{
|
|
fEnableCmdDirective = !!dwValue;
|
|
}
|
|
|
|
RegCloseKey( hKeyParam );
|
|
}
|
|
}
|
|
|
|
//
|
|
// ISAPI DLL Required Entry Points
|
|
//
|
|
|
|
DWORD
|
|
WINAPI
|
|
HttpExtensionProc(
|
|
EXTENSION_CONTROL_BLOCK * pecb
|
|
)
|
|
{
|
|
BOOL fRet;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
SSI_REQUEST * pssiReq = new SSI_REQUEST(pecb);
|
|
|
|
if( pssiReq == NULL || !pssiReq->IsValid())
|
|
{
|
|
goto failed;
|
|
}
|
|
|
|
if ( !pecb->ServerSupportFunction( pecb->ConnID,
|
|
HSE_REQ_IO_COMPLETION,
|
|
SSI_REQUEST::HseIoCompletion,
|
|
0,
|
|
(LPDWORD ) pssiReq)
|
|
)
|
|
{
|
|
goto failed;
|
|
}
|
|
|
|
hr = pssiReq->DoWork();
|
|
if ( hr == HRESULT_FROM_WIN32(ERROR_IO_PENDING) )
|
|
{
|
|
//
|
|
// No Cleanup, there is pending IO request.
|
|
// completion routine is responsible perform proper cleanup
|
|
//
|
|
return HSE_STATUS_PENDING;
|
|
}
|
|
|
|
if ( SUCCEEDED( hr ) )
|
|
{
|
|
//
|
|
// This request is completed. Do Cleanup before returning
|
|
//
|
|
delete pssiReq;
|
|
return HSE_STATUS_SUCCESS;
|
|
}
|
|
|
|
failed:
|
|
{
|
|
LPCSTR apsz[ 1 ];
|
|
DWORD cch;
|
|
LPSTR pchBuff;
|
|
CHAR chTemp = '\0';
|
|
|
|
apsz[ 0 ] = pecb->lpszPathInfo;
|
|
|
|
//
|
|
// Since FormatMessage() is lame and actually uses SEH and an
|
|
// AV to determine when to resize its buffers, we have to truncate
|
|
// the URL here. Otherwise, on a machine with a UM debugger attached
|
|
// , we will break on the 1st chance exception (unless debugger is
|
|
// configured to ignore these exceptions which isn't the case.
|
|
//
|
|
|
|
if ( strlen( pecb->lpszPathInfo ) > SSI_MAX_FORMAT_LEN )
|
|
{
|
|
chTemp = pecb->lpszPathInfo[ SSI_MAX_FORMAT_LEN ];
|
|
pecb->lpszPathInfo[ SSI_MAX_FORMAT_LEN ] = '\0';
|
|
}
|
|
|
|
cch = ::FormatMessageA( FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_ARGUMENT_ARRAY |
|
|
FORMAT_MESSAGE_FROM_HMODULE,
|
|
GetModuleHandle( SSI_DLL_NAME ),
|
|
SSINCMSG_LOG_ERROR,
|
|
0,
|
|
( LPSTR ) &pchBuff,
|
|
0,
|
|
( va_list *) apsz );
|
|
|
|
if ( chTemp != '\0' )
|
|
{
|
|
pecb->lpszPathInfo[ SSI_MAX_FORMAT_LEN ] = chTemp;
|
|
}
|
|
|
|
if( cch )
|
|
{
|
|
strncpy( pecb->lpszLogData,
|
|
pchBuff,
|
|
(cch>HSE_LOG_BUFFER_LEN)?HSE_LOG_BUFFER_LEN:cch );
|
|
pecb->lpszLogData[HSE_LOG_BUFFER_LEN-1] = 0;
|
|
::LocalFree( ( VOID * )pchBuff );
|
|
}
|
|
|
|
//
|
|
// This request is completed. Do Cleanup before returning
|
|
//
|
|
delete pssiReq;
|
|
return HSE_STATUS_ERROR;
|
|
}
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
GetExtensionVersion(
|
|
HSE_VERSION_INFO * pver
|
|
)
|
|
{
|
|
pver->dwExtensionVersion = MAKELONG( 0, 1 );
|
|
strcpy( pver->lpszExtensionDesc,
|
|
"Server Side Include Extension DLL" );
|
|
|
|
//
|
|
// Get the cache instance for W3CORE.DLL
|
|
//
|
|
|
|
g_pFileCache = W3_FILE_INFO_CACHE::GetFileCache();
|
|
if ( g_pFileCache == NULL )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if ( FAILED( SSI_REQUEST::Initialize() ) )
|
|
{
|
|
return FALSE;
|
|
}
|
|
if ( FAILED( SSI_INCLUDE_FILE::Initialize() ) )
|
|
{
|
|
SSI_REQUEST::Terminate();
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
TerminateExtension(
|
|
DWORD dwFlags
|
|
)
|
|
{
|
|
SSI_REQUEST::Terminate();
|
|
SSI_INCLUDE_FILE::Terminate();
|
|
|
|
W3CacheFlushAllCaches();
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL
|
|
WINAPI
|
|
DLLEntry(
|
|
HINSTANCE hDll,
|
|
DWORD dwReason,
|
|
LPVOID lpvReserved
|
|
)
|
|
{
|
|
switch ( dwReason )
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
|
|
CREATE_DEBUG_PRINT_OBJECT( "ssinc" );
|
|
|
|
DisableThreadLibraryCalls( hDll );
|
|
InitializeSSIGlobals();
|
|
break;
|
|
|
|
case DLL_PROCESS_DETACH:
|
|
DELETE_DEBUG_PRINT_OBJECT();
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|