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

1554 lines
30 KiB
C

#include <windows.h>
#include <stdio.h>
#include <urlcache.h>
#include <histapi.h>
//#include "cache.hxx"
//#include "history.h"
#include "generic.h"
#define DEFAULT_CEI_BUFFER_SIZE 512
// 1k ~> sizeof (CEI) + lpszSourceUrlName + lpHeaderInfo(~<255) + lpszLocalFileName(<255)
#define ASSERT(x) if (!(x)) DebugBreak();
LPCTSTR lpszHistoryPrefix = "Hist:";
DWORD cbHistoryPrefix = sizeof("Hist:") -1;
LPCTSTR lpszTitleHeader = "Title: ";
DWORD cbTitleHeader = sizeof("Title: ") -1;
LPCTSTR lpszFragmentHeader = "Frags: ";
DWORD cbFragmentHeader = sizeof("Frags: ") -1;
LPCTSTR lpszHistoryFileExtension = "HSD";
LPTSTR szCRLF = "\r\n";
DWORD cbCRLF = sizeof("\r\n") -1;
LPTSTR szSPC = " ";
LPTSTR szPND = "#";
LPTSTR szFRAGB = " (#";
LPTSTR szFRAGE = ")";
typedef struct _HISTORY_SEARCH_OBJ
{
HANDLE hEnum;
LPTSTR lpszPrefixedUrl;
LPTSTR lpszFragment;
LPCACHE_ENTRY_INFO lpCEI;
LPTSTR *aFrags;
DWORD cFrags;
DWORD iFrags;
} HISTORY_SEARCH_OBJ, *LPHISTORY_SEARCH_OBJ;
typedef struct _HISTORY_ITEM_INFO
{
DWORD dwVersion; //Version of History System
LPSTR lpszSourceUrlName; // embedded pointer to the URL name string.
DWORD HistoryItemType; // cache type bit mask.
FILETIME LastAccessTime; // last accessed time in GMT format
LPSTR lpszTitle; // embedded pointer to the History-Title: info.
LPSTR lpszDependancies; // list of URLs that this page requires to be functional, SPC delimited
DWORD dwReserved; // reserved for future use.
} HISTORY_ITEM_INFO, *LPHISTORY_ITEM_INFO;
LPTSTR
GetDependanciesFromCEI (LPCACHE_ENTRY_INFO lpCEI)
{
LPTSTR buf = NULL;
HANDLE file = NULL;
DWORD size = 0;
LPTSTR pch = NULL;
ASSERT (lpCEI);
file = CreateFile(lpCEI->lpszLocalFileName,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);
if (file == INVALID_HANDLE_VALUE)
return NULL;
size = GetFileSize(file, NULL);
buf = (LPTSTR) LocalAlloc (LPTR, size + 1);
if (!buf)
{
CloseHandle(file);
return NULL;
}
buf[size] = '\0';
// we are going to store these as URL\nURL\nURL\n so just look for \n and replace with space
for (pch = buf; *pch; pch++)
{
if (*pch == '\n')
*pch = ' ';
}
CloseHandle (file);
return buf;
}
LPTSTR
MakeDependanciesFile (LPCTSTR lpszDeps)
{
HANDLE file = NULL;
LPTSTR pch = NULL;
LPTSTR path = NULL;
LPTSTR temp = NULL;
DWORD size = 0;
path = _tempnam (NULL, "HS");
if (!path)
return NULL;
file = CreateFile(path,
GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL,
NULL);
if (file == INVALID_HANDLE_VALUE)
{
LocalFree (path);
return NULL;
}
// we are going to store these as URL\nURL\nURL\n
if(lpszDeps)
{
size = lstrlen (lpszDeps);
temp = LocalAlloc (LPTR, size + 1);
if (!temp)
{
LocalFree (path);
CloseHandle (file);
return NULL;
}
lstrcpy (temp, lpszDeps);
for (pch = temp; *pch; pch++)
{
if (*pch == ' ')
*pch = '\n';
}
WriteFile (file, temp, size, &size, NULL);
}
CloseHandle (file);
return path;
}
LPTSTR
ConvertToUnprefixedUrl (
LPCTSTR lpszPrefixedUrl,
LPCTSTR lpszFragment
)
{
DWORD size = 0;
LPTSTR lpszUrl = NULL;
LPTSTR temp = NULL;
temp = (LPTSTR) (lpszPrefixedUrl + cbHistoryPrefix) ;
size = lstrlen(temp);
if (lpszFragment)
{
size += lstrlen (lpszFragment);
size += 1; // for the fragment '#'
}
lpszUrl = (LPTSTR) LocalAlloc (LPTR, size + 1);
if (!lpszUrl)
return NULL;
lstrcpy (lpszUrl, temp);
if (lpszFragment)
{
lstrcat (lpszUrl, szPND);
lstrcat (lpszUrl, lpszFragment);
}
return lpszUrl;
}
BOOL
ConvertToPrefixedUrl (IN LPCTSTR lpszUrlName,
OUT LPTSTR *lplpszPrefixedUrl,
OUT LPTSTR *lplpszFragment)
{
if (!lpszUrlName || !*lpszUrlName)
{
*lplpszPrefixedUrl = (LPTSTR) LocalAlloc (LPTR, cbHistoryPrefix + 1);
if (!*lplpszPrefixedUrl)
return FALSE;
lstrcpy (*lplpszPrefixedUrl, lpszHistoryPrefix);
return TRUE;
}
*lplpszPrefixedUrl = (LPTSTR) LocalAlloc (LPTR, cbHistoryPrefix + strlen (lpszUrlName) + 1);
if (!*lplpszPrefixedUrl)
return FALSE;
lstrcpy (*lplpszPrefixedUrl, lpszHistoryPrefix);
lstrcat (*lplpszPrefixedUrl, lpszUrlName);
*lplpszFragment = strchr (*lplpszPrefixedUrl, '#');
if(*lplpszFragment)
*((*lplpszFragment)++) = '\0';
return TRUE;
}
LPCACHE_ENTRY_INFO
RetrievePrefixedUrl (IN LPTSTR lpszUrl)
/*++
The CEI returned must be freed and the lpszUrl unlocked
--*/
{
LPCACHE_ENTRY_INFO lpCEI = NULL;
DWORD cbCEI = 0;
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, DEFAULT_CEI_BUFFER_SIZE);
if (!lpCEI)
return NULL;
cbCEI = DEFAULT_CEI_BUFFER_SIZE;
while (!RetrieveUrlCacheEntryFile (lpszUrl,
lpCEI,
&cbCEI,
0))
{
if (GetLastError () == ERROR_NOT_ENOUGH_MEMORY)
{
LocalFree (lpCEI);
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, cbCEI);
if (!lpCEI)
return NULL;
}
else
return NULL;
}
return lpCEI;
}
LPCACHE_ENTRY_INFO
RetrievePrefixedUrlInfo (IN LPTSTR lpszUrl)
/*++
The CEI returned must be freed and the lpszUrl unlocked
--*/
{
LPCACHE_ENTRY_INFO lpCEI = NULL;
DWORD cbCEI = 0;
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, DEFAULT_CEI_BUFFER_SIZE);
if (!lpCEI)
return NULL;
cbCEI = DEFAULT_CEI_BUFFER_SIZE;
while (!GetUrlCacheEntryInfo (lpszUrl,
lpCEI,
&cbCEI
))
{
if (GetLastError () == ERROR_NOT_ENOUGH_MEMORY)
{
LocalFree (lpCEI);
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, cbCEI);
if (!lpCEI)
return NULL;
}
else
return NULL;
}
return lpCEI;
}
LPTSTR
GetTitleFromCEI (IN LPCACHE_ENTRY_INFO lpCEI, LPCTSTR lpszFragment)
{
LPTSTR pHeader, pCurr;
DWORD size = 0;
pHeader = (LPTSTR) MemFind ((LPVOID) lpCEI->lpHeaderInfo,
lpCEI->dwHeaderInfoSize,
(LPVOID) lpszTitleHeader,
cbTitleHeader);
if (!pHeader)
{
SetLastError (ERROR_FILE_NOT_FOUND);
return NULL;
}
//Header was found
pCurr = (LPTSTR) MemFind ( (LPVOID) pHeader,
(lpCEI->dwHeaderInfoSize) - ((DWORD) (pHeader - (LPTSTR)lpCEI->lpHeaderInfo)),
(LPVOID) szCRLF,
cbCRLF);
if (!pCurr)
{
// BUGBUG do what now?? found the header, but the title is not in a recognized
// format. lets bail with a internal prob
ASSERT (FALSE);
SetLastError (ERROR_FILE_NOT_FOUND);
return NULL;
}
*pCurr = '\0';
pCurr = pHeader + cbTitleHeader;
while (*pCurr == ' ')
pCurr++;
size = lstrlen (pCurr) ;
if (lpszFragment) //must also include the fragment in Title
size += lstrlen (lpszFragment) + 4;
pHeader = (LPTSTR) LocalAlloc (LPTR, size + 1);
if (!pHeader)
return NULL;
lstrcpy (pHeader, pCurr);
if (lpszFragment)
{
lstrcat (pHeader, szFRAGB);
lstrcat (pHeader, lpszFragment);
lstrcat (pHeader, szFRAGE);
}
return pHeader;
}
DWORD
GetFragmentsFromCEI(IN LPCACHE_ENTRY_INFO lpCEI,
OUT LPTSTR **paFrags,
OUT DWORD *pcFrags)
{
LPTSTR pHeader, pCurr;
//need to get the string from the CEI, then parse into args
pHeader = (LPTSTR) MemFind (lpCEI->lpHeaderInfo,
lpCEI->dwHeaderInfoSize,
(LPVOID) lpszFragmentHeader,
cbFragmentHeader);
if (!pHeader)
return ERROR_FILE_NOT_FOUND;
//Header was found
pCurr = (LPTSTR) MemFind ( (LPVOID) pHeader,
lpCEI->dwHeaderInfoSize - (pHeader - lpCEI->lpHeaderInfo),
(LPVOID) szCRLF,
cbCRLF);
if (!pCurr)
{
//this is a corrupted Entry
ASSERT (FALSE);
return ERROR_FILE_NOT_FOUND;
}
*pCurr = '\0';
//
// pHeader is now zero terminated string
// we want to parse the args of that string
//
if (!ParseArgsDyn(pHeader + cbFragmentHeader, paFrags, pcFrags))
return ERROR_NOT_ENOUGH_MEMORY;
return ERROR_SUCCESS;
}
LPBYTE
GenerateHeaderInfo(LPCTSTR lpszTitle, LPCTSTR *aFrags, DWORD cFrags)
{
DWORD size = 0;
LPBYTE hi = NULL;
LPTSTR curr;
DWORD i;
//first need to find the size required of HeaderInfo
if (lpszTitle)
{
size += lstrlen (lpszTitle);
size += cbTitleHeader;
size += cbCRLF;
}
if (cFrags)
{
size += cbFragmentHeader;
size += cFrags;
for (i = 0; i < cFrags; i++)
size += lstrlen(aFrags[i]);
size += cbCRLF;
}
hi = (LPBYTE) LocalAlloc (LPTR, ++size);
if (!hi)
return NULL;
curr = (LPTSTR) hi;
*curr = '\0';
if (lpszTitle)
{
lstrcat (curr, lpszTitleHeader);
lstrcat (curr, lpszTitle);
lstrcat (curr, szCRLF);
}
if (cFrags)
{
lstrcat (curr, lpszFragmentHeader);
for(i = 0; i < cFrags; i++)
{
if (!*(aFrags[i]))
continue;
lstrcat(curr, szSPC);
lstrcat(curr, aFrags[i]);
}
lstrcat (curr, szCRLF);
}
return hi;
}
DWORD
CopyCEItoHII (
LPCTSTR lpszFragment,
LPHISTORY_ITEM_INFO lpHII,
LPDWORD lpcbHII,
LPCACHE_ENTRY_INFO lpCEI
)
{
DWORD Error = ERROR_SUCCESS;
DWORD cbNeeded = sizeof (HISTORY_ITEM_INFO);
DWORD cbUsed = cbNeeded;
LPTSTR lpszUrl = NULL;
DWORD cbUrl = 0;
LPTSTR lpszTitle = NULL;
DWORD cbTitle = 0;
LPTSTR lpszDependancies = NULL;
DWORD cbDependancies = 0;
ASSERT (lpCEI->lpszSourceUrlName);
//
// need to determine the necessary size
//
// need the unprefixed name
lpszUrl = ConvertToUnprefixedUrl (lpCEI->lpszSourceUrlName, (LPCTSTR) lpszFragment);
if (!lpszUrl)
{
Error = ERROR_INTERNAL_ERROR;
goto quit;
}
cbUrl = lstrlen (lpszUrl);
cbNeeded += cbUrl + 1;
lpszTitle = GetTitleFromCEI (lpCEI, (LPCTSTR) lpszFragment);
if (lpszTitle)
{
cbTitle = lstrlen (lpszTitle);
cbNeeded += cbTitle + 1;
}
lpszDependancies = GetDependanciesFromCEI (lpCEI);
if (lpszDependancies)
{
cbDependancies = lstrlen (lpszDependancies);
cbNeeded += cbDependancies + 1;
}
if (cbNeeded > *lpcbHII)
{
Error = ERROR_NOT_ENOUGH_MEMORY;
*lpcbHII = cbNeeded;
goto quit;
}
//
// Add the other pieces
//
lpHII->lpszSourceUrlName = (LPTSTR) (lpHII + cbUsed + 1);
lstrcpy (lpHII->lpszSourceUrlName, lpszUrl);
cbUsed += cbUrl + 1;
if (lpszTitle)
{
lpHII->lpszTitle = (LPTSTR) (lpHII + cbUsed + 1);
lstrcpy (lpHII->lpszTitle, lpszTitle);
cbUsed += cbTitle + 1;
}
else
lpHII->lpszTitle = NULL;
if (lpszDependancies)
{
lpHII->lpszDependancies = (LPTSTR) (lpHII + cbUsed + 1);
lstrcpy (lpHII->lpszDependancies, lpszDependancies);
cbUsed += cbDependancies + 1;
}
else
lpHII->lpszDependancies = NULL;
lpHII->dwVersion = lpCEI->dwVersion;
lpHII->HistoryItemType = lpCEI->CacheEntryType;
lpHII->LastAccessTime.dwLowDateTime = lpCEI->LastAccessTime.dwLowDateTime;
lpHII->LastAccessTime.dwHighDateTime = lpCEI->LastAccessTime.dwHighDateTime;
lpHII->dwReserved = lpCEI->dwReserved;
quit:
if (lpszUrl)
LocalFree(lpszUrl);
if (lpszTitle)
LocalFree(lpszTitle);
if (lpszDependancies)
LocalFree(lpszDependancies);
if (Error == ERROR_SUCCESS)
*lpcbHII = cbUsed;
return Error;
}
HISTORYAPI_(BOOL)
AddHistoryItem(
IN LPCTSTR lpszUrlName, //direct correspondence in URLCACHE
IN LPCTSTR lpszHistoryTitle, // this needs to be added to lpHeaderInfo
IN LPCTSTR lpszDependancies,
IN DWORD dwFlags,
IN DWORD dwReserved
)
/*++
Routine Description:
Places the specified URL into the history.
If it does not exist, then it is created. If it does exist it is overwritten.
Arguments:
lpszUrlName - The URL in question.
lpszHistoryTitle - pointer to the friendly title that should be associated
with this URL. If NULL, no title will be added.
Reserved - Unused, for future implementations
Return Value:
BOOL
Success - TRUE
Failure - FALSE. Extended error can be retrieved from GetLastError()
--*/
{
LPBYTE NewHeaderInfo = NULL;
DWORD cbNHI = 0;
BOOL New = FALSE;
LPTSTR lpszPrefixedUrl = NULL;
LPTSTR lpszFragment = NULL;
DWORD Error = ERROR_SUCCESS;
LPCACHE_ENTRY_INFO lpCEI = NULL;
FILETIME ftExpires;
FILETIME ftModified;
SYSTEMTIME st;
LPTSTR *aFrags = NULL;
DWORD cFrags = 0;
DWORD i;
BOOL found = FALSE;
LPTSTR lpszDepsPath = NULL;
DWORD type = NORMAL_CACHE_ENTRY;
LPTSTR lpszOldTitle = NULL;
if (!ConvertToPrefixedUrl (lpszUrlName, &lpszPrefixedUrl, &lpszFragment))
{
Error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
lpCEI = RetrievePrefixedUrl (lpszPrefixedUrl);
if (!lpCEI)
New = TRUE;
// Buffer filled with data now
// BUGBUG must handle fragments
if (!New)
{
type = lpCEI->CacheEntryType;
GetFragmentsFromCEI (lpCEI, &aFrags, &cFrags);
lpszOldTitle = GetTitleFromCEI (lpCEI, NULL);
}
// if (Error != ERROR_SUCCESS)
if (lpszFragment)
{
for (i = 0; i < cFrags; i++)
{
if (lstrcmp (aFrags[i], lpszFragment) == 0)
{
found = TRUE;
break;
}
}
if (!found)
AddArgvDyn (&aFrags, &cFrags, lpszFragment);
}
NewHeaderInfo = GenerateHeaderInfo (lpszHistoryTitle ? lpszHistoryTitle : lpszOldTitle, aFrags, cFrags);
cbNHI = lstrlen (NewHeaderInfo);
lpszDepsPath = MakeDependanciesFile (lpszDependancies);
if (!lpszDepsPath)
{
ASSERT(FALSE);
Error = ERROR_INTERNAL_ERROR;
goto quit;
}
GetLocalTime (&st);
SystemTimeToFileTime(&st, &ftModified);
st.wDay += 7; //BUGBUG must get this setting from registry
if(!SystemTimeToFileTime(&st, &ftExpires))
{
Error = GetLastError ();
goto quit;
}
if (lpCEI)
{
UnlockUrlCacheEntryFile (lpCEI->lpszSourceUrlName, 0);
LocalFree (lpCEI);
lpCEI = NULL;
}
if (!CommitUrlCacheEntry(
lpszPrefixedUrl,
lpszDepsPath,
ftExpires,
ftModified, //we dont care about last modified time
type, //this is set from dwFlags i think
NewHeaderInfo,
cbNHI ,
lpszHistoryFileExtension,
0))
{
Error = GetLastError ();
goto quit;
}
// if we made it to here, we win!
quit:
if (aFrags)
LocalFree (aFrags);
if (lpszDepsPath)
LocalFree (lpszDepsPath);
if (lpCEI)
{
UnlockUrlCacheEntryFile (lpCEI->lpszSourceUrlName, 0);
LocalFree (lpCEI);
}
if (lpszPrefixedUrl)
LocalFree (lpszPrefixedUrl);
if (NewHeaderInfo)
LocalFree (NewHeaderInfo);
if (lpszOldTitle)
LocalFree (lpszOldTitle);
if (Error != ERROR_SUCCESS)
{
SetLastError (Error);
return FALSE;
}
else
return TRUE;
}
HISTORYAPI_(BOOL)
IsHistorical(
IN LPCTSTR lpszUrlName
)
/*++
Routine Description:
Checks to see if Url is a valid History item
Arguments:
lpszUrlName - The URL in question.
Return Value:
BOOL
Success - TRUE. Item is in History
Failure - FALSE. Extended error can be retrieved from GetLastError()
ERROR_FILE_NOT_FOUND indicates the URL is not available
--*/
{
LPTSTR lpszPrefixedUrl = NULL;
LPTSTR lpszFragment = NULL;
DWORD Error = ERROR_SUCCESS;
LPCACHE_ENTRY_INFO lpCEI = NULL;
LPTSTR *aFrags = NULL;
DWORD cFrags = 0;
DWORD i;
if (!ConvertToPrefixedUrl (lpszUrlName, &lpszPrefixedUrl, &lpszFragment))
{
Error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
lpCEI = RetrievePrefixedUrlInfo (lpszPrefixedUrl);
if (!lpCEI)
{
Error = GetLastError ();
goto quit;
}
if (lpszFragment)
{
//
// Need to check for IntraDocFrags
//
Error = GetFragmentsFromCEI(lpCEI, & aFrags, & cFrags);
if (Error != ERROR_SUCCESS)
goto quit;
for (i = 0; i < cFrags; i++)
{
if (strcmp(aFrags[i], lpszFragment) == 0)
goto quit;
}
Error = ERROR_FILE_NOT_FOUND;
}
quit:
if (aFrags)
LocalFree (aFrags);
if (lpszPrefixedUrl)
LocalFree (lpszPrefixedUrl);
if (lpCEI)
{
LocalFree (lpCEI);
}
if (Error != ERROR_SUCCESS)
{
SetLastError (Error);
return FALSE;
}
else
return TRUE;
}
HISTORYAPI_(BOOL)
RemoveHistoryItem (
IN LPCTSTR lpszUrlName,
IN DWORD dwReserved
)
/*++
Routine Description:
Changes an entry from an History Item to a normal cache entry. Removing
the Title at the same time.
Arguments:
lpszUrlName - The URL in question.
dwReserved - Unused. for future usage
Return Value:
BOOL
Success - TRUE. Item found and removed
Failure - FALSE. Extended error can be retrieved from GetLastError()
ERROR_FILE_NOT_FOUND indicates the URL is not available
--*/
{
LPTSTR *aFrags = NULL;
DWORD cFrags = 0;
DWORD i;
LPTSTR lpszTitle = NULL;
LPBYTE NewHeaderInfo = NULL;
LPTSTR lpszPrefixedUrl = NULL;
LPTSTR lpszFragment = NULL;
DWORD Error = ERROR_SUCCESS;
LPCACHE_ENTRY_INFO lpCEI = NULL;
if (!ConvertToPrefixedUrl (lpszUrlName, &lpszPrefixedUrl, &lpszFragment))
{
Error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
lpCEI = RetrievePrefixedUrl (lpszPrefixedUrl);
if (!lpCEI)
{
Error = GetLastError ();
goto quit;
}
if (lpszFragment)
{
BOOL found = FALSE;
//
// Need to check for IntraDocFrags
//
Error = GetFragmentsFromCEI(lpCEI, & aFrags, & cFrags);
if (Error != ERROR_SUCCESS)
goto quit;
for (i = 0; i < cFrags; i++)
{
if (strcmp(aFrags[i], lpszFragment) == 0)
{
//we need to delete this and reinsert
*(aFrags[i]) = '\0';
found = TRUE;
break;
}
}
if (cFrags - 1 && found)
{
lpszTitle = GetTitleFromCEI (lpCEI, NULL);
NewHeaderInfo = GenerateHeaderInfo (lpszTitle, aFrags, cFrags);
if(!NewHeaderInfo)
{
Error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
UnlockUrlCacheEntryFile(lpCEI->lpszSourceUrlName, 0);
if (!CommitUrlCacheEntry(
lpszPrefixedUrl,
lpCEI->lpszLocalFileName,
lpCEI->ExpireTime,
lpCEI->LastModifiedTime,
lpCEI->CacheEntryType , // only changes
NewHeaderInfo, //
lstrlen (NewHeaderInfo),
lpCEI->lpszFileExtension,
0))
{
Error = GetLastError ();
}
goto quit;
}
if (!found)
{
Error = ERROR_FILE_NOT_FOUND;
goto quit;
}
}
//BUGBUG looks like this will always delete a history item if there is only fragment
//problem is we could have a frag and a unfragged Item
UnlockUrlCacheEntryFile(lpCEI->lpszSourceUrlName, 0);
if (!DeleteUrlCacheEntry(lpszPrefixedUrl))
{
Error = GetLastError ();
goto quit;
}
quit:
if (aFrags)
LocalFree (aFrags);
if (lpszTitle)
LocalFree (lpszTitle);
if (lpCEI)
{
LocalFree (lpCEI);
}
if (NewHeaderInfo)
LocalFree (NewHeaderInfo);
if (lpszPrefixedUrl)
LocalFree (lpszPrefixedUrl);
if (Error != ERROR_SUCCESS)
{
SetLastError (Error);
return FALSE;
}
else
return TRUE;
}
HISTORYAPI_(BOOL)
GetHistoryItemInfo (
IN LPCTSTR lpszUrlName,
OUT LPHISTORY_ITEM_INFO lpHistoryItemInfo,
IN OUT LPDWORD lpdwHistoryItemInfoBufferSize
)
/*++
Routine Description:
Fills a buffer with a HISTORY_ITEM_INFO struct.
Arguments:
lpszUrlName - The URL in question.
lpHistoryItemInfo - Buffer that will hold the HISTORY_ITEM_INFO
lpdwHistoryItemInfoBufferSize - IN: size of the lpHistoryItemInfo buffer
OUT: size of filled struct when successful
or necessary buffer size when failed
Return Value:
BOOL
Success - TRUE.
Failure - FALSE. Extended error can be retrieved from GetLastError()
ERROR_NOT_ENOUGH_MEMORY indicates the buffer is insufficient
--*/
{
LPTSTR lpszPrefixedUrl = NULL;
LPTSTR lpszFragment = NULL;
DWORD Error = ERROR_SUCCESS;
LPCACHE_ENTRY_INFO lpCEI = NULL;
if (!ConvertToPrefixedUrl (lpszUrlName, &lpszPrefixedUrl, &lpszFragment))
{
Error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
lpCEI = RetrievePrefixedUrlInfo (lpszPrefixedUrl);
if (!lpCEI)
{
Error = GetLastError ();
goto quit;
}
Error = CopyCEItoHII (lpszFragment, lpHistoryItemInfo, lpdwHistoryItemInfoBufferSize, lpCEI);
quit:
if (lpszPrefixedUrl)
LocalFree (lpszPrefixedUrl);
if (lpCEI)
{
LocalFree (lpCEI);
}
if (Error != ERROR_SUCCESS)
{
SetLastError (Error);
return FALSE;
}
else
return TRUE;
}
HISTORYAPI_(HANDLE)
FindFirstHistoryItem(
IN LPCTSTR lpszUrlSearchPattern,
OUT LPHISTORY_ITEM_INFO lpFirstHistoryItemInfo,
IN OUT LPDWORD lpdwFirstHistoryItemInfoBufferSize
)
/*++
Routine Description:
Searches through the History looking for URLs that match the search pattern,
and copies the HISTORY_ITEM_INFO into the buffer.
Arguments:
lpszUrlSearchPattern - The URL in question.
lpFirstHistoryItemInfo - Buffer that will hold the HISTORY_ITEM_INFO
lpdwFirstHistoryItemInfoBufferSize - IN: size of the lpHistoryItemInfo buffer
OUT: size of filled struct when successful
or necessary buffer size when failed
Return Value:
HANDLE
Success - Valid enumeration handle to pass into subsequent calls to
FindNextHistoryItem ().
Failure - NULL. Extended error can be retrieved from GetLastError()
ERROR_NOT_ENOUGH_MEMORY indicates the buffer is insufficient
--*/
{
LPHISTORY_SEARCH_OBJ hso = NULL;
LPCACHE_ENTRY_INFO lpCEI = NULL;
DWORD cbCEI = 0;
LPTSTR lpszFoundFragment = NULL;
DWORD Error = ERROR_SUCCESS;
BOOL found = FALSE;
hso = (LPHISTORY_SEARCH_OBJ) LocalAlloc (LPTR, sizeof (HISTORY_SEARCH_OBJ));
if (!hso)
{
Error = GetLastError ();
goto quit;
}
hso->aFrags = NULL;
hso->cFrags = 0;
hso->iFrags = 0;
hso->lpszPrefixedUrl = NULL;
hso->lpszFragment = NULL;
if (!ConvertToPrefixedUrl (lpszUrlSearchPattern, &(hso->lpszPrefixedUrl), &(hso->lpszFragment)))
{
Error = ERROR_NOT_ENOUGH_MEMORY;
goto quit;
}
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, DEFAULT_CEI_BUFFER_SIZE);
if (!lpCEI)
{
Error = GetLastError ();
goto quit;
}
while (TRUE)
{
hso->hEnum = FindFirstUrlCacheEntry (hso->lpszPrefixedUrl,
lpCEI,
&cbCEI);
if (!hso->hEnum)
{
Error = GetLastError ();
if (Error == ERROR_NOT_ENOUGH_MEMORY)
{
LocalFree (lpCEI);
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, cbCEI);
if (!lpCEI)
{
Error = ERROR_INTERNAL_ERROR;
goto quit;
}
}
else
goto quit;
}
else break;
}
found = TRUE;
//BUGBUG have to handle enum of fragments
Error = GetFragmentsFromCEI (lpCEI, &(hso->aFrags), &(hso->cFrags));
switch (Error)
{
case ERROR_FILE_NOT_FOUND: //only the default URL is used
Error = ERROR_SUCCESS;
break;
case ERROR_SUCCESS: //first return the default URL next call will get frags
hso->lpCEI = lpCEI;
break;
default:
goto quit;
break;
}
if (hso->lpszFragment)
{
found = FALSE;
for (; hso->iFrags < hso->cFrags; hso->iFrags++)
{
if (strncmp (hso->aFrags[hso->iFrags], hso->lpszFragment, lstrlen (hso->lpszFragment)) == 0)
{
found = TRUE;
lpszFoundFragment = hso->aFrags[hso->iFrags];
break;
}
}
}
if (!found)
{
Error = ERROR_FILE_NOT_FOUND;
goto quit;
}
Error = CopyCEItoHII (
lpszFoundFragment,
lpFirstHistoryItemInfo,
lpdwFirstHistoryItemInfoBufferSize,
lpCEI);
quit:
if (Error != ERROR_SUCCESS)
{
SetLastError (Error);
if (hso->lpszPrefixedUrl)
LocalFree (hso->lpszPrefixedUrl);
if (hso->aFrags)
LocalFree (hso->aFrags);
if (hso->lpCEI)
{
UnlockUrlCacheEntryFile (hso->lpCEI->lpszSourceUrlName, 0);
LocalFree(hso->lpCEI);
}
if (hso)
LocalFree (hso);
return NULL;
}
if (lpCEI && !hso->lpCEI)
LocalFree (lpCEI);
return (HANDLE) hso;
}
HISTORYAPI_(BOOL)
FindNextHistoryItem(
IN HANDLE hEnumHandle,
OUT LPHISTORY_ITEM_INFO lpHistoryItemInfo,
IN OUT LPDWORD lpdwHistoryItemInfoBufferSize
)
/*++
Routine Description:
Searches through the History looking for URLs that match the search pattern,
and copies the HISTORY_ITEM_INFO into the buffer.
Arguments:
lpszUrlSearchPattern - The URL in question.
lpFirstHistoryItemInfo - Buffer that will hold the HISTORY_ITEM_INFO
lpdwFirstHistoryItemInfoBufferSize - IN: size of the lpHistoryItemInfo buffer
OUT: size of filled struct when successful
or necessary buffer size when failed
Return Value:
HANDLE
Success - Valid enumeration handle to pass into subsequent calls to
FindNextHistoryItem ().
Failure - NULL. Extended error can be retrieved from GetLastError()
ERROR_NOT_ENOUGH_MEMORY indicates the buffer is insufficient
--*/
{
DWORD Error = ERROR_SUCCESS;
LPCACHE_ENTRY_INFO lpCEI = NULL;
DWORD cbCEI = 0;
LPHISTORY_SEARCH_OBJ hso = NULL;
BOOL found = FALSE;
LPTSTR lpszFoundFragment;
if (!hEnumHandle)
{
SetLastError (ERROR_INVALID_PARAMETER);
return FALSE;
}
hso = (LPHISTORY_SEARCH_OBJ) hEnumHandle;
while (!found)
{
if (hso->aFrags)
{
//this means that there are only fragments to find
for (lpszFoundFragment = NULL; hso->iFrags < hso->cFrags; hso->iFrags++)
{
if (hso->lpszFragment)
{
if (strncmp (hso->aFrags[hso->iFrags], hso->lpszFragment, lstrlen (hso->lpszFragment)) == 0)
{
found = TRUE;
lpCEI = hso->lpCEI;
break;
}
}
else
{
found = TRUE;
break;
}
}
if (!found)
{
if (hso->lpszFragment)
{
Error = ERROR_FILE_NOT_FOUND;
goto quit;
}
else
{
//this means that we went through all the frags
//we need to drop through and find the Cache Entry that matches
Error = ERROR_SUCCESS;
ASSERT (hso->lpCEI);
ASSERT (hso->aFrags);
lpCEI = hso->lpCEI; //reuse the buffer if possible
LocalFree (hso->aFrags);
hso->lpCEI = NULL;
hso->aFrags = NULL;
}
}
else
{
lpszFoundFragment = hso->aFrags[hso->iFrags];
lpCEI = hso->lpCEI;
}
}
else
{
if (!lpCEI)
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, DEFAULT_CEI_BUFFER_SIZE);
if (!lpCEI)
{
Error = ERROR_INTERNAL_ERROR;
goto quit;
}
while (TRUE)
{
if (!FindNextUrlCacheEntry (hso->hEnum,
lpCEI,
&cbCEI))
{
Error = GetLastError ();
if (Error == ERROR_NOT_ENOUGH_MEMORY)
{
LocalFree (lpCEI);
lpCEI = (LPCACHE_ENTRY_INFO) LocalAlloc (LPTR, cbCEI);
if (!lpCEI)
{
Error = ERROR_INTERNAL_ERROR;
goto quit;
}
}
else
goto quit;
}
else
break;
}
Error = GetFragmentsFromCEI (lpCEI, &(hso->aFrags), &(hso->cFrags));
switch (Error)
{
case ERROR_FILE_NOT_FOUND: //only the default URL is used
found = TRUE;
Error = ERROR_SUCCESS;
break;
case ERROR_SUCCESS: //first return the default URL next call will get frags
hso->lpCEI = lpCEI;
found = TRUE;
break;
default:
goto quit;
break;
}
}
}
Error = CopyCEItoHII(
lpszFoundFragment,
lpHistoryItemInfo,
lpdwHistoryItemInfoBufferSize,
lpCEI);
quit:
if (lpCEI && !hso->lpCEI)
{
UnlockUrlCacheEntryFile (lpCEI->lpszSourceUrlName, 0);
LocalFree (lpCEI);
}
if (Error != ERROR_SUCCESS)
{
SetLastError (Error);
return FALSE;
}
else
return TRUE;
}
HISTORYAPI_(BOOL)
FindCloseHistory (
IN HANDLE hEnumHandle
)
{
LPHISTORY_SEARCH_OBJ hso;
HANDLE hEnum;
//possibly we should be keeping track of valid hso's i dunno
if (!hEnumHandle)
{
SetLastError (ERROR_INVALID_PARAMETER);
return FALSE;
}
hso = (LPHISTORY_SEARCH_OBJ) hEnumHandle;
hEnum = hso->hEnum;
if (hso->aFrags)
LocalFree (hso->aFrags);
if (hso->lpszPrefixedUrl)
LocalFree(hso->lpszPrefixedUrl);
if (hso->lpCEI)
{
LocalFree (hso->lpCEI);
}
LocalFree (hso);
return FindCloseUrlCache (hEnum);
}
BOOL
DLLHistoryEntry(
IN HINSTANCE DllHandle,
IN DWORD Reason,
IN LPVOID Reserved
)
/*++
Routine Description:
Performs global initialization and termination for all protocol modules.
This function only handles process attach and detach which are required for
global initialization and termination, respectively. We disable thread
attach and detach. New threads calling Wininet APIs will get an
INTERNET_THREAD_INFO structure created for them by the first API requiring
this structure
Arguments:
DllHandle - handle of this DLL. Unused
Reason - process attach/detach or thread attach/detach
Reserved - if DLL_PROCESS_ATTACH, NULL means DLL is being dynamically
loaded, else static. For DLL_PROCESS_DETACH, NULL means DLL
is being freed as a consequence of call to FreeLibrary()
else the DLL is being freed as part of process termination
Return Value:
BOOL
Success - TRUE
Failure - FALSE. Failed to initialize
--*/
{
BOOL ok;
DWORD error;
// UNREFERENCED_PARAMETER(DllHandle);
//
// perform global dll initialization, if any.
//
switch (Reason) {
case DLL_PROCESS_ATTACH:
// error = DllProcessAttachDiskCache();
//
// we switch off thread library calls to avoid taking a hit for every
// thread creation/termination that happens in this process, regardless
// of whether Internet APIs are called in the thread.
//
// If a new thread does make Internet API calls that require a per-thread
// structure then the individual API will create one
//
// DisableThreadLibraryCalls(DllHandle);
break;
case DLL_PROCESS_DETACH:
if (Reserved != NULL) {
//
// Only Cleanup if there is a FreeLibrary() call.
//
break;
}
// DllProcessDetachDiskCache();
break;
}
return (TRUE);
}