/* */ #include "AsyncNameResolver.h" #include #include "A2STR.h" #include "LogFactory.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) { #ifdef HAVE_INET_NTOP char addrstring[INET6_ADDRSTRLEN]; const char* dst = inet_ntop(host->h_addrtype, *ap, addrstring, sizeof(addrstring)); if(dst) { resolverPtr->resolvedAddresses_.push_back(dst); } #else // !HAVE_INET_NTOP if(host->h_addrtype != AF_INET) { continue; } struct in_addr addr; memcpy(&addr, *ap, sizeof(in_addr)); resolverPtr->resolvedAddresses_.push_back(inet_ntoa(addr)); #endif // !HAVE_INET_NTOP } 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_); } } // namespace aria2