123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381 |
- /* <!-- copyright */
- /*
- * aria2 - The high speed download utility
- *
- * Copyright (C) 2006 Tatsuhiro Tsujikawa
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- *
- * In addition, as a special exception, the copyright holders give
- * permission to link the code of portions of this program with the
- * OpenSSL library under certain conditions as described in each
- * individual source file, and distribute linked combinations
- * including the two.
- * You must obey the GNU General Public License in all respects
- * for all of the code used other than OpenSSL. If you modify
- * file(s) with this exception, you may extend this exception to your
- * version of the file(s), but you are not obligated to do so. If you
- * do not wish to do so, delete this exception statement from your
- * version. If you delete this exception statement from all source
- * files in the program, then also delete it here.
- */
- /* copyright --> */
- #include "ConsoleStatCalc.h"
- #ifdef HAVE_TERMIOS_H
- #include <termios.h>
- #endif // HAVE_TERMIOS_H
- #ifdef HAVE_SYS_IOCTL_H
- #include <sys/ioctl.h>
- #endif // HAVE_SYS_IOCTL_H
- #include <unistd.h>
- #include <cstdio>
- #include <iomanip>
- #include <iostream>
- #include <algorithm>
- #include <cstring>
- #include <sstream>
- #include <iterator>
- #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"
- #include "wallclock.h"
- #include "FileEntry.h"
- #include "console.h"
- #ifdef ENABLE_BITTORRENT
- # include "bittorrent_helper.h"
- # include "PeerStorage.h"
- # include "BtRegistry.h"
- #endif // ENABLE_BITTORRENT
- namespace aria2 {
- std::string SizeFormatter::operator()(int64_t size) const
- {
- return format(size);
- }
- namespace {
- class AbbrevSizeFormatter:public SizeFormatter {
- protected:
- virtual std::string format(int64_t size) const
- {
- return util::abbrevSize(size);
- }
- };
- } // namespace
- namespace {
- class PlainSizeFormatter:public SizeFormatter {
- protected:
- virtual std::string format(int64_t size) const
- {
- return util::itos(size);
- }
- };
- } // namespace
- namespace {
- void printSizeProgress(std::ostream& o, const SharedHandle<RequestGroup>& rg,
- const TransferStat& stat,
- const SizeFormatter& sizeFormatter)
- {
- #ifdef ENABLE_BITTORRENT
- if(rg->getDownloadContext()->hasAttribute(CTX_ATTR_BT) &&
- !bittorrent::getTorrentAttrs(rg->getDownloadContext())
- ->metadata.empty() && rg->downloadFinished()) {
- o << "SEED(";
- if(rg->getCompletedLength() > 0) {
- std::streamsize oldprec = o.precision();
- o << std::fixed << std::setprecision(1)
- << ((stat.allTimeUploadLength*10)/rg->getCompletedLength())/10.0
- << std::setprecision(oldprec)
- << std::resetiosflags(std::ios::fixed);
- } else {
- o << "--";
- }
- o << ")";
- } else
- #endif // ENABLE_BITTORRENT
- {
- o << sizeFormatter(rg->getCompletedLength()) << "B/"
- << sizeFormatter(rg->getTotalLength()) << "B";
- if(rg->getTotalLength() > 0) {
- o << "(" << 100*rg->getCompletedLength()/rg->getTotalLength() << "%)";
- }
- }
- }
- } // namespace
- namespace {
- void printProgressCompact(std::ostream& o, const DownloadEngine* e,
- const SizeFormatter& sizeFormatter)
- {
- if(!e->getRequestGroupMan()->downloadFinished()) {
- NetStat& netstat = e->getRequestGroupMan()->getNetStat();
- int dl = netstat.calculateDownloadSpeed();
- int ul = netstat.calculateUploadSpeed();
- o << "[DL:" << sizeFormatter(dl) << "B UL:" << sizeFormatter(ul) << "B]";
- }
- const RequestGroupList& groups =
- e->getRequestGroupMan()->getRequestGroups();
- size_t cnt = 0;
- const size_t MAX_ITEM = 5;
- for(RequestGroupList::SeqType::const_iterator i = groups.begin(),
- eoi = groups.end(); i != eoi && cnt < MAX_ITEM; ++i, ++cnt) {
- const SharedHandle<RequestGroup>& rg = (*i).second;
- TransferStat stat = rg->calculateStat();
- o << "[#" << GroupId::toAbbrevHex(rg->getGID()) << " ";
- printSizeProgress(o, rg, stat, sizeFormatter);
- o << "]";
- }
- if(cnt < groups.size()) {
- o << "(+" << groups.size()-cnt << ")";
- }
- }
- } // namespace
- namespace {
- void printProgress
- (std::ostream& o, const SharedHandle<RequestGroup>& rg, const DownloadEngine* e,
- const SizeFormatter& sizeFormatter)
- {
- TransferStat stat = rg->calculateStat();
- int eta = 0;
- if(rg->getTotalLength() > 0 && stat.downloadSpeed > 0) {
- eta = (rg->getTotalLength()-rg->getCompletedLength())/stat.downloadSpeed;
- }
- o << "[#" << GroupId::toAbbrevHex(rg->getGID()) << " ";
- printSizeProgress(o, rg, stat, sizeFormatter);
- o << " CN:"
- << rg->getNumConnection();
- #ifdef ENABLE_BITTORRENT
- const SharedHandle<BtObject>& btObj = e->getBtRegistry()->get(rg->getGID());
- if(btObj) {
- const PeerSet& peers = btObj->peerStorage->getUsedPeers();
- o << " SD:"
- << countSeeder(peers.begin(), peers.end());
- }
- #endif // ENABLE_BITTORRENT
- if(!rg->downloadFinished()) {
- o << " DL:"
- << sizeFormatter(stat.downloadSpeed) << "B";
- }
- if(stat.sessionUploadLength > 0) {
- o << " UL:"
- << sizeFormatter(stat.uploadSpeed) << "B"
- << "(" << sizeFormatter(stat.allTimeUploadLength) << "B)";
- }
- if(eta > 0) {
- o << " ETA:"
- << util::secfmt(eta);
- }
- o << "]";
- }
- } // namespace
- namespace {
- class PrintSummary
- {
- private:
- size_t cols_;
- const DownloadEngine* e_;
- const SizeFormatter& sizeFormatter_;
- public:
- PrintSummary
- (size_t cols, const DownloadEngine* e,
- const SizeFormatter& sizeFormatter):
- cols_(cols), e_(e), sizeFormatter_(sizeFormatter) {}
- void operator()(const RequestGroupList::SeqType::value_type& val)
- {
- const SharedHandle<RequestGroup>& rg = val.second;
- const char SEP_CHAR = '-';
- std::stringstream o;
- printProgress(o, rg, e_, sizeFormatter_);
- const std::vector<SharedHandle<FileEntry> >& fileEntries =
- rg->getDownloadContext()->getFileEntries();
- o << "\nFILE: ";
- writeFilePath(fileEntries.begin(), fileEntries.end(),
- o, rg->inMemoryDownload());
- o << "\n"
- << std::setfill(SEP_CHAR) << std::setw(cols_) << SEP_CHAR << "\n";
- global::cout()->write(o.str().c_str());
- }
- };
- } // namespace
- namespace {
- void printProgressSummary
- (const RequestGroupList& groups, size_t cols, const DownloadEngine* e,
- const SizeFormatter& sizeFormatter)
- {
- const char SEP_CHAR = '=';
- time_t now;
- time(&now);
- std::stringstream o;
- o << " *** 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';
- }
- o << " as of " << buf;
- }
- }
- o << " *** \n"
- << std::setfill(SEP_CHAR) << std::setw(cols) << SEP_CHAR << "\n";
- global::cout()->write(o.str().c_str());
- std::for_each(groups.begin(), groups.end(),
- PrintSummary(cols, e, sizeFormatter));
- }
- } // namespace
- ConsoleStatCalc::ConsoleStatCalc(time_t summaryInterval, bool humanReadable):
- summaryInterval_(summaryInterval),
- readoutVisibility_(true),
- truncate_(true),
- #ifdef __MINGW32__
- isTTY_(true)
- #else // !__MINGW32__
- isTTY_(isatty(STDOUT_FILENO) == 1)
- #endif // !__MINGW32__
- {
- if(humanReadable) {
- sizeFormatter_.reset(new AbbrevSizeFormatter());
- } else {
- sizeFormatter_.reset(new PlainSizeFormatter());
- }
- }
- void
- ConsoleStatCalc::calculateStat(const DownloadEngine* e)
- {
- if(cp_.differenceInMillis(global::wallclock())+A2_DELTA_MILLIS < 1000) {
- return;
- }
- cp_ = global::wallclock();
- const SizeFormatter& sizeFormatter = *sizeFormatter_.get();
- // Some terminals (e.g., Windows terminal) prints next line when the
- // character reached at the last column.
- unsigned short int cols = 79;
- if(isTTY_) {
- #ifndef __MINGW32__
- #ifdef HAVE_TERMIOS_H
- struct winsize size;
- if(ioctl(STDOUT_FILENO, TIOCGWINSZ, &size) == 0) {
- cols = std::max(0, (int)size.ws_col-1);
- }
- #endif // HAVE_TERMIOS_H
- #endif // !__MINGW32__
- std::string line(cols, ' ');
- global::cout()->printf("\r%s\r", line.c_str());
- }
- std::ostringstream o;
- if(e->getRequestGroupMan()->countRequestGroup() > 0) {
- if((summaryInterval_ > 0) &&
- lastSummaryNotified_.differenceInMillis(global::wallclock())+
- A2_DELTA_MILLIS >= summaryInterval_*1000) {
- lastSummaryNotified_ = global::wallclock();
- printProgressSummary(e->getRequestGroupMan()->getRequestGroups(), cols, e,
- sizeFormatter);
- global::cout()->write("\n");
- global::cout()->flush();
- }
- }
- if(!readoutVisibility_) {
- return;
- }
- size_t numGroup = e->getRequestGroupMan()->countRequestGroup();
- if(numGroup == 1) {
- const SharedHandle<RequestGroup>& rg =
- (*e->getRequestGroupMan()->getRequestGroups().begin()).second;
- printProgress(o, rg, e, sizeFormatter);
- } else if(numGroup > 1) {
- // For more than 2 RequestGroups, use compact readout form
- printProgressCompact(o, e, sizeFormatter);
- }
- {
- const SharedHandle<FileAllocationEntry>& entry =
- e->getFileAllocationMan()->getPickedEntry();
- if(entry) {
- o << " [FileAlloc:#"
- << GroupId::toAbbrevHex(entry->getRequestGroup()->getGID()) << " "
- << sizeFormatter(entry->getCurrentLength()) << "B/"
- << sizeFormatter(entry->getTotalLength()) << "B(";
- if(entry->getTotalLength() > 0) {
- o << 100LL*entry->getCurrentLength()/entry->getTotalLength();
- } else {
- o << "--";
- }
- o << "%)]";
- if(e->getFileAllocationMan()->hasNext()) {
- o << "(+" << e->getFileAllocationMan()->countEntryInQueue() << ")";
- }
- }
- }
- #ifdef ENABLE_MESSAGE_DIGEST
- {
- const SharedHandle<CheckIntegrityEntry>& entry =
- e->getCheckIntegrityMan()->getPickedEntry();
- if(entry) {
- o << " [Checksum:#"
- << GroupId::toAbbrevHex(entry->getRequestGroup()->getGID()) << " "
- << sizeFormatter(entry->getCurrentLength()) << "B/"
- << sizeFormatter(entry->getTotalLength()) << "B(";
- if(entry->getTotalLength() > 0) {
- o << 100LL*entry->getCurrentLength()/entry->getTotalLength();
- } else {
- o << "--";
- }
- o << "%)]";
- if(e->getCheckIntegrityMan()->hasNext()) {
- o << "(+" << e->getCheckIntegrityMan()->countEntryInQueue() << ")";
- }
- }
- }
- #endif // ENABLE_MESSAGE_DIGEST
- std::string readout = o.str();
- if(isTTY_) {
- if(truncate_ && readout.size() > cols) {
- readout[cols] = '\0';
- }
- global::cout()->write(readout.c_str());
- global::cout()->flush();
- } else {
- global::cout()->write(readout.c_str());
- global::cout()->write("\n");
- }
- }
- } // namespace aria2
|