/* */ #include "ConsoleStatCalc.h" #ifdef HAVE_TERMIOS_H #include #endif // HAVE_TERMIOS_H #ifdef HAVE_SYS_IOCTL_H #include #endif // HAVE_SYS_IOCTL_H #include #include #include #include #include #include #include #include "DownloadEngine.h" #include "RequestGroupMan.h" #include "RequestGroup.h" #include "FileAllocationMan.h" #include "FileAllocationEntry.h" #include "CheckIntegrityMan.h" #include "CheckIntegrityEntry.h" #include "util.h" #include "DownloadContext.h" #ifdef ENABLE_BITTORRENT # include "bittorrent_helper.h" # include "Peer.h" # include "PeerStorage.h" # include "BtRegistry.h" # include "BtProgressInfoFile.h" # include "BtRuntime.h" # include "BtAnnounce.h" # include "PieceStorage.h" #endif // ENABLE_BITTORRENT namespace aria2 { static void printProgress (std::ostream& o, const SharedHandle& rg, const DownloadEngine* e) { TransferStat stat = rg->calculateStat(); unsigned int eta = 0; if(rg->getTotalLength() > 0 && stat.getDownloadSpeed() > 0) { eta = (rg->getTotalLength()-rg->getCompletedLength())/stat.getDownloadSpeed(); } o << "[" << "#" << rg->getGID() << " "; #ifdef ENABLE_BITTORRENT if(rg->getDownloadContext()->hasAttribute(bittorrent::BITTORRENT) && rg->downloadFinished()) { o << "SEEDING" << "(" << "ratio:"; if(rg->getCompletedLength() > 0) { o << std::fixed << std::setprecision(1) << ((stat.getAllTimeUploadLength()*10)/rg->getCompletedLength())/10.0; } else { o << "--"; } o << ")"; } else #endif // ENABLE_BITTORRENT { o << "SIZE:" << util::abbrevSize(rg->getCompletedLength()) << "B" << "/" << util::abbrevSize(rg->getTotalLength()) << "B"; if(rg->getTotalLength() > 0) { o << "(" << 100*rg->getCompletedLength()/rg->getTotalLength() << "%)"; } } o << " " << "CN:" << rg->getNumConnection(); #ifdef ENABLE_BITTORRENT if(rg->getDownloadContext()->hasAttribute(bittorrent::BITTORRENT)) { const BDE& torrentAttrs = rg->getDownloadContext()->getAttribute(bittorrent::BITTORRENT); SharedHandle ps = e->getBtRegistry()->get(torrentAttrs[bittorrent::INFO_HASH].s())._peerStorage; std::deque > peers; ps->getActivePeers(peers); o << " " << "SEED:" << countSeeder(peers.begin(), peers.end()); } #endif // ENABLE_BITTORRENT if(!rg->downloadFinished()) { o << " " << "SPD:" << util::abbrevSize(stat.getDownloadSpeed()) << "Bs"; } if(stat.getSessionUploadLength() > 0) { o << " " << "UP:" << util::abbrevSize(stat.getUploadSpeed()) << "Bs" << "(" << util::abbrevSize(stat.getAllTimeUploadLength()) << "B)"; } if(eta > 0) { o << " " << "ETA:" << util::secfmt(eta); } o << "]"; } class PrintSummary { private: size_t _cols; const DownloadEngine* _e; public: PrintSummary(size_t cols, const DownloadEngine* e):_cols(cols), _e(e) {} void operator()(const SharedHandle& rg) { const char SEP_CHAR = '-'; printProgress(std::cout, rg, _e); const std::vector >& fileEntries = rg->getDownloadContext()->getFileEntries(); std::cout << "\n" << "FILE: "; writeFilePath(fileEntries.begin(), fileEntries.end(), std::cout, rg->inMemoryDownload()); std::cout << "\n" << std::setfill(SEP_CHAR) << std::setw(_cols) << SEP_CHAR << "\n"; } }; static void printProgressSummary (const std::deque >& groups, size_t cols, const DownloadEngine* e) { const char SEP_CHAR = '='; time_t now; time(&now); std::cout << " *** Download Progress Summary"; { time_t now; struct tm* staticNowtmPtr; char buf[26]; if(time(&now) != (time_t)-1 && (staticNowtmPtr = localtime(&now)) != 0 && asctime_r(staticNowtmPtr, buf) != 0) { char* lfptr = strchr(buf, '\n'); if(lfptr) { *lfptr = '\0'; } std::cout << " as of " << buf; } } std::cout << " *** " << "\n" << std::setfill(SEP_CHAR) << std::setw(cols) << SEP_CHAR << "\n"; std::for_each(groups.begin(), groups.end(), PrintSummary(cols, e)); } ConsoleStatCalc::ConsoleStatCalc(time_t summaryInterval): _summaryInterval(summaryInterval) {} void ConsoleStatCalc::calculateStat(const DownloadEngine* e) { if(!_cp.elapsed(1)) { return; } _cp.reset(); bool isTTY = isatty(STDOUT_FILENO) == 1; unsigned short int cols = 80; if(isTTY) { #ifdef HAVE_TERMIOS_H struct winsize size; if(ioctl(STDOUT_FILENO, TIOCGWINSZ, &size) == 0) { cols = size.ws_col; } #endif // HAVE_TERMIOS_H std::cout << '\r' << std::setfill(' ') << std::setw(cols) << ' ' << '\r'; } std::ostringstream o; if(e->_requestGroupMan->countRequestGroup() > 0) { if((_summaryInterval > 0) && _lastSummaryNotified.elapsed(_summaryInterval)) { _lastSummaryNotified.reset(); printProgressSummary(e->_requestGroupMan->getRequestGroups(), cols, e); std::cout << "\n"; } RequestGroupHandle firstRequestGroup = e->_requestGroupMan->getRequestGroup(0); printProgress(o, firstRequestGroup, e); if(e->_requestGroupMan->countRequestGroup() > 1) { o << "(" << e->_requestGroupMan->countRequestGroup()-1 << "more...)"; } } if(e->_requestGroupMan->countRequestGroup() > 1 && !e->_requestGroupMan->downloadFinished()) { TransferStat stat = e->_requestGroupMan->calculateStat(); o << " " << "[TOTAL SPD:" << util::abbrevSize(stat.getDownloadSpeed()) << "Bs" << "]"; } { SharedHandle entry=e->_fileAllocationMan->getPickedEntry(); if(!entry.isNull()) { o << " " << "[FileAlloc:" << "#" << entry->getRequestGroup()->getGID() << " " << util::abbrevSize(entry->getCurrentLength()) << "B" << "/" << util::abbrevSize(entry->getTotalLength()) << "B" << "("; if(entry->getTotalLength() > 0) { o << 100*entry->getCurrentLength()/entry->getTotalLength(); } else { o << "--"; } o << "%)" << "]"; if(e->_fileAllocationMan->hasNext()) { o << "(" << e->_fileAllocationMan->countEntryInQueue() << "waiting...)"; } } } #ifdef ENABLE_MESSAGE_DIGEST { CheckIntegrityEntryHandle entry = e->_checkIntegrityMan->getPickedEntry(); if(!entry.isNull()) { o << " " << "[Checksum:" << "#" << entry->getRequestGroup()->getGID() << " " << util::abbrevSize(entry->getCurrentLength()) << "B" << "/" << util::abbrevSize(entry->getTotalLength()) << "B" << "(" << 100*entry->getCurrentLength()/entry->getTotalLength() << "%)" << "]"; if(e->_checkIntegrityMan->hasNext()) { o << "(" << e->_checkIntegrityMan->countEntryInQueue() << "waiting...)"; } } } #endif // ENABLE_MESSAGE_DIGEST std::string readout = o.str(); if(isTTY) { std::string::iterator last = readout.begin(); if(readout.size() > cols) { std::advance(last, cols); } else { last = readout.end(); } std::copy(readout.begin(), last, std::ostream_iterator(std::cout)); std::cout << std::flush; } else { std::copy(readout.begin(), readout.end(), std::ostream_iterator(std::cout)); std::cout << std::endl; } } } // namespace aria2