/* */ #include "AnnounceList.h" #include #include "A2STR.h" #include "SimpleRandomizer.h" #include "a2algo.h" namespace aria2 { const std::string AnnounceList::STARTED("started"); const std::string AnnounceList::STOPPED("stopped"); const std::string AnnounceList::COMPLETED("completed"); AnnounceList::AnnounceList():currentTrackerInitialized_(false) {} AnnounceList::AnnounceList (const std::vector >& announceList): currentTrackerInitialized_(false) { reconfigure(announceList); } AnnounceList::AnnounceList (const std::deque >& announceTiers): tiers_(announceTiers), currentTrackerInitialized_(false) { resetIterator(); } AnnounceList::~AnnounceList() {} void AnnounceList::reconfigure (const std::vector >& announceList) { for(std::vector >::const_iterator itr = announceList.begin(), eoi = announceList.end(); itr != eoi; ++itr) { if((*itr).empty()) { continue; } std::deque urls((*itr).begin(), (*itr).end()); SharedHandle tier(new AnnounceTier(urls)); tiers_.push_back(tier); } resetIterator(); } void AnnounceList::reconfigure(const std::string& url) { std::deque urls; urls.push_back(url); SharedHandle tier(new AnnounceTier(urls)); tiers_.push_back(tier); 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; } } namespace { 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; } } }; } // namespace namespace { class FindCompletedAllowedTier { public: bool operator()(const SharedHandle& tier) const { switch(tier->event) { case AnnounceTier::DOWNLOADING: case AnnounceTier::COMPLETED: return true; default: return false; } } }; } // namespace 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(); } } 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; } } size_t AnnounceList::countTier() const { return tiers_.size(); } } // namespace aria2