/* */ #include "DNSCache.h" #include "A2STR.h" namespace aria2 { DNSCache::AddrEntry::AddrEntry(const std::string& addr) : addr_(addr), good_(true) { } DNSCache::AddrEntry::AddrEntry(const AddrEntry& c) = default; DNSCache::AddrEntry::~AddrEntry() = default; DNSCache::AddrEntry& DNSCache::AddrEntry::operator=(const AddrEntry& c) { if (this != &c) { addr_ = c.addr_; good_ = c.good_; } return *this; } DNSCache::CacheEntry::CacheEntry(const std::string& hostname, uint16_t port) : hostname_(hostname), port_(port) { } DNSCache::CacheEntry::CacheEntry(const CacheEntry& c) = default; DNSCache::CacheEntry::~CacheEntry() = default; DNSCache::CacheEntry& DNSCache::CacheEntry::operator=(const CacheEntry& c) { if (this != &c) { hostname_ = c.hostname_; port_ = c.port_; addrEntries_ = c.addrEntries_; } return *this; } bool DNSCache::CacheEntry::add(const std::string& addr) { for (std::vector::const_iterator i = addrEntries_.begin(), eoi = addrEntries_.end(); i != eoi; ++i) { if ((*i).addr_ == addr) { return false; } } addrEntries_.push_back(AddrEntry(addr)); return true; } std::vector::iterator DNSCache::CacheEntry::find(const std::string& addr) { for (auto i = addrEntries_.begin(), eoi = addrEntries_.end(); i != eoi; ++i) { if ((*i).addr_ == addr) { return i; } } return addrEntries_.end(); } std::vector::const_iterator DNSCache::CacheEntry::find(const std::string& addr) const { for (auto i = addrEntries_.begin(), eoi = addrEntries_.end(); i != eoi; ++i) { if ((*i).addr_ == addr) { return i; } } return addrEntries_.end(); } bool DNSCache::CacheEntry::contains(const std::string& addr) const { return find(addr) != addrEntries_.end(); } const std::string& DNSCache::CacheEntry::getGoodAddr() const { for (auto& elem : addrEntries_) { if (elem.good_) { return (elem).addr_; } } return A2STR::NIL; } void DNSCache::CacheEntry::markBad(const std::string& addr) { auto i = find(addr); if (i != addrEntries_.end()) { i->good_ = false; } } bool DNSCache::CacheEntry::operator<(const CacheEntry& e) const { int r = hostname_.compare(e.hostname_); if (r != 0) { return r < 0; } return port_ < e.port_; } bool DNSCache::CacheEntry::operator==(const CacheEntry& e) const { return hostname_ == e.hostname_ && port_ == e.port_; } DNSCache::DNSCache() {} DNSCache::DNSCache(const DNSCache& c) = default; DNSCache::~DNSCache() = default; DNSCache& DNSCache::operator=(const DNSCache& c) { if (this != &c) { entries_ = c.entries_; } return *this; } const std::string& DNSCache::find(const std::string& hostname, uint16_t port) const { auto target = std::make_shared(hostname, port); auto i = entries_.find(target); if (i == entries_.end()) { return A2STR::NIL; } else { return (*i)->getGoodAddr(); } } void DNSCache::put(const std::string& hostname, const std::string& ipaddr, uint16_t port) { auto target = std::make_shared(hostname, port); auto i = entries_.lower_bound(target); if (i != entries_.end() && *(*i) == *target) { (*i)->add(ipaddr); } else { target->add(ipaddr); entries_.insert(i, target); } } void DNSCache::markBad(const std::string& hostname, const std::string& ipaddr, uint16_t port) { auto target = std::make_shared(hostname, port); auto i = entries_.find(target); if (i != entries_.end()) { (*i)->markBad(ipaddr); } } void DNSCache::remove(const std::string& hostname, uint16_t port) { auto target = std::make_shared(hostname, port); entries_.erase(target); } } // namespace aria2