/* */ #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) : addr_(c.addr_), good_(c.good_) {} DNSCache::AddrEntry::~AddrEntry() {} 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) : hostname_(c.hostname_), port_(c.port_), addrEntries_(c.addrEntries_) {} DNSCache::CacheEntry::~CacheEntry() {} DNSCache::CacheEntry& DNSCache::CacheEntry::operator=(const CacheEntry& c) { if(this != &c) { hostname_ = c.hostname_; port_ = c.port_; addrEntries_ = c.addrEntries_; } return *this; } void DNSCache::CacheEntry::add(const std::string& addr) { addrEntries_.push_back(AddrEntry(addr)); } std::vector::iterator DNSCache::CacheEntry::find (const std::string& addr) { for(std::vector::iterator 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(std::vector::const_iterator 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(std::vector::const_iterator i = addrEntries_.begin(), eoi = addrEntries_.end(); i != eoi; ++i) { if((*i).good_) { return (*i).addr_; } } return A2STR::NIL; } void DNSCache::CacheEntry::markBad(const std::string& addr) { std::vector::iterator 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):entries_(c.entries_) {} DNSCache::~DNSCache() {} 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 { CacheEntry target(hostname, port); std::deque::const_iterator i = std::lower_bound(entries_.begin(), entries_.end(), target); if(i != entries_.end() && (*i) == target) { return (*i).getGoodAddr(); } return A2STR::NIL; } void DNSCache::put (const std::string& hostname, const std::string& ipaddr, uint16_t port) { CacheEntry target(hostname, port); std::deque::iterator i = std::lower_bound(entries_.begin(), entries_.end(), target); if(i == entries_.end() || !((*i) == target)) { target.add(ipaddr); entries_.insert(i, target); } else { if(!(*i).contains(ipaddr)) { (*i).add(ipaddr); } } } void DNSCache::markBad (const std::string& hostname, const std::string& ipaddr, uint16_t port) { CacheEntry target(hostname, port); std::deque::iterator i = std::lower_bound(entries_.begin(), entries_.end(), target); if(i != entries_.end() && (*i) == target) { (*i).markBad(ipaddr); } } void DNSCache::remove(const std::string& hostname, uint16_t port) { CacheEntry target(hostname, port); std::deque::iterator i = std::lower_bound(entries_.begin(), entries_.end(), target); if(i != entries_.end() && (*i) == target) { entries_.erase(i); } } } // namespace aria2