// OpenVPN -- An application to securely tunnel IP networks // over a single port, with support for SSL/TLS-based // session authentication and key exchange, // packet encryption, packet authentication, and // packet compression. // // Copyright (C) 2012-2017 OpenVPN Technologies, Inc. // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License Version 3 // as published by the Free Software Foundation. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program in the COPYING file. // If not, see . // Name Resolution Policy Table (NRPT) utilities for Windows #ifndef OPENVPN_TUN_WIN_NRPT_H #define OPENVPN_TUN_WIN_NRPT_H #include #include #include #include #include #include #include #include #include namespace openvpn { namespace TunWin { // NRPT rules described here: https://msdn.microsoft.com/en-us/library/ff957356.aspx class NRPT { public: OPENVPN_EXCEPTION(nrpt_error); static void create_rule(const std::vector names, const std::vector dns_servers) { Win::RegKey key; // open/create the key { const LONG status = ::RegCreateKeyA(HKEY_LOCAL_MACHINE, subkey(), key.ref()); if (status != ERROR_SUCCESS) { const Win::Error err(status); OPENVPN_THROW(nrpt_error, "cannot open/create registry key " << subkey << " : " << err.message()); } } // Name { const std::wstring names_packed = wstring::pack_string_vector(names); const LONG status = ::RegSetValueExW(key(), L"Name", 0, REG_MULTI_SZ, (const BYTE *)names_packed.c_str(), (names_packed.length()+1)*2); if (status != ERROR_SUCCESS) { const Win::Error err(status); OPENVPN_THROW(nrpt_error, "cannot set registry value for 'Name' : " << err.message()); } } // GenericDNSServers { const std::wstring dns_servers_joined = wstring::from_utf8(string::join(dns_servers, ";")); const LONG status = ::RegSetValueExW(key(), L"GenericDNSServers", 0, REG_SZ, (const BYTE *)dns_servers_joined.c_str(), (dns_servers_joined.length()+1)*2); if (status != ERROR_SUCCESS) { const Win::Error err(status); OPENVPN_THROW(nrpt_error, "cannot set registry value for 'GenericDNSServers' : " << err.message()); } } // ConfigOptions { const DWORD value = 0x8; // Only the Generic DNS server option (that is, the option defined in section 2.2.2.13) is specified. const LONG status = ::RegSetValueExW(key(), L"ConfigOptions", 0, REG_DWORD, (const BYTE *)&value, sizeof(value)); if (status != ERROR_SUCCESS) { const Win::Error err(status); OPENVPN_THROW(nrpt_error, "cannot set registry value for 'ConfigOptions' : " << err.message()); } } // Version { const DWORD value = 0x2; const LONG status = ::RegSetValueExW(key(), L"Version", 0, REG_DWORD, (const BYTE *)&value, sizeof(value)); if (status != ERROR_SUCCESS) { const Win::Error err(status); OPENVPN_THROW(nrpt_error, "cannot set registry value for 'Version' : " << err.message()); } } } static bool delete_rule() { return ::RegDeleteTreeA(HKEY_LOCAL_MACHINE, subkey()) == ERROR_SUCCESS; } private: static const char *subkey() { static const char subkey[] = "SYSTEM\\CurrentControlSet\\Services\\Dnscache\\Parameters\\DnsPolicyConfig\\OpenVPNDNSRouting"; return subkey; } public: class ActionCreate : public Action { public: ActionCreate(const std::vector& names_arg, const std::vector& dns_servers_arg) : names(names_arg), dns_servers(dns_servers_arg) { } virtual void execute(std::ostream& log) override { log << to_string() << std::endl; create_rule(names, dns_servers); } virtual std::string to_string() const override { std::ostringstream os; os << "NRPT::ActionCreate" << " names=[" << string::join(names, ",") << "]" << " dns_servers=[" << string::join(dns_servers, ",") << "]"; return os.str(); } private: const std::vector names; const std::vector dns_servers; }; class ActionDelete : public Action { public: virtual void execute(std::ostream& log) override { log << to_string() << std::endl; delete_rule(); } virtual std::string to_string() const override { return "NRPT::ActionDelete"; } }; }; } } #endif