/* */ #include "AsyncNameResolver.h" #include #include "A2STR.h" #include "LogFactory.h" #include "SocketCore.h" #include "util.h" namespace aria2 { void callback(void* arg, int status, int timeouts, struct hostent* host) { AsyncNameResolver* resolverPtr = reinterpret_cast(arg); if(status != ARES_SUCCESS) { resolverPtr->error_ = ares_strerror(status); resolverPtr->status_ = AsyncNameResolver::STATUS_ERROR; return; } for(char** ap = host->h_addr_list; *ap; ++ap) { char addrstring[NI_MAXHOST]; if(inetNtop(host->h_addrtype, *ap, addrstring, sizeof(addrstring)) == 0) { resolverPtr->resolvedAddresses_.push_back(addrstring); } } if(resolverPtr->resolvedAddresses_.empty()) { resolverPtr->error_ = "address conversion failed"; resolverPtr->status_ = AsyncNameResolver::STATUS_ERROR; } else { resolverPtr->status_ = AsyncNameResolver::STATUS_SUCCESS; } } AsyncNameResolver::AsyncNameResolver (int family #ifdef HAVE_ARES_ADDR_NODE , ares_addr_node* servers #endif // HAVE_ARES_ADDR_NODE ) : status_(STATUS_READY), family_(family) { // TODO evaluate return value ares_init(&channel_); #if defined HAVE_ARES_SET_SERVERS && HAVE_ARES_ADDR_NODE if(servers) { // ares_set_servers has been added since c-ares 1.7.1 if(ares_set_servers(channel_, servers) != ARES_SUCCESS) { A2_LOG_DEBUG("ares_set_servers failed"); } } #endif // HAVE_ARES_SET_SERVERS && HAVE_ARES_ADDR_NODE } AsyncNameResolver::~AsyncNameResolver() { ares_destroy(channel_); } void AsyncNameResolver::resolve(const std::string& name) { hostname_ = name; status_ = STATUS_QUERYING; ares_gethostbyname(channel_, name.c_str(), family_, callback, this); } int AsyncNameResolver::getFds(fd_set* rfdsPtr, fd_set* wfdsPtr) const { return ares_fds(channel_, rfdsPtr, wfdsPtr); } void AsyncNameResolver::process(fd_set* rfdsPtr, fd_set* wfdsPtr) { ares_process(channel_, rfdsPtr, wfdsPtr); } #ifdef HAVE_LIBCARES int AsyncNameResolver::getsock(sock_t* sockets) const { return ares_getsock(channel_, reinterpret_cast(sockets), ARES_GETSOCK_MAXNUM); } void AsyncNameResolver::process(ares_socket_t readfd, ares_socket_t writefd) { ares_process_fd(channel_, readfd, writefd); } #endif // HAVE_LIBCARES bool AsyncNameResolver::operator==(const AsyncNameResolver& resolver) const { return this == &resolver; } void AsyncNameResolver::reset() { hostname_ = A2STR::NIL; resolvedAddresses_.clear(); status_ = STATUS_READY; ares_destroy(channel_); // TODO evaluate return value ares_init(&channel_); } #ifdef HAVE_ARES_ADDR_NODE ares_addr_node* parseAsyncDNSServers(const std::string& serversOpt) { std::vector servers; util::split(serversOpt.begin(), serversOpt.end(), std::back_inserter(servers), ',', true /* doStrip */); ares_addr_node root; root.next = 0; ares_addr_node* tail = &root; ares_addr_node* node = 0; for(std::vector::const_iterator i = servers.begin(), eoi = servers.end(); i != eoi; ++i) { if(node == 0) { node = new ares_addr_node(); } size_t len = net::getBinAddr(&node->addr, (*i).c_str()); if(len != 0) { node->next = 0; node->family = (len == 4 ? AF_INET : AF_INET6); tail->next = node; tail = node; node = 0; } } if(node) { delete node; } return root.next; } #endif // HAVE_ARES_ADDR_NODE } // namespace aria2