123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372 |
- /* <!-- 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"
- #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 printProgress
- (std::ostream& o, const SharedHandle<RequestGroup>& rg, const DownloadEngine* e,
- const SizeFormatter& sizeFormatter)
- {
- 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) &&
- !bittorrent::getTorrentAttrs(rg->getDownloadContext())->metadata.empty() &&
- 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:"
- << sizeFormatter(rg->getCompletedLength())
- << "B"
- << "/"
- << sizeFormatter(rg->getTotalLength())
- << "B";
- if(rg->getTotalLength() > 0) {
- o << "("
- << 100*rg->getCompletedLength()/rg->getTotalLength()
- << "%)";
- }
- }
- o << " "
- << "CN:"
- << rg->getNumConnection();
- #ifdef ENABLE_BITTORRENT
- SharedHandle<PeerStorage> ps =
- e->getBtRegistry()->get(rg->getGID()).peerStorage_;
- if(ps) {
- std::vector<SharedHandle<Peer> > peers;
- ps->getActivePeers(peers);
- o << " " << "SEED:"
- << countSeeder(peers.begin(), peers.end());
- }
- #endif // ENABLE_BITTORRENT
- if(!rg->downloadFinished()) {
- o << " "
- << "SPD:"
- << sizeFormatter(stat.getDownloadSpeed()) << "Bs";
- }
- if(stat.getSessionUploadLength() > 0) {
- o << " "
- << "UP:"
- << sizeFormatter(stat.getUploadSpeed()) << "Bs"
- << "(" << sizeFormatter(stat.getAllTimeUploadLength()) << "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 SharedHandle<RequestGroup>& rg)
- {
- const char SEP_CHAR = '-';
- printProgress(std::cout, rg, e_, sizeFormatter_);
- const std::vector<SharedHandle<FileEntry> >& 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";
- }
- };
- } // namespace
- namespace {
- void printProgressSummary
- (const std::deque<SharedHandle<RequestGroup> >& groups, size_t cols,
- const DownloadEngine* e,
- const SizeFormatter& sizeFormatter)
- {
- 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, sizeFormatter));
- }
- } // namespace
- ConsoleStatCalc::ConsoleStatCalc(time_t summaryInterval, bool humanReadable):
- summaryInterval_(summaryInterval)
- {
- if(humanReadable) {
- sizeFormatter_.reset(new AbbrevSizeFormatter());
- } else {
- sizeFormatter_.reset(new PlainSizeFormatter());
- }
- }
- void
- ConsoleStatCalc::calculateStat(const DownloadEngine* e)
- {
- if(cp_.differenceInMillis(global::wallclock) < 900) {
- return;
- }
- cp_ = global::wallclock;
- const SizeFormatter& sizeFormatter = *sizeFormatter_.get();
- #ifdef __MINGW32__
- bool isTTY = true;
- // Windows terminal cannot handle at the end of line properly.
- unsigned short int cols = 79;
- #else // !__MINGW32__
- bool isTTY = isatty(STDOUT_FILENO) == 1;
- unsigned short int cols = 80;
- #endif // !__MINGW32__
- if(isTTY) {
- #ifndef __MINGW32__
- #ifdef HAVE_TERMIOS_H
- struct winsize size;
- if(ioctl(STDOUT_FILENO, TIOCGWINSZ, &size) == 0) {
- cols = size.ws_col;
- }
- #endif // HAVE_TERMIOS_H
- #endif // !__MINGW32__
- std::cout << '\r' << std::setfill(' ') << std::setw(cols) << ' ' << '\r';
- }
- std::ostringstream o;
- if(e->getRequestGroupMan()->countRequestGroup() > 0) {
- if((summaryInterval_ > 0) &&
- lastSummaryNotified_.difference(global::wallclock) >= summaryInterval_) {
- lastSummaryNotified_ = global::wallclock;
- printProgressSummary(e->getRequestGroupMan()->getRequestGroups(), cols, e,
- sizeFormatter);
- std::cout << "\n";
- }
- SharedHandle<RequestGroup> firstRequestGroup =
- e->getRequestGroupMan()->getRequestGroup(0);
- printProgress(o, firstRequestGroup, e, sizeFormatter);
- if(e->getRequestGroupMan()->countRequestGroup() > 1) {
- o << "("
- << e->getRequestGroupMan()->countRequestGroup()-1
- << "more...)";
- }
- }
- if(e->getRequestGroupMan()->countRequestGroup() > 1 &&
- !e->getRequestGroupMan()->downloadFinished()) {
- TransferStat stat = e->getRequestGroupMan()->calculateStat();
- o << " "
- << "[TOTAL SPD:"
- << sizeFormatter(stat.getDownloadSpeed()) << "Bs" << "]";
- }
- {
- SharedHandle<FileAllocationEntry> entry =
- e->getFileAllocationMan()->getPickedEntry();
- if(entry) {
- o << " "
- << "[FileAlloc:"
- << "#" << 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()
- << "waiting...)";
- }
- }
- }
- #ifdef ENABLE_MESSAGE_DIGEST
- {
- SharedHandle<CheckIntegrityEntry> entry =
- e->getCheckIntegrityMan()->getPickedEntry();
- if(entry) {
- o << " "
- << "[Checksum:"
- << "#" << 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()
- << "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<char>(std::cout));
- std::cout << std::flush;
- } else {
- std::copy(readout.begin(), readout.end(), std::ostream_iterator<char>(std::cout));
- std::cout << std::endl;
- }
- }
- } // namespace aria2
|