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

1138 lines
23 KiB
C++

/*
* store.cpp
*/
#define STORE_DOT_CPP
#include "pch.h"
#include "store.hpp"
#define MAX_STORES 50
#define CHUNK_SIZE 4096L
static SRVTYPEINFO gtypeinfo[stError] =
{
{"http://", 0, 7}, // inetCopy, 7},
{"https://", 0, 8}, // inetCopy, 8},
{"", 0, 0} // fileCopy, 0}
};
static HINTERNET ghint = INVALID_HANDLE_VALUE;
static HWND ghwnd = 0;
static UINT_PTR gopts;
static char *gproxy = NULL;
static char *gdstore = NULL;
static BOOL gcancel = false;
void disperror(
DWORD err,
char *file
)
{
if (!err || err == ERROR_REQUEST_ABORTED)
return;
if (err == ERROR_FILE_NOT_FOUND)
dprint("%s - file not found\n", file);
else
dprint("%s\n %s\n", file, FormatStatus(err));
}
void
dumpproxyinfo(
VOID
)
{
HINTERNET hint;
INTERNET_PROXY_INFO *pi;
DWORD size;
BOOL rc;
DWORD err;
hint = (gproxy) ? ghint : NULL;
size = 0;
rc = InternetQueryOption(hint, INTERNET_OPTION_PROXY, NULL, &size);
if (rc) {
SetLastError(ERROR_INVALID_DATA);
return;
}
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
return;
pi = (INTERNET_PROXY_INFO *)LocalAlloc(LPTR, size);
if (!pi)
return;
ZeroMemory(pi, size);
rc = InternetQueryOption(hint, INTERNET_OPTION_PROXY, pi, &size);
if (rc && pi->lpszProxy && *pi->lpszProxy)
dprint("Using proxy server: %s\n", pi->lpszProxy);
if (pi)
LocalFree(pi);
}
DWORD
fixerror(
DWORD err
)
{
if (err == ERROR_PATH_NOT_FOUND)
return ERROR_FILE_NOT_FOUND;
return err;
}
void
setdprint(
DBGPRINT fndprint
)
{
gdprint = fndprint;
}
void
setproxy(
char *proxy
)
{
gproxy = proxy;
}
void
setdstore(
char *dstore
)
{
gdstore = dstore;
}
void
SetParentWindow(
HWND hwnd
)
{
ghwnd = hwnd;
}
void
SetStoreOptions(
UINT_PTR opts
)
{
gopts = opts;
}
DWORD
GetStoreType(
LPCSTR sz
)
{
DWORD i;
for (i = 0; i < stError; i++) {
if (!_strnicmp(sz, gtypeinfo[i].tag, gtypeinfo[i].taglen)) {
return i;
}
}
return stError;
}
BOOL
ParsePath(
IN LPCSTR ipath,
OUT LPSTR site,
OUT LPSTR path,
OUT LPSTR file,
IN BOOL striptype
)
{
char sz[_MAX_PATH + 1];
char *c;
char *p;
DWORD type;
assert(ipath && site && path);
*site = 0;
*path = 0;
if (file)
*file = 0;
if (!CopyString(sz, ipath, _MAX_PATH))
return false;
ConvertBackslashes(sz);
// get start of site string
type = GetStoreType(sz);
p = sz + gtypeinfo[type].taglen;
// there has to be at least a site
c = strchr(p, '/');
if (!c) {
strcpy(site, p); // SECURITy: ParsePath is a safe function.
return true;
}
// copy site name
*c = 0;
strcpy(site, (striptype) ? p : sz); // SECURITy: ParsePath is a safe function.
p = c + 1;
// if no file parameter, include the file in the path parameter
if (!file) {
strcpy(path, p); // SECURITy: ParsePath is a safe function.
return true;
}
// look for path in the middle
for (c = p + strlen(p); p < c; c--) {
if (*c == '/') {
*c = 0;
strcpy(path, p); // SECURITy: ParsePath is a safe function.
p = c + 1;
break;
}
}
strcpy(file, p); // SECURITy: ParsePath is a safe function.
return true;
}
static char
ChangeLastChar(
LPSTR sz,
char newchar
)
{
char c;
DWORD len;
len = strlen(sz) - 1;
c = sz[len];
sz[len] = newchar;
return c;
}
static BOOL
ReplaceFileName(
LPSTR path,
LPCSTR file,
DWORD size
)
{
char *p;
assert(path && *path && file && *file);
for (p = path + strlen(path) - 1; *p; p--) {
if (*p == '\\' || *p == '/') {
CopyString(++p, file, size - (ULONG)(ULONG_PTR)(p - path));
return true;
}
}
return false;
}
DWORD CALLBACK cbCopyProgress(
LARGE_INTEGER TotalFileSize, // file size
LARGE_INTEGER TotalBytesTransferred, // bytes transferred
LARGE_INTEGER StreamSize, // bytes in stream
LARGE_INTEGER StreamBytesTransferred, // bytes transferred for stream
DWORD dwStreamNumber, // current stream
DWORD dwCallbackReason, // callback reason
HANDLE hSourceFile, // handle to source file
HANDLE hDestinationFile, // handle to destination file
LPVOID lpData // from CopyFileEx
)
{
Store *store = (Store *)lpData;
store->setsize(TotalFileSize.QuadPart);
store->setbytes(TotalBytesTransferred.QuadPart);
store->progress();
if (querycancel()) {
store->setbytes((LONGLONG)-1);
store->progress();
return PROGRESS_CANCEL;
}
return PROGRESS_CONTINUE;
}
BOOL
ReadFilePtr(
LPSTR path,
DWORD size
)
{
BOOL rc;
HANDLE hptr;
DWORD fsize;
DWORD cb;
LPSTR p;
char ptrfile[MAX_PATH + 1];
char file[MAX_PATH + 1];
assert(path && *path);
rc = false;
// check for existance of file pointer
if (!CopyString(ptrfile, path, _MAX_PATH))
return false;
if (!ReplaceFileName(ptrfile, "file.ptr", DIMA(ptrfile)))
return false;
if (FileStatus(ptrfile))
return false;
hptr = CreateFile(ptrfile,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);
if (hptr == INVALID_HANDLE_VALUE)
return false;
// test validity of file pointer
fsize = GetFileSize(hptr, NULL);
if (!fsize || fsize > MAX_PATH)
goto cleanup;
// read it
ZeroMemory(file, _MAX_PATH * sizeof(path[0]));
if (!ReadFile(hptr, file, fsize, &cb, 0))
goto cleanup;
if (cb != fsize)
goto cleanup;
rc = true;
// trim string down to the CR
for (p = file; *p; p++) {
if (*p == 10 || *p == 13)
{
*p = 0;
break;
}
}
CopyString(path, file, size);
dprint("%s\n", ptrfile);
cleanup:
// done
if (hptr)
CloseHandle(hptr);
return rc;
}
#ifdef USE_INERROR
DWORD inerror(DWORD error)
{
char *detail = NULL;
DWORD iErr;
DWORD len = 0;
static char message[256]="";
if (error == ERROR_SUCCESS)
return error;
FormatMessage(FORMAT_MESSAGE_FROM_HMODULE,
GetModuleHandle("wininet.dll"),
error,
0,
message,
256,
NULL);
EnsureTrailingCR(message);
dprint("Internet error code: %d\n Message: %s\n", error, message);
if (error != ERROR_INTERNET_EXTENDED_ERROR)
return error;
InternetGetLastResponseInfo(&iErr, NULL, &len);
if (!len)
return error;
detail = (char *)LocalAlloc(LPTR, len + 1000);
if (!detail)
return error;
if (!InternetGetLastResponseInfo(&iErr, (LPTSTR)detail, &len))
return error;
dprint(detail);
LocalFree(detail);
return error;
}
#endif
Store *gstores[MAX_STORES];
DWORD gcstores = 0;
Store *
FindStore(
PCSTR name
)
{
DWORD i;
for (i = 0; i < gcstores; i++) {
if (gstores[i] && !strcmp((gstores[i])->name(), name))
return gstores[i];
}
return NULL;
}
Store *
AddStore(
PCSTR name
)
{
DWORD type;
Store *store;
type = GetStoreType(name);
switch (type) {
case stUNC:
store = new StoreUNC();
break;
case stHTTP:
case stHTTPS:
store = new StoreHTTP();
break;
case stError:
default:
SetLastError(ERROR_INVALID_PARAMETER);
return NULL;
}
if (!store) {
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
return NULL;
}
store->assign(name);
if (gcstores >= MAX_STORES) {
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
return false;
}
if (!gcstores)
ZeroMemory(gstores, sizeof(gstores));
gstores[gcstores] = store;
gcstores++;
return store;
}
Store *
GetStore(
PCSTR name
)
{
Store *store;
store = FindStore(name);
if (!store)
store = AddStore(name);
return store;
}
BOOL
DeleteStore(
Store *store
)
{
DWORD i;
for (i = 0; i < gcstores; i++) {
if (gstores[i] == store) {
gstores[i] = NULL;
delete store;
return true;
}
}
return false;
}
DWORD Store::assign(PCSTR name)
{
CopyStrArray(m_name, name);
m_type = GetStoreType(name);
m_flags = 0;
return m_type;
}
char *Store::target()
{
return m_tpath;
}
BOOL Store::init()
{
if (m_flags & SF_DISABLED)
return false;
*m_tpath = 0;
return true;
}
BOOL Store::ping()
{
return false;
}
BOOL Store::open(PCSTR rpath, PCSTR file)
{
CopyStrArray(m_rpath, rpath ? rpath : "");
CopyStrArray(m_file, file ? file : "");
return true;
}
VOID Store::close()
{
return;
}
BOOL Store::get(PCSTR trg)
{
pathcpy(m_tpath, trg, m_rpath, DIMA(m_tpath));
pathcat(m_tpath, m_file, DIMA(m_tpath));
EnsurePathExists(m_tpath, m_epath, DIMA(m_epath));
return true;
}
BOOL Store::copy(PCSTR rpath, PCSTR file, PCSTR trg)
{
if (m_flags & SF_DISABLED)
return false;
return true;
}
BOOL Store::progress()
{
SYSTEMTIME st;
if (!*m_file)
return true;
// Do not condense these print statements into single lines.
// They are split up to work with dbghelp's backspace filtering.
switch (m_bytes)
{
case (LONGLONG)-1:
eprint("\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
eprint("cancelled \n");
SetLastError(ERROR_REQUEST_ABORTED);
break;
case 0:
dprint("%s from %s: %ld bytes - ", m_file, m_name, m_size);
eprint("\b%12ld ", 0);
break;
default:
GetSystemTime(&st);
if (st.wSecond != m_tic)
eprint("\b\b\b\b\b\b\b\b\b\b\b\b%12ld", m_bytes);
m_tic = st.wSecond;
break;
}
if (m_bytes && (m_bytes == m_size)) {
eprint("\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
eprint("copied \n");
}
return true;
}
void Store::setsize(LONGLONG size)
{
m_size = size;
}
void Store::setbytes(LONGLONG bytes)
{
m_bytes = bytes;
}
BOOL StoreUNC::get(PCSTR trg)
{
BOOL rc;
DWORD err;
char c;
pathcpy(m_spath, m_name, m_rpath, DIMA(m_spath));
pathcat(m_spath, m_file, DIMA(m_spath));
Store::get(trg);
do {
// try to copy the file, first
if (gdprint)
rc = CopyFileEx(m_spath, m_tpath, cbCopyProgress, this, &gcancel, COPY_FILE_FAIL_IF_EXISTS);
else
rc = CopyFile(m_spath, m_tpath, true);
if (rc)
return true;
err = GetLastError();
err = fixerror(err);
if (err != ERROR_FILE_NOT_FOUND) {
disperror(err, m_spath);
return false;
}
// now try to uncompress the file
c = ChangeLastChar(m_spath, '_');
if (!FileStatus(m_spath)) {
rc = UncompressFile(m_spath, m_tpath);
if (!rc)
disperror(GetLastError(), m_spath);
return rc;
}
ChangeLastChar(m_spath, c);
// try to read from a file pointer
} while (ReadFilePtr(m_spath, DIMA(m_spath)));
err = GetLastError();
err = fixerror(err);
disperror(err, m_spath);
return false;
}
BOOL StoreUNC::copy(PCSTR rpath, PCSTR file, PCSTR trg)
{
if (!Store::copy(rpath, file, trg))
return false;
if (!open(rpath, file))
return false;
return get(trg);
}
BOOL StoreUNC::ping()
{
BOOL rc;
CopyStrArray(m_spath, m_name);
EnsureTrailingBackslash(m_spath);
CatStrArray(m_spath, "pingme.txt");
rc = (GetFileAttributes(m_spath) == 0xFFFFFFFF);
if (!rc)
m_flags |= SF_DISABLED;
return rc;
}
BOOL StoreInet::init()
{
char sz[_MAX_PATH];
static char uasz[_MAX_PATH] = "";
// internet handle is null, then we know from previous
// attempts that it can't be opened, so bail
if (!ghint)
return false;
if (!*uasz) {
CopyStrArray(uasz, "Microsoft-Symbol-Server/");
CatStrArray(uasz, VER_PRODUCTVERSION_STR);
}
*m_spath = 0;
*m_rpath = 0;
*m_file = 0;
ParsePath(m_name, m_site, sz, NULL, true);
CopyStrArray(m_srpath, "/");
CatStrArray(m_srpath, sz);
if (ghint == INVALID_HANDLE_VALUE) {
ghint = InternetOpen(uasz,
(gproxy) ? INTERNET_OPEN_TYPE_PROXY : INTERNET_OPEN_TYPE_PRECONFIG,
gproxy,
NULL,
0);
if (!ghint)
return false;
dumpproxyinfo();
}
if (m_hsite)
return true;
m_hsite = InternetConnect(ghint,
m_site,
m_port,
NULL,
NULL,
m_service,
0,
NULL);//m_context ? (DWORD_PTR)&m_context : NULL);
if (!m_hsite)
return false;
return true;
}
BOOL StoreInet::open(PCSTR rpath, PCSTR file)
{
Store::open(rpath, file);
CopyStrArray(m_spath, m_srpath);
pathcat(m_spath, m_rpath, DIMA(m_spath));
return true;
}
BOOL StoreInet::copy(PCSTR rpath, PCSTR file, PCSTR trg)
{
BOOL rc;
DWORD err;
char cfile[MAX_PATH + 1];
char c;
if (!Store::copy(rpath, file, trg))
return false;
if (m_flags & SF_INTERNET_DISABLED)
return false;
// open and copy the file
if (open(rpath, file)) {
rc = get(trg);
close();
return rc;
}
// if file wasn't found, look for a compressed version
err = GetLastError();
if (err != ERROR_FILE_NOT_FOUND)
return false;
CopyStrArray(cfile, file);
c = ChangeLastChar(cfile, '_');
if (!open(rpath, cfile)) {
dprint("%s%s%s not found\n", gtypeinfo[m_type].tag, m_site, m_spath);
return false;
}
rc = get(trg);
close();
if (!rc)
return false;
// if we found a compressed version, expand it
CopyStrArray(cfile, m_tpath);
ChangeLastChar(m_tpath, c);
rc = UncompressFile(cfile, m_tpath);
DeleteFile(cfile);
if (!rc)
DeleteFile(m_tpath);
return rc;
}
BOOL StoreInet::ping()
{
return open("", "pingme.txt");
}
BOOL StoreHTTP::init()
{
BOOL rc;
m_iflags = INTERNET_FLAG_RELOAD | INTERNET_FLAG_DONT_CACHE | INTERNET_FLAG_KEEP_CONNECTION;
m_service = INTERNET_SERVICE_HTTP;
m_context = 0;
if (m_type == stHTTPS) {
m_port = INTERNET_DEFAULT_HTTPS_PORT;
m_iflags |= INTERNET_FLAG_SECURE;
} else {
m_port = INTERNET_DEFAULT_HTTP_PORT;
}
*m_srpath = 0;
return StoreInet::init();
}
BOOL StoreHTTP::open(PCSTR rpath, PCSTR file)
{
DWORD err = ERROR_NOT_FOUND;
Store::open(rpath, file);
CopyStrArray(m_spath, m_srpath);
pathcat(m_spath, m_rpath, DIMA(m_spath));
pathcat(m_spath, m_file, DIMA(m_spath));
ConvertBackslashes(m_spath);
close();
m_hfile = HttpOpenRequest(m_hsite,
"GET",
m_spath,
HTTP_VERSION,
NULL,
NULL,
m_iflags,
0);
if (!m_hfile)
goto error;
err = fileinfo();
if (!err)
return true;
error:
close();
SetLastError(err);
return false;
}
VOID StoreHTTP::close()
{
DWORD err;
if (!m_hfile)
return;
// InternetCloseHandle resets last error to zero.
// Preserve it and restore it afterwards.
err = GetLastError();
InternetCloseHandle(m_hfile);
if (err)
SetLastError(err);
m_hfile = 0;
}
DWORD StoreHTTP::fileinfo()
{
BOOL rc;
DWORD err;
DWORD status;
DWORD cbstatus;
DWORD index;
DWORD cbsize;
#ifdef PROXYTEST
dprint("FILE %s\n", m_spath);
#endif
do {
err = request();
if (err != ERROR_SUCCESS)
return err;
index = 0;
m_size = 0;
cbsize = sizeof(m_size);
rc = HttpQueryInfo(m_hfile,
HTTP_QUERY_CONTENT_LENGTH | HTTP_QUERY_FLAG_NUMBER,
&m_size,
&cbsize,
&index);
if (!rc) {
if (GetLastError())
return err;
return ERROR_INTERNET_EXTENDED_ERROR;
}
index = 0;
cbstatus = sizeof(status);
rc = HttpQueryInfo(m_hfile,
HTTP_QUERY_STATUS_CODE | HTTP_QUERY_FLAG_NUMBER,
&status,
&cbstatus,
&index);
if (!rc) {
if (GetLastError())
return err;
return ERROR_INTERNET_EXTENDED_ERROR;
}
switch (status)
{
case HTTP_STATUS_DENIED: // need a valid login?
// dprint("status HTTP_STATUS_DENIED\n");
err = prompt(m_hfile, ERROR_INTERNET_INCORRECT_PASSWORD);
// user entered a password - try again
if (err == ERROR_INTERNET_FORCE_RETRY)
break;
// user cancelled
m_flags |= SF_DISABLED;
return ERROR_NOT_READY;
case HTTP_STATUS_PROXY_AUTH_REQ:
// dprint("status HTTP_STATUS_PROXY_AUTH_REQ\n");
err = prompt(m_hfile, err);
// user entered a password - try again
if (err == ERROR_INTERNET_FORCE_RETRY)
break;
// user cancelled
m_flags |= SF_INTERNET_DISABLED;
return ERROR_NOT_READY;
case HTTP_STATUS_FORBIDDEN:
// dprint("status HTTP_STATUS_FORBIDDEN\n");
m_flags |= SF_DISABLED;
return ERROR_ACCESS_DENIED;
case HTTP_STATUS_NOT_FOUND:
// dprint("status HTTP_STATUS_NOT_FOUND\n");
return ERROR_FILE_NOT_FOUND;
case HTTP_STATUS_OK:
// dprint("status HTTP_STATUS_OK\n");
return ERROR_SUCCESS;
}
} while (err == ERROR_INTERNET_FORCE_RETRY);
return ERROR_INTERNET_EXTENDED_ERROR;
}
DWORD StoreHTTP::request()
{
DWORD err = ERROR_SUCCESS;
while (!HttpSendRequest(m_hfile, NULL, 0, NULL, 0))
{
err = GetLastError();
switch (err)
{
// These cases get input from the user in oder to try again.
case ERROR_INTERNET_INCORRECT_PASSWORD:
case ERROR_INTERNET_CLIENT_AUTH_CERT_NEEDED:
err = prompt(m_hfile, err);
if (err != ERROR_SUCCESS && err != ERROR_INTERNET_FORCE_RETRY)
{
err = ERROR_ACCESS_DENIED;
return err;
}
break;
// These cases get input from the user in order to try again.
// However, if the user bails, don't use this internet
// connection again in this session.
case ERROR_INTERNET_INVALID_CA:
case ERROR_INTERNET_SEC_CERT_DATE_INVALID:
case ERROR_INTERNET_SEC_CERT_CN_INVALID:
case ERROR_INTERNET_POST_IS_NON_SECURE:
err = prompt(m_hfile, err);
if (err != ERROR_SUCCESS && err != ERROR_INTERNET_FORCE_RETRY)
{
m_flags |= SF_DISABLED;
err = ERROR_NOT_READY;
return err;
}
break;
// no go - give up the channel
case ERROR_INTERNET_SECURITY_CHANNEL_ERROR:
m_flags |= SF_DISABLED;
err = ERROR_NOT_READY;
return err;
// Tell the user something went wrong and get out of here.
case ERROR_INTERNET_HTTP_TO_HTTPS_ON_REDIR:
default:
prompt(m_hfile, err);
return err;
}
}
return err;
}
DWORD StoreHTTP::prompt(HINTERNET hreq, DWORD err)
{
if (gopts & SSRVOPT_UNATTENDED)
return err;
if (!ghwnd)
ghwnd = GetDesktopWindow();
if (!ghwnd)
return err;
err = InternetErrorDlg(ghwnd,
hreq,
err,
FLAGS_ERROR_UI_FILTER_FOR_ERRORS |
FLAGS_ERROR_UI_FLAGS_GENERATE_DATA |
FLAGS_ERROR_UI_FLAGS_CHANGE_OPTIONS,
NULL);
return err;
}
BOOL StoreHTTP::get(PCSTR trg)
{
DWORD read;
DWORD written;
DWORD err = 0;
BYTE *buf;
BOOL rc = false;
HANDLE hf = INVALID_HANDLE_VALUE;
ULONG64 copied;
buf = (BYTE *)LocalAlloc(LPTR, CHUNK_SIZE);
if (!buf) {
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
return false;
}
Store::get(trg);
hf = CreateFile(m_tpath,
GENERIC_WRITE,
0,
NULL,
CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL,
NULL);
if (hf == INVALID_HANDLE_VALUE)
goto cleanup;
m_bytes = 0;
do
{
if (!progress())
goto cleanup;
rc = InternetReadFile(m_hfile,
(LPVOID)buf,
CHUNK_SIZE,
&read);
if (!rc || !read)
break;
rc = WriteFile(hf, (LPVOID)buf, read, &written, NULL);
m_bytes += written;
}
while (rc);
cleanup:
// if there was an error, save it and set it later
if (!err)
err = GetLastError();
disperror(err || GetLastError(), m_spath);
// If target file is open, close it.
if (hf != INVALID_HANDLE_VALUE)
CloseHandle(hf);
// free the memory
LocalFree(buf);
SetLastError(err);
return err ? false : true;
}
BOOL StoreHTTP::progress()
{
Store::progress();
if (!m_bytes || !querycancel())
return true;
setbytes((LONGLONG)-1);
Store::progress();
return false;
}