/* */ #include "AnnounceList.h" #include "List.h" #include "Data.h" 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(MetaList::const_iterator itr = l->getList().begin(); itr != l->getList().end(); itr++) { const List* elem = (List*)*itr; Strings urls; for(MetaList::const_iterator elemItr = elem->getList().begin(); elemItr != elem->getList().end(); elemItr++) { const Data* data = (Data*)*elemItr; urls.push_back(data->toString()); } if(urls.size()) { AnnounceTierHandle tier(new AnnounceTier(urls)); tiers.push_back(tier); } } resetIterator(); } } void AnnounceList::reconfigure(const string& url) { Strings 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; } } string AnnounceList::getAnnounce() const { if(currentTrackerInitialized) { return *currentTracker; } else { return ""; } } void AnnounceList::announceSuccess() { if(currentTrackerInitialized) { (*currentTier)->nextEvent(); 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()) { 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; } } 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 ""; } } else { return ""; } } 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; } } }; int32_t AnnounceList::countStoppedAllowedTier() const { return count_if(tiers.begin(), tiers.end(), FindStoppedAllowedTier()); } int32_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 = find_if(current, last, pred); if(itr == last) { itr = 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++) { Strings& urls = (*itr)->urls; random_shuffle(urls.begin(), urls.end()); } } bool AnnounceList::allTiersFailed() const { return currentTier == tiers.end(); } void AnnounceList::resetTier() { resetIterator(); }