/* * Copyright (c) 2010-2011 Thilo Cestonaro * Licensed under the simplified BSD license. * See Documentation/Licenses/BSD-simplified.txt for more information. */ /* * Copyright (c) 2016 Isode Limited. * All rights reserved. * See the COPYING file for more information. */ #include #include #include #include #include #include #include #include #include #include #include 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(vecptr(dataBuffer)), &dataSize); if(result == ERROR_SUCCESS) { std::vector proxies = String::split(byteArrayToString(dataBuffer), ';'); std::pair protocolAndProxy; for(auto&& proxy : proxies) { if(proxy.find('=') != std::string::npos) { protocolAndProxy = String::getSplittedAtFirst(proxy, '='); SWIFT_LOG(debug) << "Found proxy: " << protocolAndProxy.first << " => " << protocolAndProxy.second; 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 tmp; unsigned short 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::numeric_cast(boost::lexical_cast (tmp.second.c_str())); ret = HostAddressPort(HostAddress::fromString(tmp.first).get(), port); } catch(...) { SWIFT_LOG(error) << "Exception occured while parsing windows proxy \"getHostAddressPort\"."; } 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(vecptr(dataBuffer)), &dataSize); if(result != ERROR_SUCCESS) { return ret; } for(size_t t = 0; t < dataBuffer.size(); t++) { data += static_cast (dataBuffer[t]) * pow(256, static_cast(t)); } return (data == 1); } }