/* */ #include "AnnounceList.h" #include #include "List.h" #include "Data.h" #include "A2STR.h" #include "SimpleRandomizer.h" namespace aria2 { const std::string AnnounceList::STARTED("started"); const std::string AnnounceList::STOPPED("stopped"); const std::string AnnounceList::COMPLETED("completed"); AnnounceList::AnnounceList(const MetaEntry* announceListEntry): currentTrackerInitialized(false) { reconfigure(announceListEntry); } AnnounceList::AnnounceList(const AnnounceTiers& announceTiers): tiers(announceTiers), currentTrackerInitialized(false) { resetIterator(); } void AnnounceList::reconfigure(const MetaEntry* announceListEntry) { const List* l = dynamic_cast(announceListEntry); if(l) { for(std::deque::const_iterator itr = l->getList().begin(); itr != l->getList().end(); itr++) { const List* elem = dynamic_cast(*itr); if(!elem) { continue; } std::deque urls; for(std::deque::const_iterator elemItr = elem->getList().begin(); elemItr != elem->getList().end(); elemItr++) { const Data* data = dynamic_cast(*elemItr); if(data) { urls.push_back(data->toString()); } } if(urls.size()) { AnnounceTierHandle 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(AnnounceTierHandle(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 AnnounceTierHandle& 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 AnnounceTierHandle& 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 AnnounceTiers::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() { AnnounceTiers::iterator itr = find_wrap_if(tiers.begin(), tiers.end(), currentTier, FindStoppedAllowedTier()); setCurrentTier(itr); } void AnnounceList::moveToCompletedAllowedTier() { AnnounceTiers::iterator itr = find_wrap_if(tiers.begin(), tiers.end(), currentTier, FindCompletedAllowedTier()); setCurrentTier(itr); } void AnnounceList::shuffle() { for(AnnounceTiers::iterator itr = tiers.begin(); itr != tiers.end(); 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