winamp/Src/Plugins/Input/in_nsv/BigLib.cpp

1428 lines
37 KiB
C++
Raw Permalink Normal View History

2024-09-24 13:54:57 +01:00
#ifdef WINAMPX
#define WIN32_LEAN_AND_MEAN 1
#include <windows.h>
#include "stdio.h"
#include "proxydt.h"
#include <winsock2.h>
#include ".\ws2tcpip.h"
#include ".\wininet.h"
#include "../jnetlib/jnetlib.h"
#include <urlmon.h>
extern void SendMetadata( char *data, int arg );
HRESULT JNetLibDownloadToFile(LPVOID lpUnused1, LPSTR lpWPADLocation, LPSTR lpTempFile, LPVOID lpUnused2, LPVOID lpUnused3);
enum {
BK_UNKNOWN = 0,
BK_IE4 = 1,
BK_NETSCAPE4 = 2,
BK_NETSCAPE6 = 3,
BK_MOZILLA = 4,
BK_FIREFOX = 5
};
// browser info struct
typedef struct{
LPSTR lpName;
BOOL bSupported;
}BK_INFO;
// browser info
BK_INFO BrowserInfo[] = {
"Unknown", FALSE,
"IE 4.0+", TRUE,
"Netscape 4 or 5", FALSE,
"Netscape 6+", TRUE,
"Mozilla", TRUE,
"Firefox", TRUE
};
// Global variables
int gBrowserKind = BK_UNKNOWN;
int gTryAuto = 1;
// Exported C functions
extern "C" BOOL ProxyInit();
extern "C" void ProxyDeInit();
extern "C" int ResolvProxyFromURL(LPSTR lpURL, LPSTR lpHostname, LPSTR lpDest);
// Global C functions
BOOL IsIEProxySet();
int GetIESettings();
int ResolveURL_IE(LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort);
BOOL IsFirefoxProxySet();
int GetFirefoxSettings();
int ResolveURL_Firefox(LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort);
BOOL IsMozillaProxySet();
int GetMozillaSettings();
int ResolveURL_Mozilla(LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort);
int GetDefaultBrowser();
int ReadWPADFile(LPSTR lpWPADLocation, LPSTR pIPAddress, int *pnPort);
int GetFirefoxOrMozillaSettings(BOOL bFirefox);
BOOL IsFirefoxOrMozillaProxySet(BOOL bFirefox);
int ResolveURL_MozillaOrFirefox(BOOL bFirefox, LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort);
// exported functions
extern "C" BOOL ProxyInit()
{
BOOL bRet;
bRet = FALSE;
gBrowserKind = GetDefaultBrowser();
switch(gBrowserKind) {
case BK_IE4:
bRet = IsIEProxySet();
break;
case BK_MOZILLA:
bRet = IsMozillaProxySet();
break;
case BK_FIREFOX:
bRet = IsFirefoxProxySet();
break;
}
return bRet;
}
extern "C" void ProxyDeInit()
{
}
extern "C" int ResolvProxyFromURL(LPSTR lpURL, LPSTR lpHostname, LPSTR lpDest)
{
// lpURL = URL to resolve
// lpHostname = hostname
// lpDest = where to store the result, such as "www.proxyserver.com:8080"
char szIPAddress[MAX_PATH] = {0};
int ret, nPort=0;
lpDest[0]=0;
if(lpURL && lpHostname && lpDest) {
switch(gBrowserKind) {
case BK_IE4:
ret = ResolveURL_IE(lpURL, lpHostname, szIPAddress, sizeof(szIPAddress), &nPort);
break;
case BK_MOZILLA:
ret = ResolveURL_Mozilla(lpURL, lpHostname, szIPAddress, sizeof(szIPAddress), &nPort);
break;
case BK_FIREFOX:
ret = ResolveURL_Firefox(lpURL, lpHostname, szIPAddress, sizeof(szIPAddress), &nPort);
break;
}
if(ret == 0) {
if ( szIPAddress[0] )
{
wsprintf(lpDest, "%s:%d", szIPAddress, nPort);
return 1;
}
else return 0;
}
else return 0;
}
else return -1;
}
int GetDefaultBrowser()
{
DWORD dwSize, dwType;
TCHAR valueBuf[MAX_PATH] = {0};
DWORD valueSize = sizeof(valueBuf);
HKEY hKey;
long lRet;
memset(valueBuf, 0, sizeof(valueBuf));
lRet = RegOpenKeyEx(HKEY_CLASSES_ROOT, "http\\shell\\open\\ddeexec\\Application", 0, KEY_READ, &hKey);
if (lRet == ERROR_SUCCESS) {
dwSize = valueSize;
lRet = RegQueryValueEx(hKey, "", NULL, &dwType, (LPBYTE)valueBuf, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_SZ) {
if (_tcsicmp(_T("NSShell"), valueBuf) == 0) { //NS 4.x
return BK_NETSCAPE4;
} else if (_tcsicmp(_T("IExplore"), valueBuf) == 0) { //IE 4+
return BK_IE4;
} else if (_tcsicmp(_T("Mozilla"), valueBuf) == 0) { //Mozilla
return BK_MOZILLA;
} else if (_tcsicmp(_T("Firefox"), valueBuf) == 0) { //Firefox
return BK_FIREFOX;
}
}
}
RegCloseKey(hKey);
lRet = RegOpenKeyEx(HKEY_CLASSES_ROOT, "http\\shell\\open\\command", 0, KEY_READ, &hKey);
if(lRet == ERROR_SUCCESS) {
dwSize = valueSize;
lRet = RegQueryValueEx(hKey, "", NULL, &dwType, (LPBYTE)valueBuf, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_SZ) {
if(strstr(valueBuf, "MOZILLA")) {
return BK_MOZILLA;
}
if(strstr(valueBuf, "NETSCAPE")) {
return BK_MOZILLA;
}
if(strstr(valueBuf, "FIREFOX")) {
return BK_FIREFOX;
}
}
}
RegCloseKey(hKey);
return BK_UNKNOWN;
}
////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// helper functions for JSProxy.dll
DWORD __stdcall ResolveHostName(LPSTR lpszHostName, LPSTR lpszIPAddress, LPDWORD lpdwIPAddressSize);
BOOL __stdcall IsResolvable(LPSTR lpszHost);
DWORD __stdcall GetIPAddress(LPSTR lpszIPAddress, LPDWORD lpdwIPAddressSize);
BOOL __stdcall IsInNet(LPSTR lpszIPAddress, LPSTR lpszDest, LPSTR lpszMask);
// functions to get IE checkbox state
BOOL GetAutomaticallyDetectSettingsCheckboxState();
BOOL GetUseAProxyServerForYourLanCheckboxState();
BOOL GetAutomaticConfigurationScriptCheckboxState();
BOOL GetBypassProxyServerForLocalAddressesCheckboxState();
// functions to actually get an IP address and port # of the proxy server
int GetAutomaticDetectSettings(LPSTR lpIPAddress, int *pnPort);
int GetProxyServerForLanProxySettings(LPSTR lpIPAddress, int *pnPort);
int GetAutoConfigScriptProxySettings(LPSTR lpIPAddress, int *pnPort);
// various helper functions
BOOL IsDirect(LPSTR proxy);
BOOL IsAProxy(LPSTR proxy);
void reportFuncErr(TCHAR* funcName);
char * strstri(LPSTR lpOne, LPSTR lpTwo);
int GetProxyIP(LPSTR proxy, LPSTR szProxyIP);
int GetProxyPort(LPSTR proxy);
// some global variables
char gszURL[1025] = {0};
char gszHost[256] = {0};
// returns TRUE if the user has set a proxy in IE
BOOL IsIEProxySet()
{
BOOL bAutomaticallyDetectSettings = GetAutomaticallyDetectSettingsCheckboxState();
BOOL bUseAutomaticConfigurationScript = GetAutomaticConfigurationScriptCheckboxState();
BOOL bUseAProxyServerForYourLan = GetUseAProxyServerForYourLanCheckboxState();
if(bAutomaticallyDetectSettings || bUseAutomaticConfigurationScript || bUseAProxyServerForYourLan) {
return TRUE;
}
return FALSE;
}
int ResolveURL_IE(LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort)
{
// get the state of the four checkboxes in the proxy settings dialog for IE
BOOL bAutomaticallyDetectSettings = GetAutomaticallyDetectSettingsCheckboxState();
BOOL bUseAutomaticConfigurationScript = GetAutomaticConfigurationScriptCheckboxState();
BOOL bUseAProxyServerForYourLan = GetUseAProxyServerForYourLanCheckboxState();
//BOOL bBypassProxyServerForLocalAddresses = GetBypassProxyServerForLocalAddressesCheckboxState();
int ret;
lstrcpyn(gszURL, lpURL, 1025);
lstrcpyn(gszHost, lpHostname, 256);
// if nothing checked, return
if(!bAutomaticallyDetectSettings && !bUseAutomaticConfigurationScript && !bUseAProxyServerForYourLan) {
return 0;
}
// if all three checkboxes on...
if(bAutomaticallyDetectSettings && gTryAuto)
{
// try the automatic configuration next
ret = GetAutomaticDetectSettings(lpIPAddress, pnPort);
if(ret == 0 && *pnPort) {
return 0;
}
gTryAuto = 0;
}
if ( bUseAutomaticConfigurationScript)
{
// try the automatic config script method first
ret = GetAutoConfigScriptProxySettings(lpIPAddress, pnPort);
if(ret == 0 && *pnPort ) {
return 0;
}
}
if ( bUseAProxyServerForYourLan)
{
// if still no success, try the "Use a proxy server for your lan" settings
ret = GetProxyServerForLanProxySettings(lpIPAddress, pnPort);
if(ret == 0 && *pnPort) {
return 0;
}
}
// no success...
return 0;
}
// handles the "Automatically Detect" checkbox
int GetAutomaticDetectSettings(LPSTR lpIPAddress, int *pnPort)
{
// By not specifying a domain name, Windows uses the local domain name,
// so form an http request to go to http://wpad/wpad.dat,
// store results in szWPADLocation and call URLDownloadToFileA()
if(lpIPAddress && pnPort) {
// download wpad.dat from the URL in szURL
return ReadWPADFile("http://wpad/wpad.dat", lpIPAddress, pnPort);
}
return -1;
}
// handles the "Use automatic configuration script" checkbox
int GetAutoConfigScriptProxySettings(LPSTR lpIPAddress, int *pnPort)
{
DWORD dwType, dwSize;
HKEY hKey;
char szWPADLocation[MAX_PATH] = {0};
long lRet;
int retval = -1;
if(!lpIPAddress) {
return retval;
}
if(!pnPort) {
return retval;
}
// use the registry read of HKCU\\software\microsoft\windows\current version\internet settings to see if "Use Automatic Configuration Script" is checked
lstrcpyn(szWPADLocation, "", MAX_PATH);
lRet = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings", 0, KEY_READ, &hKey);
if (lRet == ERROR_SUCCESS) {
dwSize = sizeof(szWPADLocation);
lRet = RegQueryValueEx(hKey, "AutoConfigURL", NULL, &dwType, (LPBYTE)szWPADLocation, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_SZ) {
retval = ReadWPADFile(szWPADLocation, lpIPAddress, pnPort);
}
}
RegCloseKey(hKey);
return retval; //0 = success
}
// handles the "Use a proxy server for your LAN" checkbox
int GetProxyServerForLanProxySettings(LPSTR lpIPAddress, int *pnPort)
{
DWORD dwType, dwSize;
HKEY hKey;
BOOL bDirectOrProxy;
char szProxy[MAX_PATH] = {0};
long lRet;
int retval = -1;
if(lpIPAddress) {
strcpy(lpIPAddress, "");
}
if(pnPort) {
*pnPort = 0;
}
lRet = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings", 0, KEY_READ, &hKey);
if (lRet == ERROR_SUCCESS) {
dwSize = sizeof(szProxy);
lRet = RegQueryValueEx(hKey, "ProxyServer", NULL, &dwType, (LPBYTE)szProxy, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_SZ) {
retval = 0;
bDirectOrProxy = FALSE;
if(IsDirect(szProxy)) {
// string is something like "DIRECT"
// It's a 'direct' kind of proxy.
bDirectOrProxy = TRUE;
// set the 'out' parameters
if(lpIPAddress) {
strcpy(lpIPAddress, "");
}
if(pnPort) {
*pnPort = 0;
}
}
if(IsAProxy(szProxy)) {
char szProxyIP[MAX_PATH] = {0};
// string is something like "D
bDirectOrProxy = TRUE;
GetProxyIP(szProxy, szProxyIP);
// It's a 'regular' kind of proxy, with an IP of %s and a port of %d\n", szProxyIP, GetProxyPort(szProxy)
// set the 'out' parameters
if(lpIPAddress) {
strcpy(lpIPAddress, szProxyIP);
}
if(pnPort) {
*pnPort = GetProxyPort(szProxy);
}
}
if(!bDirectOrProxy) {
// string is something like "10.0.0.1:4543"
LPSTR lpColon = NULL;
if ( isdigit(szProxy[0]) )
{
lpColon = strchr(szProxy, ':');
if(lpColon) {
*lpColon = '\0';
// set the 'out' parameters
if(lpIPAddress) {
strcpy(lpIPAddress, szProxy);
}
*lpColon = ':';
if(pnPort) {
*pnPort = GetProxyPort(szProxy);
}
}
}
else if ( strstr(szProxy,"http=") )
{
char *p = strstr(szProxy,"http=");
int offset= strlen("http=");
char *semi = strchr(p+offset, ';');
if(semi) {
*semi= '\0';
}
lpColon = strchr(p+offset, ':');
if(lpColon) {
*lpColon = '\0';
}
// set the 'out' parameters
if(lpIPAddress) {
strcpy(lpIPAddress, p+offset);
}
if (lpColon)
if(pnPort) {
*pnPort = (short)atoi(lpColon+1);
}
if ( !*pnPort ) *pnPort = 80;
}
else
{
if(lpIPAddress) {
strcpy(lpIPAddress, "");
}
if(pnPort) {
*pnPort = 0;
}
}
}
}
}
RegCloseKey(hKey);
return retval;
}
int ReadWPADFile(LPSTR lpWPADLocation, LPSTR lpIPAddress, int *pnPort)
{
// Declare function pointers for the three autoproxy functions
pfnInternetInitializeAutoProxyDll pInternetInitializeAutoProxyDll;
pfnInternetDeInitializeAutoProxyDll pInternetDeInitializeAutoProxyDll;
pfnInternetGetProxyInfo pInternetGetProxyInfo;
// Declare and populate an AutoProxyHelperVtbl structure, and then
// place a pointer to it in a containing AutoProxyHelperFunctions
// structure, which will be passed to InternetInitializeAutoProxyDll:
AutoProxyHelperVtbl Vtbl = {IsResolvable, GetIPAddress, ResolveHostName, IsInNet };
AutoProxyHelperFunctions HelperFunctions = { &Vtbl };
HMODULE hModJS;
HRESULT hr;
char szTempPath[MAX_PATH] = {0};
char szTempFile[MAX_PATH] = {0};
int retval = 0;
if(!(hModJS = LoadLibrary("jsproxy.dll"))) {
reportFuncErr("LoadLibrary");
return -1;
}
if(!(pInternetInitializeAutoProxyDll = (pfnInternetInitializeAutoProxyDll)
GetProcAddress(hModJS, "InternetInitializeAutoProxyDll")) ||
!(pInternetDeInitializeAutoProxyDll = (pfnInternetDeInitializeAutoProxyDll)
GetProcAddress(hModJS, "InternetDeInitializeAutoProxyDll")) ||
!(pInternetGetProxyInfo = (pfnInternetGetProxyInfo)
GetProcAddress(hModJS, "InternetGetProxyInfo"))) {
FreeLibrary(hModJS);
reportFuncErr("GetProcAddress");
return -1;
}
if(lpIPAddress)
{
strcpy(lpIPAddress, "");
}
if(pnPort) {
*pnPort = 0;
}
GetTempPathA(sizeof(szTempPath)/sizeof(szTempPath[0]), szTempPath);
GetTempFileNameA(szTempPath, "X", 2, szTempFile);
//printf(" Downloading %s ...\n", lpWPADLocation);
hr = JNetLibDownloadToFile(NULL, lpWPADLocation, szTempFile, NULL, NULL);
if(hr == S_OK) {
if(!pInternetInitializeAutoProxyDll(0, szTempFile, NULL, &HelperFunctions, NULL)) {
//printf(" Calling 'InternetInitializeAutoProxyDll' in JSPROXY.DLL failed\n (usually because 'Use Automatic Configuration Script' checkbox is OFF)\n");
pInternetDeInitializeAutoProxyDll(NULL, 0);
FreeLibrary(hModJS);
retval = -1;
}else{
// printf("\n InternetInitializeAutoProxyDll returned: %d\n", returnVal);
// Delete the temporary file
// (or, to examine the auto-config script, comment out the
// file delete and substitute the following printf call)
// printf("\n The auto-config script temporary file is:\n %s\n", szTempFile);
DeleteFileA(szTempFile);
DWORD dwSize = 0;
LPSTR pProxy = NULL;
if(!pInternetGetProxyInfo((LPSTR)gszURL, sizeof(gszURL), (LPSTR)gszHost, sizeof(gszHost), &pProxy, &dwSize)) {
reportFuncErr("InternetGetProxyInfo");
retval = -1;
}else{
// printf("\n Proxy is: %s\n", proxy);
if(IsDirect(pProxy)) {
//printf(" It's a 'direct' kind of proxy.\n");
// set the 'out' parameters
if(lpIPAddress) {
strcpy(lpIPAddress, "");
}
if(pnPort) {
*pnPort = 0;
}
}
if(IsAProxy(pProxy)) {
char szProxyIP[MAX_PATH] = {0};
GetProxyIP(pProxy, szProxyIP);
//printf(" It's a 'regular' kind of proxy, with an IP of %s and a port of %d\n", szProxyIP, GetProxyPort(szProxy));
// set the 'out' parameters
if(lpIPAddress) {
strcpy(lpIPAddress, szProxyIP);
}
if(pnPort) {
*pnPort = GetProxyPort(pProxy);
}
}
}
}
}else{
//printf(" Error downloading %s (hr=0x%X)\n", lpWPADLocation, hr);
// there is no proxy, go direct
if(lpIPAddress) {
strcpy(lpIPAddress, "");
}
if(pnPort) {
*pnPort = 0;
}
retval = 0;
}
if(!pInternetDeInitializeAutoProxyDll(NULL, 0)) {
reportFuncErr("InternetDeInitializeAutoProxyDll");
}
return retval; // 0 = success
}
// Puts "10.0.0.1" into lpDest from a string like "PROXY 10.0.0.1:8088"
// Returns 0 if success, -1 if an error
int GetProxyIP(LPSTR lpProxy, LPSTR lpDest)
{
LPSTR lpData;
LPSTR lpLastColon;
BOOL bDone;
char szProxy[MAX_PATH] = {0};
int ret = 0;
if(lpProxy && lpDest) {
lstrcpyn(szProxy, lpProxy, MAX_PATH);
// find the last ":" in the string
lpLastColon = NULL;
lpData = szProxy;
while(*lpData) {
if(*lpData == ':') {
lpLastColon = lpData;
}
lpData++;
}
if(lpLastColon) {
// truncate the string at the last colon
*lpLastColon = '\0';
bDone = FALSE;
while(lpData > szProxy && !bDone) {
if(*lpData == ' ') {
bDone = TRUE;
lpData++;
}else{
lpData--;
}
}
strcpy(lpDest, lpData);
ret = 0;
}else {
strcpy(lpDest, lpProxy);
ret =0;
}
}else{
ret = -1;
}
return ret;
}
// Returns 8088 from a string like "PROXY 10.0.0.1:8088"
// Returns a port # if success, -1 if an error
int GetProxyPort(LPSTR lpProxy)
{
LPSTR lpData;
LPSTR lpLastColon = NULL;
char szProxy[MAX_PATH] = {0};
int ret = -1;
if(lpProxy) {
lstrcpyn(szProxy, lpProxy, MAX_PATH);
// find the last ":" in the string
lpData = szProxy;
while(*lpData) {
if(*lpData == ':') {
lpLastColon = lpData;
}
lpData++;
}
// from the last colon to the end of the string is the port number
if ( lpLastColon )
{
lpLastColon++;
ret = (unsigned short)atoi(lpLastColon);
}
else ret = 80;
}
return ret;
}
BOOL IsDirect(LPSTR proxy)
{
BOOL bRet = FALSE;
if(proxy) {
if(strstri("DIRECT", proxy)) {
bRet = TRUE;
}
}
return bRet;
}
BOOL IsAProxy(LPSTR proxy)
{
BOOL bRet = FALSE;
if(proxy) {
if(strstri("PROXY", proxy)) {
bRet = TRUE;
}
}
return bRet;
}
// like strstr() but case-insensitive
char * strstri(LPSTR lpOne, LPSTR lpTwo)
{
unsigned int b;
char szOne[MAX_PATH] = {0}, szTwo[MAX_PATH] = {0};
if(lpOne && lpTwo) {
strcpy(szOne, lpOne);
strcpy(szTwo, lpTwo);
for(b=0; b<strlen(szOne); b++) {
szOne[b] = tolower(szOne[b]);
}
for(b=0; b<strlen(szTwo); b++) {
szTwo[b] = tolower(szTwo[b]);
}
}
return strstr(szTwo, szOne);
}
BOOL GetAutomaticallyDetectSettingsCheckboxState()
{
DWORD dwSize, dwType;
HKEY hKey;
BOOL bAutomaticallyDetectSettings = FALSE;
long lRet;
// see if the "Automatically Detect Settings" checkbox is on (I know, it's ugly)
// I noticed that the 9th byte in a binary struct at HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Connections\DefaultConnectionSettings"
// changes a bit to 1 or 0 based on the state of the checkbox. I'm using Windows XP. Not sure what byte to check on other Windows versions.
BYTE Buffer[200] = {0};
lRet = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Connections", 0, KEY_READ, &hKey);
if (lRet == ERROR_SUCCESS) {
dwSize = sizeof(Buffer);
lRet = RegQueryValueEx(hKey, "DefaultConnectionSettings", NULL, &dwType, (LPBYTE)&Buffer, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_BINARY) {
if(Buffer[8] & 8) {
bAutomaticallyDetectSettings = TRUE;
}
}
}
RegCloseKey(hKey);
return bAutomaticallyDetectSettings;
}
BOOL GetUseAProxyServerForYourLanCheckboxState()
{
DWORD dwSize, dwValue, dwType;
HKEY hKey;
BOOL bUseAProxyServerForYourLan = FALSE;
long lRet;
// see if the "Use a proxy server for your LAN" checkbox is on
lRet = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings", 0, KEY_READ, &hKey);
if (lRet == ERROR_SUCCESS) {
dwSize = sizeof(DWORD);
lRet = RegQueryValueEx(hKey, "ProxyEnable", NULL, &dwType, (LPBYTE)&dwValue, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_DWORD) {
bUseAProxyServerForYourLan = dwValue;
}
}
RegCloseKey(hKey);
return bUseAProxyServerForYourLan;
}
BOOL GetAutomaticConfigurationScriptCheckboxState()
{
DWORD dwType, dwSize;
HKEY hKey;
BOOL bUseAutomaticConfigurationScript = FALSE;
char szWPAD[MAX_PATH] = {0};
long lRet;
#if 1
// use the registry read of HKCU\\software\microsoft\windows\current version\internet settings to see if "Use Automatic Configuration Script" is checked
szWPAD[0] = '\0';
lRet = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings", 0, KEY_READ, &hKey);
if (lRet == ERROR_SUCCESS) {
dwSize = sizeof(szWPAD);
lRet = RegQueryValueEx(hKey, "AutoConfigURL", NULL, &dwType, (LPBYTE)szWPAD, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_SZ) {
}
}
RegCloseKey(hKey);
#else
// use DetectAutoProxyURL
if(!DetectAutoProxyUrl(szWPADLocation, sizeof(szWPADLocation), PROXY_AUTO_DETECT_TYPE_DHCP | PROXY_AUTO_DETECT_TYPE_DNS_A)) {
reportFuncErr("DetectAutoProxyUrl");
}
#endif
if(strlen(szWPAD)) {
bUseAutomaticConfigurationScript = TRUE;
}
return bUseAutomaticConfigurationScript;
}
BOOL GetBypassProxyServerForLocalAddressesCheckboxState()
{
DWORD dwSize, dwType;
HKEY hKey;
BOOL bBypassProxyServerForLocalAddresses = FALSE;
char szBuffer[MAX_PATH] = {0};
long lRet;
dwSize = sizeof(szBuffer);
lRet = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings", 0, KEY_READ, &hKey);
if (lRet == ERROR_SUCCESS) {
lRet = RegQueryValueEx(hKey, "ProxyOverride", NULL, &dwType, (LPBYTE)&szBuffer, &dwSize);
if(lRet == ERROR_SUCCESS && dwType == REG_SZ) {
}
}
RegCloseKey(hKey);
if(strcmp(szBuffer, "<local>") == 0) {
bBypassProxyServerForLocalAddresses = TRUE;
}
return bBypassProxyServerForLocalAddresses;
}
/* ==================================================================
HELPER FUNCTIONS
================================================================== */
// ResolveHostName (a helper function)
DWORD __stdcall ResolveHostName(LPSTR lpszHostName, LPSTR lpszIPAddress, LPDWORD lpdwIPAddressSize)
{
DWORD dwIPAddressSize;
addrinfo Hints;
LPADDRINFO lpAddrInfo;
LPADDRINFO IPv4Only;
DWORD error;
// Figure out first whether to resolve a name or an address literal.
// If getaddrinfo() with the AI_NUMERICHOST flag succeeds, then
// lpszHostName points to a string representation of an IPv4 or IPv6
// address. Otherwise, getaddrinfo() should return EAI_NONAME.
ZeroMemory(&Hints, sizeof(addrinfo));
Hints.ai_flags = AI_NUMERICHOST; // Only check for address literals.
Hints.ai_family = PF_UNSPEC; // Accept any protocol family.
Hints.ai_socktype = SOCK_STREAM; // Constrain results to stream socket.
Hints.ai_protocol = IPPROTO_TCP; // Constrain results to TCP.
error = getaddrinfo(lpszHostName, NULL, &Hints, &lpAddrInfo);
if(error != EAI_NONAME) {
if(error != 0) {
error = (error == EAI_MEMORY) ?
ERROR_NOT_ENOUGH_MEMORY : ERROR_INTERNET_NAME_NOT_RESOLVED;
goto quit;
}
freeaddrinfo(lpAddrInfo);
// An IP address (either v4 or v6) was passed in, so if there is
// room in the lpszIPAddress buffer, copy it back out and return.
dwIPAddressSize = lstrlen(lpszHostName);
if((*lpdwIPAddressSize < dwIPAddressSize) || (lpszIPAddress == NULL)) {
*lpdwIPAddressSize = dwIPAddressSize + 1;
error = ERROR_INSUFFICIENT_BUFFER;
goto quit;
}
lstrcpy(lpszIPAddress, lpszHostName);
goto quit;
}
// Call getaddrinfo() again, this time with no flag set.
Hints.ai_flags = 0;
error = getaddrinfo(lpszHostName, NULL, &Hints, &lpAddrInfo);
if(error != 0) {
error = (error == EAI_MEMORY) ?
ERROR_NOT_ENOUGH_MEMORY : ERROR_INTERNET_NAME_NOT_RESOLVED;
goto quit;
}
// Convert the IP address in addrinfo into a string.
// (the following code only handles IPv4 addresses)
IPv4Only = lpAddrInfo;
while(IPv4Only->ai_family != AF_INET) {
IPv4Only = IPv4Only->ai_next;
if(IPv4Only == NULL)
{
error = ERROR_INTERNET_NAME_NOT_RESOLVED;
goto quit;
}
}
error = getnameinfo(IPv4Only->ai_addr, (socklen_t)IPv4Only->ai_addrlen, lpszIPAddress, *lpdwIPAddressSize, NULL, 0, NI_NUMERICHOST);
if(error != 0)
error = ERROR_INTERNET_NAME_NOT_RESOLVED;
quit:
return(error);
}
// IsResolvable (a helper function)
BOOL __stdcall IsResolvable(LPSTR lpszHost)
{
char szDummy[255] = {0};
DWORD dwDummySize = sizeof(szDummy) - 1;
if(ResolveHostName(lpszHost, szDummy, &dwDummySize))
return(FALSE);
return TRUE;
}
// GetIPAddress (a helper function)
DWORD __stdcall GetIPAddress(LPSTR lpszIPAddress, LPDWORD lpdwIPAddressSize)
{
char szHostBuffer[255] = {0};
if(gethostname(szHostBuffer, sizeof(szHostBuffer) - 1) != ERROR_SUCCESS)
return(ERROR_INTERNET_INTERNAL_ERROR);
return(ResolveHostName(szHostBuffer, lpszIPAddress, lpdwIPAddressSize));
}
// IsInNet (a helper function)
BOOL __stdcall IsInNet(LPSTR lpszIPAddress, LPSTR lpszDest, LPSTR lpszMask)
{
DWORD dwDest;
DWORD dwIpAddr;
DWORD dwMask;
dwIpAddr = inet_addr(lpszIPAddress);
dwDest = inet_addr(lpszDest);
dwMask = inet_addr(lpszMask);
if((dwDest == INADDR_NONE) || (dwIpAddr == INADDR_NONE) || ((dwIpAddr & dwMask) != dwDest))
return(FALSE);
return(TRUE);
}
// reportFuncErr (simple error reporting)
void reportFuncErr(TCHAR* funcName)
{
//printf(" ERROR: %s failed with error number %d.\n", funcName, GetLastError());
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////
class CProfileFolder
{
public:
int GetProfileFolder(LPSTR lpProfileFolder, BOOL bFirefox);
private:
int GetProfileFolder_9598ME(LPSTR lpProfileFolder, BOOL bFirefox);
int GetProfileFolder_2000XP(LPSTR lpProfileFolder, BOOL bFirefox);
};
/////////////////////////////////////////////////////////////////////////////////////////////////////////
class CMozSettings
{
public:
CMozSettings(BOOL bFirefox);
virtual ~CMozSettings();
int GetPreference(LPSTR lpPreferenceWanted, int *pnDest);
int GetPreference(LPSTR lpPreferenceWanted, LPSTR lpDest, int sizeof_dest);
private:
CProfileFolder m_pf;
HGLOBAL m_hData;
LPSTR m_lpData;
int m_sizeof_data;
};
int CProfileFolder::GetProfileFolder(LPSTR lpProfileFolder, BOOL bFirefox)
{
// See http://www.mozilla.org/support/firefox/edit for where I got this info:
// On Windows XP/2000, the path is usually %AppData%\Mozilla\Firefox\Profiles\xxxxxxxx.default\, where xxxxxxxx is a random string of characters. Just browse to C:\Documents and Settings\[User Name]\Application Data\Mozilla\Firefox\Profiles\ and the rest should be obvious.
// On Windows 95/98/Me, the path is usually C:\WINDOWS\Application Data\Mozilla\Firefox\Profiles\xxxxxxxx.default\
// On Linux, the path is usually ~/.mozilla/firefox/xxxxxxxx.default/
// On Mac OS X, the path is usually ~/Library/Application Support/Firefox/Profiles/xxxxxxxx.default/
OSVERSIONINFO version;
ZeroMemory(&version, sizeof(version));
version.dwOSVersionInfoSize = sizeof(version);
GetVersionEx(&version);
if(version.dwMajorVersion == 3) {
return -1; // NT 3.51 not supported
}
if(version.dwMajorVersion == 4) {
return GetProfileFolder_9598ME(lpProfileFolder, bFirefox);
}
if(version.dwMajorVersion >= 5) {
return GetProfileFolder_2000XP(lpProfileFolder, bFirefox);
}
return -1;
}
// private function for GetProfileFolder()
// on my test system, the folder to get is c:\windows\application data\mozilla\profiles\default\y3h9azmi.slt
int CProfileFolder::GetProfileFolder_9598ME(LPSTR lpProfileFolder, BOOL bFirefox)
{
WIN32_FIND_DATA fd;
HANDLE hFind;
BOOL bDone, bFound;
char szHomePath[MAX_PATH] = {0};
char szTemp[MAX_PATH] = {0};
if(lpProfileFolder) {
GetEnvironmentVariable("WINDIR", szHomePath, sizeof(szHomePath));
strcpy(lpProfileFolder, szHomePath);
if(bFirefox) {
strcat(lpProfileFolder, "\\Application Data\\Mozilla\\Firefox\\Profiles\\");
}else{
strcat(lpProfileFolder, "\\Application Data\\Mozilla\\Profiles\\default\\");
}
// find the first folder in the the path specified in szProfileFolder
lstrcpyn(szTemp, lpProfileFolder, MAX_PATH-4);
strcat(szTemp, "*.*");
bDone = FALSE;
bFound = FALSE;
hFind = FindFirstFile(szTemp, &fd);
while(hFind != INVALID_HANDLE_VALUE && !bDone) {
if(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
// we're at a directory.
// make sure it's not "." or ".."
if(fd.cFileName[0] != '.') {
bFound = TRUE;
}
}
bDone = !FindNextFile(hFind, &fd);
}
FindClose(hFind);
if(bFound) {
strcat(lpProfileFolder, fd.cFileName);
return 0;
}
}
return -1;
}
// private function for GetProfileFolder()
int CProfileFolder::GetProfileFolder_2000XP(LPSTR lpProfileFolder, BOOL bFirefox)
{
WIN32_FIND_DATA fd;
HANDLE hFind;
BOOL bDone, bFound;
char szHomePath[MAX_PATH] = {0};
char szTemp[MAX_PATH] = {0};
if(lpProfileFolder) {
GetEnvironmentVariable("APPDATA", szHomePath, sizeof(szHomePath));
strcpy(lpProfileFolder, szHomePath);
if(bFirefox) {
strcat(lpProfileFolder, "\\Mozilla\\Firefox\\Profiles\\");
}else{
strcat(lpProfileFolder, "\\Mozilla\\Profiles\\default\\");
}
// find the first folder in the the path specified in szProfileFolder
strcpy(szTemp, lpProfileFolder);
strcat(szTemp, "*.*");
bDone = FALSE;
bFound = FALSE;
hFind = FindFirstFile(szTemp, &fd);
while(hFind != INVALID_HANDLE_VALUE && !bDone) {
if(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
// we're at a directory.
// make sure it's not "." or ".."
if(fd.cFileName[0] != '.') {
bFound = TRUE;
}
}
bDone = !FindNextFile(hFind, &fd);
}
FindClose(hFind);
if(bFound) {
strcat(lpProfileFolder, fd.cFileName);
return 0;
}
}
return -1;
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////
CMozSettings::CMozSettings(BOOL bFirefox)
{
WIN32_FIND_DATA fd;
OFSTRUCT of;
HANDLE hFind;
HFILE hPrefsFile;
char szProfileFolder[MAX_PATH] = {0};
char szPrefsFile[MAX_PATH] = {0};
int ret;
m_hData = NULL;
m_lpData = NULL;
ret = m_pf.GetProfileFolder(szProfileFolder, bFirefox);
if(ret == 0) {
// We found the folder where prefs.js lives. Read it in.
strcpy(szPrefsFile, szProfileFolder);
strcat(szPrefsFile, "\\prefs.js");
// get the size of the file and alloc memory for it
hFind = FindFirstFile(szPrefsFile, &fd);
if(hFind != INVALID_HANDLE_VALUE) {
m_hData = GlobalAlloc(GHND, fd.nFileSizeLow + 256);
if(m_hData) {
m_lpData = (LPSTR)GlobalLock(m_hData);
if(m_lpData) {
hPrefsFile = OpenFile(szPrefsFile, &of, OF_READ);
if(hPrefsFile) {
m_sizeof_data = fd.nFileSizeLow;
_lread(hPrefsFile, m_lpData, m_sizeof_data);
_lclose(hPrefsFile);
hPrefsFile = NULL;
}
}
}
FindClose(hFind);
}
}
}
CMozSettings::~CMozSettings()
{
if(m_lpData) {
GlobalUnlock(m_hData);
m_lpData = NULL;
}
if(m_hData) {
GlobalFree(m_hData);
m_hData = NULL;
}
}
int CMozSettings::GetPreference(LPSTR lpPreferenceWanted, LPSTR lpDest, int sizeof_dest)
{
LPSTR lpPointer, lpPointerEnd, lpData;
LPSTR lpLineStart, lpSearchStart, lpFoundString, lpResult;
BOOL bDone;
int nDoubleQuoteCount, retval;
retval = -1;
if(m_lpData) {
if(lpPreferenceWanted) {
if(lpDest) {
*lpDest = '\0';
bDone = FALSE;
lpPointer = m_lpData;
lpPointerEnd = lpPointer + m_sizeof_data;
while(lpPointer < lpPointerEnd && !bDone) {
if(strncmp(lpPointer, "user_pref(", 10) == 0) {
lpLineStart = lpPointer;
lpSearchStart = lpLineStart + 11;
if(strncmp(lpSearchStart, lpPreferenceWanted, strlen(lpPreferenceWanted)) == 0) {
lpFoundString = lpSearchStart + strlen(lpPreferenceWanted);
// lpFoundString almost points to what we want. Skip over the " character it's at now, skip over the " character
// starting the value we want and null-terminate what we want when we find the 3rd " character
lpData = lpFoundString;
nDoubleQuoteCount = 0;
while(nDoubleQuoteCount <= 3 && !bDone && lpData < lpPointerEnd) {
if(*lpData == '"') {
nDoubleQuoteCount++;
if(nDoubleQuoteCount == 2) {
// we're at the starting quote
lpResult = lpData;
lpResult++;
}
if(nDoubleQuoteCount == 3) {
// we're at the ending quote
// null-terminate what we want, and copy it to the dest buffer
*lpData = '\0';
lstrcpyn(lpDest, lpResult, sizeof_dest);
bDone = TRUE;
retval = 0;
}
}
lpData++;
}
}
}
lpPointer++;
}
}
}
}
return retval;
}
int CMozSettings::GetPreference(LPSTR lpPreferenceWanted, int *pnDest)
{
LPSTR lpPointer, lpPointerEnd, lpData;
LPSTR lpLineStart, lpSearchStart, lpFoundString;
BOOL bDone;
int retval;
retval = -1;
if(m_lpData) {
if(lpPreferenceWanted) {
if(pnDest) {
bDone = FALSE;
lpPointer = m_lpData;
lpPointerEnd = lpPointer + m_sizeof_data;
while(lpPointer < lpPointerEnd && !bDone) {
if(strncmp(lpPointer, "user_pref(", 10) == 0) {
lpLineStart = lpPointer;
lpSearchStart = lpLineStart + 11;
if(strncmp(lpSearchStart, lpPreferenceWanted, strlen(lpPreferenceWanted)) == 0) {
lpFoundString = lpSearchStart + strlen(lpPreferenceWanted);
// lpFoundString almost points to what we want. Skip over the " character it's at now, skip over the ","
// starting the value we want and null-terminate what we want when we find the 3rd " character
lpData = lpFoundString;
while(*lpData != ',' && lpData < lpPointerEnd) {
lpData++;
}
if(*lpData == ',') {
lpData++;
lpFoundString = lpData;
while(*lpData != ')' && lpData < lpPointerEnd) {
lpData++;
}
if(*lpData == ')') {
// null-terminate what we want, and copy it to the dest buffer
*lpData = '\0';
*pnDest = atoi(lpFoundString);
bDone = TRUE;
retval = 0;
}
}
}
}
lpPointer++;
}
}
}
}
return retval;
}
////////////////////////////////////////////////////////////////////////
BOOL IsFirefoxProxySet()
{
return IsFirefoxOrMozillaProxySet(TRUE);
}
BOOL IsMozillaProxySet()
{
return IsFirefoxOrMozillaProxySet(FALSE);
}
BOOL IsFirefoxOrMozillaProxySet(BOOL bFirefox)
{
CMozSettings settings(bFirefox);
int ret, nValue;
ret = settings.GetPreference("network.proxy.type", &nValue);
if(ret == 0) {
switch(nValue) {
case 0: // shouldn't get here
break;
case 1: // manual configuration
return TRUE;
case 2: // automatic configuration
return TRUE;
case 4: // auto-detect
return TRUE;
default: // don't know
break;
}
}
return FALSE;
}
int ResolveURL_Mozilla(LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort)
{
return ResolveURL_MozillaOrFirefox(FALSE, lpURL, lpHostname, lpIPAddress, sizeof_address, pnPort);
}
int ResolveURL_Firefox(LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort)
{
return ResolveURL_MozillaOrFirefox(TRUE, lpURL, lpHostname, lpIPAddress, sizeof_address, pnPort);
}
int ResolveURL_MozillaOrFirefox(BOOL bFirefox, LPSTR lpURL, LPSTR lpHostname, LPSTR lpIPAddress, int sizeof_address, int *pnPort)
{
CMozSettings setting(bFirefox);
int ret, nValue;
// search for the "network.proxy.http" preference
ret = setting.GetPreference("network.proxy.type", &nValue);
if(ret == 0) {
switch(nValue) {
case 0:
// shouldn't get here
break;
case 1:
// manual configuration
setting.GetPreference("network.proxy.http", lpIPAddress, sizeof_address);
setting.GetPreference("network.proxy.http_port", pnPort);
break;
case 2:
// automatic configuration
{
char szWPADLocation[MAX_PATH] = {0};
setting.GetPreference("network.proxy.autoconfig_url", szWPADLocation, sizeof(szWPADLocation));
ret = ReadWPADFile(szWPADLocation, lpIPAddress, pnPort);
}
break;
case 4:
// Auto-detect proxy settings for this network
ret = ReadWPADFile("http://wpad/wpad.dat", lpIPAddress, pnPort);
break;
default:
break;
}
}
return ret;
}
// My function that downloads from a URL to a file using the Nullsoft JNetLib library instead of using
// URLDownloadToFile(). Only parameters 2 and 3 are used, to mimick the parameters of URLDownloadToFile().
HRESULT JNetLibDownloadToFile(LPVOID lpUnused1, LPSTR lpWPADLocation, LPSTR lpTempFile, LPVOID lpUnused2, LPVOID lpUnused3)
{
api_httpreceiver *http=0;
waServiceFactory *sf=0;
OFSTRUCT of;
HGLOBAL hData;
HRESULT hRet = S_FALSE; // default return value
LPSTR lpData;
DWORD dwSize;
HFILE hFile;
BOOL bDone;
//JNL jNetLib;
int ret;
if(lpWPADLocation && lpTempFile)
{
if (WASABI_API_SVC)
{
sf = WASABI_API_SVC->service_getServiceByGuid(httpreceiverGUID);
if (sf) http = (api_httpreceiver *)sf->getInterface();
}
if (!http)
return S_FALSE;
// init the library and open a connection to the URL
http->Open();
http->Connect(lpWPADLocation);
// loop until JNetLib gets the data.
// run() returns 0 if OK, -1 if error (call geterrorstr()), or 1 if connection closed.
bDone = FALSE;
while(!bDone) {
ret = http->Run();
if(ret == -1 || ret == 1) {
bDone = TRUE;
}
Sleep(50);
}
dwSize = (DWORD)http->GetContentLength();
if(dwSize && ret == 1) {
// Got something!
// Allocate memory for it and write it to lpTempFile
hData = GlobalAlloc(GHND, dwSize + 100);
if(hData) {
lpData = (LPSTR)GlobalLock(hData);
if(lpData) {
http->GetBytes(lpData, (int)dwSize);
// create the output file and write to it
hFile = OpenFile(lpTempFile, &of, OF_CREATE);
if(hFile != HFILE_ERROR) {
_lwrite(hFile, lpData, (UINT)dwSize);
_lclose(hFile);
hRet = S_OK; // success
}
GlobalUnlock(hData);
lpData = NULL;
}
GlobalFree(hData);
hData = NULL;
}
}
}
if (http && sf)
sf->releaseInterface(http);
return hRet;
}
#endif