/* */ #include "AnnounceList.h" #include #include "A2STR.h" #include "SimpleRandomizer.h" #include "bencode.h" #include "util.h" namespace aria2 { const std::string AnnounceList::STARTED("started"); const std::string AnnounceList::STOPPED("stopped"); const std::string AnnounceList::COMPLETED("completed"); AnnounceList::AnnounceList(const BDE& announceList): currentTrackerInitialized(false) { reconfigure(announceList); } AnnounceList::AnnounceList (const std::deque >& announceTiers): tiers(announceTiers), currentTrackerInitialized(false) { resetIterator(); } void AnnounceList::reconfigure(const BDE& announceList) { if(announceList.isList()) { for(BDE::List::const_iterator itr = announceList.listBegin(), eoi = announceList.listEnd(); itr != eoi; ++itr) { const BDE& elemList = *itr; if(!elemList.isList()) { continue; } std::deque urls; for(BDE::List::const_iterator elemItr = elemList.listBegin(), eoi2 = elemList.listEnd(); elemItr != eoi2; ++elemItr) { const BDE& data = *elemItr; if(data.isString()) { urls.push_back(data.s()); } } if(!urls.empty()) { SharedHandle tier(new AnnounceTier(urls)); tiers.push_back(tier); } } resetIterator(); } } void AnnounceList::reconfigure(const std::string& url) { std::deque urls; urls.push_back(url); tiers.push_back(SharedHandle(new AnnounceTier(urls))); resetIterator(); } void AnnounceList::resetIterator() { currentTier = tiers.begin(); if(currentTier != tiers.end() && (*currentTier)->urls.size()) { currentTracker = (*currentTier)->urls.begin(); currentTrackerInitialized = true; } else { currentTrackerInitialized = false; } } std::string AnnounceList::getAnnounce() const { if(currentTrackerInitialized) { return *currentTracker; } else { return A2STR::NIL; } } void AnnounceList::announceSuccess() { if(currentTrackerInitialized) { (*currentTier)->nextEvent(); std::string url = *currentTracker; (*currentTier)->urls.erase(currentTracker); (*currentTier)->urls.push_front(url); currentTier = tiers.begin(); currentTracker = (*currentTier)->urls.begin(); } } void AnnounceList::announceFailure() { if(currentTrackerInitialized) { ++currentTracker; if(currentTracker == (*currentTier)->urls.end()) { // force next event (*currentTier)->nextEventIfAfterStarted(); ++currentTier; if(currentTier == tiers.end()) { currentTrackerInitialized = false; } else { currentTracker = (*currentTier)->urls.begin(); } } } } AnnounceTier::AnnounceEvent AnnounceList::getEvent() const { if(currentTrackerInitialized) { return (*currentTier)->event; } else { return AnnounceTier::STARTED; } } void AnnounceList::setEvent(AnnounceTier::AnnounceEvent event) { if(currentTrackerInitialized) { (*currentTier)->event = event; } } std::string AnnounceList::getEventString() const { if(currentTrackerInitialized) { switch((*currentTier)->event) { case AnnounceTier::STARTED: case AnnounceTier::STARTED_AFTER_COMPLETION: return STARTED; case AnnounceTier::STOPPED: return STOPPED; case AnnounceTier::COMPLETED: return COMPLETED; default: return A2STR::NIL; } } else { return A2STR::NIL; } } class FindStoppedAllowedTier { public: bool operator()(const SharedHandle& tier) const { switch(tier->event) { case AnnounceTier::DOWNLOADING: case AnnounceTier::STOPPED: case AnnounceTier::COMPLETED: case AnnounceTier::SEEDING: return true; default: return false; } } }; class FindCompletedAllowedTier { public: bool operator()(const SharedHandle& tier) const { switch(tier->event) { case AnnounceTier::DOWNLOADING: case AnnounceTier::COMPLETED: return true; default: return false; } } }; size_t AnnounceList::countStoppedAllowedTier() const { return count_if(tiers.begin(), tiers.end(), FindStoppedAllowedTier()); } size_t AnnounceList::countCompletedAllowedTier() const { return count_if(tiers.begin(), tiers.end(), FindCompletedAllowedTier()); } void AnnounceList::setCurrentTier (const std::deque >::iterator& itr) { if(itr != tiers.end()) { currentTier = itr; currentTracker = (*currentTier)->urls.begin(); } } template InputIterator find_wrap_if(InputIterator first, InputIterator last, InputIterator current, Predicate pred) { InputIterator itr = std::find_if(current, last, pred); if(itr == last) { itr = std::find_if(first, current, pred); } return itr; } void AnnounceList::moveToStoppedAllowedTier() { std::deque >::iterator itr = find_wrap_if(tiers.begin(), tiers.end(), currentTier, FindStoppedAllowedTier()); setCurrentTier(itr); } void AnnounceList::moveToCompletedAllowedTier() { std::deque >::iterator itr = find_wrap_if(tiers.begin(), tiers.end(), currentTier, FindCompletedAllowedTier()); setCurrentTier(itr); } void AnnounceList::shuffle() { for(std::deque >::const_iterator itr = tiers.begin(), eoi = tiers.end(); itr != eoi; ++itr) { std::deque& urls = (*itr)->urls; std::random_shuffle(urls.begin(), urls.end(), *(SimpleRandomizer::getInstance().get())); } } bool AnnounceList::allTiersFailed() const { return currentTier == tiers.end(); } void AnnounceList::resetTier() { resetIterator(); } bool AnnounceList::currentTierAcceptsStoppedEvent() const { if(currentTrackerInitialized) { return FindStoppedAllowedTier()(*currentTier); } else { return false; } } bool AnnounceList::currentTierAcceptsCompletedEvent() const { if(currentTrackerInitialized) { return FindCompletedAllowedTier()(*currentTier); } else { return false; } } } // namespace aria2