summaryrefslogtreecommitdiffstats
blob: 3ae43e01c70fe065dc153e65c98df1212586282d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
/*
 * Copyright (c) 2010-2011 Thilo Cestonaro
 * Licensed under the simplified BSD license.
 * See Documentation/Licenses/BSD-simplified.txt for more information.
 */

#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <iostream>
#include <boost/lexical_cast.hpp>

#include <Swiften/Base/log.h>
#include <Swiften/Base/foreach.h>
#include <Swiften/Network/WindowsProxyProvider.h>
#include <Swiften/Base/ByteArray.h>

#include <windows.h>

namespace Swift {

WindowsProxyProvider::WindowsProxyProvider()
: ProxyProvider()
{
	HKEY hKey = (HKEY)INVALID_HANDLE_VALUE;
	long result;
		
	result = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings", 0, KEY_READ, &hKey);
	if (result == ERROR_SUCCESS && hKey != INVALID_HANDLE_VALUE && proxyEnabled(hKey)) {
		DWORD dataType = REG_SZ;
		DWORD dataSize = 0;
		ByteArray dataBuffer;

		result = RegQueryValueEx(hKey, "ProxyServer", NULL, &dataType, NULL, &dataSize);
		if(result != ERROR_SUCCESS) {
			return;
		}
		dataBuffer.resize(dataSize);
		result = RegQueryValueEx(hKey, "ProxyServer", NULL, &dataType, reinterpret_cast<BYTE*>(vecptr(dataBuffer)), &dataSize);
		if(result == ERROR_SUCCESS) {
			std::vector<std::string> proxies = String::split(byteArrayToString(dataBuffer), ';');
			std::pair<std::string, std::string> protocolAndProxy;
			foreach(std::string proxy, proxies) {
				if(proxy.find('=') != std::string::npos) {
					protocolAndProxy = String::getSplittedAtFirst(proxy, '=');
					SWIFT_LOG(debug) << "Found proxy: " << protocolAndProxy.first << " => " << protocolAndProxy.second << std::endl;
					if(protocolAndProxy.first.compare("socks") == 0) {
						socksProxy = getAsHostAddressPort(protocolAndProxy.second);
					}
					else if (protocolAndProxy.first.compare("http") == 0) {
						httpProxy = getAsHostAddressPort(protocolAndProxy.second);
					}
				}
			}
		}
	}
}

HostAddressPort WindowsProxyProvider::getHTTPConnectProxy() const {
	return httpProxy;
}

HostAddressPort WindowsProxyProvider::getSOCKS5Proxy() const {
	return socksProxy;
}

HostAddressPort WindowsProxyProvider::getAsHostAddressPort(std::string proxy) {
	HostAddressPort ret(HostAddress(), 0);

	try {
		std::pair<std::string, std::string> tmp;
		int port = 0;
		tmp = String::getSplittedAtFirst(proxy, ':');
		// .c_str() is needed as tmp.second can include a \0 char which will end in an exception of the lexical cast.
		// with .c_str() the \0 will not be part of the string which is to be casted
		port = boost::lexical_cast<int> (tmp.second.c_str());
		ret = HostAddressPort(HostAddress(tmp.first), port);
	}
	catch(...) {
			std::cerr << "Exception occured while parsing windows proxy \"getHostAddressPort\"." << std::endl;
	}

	return ret;
}


bool WindowsProxyProvider::proxyEnabled(HKEY hKey) const {
	bool ret = false;
	long result;
	DWORD dataType = REG_DWORD;
	DWORD dataSize = 0;
	DWORD data = 0;
	ByteArray dataBuffer;

	if(hKey == INVALID_HANDLE_VALUE)
		return ret;

	result = RegQueryValueEx(hKey, "ProxyEnable", NULL, &dataType, NULL, &dataSize);
	if(result != ERROR_SUCCESS)
		return ret;

	dataBuffer.resize(dataSize);
	result = RegQueryValueEx(hKey, "ProxyEnable", NULL, &dataType, reinterpret_cast<BYTE*>(vecptr(dataBuffer)), &dataSize);
	if(result != ERROR_SUCCESS)
		return ret;

	for(size_t t = 0; t < dataBuffer.size(); t++) {
		data += static_cast<int> (dataBuffer[t]) * pow(256, static_cast<double>(t));
	}
	return (data == 1);
}

}