aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/common/dns_utils.cpp103
-rw-r--r--src/common/dns_utils.h43
-rw-r--r--src/p2p/net_node.h7
-rw-r--r--src/p2p/net_node.inl38
-rw-r--r--tests/unit_tests/dns_resolver.cpp65
5 files changed, 224 insertions, 32 deletions
diff --git a/src/common/dns_utils.cpp b/src/common/dns_utils.cpp
index 0bacc4d46..f95a8d72b 100644
--- a/src/common/dns_utils.cpp
+++ b/src/common/dns_utils.cpp
@@ -27,29 +27,110 @@
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include "common/dns_utils.h"
+#include <ldns/rr.h> // for RR type and class defs
+#include <unbound.h>
+#include <arpa/inet.h> // for inet_ntoa (bytes to text for IPs)
namespace tools
{
-namespace dns
+
+struct DNSResolverData
+{
+ ub_ctx* m_ub_context;
+};
+
+DNSResolver::DNSResolver() : m_data(new DNSResolverData())
{
+ // init libunbound context
+ m_data->m_ub_context = ub_ctx_create();
- std::vector<std::string> get_ipv4(const std::string& uri)
+ // look for "/etc/resolv.conf" and "/etc/hosts" or platform equivalent
+ ub_ctx_resolvconf(m_data->m_ub_context, "");
+ ub_ctx_hosts(m_data->m_ub_context, "");
+}
+
+DNSResolver::~DNSResolver()
+{
+ if (m_data)
{
- std::vector<std::string retval;
- return retval;
+ if (m_data->m_ub_context != NULL)
+ {
+ ub_ctx_delete(m_data->m_ub_context);
+ }
+ delete m_data;
}
+}
- std::vector<std::string> get_ipv6(const std::string& uri)
+std::vector<std::string> DNSResolver::get_ipv4(const std::string& url)
+{
+ ub_result* result = NULL;
+ std::vector<std::string> retval;
+
+ // call DNS resolver, blocking. if return value not zero, something went wrong
+ if (!ub_resolve(m_data->m_ub_context, url.c_str(), LDNS_RR_TYPE_A, LDNS_RR_CLASS_IN, &result))
{
- std::vector<std::string retval;
- return retval;
+ if (result->havedata)
+ {
+ for (int i=0; result->data[i] != NULL; i++)
+ {
+ char as_str[INET_ADDRSTRLEN];
+
+ // convert bytes to string, append if no error
+ if (inet_ntop(AF_INET, result->data[0], as_str, sizeof(as_str)))
+ {
+ retval.push_back(as_str);
+ }
+ }
+ }
}
- std::string get_payment_address(const std::string& uri)
+ // cleanup
+ ub_resolve_free(result);
+ return retval;
+}
+
+std::vector<std::string> DNSResolver::get_ipv6(const std::string& url)
+{
+ ub_result* result = NULL;
+ std::vector<std::string> retval;
+
+ // call DNS resolver, blocking. if return value not zero, something went wrong
+ if (!ub_resolve(m_data->m_ub_context, url.c_str(), LDNS_RR_TYPE_AAAA, LDNS_RR_CLASS_IN, &result))
+ {
+ if (result->havedata)
+ {
+ for (int i=0; result->data[i] != NULL; i++)
+ {
+ char as_str[INET6_ADDRSTRLEN];
+
+ // convert bytes to string, append if no error
+ if (inet_ntop(AF_INET6, result->data[0], as_str, sizeof(as_str)))
+ {
+ retval.push_back(as_str);
+ }
+ }
+ }
+ }
+
+ // cleanup
+ ub_resolve_free(result);
+ return retval;
+}
+
+std::string DNSResolver::get_payment_address(const std::string& url)
+{
+ std::string retval;
+ return retval;
+}
+
+DNSResolver& DNSResolver::instance()
+{
+ static DNSResolver* staticInstance = NULL;
+ if (staticInstance == NULL)
{
- std::string retval;
- return retval;
+ staticInstance = new DNSResolver();
}
+ return *staticInstance;
+}
-} // namespace dns
} // namespace tools
diff --git a/src/common/dns_utils.h b/src/common/dns_utils.h
index 2919100df..6697d8fff 100644
--- a/src/common/dns_utils.h
+++ b/src/common/dns_utils.h
@@ -32,6 +32,8 @@
namespace tools
{
+struct DNSResolverData;
+
/**
* @brief Provides high-level access to DNS resolution
*
@@ -44,43 +46,64 @@ class DNSResolver
public:
/**
- * @brief Constructs and sets the URI to be resolved
+ * @brief Constructs an instance of DNSResolver
*
- * @param uri the URI to be resolved
+ * Constructs a class instance and does setup stuff for the backend resolver.
+ */
+ DNSResolver();
+
+ /**
+ * @brief takes care of freeing C pointers and such
*/
- DNSResolver(const std::string& uri);
+ ~DNSResolver();
/**
- * @brief gets ipv4 addresses from DNS query
+ * @brief gets ipv4 addresses from DNS query of a URL
*
- * returns a vector of all IPv4 "A" records for given URI.
+ * returns a vector of all IPv4 "A" records for given URL.
* If no "A" records found, returns an empty vector.
*
+ * @param url A string containing a URL to query for
+ *
* @return vector of strings containing ipv4 addresses
*/
- std::vector<std::string> get_ipv4();
+ std::vector<std::string> get_ipv4(const std::string& url);
/**
* @brief gets ipv6 addresses from DNS query
*
- * returns a vector of all IPv6 "A" records for given URI.
+ * returns a vector of all IPv6 "A" records for given URL.
* If no "A" records found, returns an empty vector.
*
+ * @param url A string containing a URL to query for
+ *
* @return vector of strings containing ipv6 addresses
*/
- std::vector<std::string> get_ipv6();
+ std::vector<std::string> get_ipv6(const std::string& url);
/**
* @brief gets a monero address from the TXT record of the DNS query response
*
- * returns a monero address string from the TXT record associated with URI
+ * returns a monero address string from the TXT record associated with URL
* if no TXT record present, or no valid monero address in TXT,
* returns an empty string.
*
+ * @param url A string containing a URL to query for
+ *
* @return
*/
- std::string get_payment_address();
+ std::string get_payment_address(const std::string& url);
+
+ /**
+ * @brief Gets the singleton instance of DNSResolver
+ *
+ * @return returns a pointer to the singleton
+ */
+ static DNSResolver& instance();
+
+private:
+ DNSResolverData *m_data;
}; // class DNSResolver
} // namespace tools
diff --git a/src/p2p/net_node.h b/src/p2p/net_node.h
index 759b21fcf..53a13f327 100644
--- a/src/p2p/net_node.h
+++ b/src/p2p/net_node.h
@@ -112,6 +112,13 @@ namespace nodetool
size_t get_outgoing_connections_count();
peerlist_manager& get_peerlist_manager(){return m_peerlist;}
private:
+ const std::vector<std::string> m_seed_nodes_list =
+ { "seeds.moneroseeds.se"
+ , "seeds.moneroseeds.ae.org"
+ , "seeds.moneroseeds.ch"
+ , "seeds.moneroseeds.li"
+ };
+
typedef COMMAND_REQUEST_STAT_INFO_T<typename t_payload_net_handler::stat_info> COMMAND_REQUEST_STAT_INFO;
CHAIN_LEVIN_INVOKE_MAP2(p2p_connection_context); //move levin_commands_handler interface invoke(...) callbacks into invoke map
diff --git a/src/p2p/net_node.inl b/src/p2p/net_node.inl
index 954d794b7..60bc870e7 100644
--- a/src/p2p/net_node.inl
+++ b/src/p2p/net_node.inl
@@ -245,17 +245,33 @@ namespace nodetool
}
else
{
- add_hardcoded_seed_node(m_seed_nodes, "62.210.78.186:18080");
- add_hardcoded_seed_node(m_seed_nodes, "195.12.60.154:18080");
- add_hardcoded_seed_node(m_seed_nodes, "54.241.246.125:18080");
- add_hardcoded_seed_node(m_seed_nodes, "107.170.157.169:18080");
- add_hardcoded_seed_node(m_seed_nodes, "54.207.112.216:18080");
- add_hardcoded_seed_node(m_seed_nodes, "78.27.112.54:18080");
- add_hardcoded_seed_node(m_seed_nodes, "209.222.30.57:18080");
- add_hardcoded_seed_node(m_seed_nodes, "80.71.13.55:18080");
- add_hardcoded_seed_node(m_seed_nodes, "107.178.112.126:18080");
- add_hardcoded_seed_node(m_seed_nodes, "107.158.233.98:18080");
- add_hardcoded_seed_node(m_seed_nodes, "64.22.111.2:18080");
+ // for each hostname in the seed nodes list, attempt to DNS resolve and
+ // add the result addresses as seed nodes
+ // TODO: at some point add IPv6 support, but that won't be relevant
+ // for some time yet.
+ for (const std::string& addr_str : m_seed_nodes_list)
+ {
+ std::vector<std::string> addr_list = tools::DNSResolver::instance().get_ipv4(addr_str);
+ for (const std::string& a : addr_list)
+ {
+ append_net_address(m_seed_nodes, a + ":18080");
+ }
+ }
+
+ if (!m_seed_nodes.size())
+ {
+ add_hardcoded_seed_node(m_seed_nodes, "62.210.78.186:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "195.12.60.154:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "54.241.246.125:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "107.170.157.169:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "54.207.112.216:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "78.27.112.54:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "209.222.30.57:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "80.71.13.55:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "107.178.112.126:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "107.158.233.98:18080");
+ add_hardcoded_seed_node(m_seed_nodes, "64.22.111.2:18080");
+ }
}
bool res = handle_command_line(vm, testnet);
diff --git a/tests/unit_tests/dns_resolver.cpp b/tests/unit_tests/dns_resolver.cpp
new file mode 100644
index 000000000..fe8fe602b
--- /dev/null
+++ b/tests/unit_tests/dns_resolver.cpp
@@ -0,0 +1,65 @@
+#include "gtest/gtest.h"
+
+#include "common/dns_utils.h"
+
+TEST(DNSResolver, IPv4Success)
+{
+ tools::DNSResolver resolver;
+
+ auto ips = resolver.get_ipv4("example.com");
+
+ ASSERT_EQ(1, ips.size());
+
+ ASSERT_STREQ("93.184.216.119", ips[0].c_str());
+
+ ips = tools::DNSResolver::instance().get_ipv4("example.com");
+
+ ASSERT_EQ(1, ips.size());
+
+ ASSERT_STREQ("93.184.216.119", ips[0].c_str());
+}
+
+TEST(DNSResolver, IPv4Failure)
+{
+ // guaranteed by IANA/ICANN/RFC to be invalid
+ tools::DNSResolver resolver;
+
+ auto ips = resolver.get_ipv4("example.invalid");
+
+ ASSERT_EQ(0, ips.size());
+
+ ips = tools::DNSResolver::instance().get_ipv4("example.invalid");
+
+ ASSERT_EQ(0, ips.size());
+}
+
+TEST(DNSResolver, IPv6Success)
+{
+ tools::DNSResolver resolver;
+
+ auto ips = resolver.get_ipv6("example.com");
+
+ ASSERT_EQ(1, ips.size());
+
+ ASSERT_STREQ("2606:2800:220:6d:26bf:1447:1097:aa7", ips[0].c_str());
+
+ ips = tools::DNSResolver::instance().get_ipv6("example.com");
+
+ ASSERT_EQ(1, ips.size());
+
+ ASSERT_STREQ("2606:2800:220:6d:26bf:1447:1097:aa7", ips[0].c_str());
+}
+
+TEST(DNSResolver, IPv6Failure)
+{
+ // guaranteed by IANA/ICANN/RFC to be invalid
+ tools::DNSResolver resolver;
+
+ auto ips = resolver.get_ipv6("example.invalid");
+
+ ASSERT_EQ(0, ips.size());
+
+ ips = tools::DNSResolver::instance().get_ipv6("example.invalid");
+
+ ASSERT_EQ(0, ips.size());
+}