diff --git a/configure.ac b/configure.ac index 54d03abf..e75e53d5 100644 --- a/configure.ac +++ b/configure.ac @@ -435,6 +435,7 @@ AM_CONDITIONAL([HAVE_POLL], [test "x$have_poll" = "xyes"]) case "$target" in *mingw*) + AM_CONDITIONAL([MINGW_BUILD], true) dnl defined in ws2tcpip.h, but only if _WIN32_WINNT >= 0x0501 AM_CONDITIONAL([HAVE_GETADDRINFO], true) dnl defined in ws2tcpip.h, but missing in C:\mingw\lib\libws2_32.a @@ -444,6 +445,7 @@ case "$target" in fi ;; *) + AM_CONDITIONAL([MINGW_BUILD], false) AM_CONDITIONAL([HAVE_TIMEGETTIME], false) ;; esac diff --git a/src/BufferedFile.cc b/src/BufferedFile.cc index 812704c9..1bca85f4 100644 --- a/src/BufferedFile.cc +++ b/src/BufferedFile.cc @@ -77,6 +77,11 @@ size_t BufferedFile::write(const void* ptr, size_t count) return fwrite(ptr, 1, count, fp_); } +size_t BufferedFile::write(const char* str) +{ + return write(str, strlen(str)); +} + char* BufferedFile::gets(char* s, int size) { return fgets(s, size, fp_); diff --git a/src/BufferedFile.h b/src/BufferedFile.h index b5308c0c..db006c8a 100644 --- a/src/BufferedFile.h +++ b/src/BufferedFile.h @@ -35,7 +35,7 @@ #ifndef D_BUFFERED_FILE_H #define D_BUFFERED_FILE_H -#include "common.h" +#include "OutputFile.h" #include #include @@ -44,14 +44,14 @@ namespace aria2 { // This is a wrapper class for fopen/fclose/fread/fwrite/fgets. -class BufferedFile { +class BufferedFile:public OutputFile { private: typedef void (BufferedFile::*unspecified_bool_type)() const; void good_state() const {} public: BufferedFile(const std::string& filename, const std::string& mode); BufferedFile(FILE* fp); - ~BufferedFile(); + virtual ~BufferedFile(); // Returns true if file is opened and both ferror and feof returns // 0. Otherwise returns false. operator unspecified_bool_type() const; @@ -59,6 +59,7 @@ public: size_t read(void* ptr, size_t count); // wrapper for fwrite. Using 1 for 2nd argument of fwrite. size_t write(const void* ptr, size_t count); + virtual size_t write(const char* str); // wrapper for fgets char* gets(char* s, int size); // wrapper for fgets, but trailing '\n' is replaced with '\0'. @@ -71,9 +72,9 @@ public: // given stream. Returns written size. size_t transfer(std::ostream& out); // wrapper for fprintf - int printf(const char* format, ...); + virtual int printf(const char* format, ...); // wrapper for fflush - int flush(); + virtual int flush(); // Mode for reading static const std::string READ; // Mode for writing diff --git a/src/ConsoleStatCalc.cc b/src/ConsoleStatCalc.cc index c8969f16..5710b184 100644 --- a/src/ConsoleStatCalc.cc +++ b/src/ConsoleStatCalc.cc @@ -61,6 +61,7 @@ #include "DownloadContext.h" #include "wallclock.h" #include "FileEntry.h" +#include "console.h" #ifdef ENABLE_BITTORRENT # include "bittorrent_helper.h" # include "PeerStorage.h" @@ -185,15 +186,17 @@ public: void operator()(const SharedHandle& rg) { const char SEP_CHAR = '-'; - printProgress(std::cout, rg, e_, sizeFormatter_); + std::stringstream o; + printProgress(o, rg, e_, sizeFormatter_); const std::vector >& fileEntries = rg->getDownloadContext()->getFileEntries(); - std::cout << "\n" - << "FILE: "; + o << "\n" + << "FILE: "; writeFilePath(fileEntries.begin(), fileEntries.end(), - std::cout, rg->inMemoryDownload()); - std::cout << "\n" - << std::setfill(SEP_CHAR) << std::setw(cols_) << SEP_CHAR << "\n"; + o, rg->inMemoryDownload()); + o << "\n" + << std::setfill(SEP_CHAR) << std::setw(cols_) << SEP_CHAR << "\n"; + global::cout->write(o.str().c_str()); } }; } // namespace @@ -207,7 +210,8 @@ void printProgressSummary const char SEP_CHAR = '='; time_t now; time(&now); - std::cout << " *** Download Progress Summary"; + std::stringstream o; + o << " *** Download Progress Summary"; { time_t now; struct tm* staticNowtmPtr; @@ -218,11 +222,12 @@ void printProgressSummary if(lfptr) { *lfptr = '\0'; } - std::cout << " as of " << buf; + o << " as of " << buf; } } - std::cout << " *** " << "\n" - << std::setfill(SEP_CHAR) << std::setw(cols) << SEP_CHAR << "\n"; + 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)); } @@ -267,7 +272,8 @@ ConsoleStatCalc::calculateStat(const DownloadEngine* e) } #endif // HAVE_TERMIOS_H #endif // !__MINGW32__ - std::cout << '\r' << std::setfill(' ') << std::setw(cols) << ' ' << '\r'; + std::string line(cols, ' '); + global::cout->printf("\r%s\r", line.c_str()); } std::ostringstream o; if(e->getRequestGroupMan()->countRequestGroup() > 0) { @@ -277,7 +283,8 @@ ConsoleStatCalc::calculateStat(const DownloadEngine* e) lastSummaryNotified_ = global::wallclock; printProgressSummary(e->getRequestGroupMan()->getRequestGroups(), cols, e, sizeFormatter); - std::cout << "\n"; + global::cout->write("\n"); + global::cout->flush(); } } if(!readoutVisibility_) { @@ -362,17 +369,14 @@ ConsoleStatCalc::calculateStat(const DownloadEngine* e) #endif // ENABLE_MESSAGE_DIGEST std::string readout = o.str(); if(isTTY) { - std::string::iterator last = readout.begin(); if(truncate_ && readout.size() > cols) { - std::advance(last, cols); - } else { - last = readout.end(); + readout[cols] = '\0'; } - std::copy(readout.begin(), last, std::ostream_iterator(std::cout)); - std::cout << std::flush; + global::cout->write(readout.c_str()); + global::cout->flush(); } else { - std::copy(readout.begin(), readout.end(), std::ostream_iterator(std::cout)); - std::cout << std::endl; + global::cout->write(readout.c_str()); + global::cout->write("\n"); } } diff --git a/src/FileEntry.h b/src/FileEntry.h index ce202d57..f7d23f38 100644 --- a/src/FileEntry.h +++ b/src/FileEntry.h @@ -317,9 +317,9 @@ void writeFilePath } } else { if(memory) { - o << "[MEMORY]" << utf8ToNative(File(e->getPath()).getBasename()); + o << "[MEMORY]" << File(e->getPath()).getBasename(); } else { - o << utf8ToNative(e->getPath()); + o << e->getPath(); } size_t count = countRequestedFileEntry(first, last); if(count > 1) { diff --git a/src/Logger.cc b/src/Logger.cc index be167f18..3f88451b 100644 --- a/src/Logger.cc +++ b/src/Logger.cc @@ -44,6 +44,7 @@ #include "a2time.h" #include "BufferedFile.h" #include "util.h" +#include "console.h" namespace aria2 { @@ -61,31 +62,30 @@ static const std::string ERROR_LABEL("ERROR"); Logger::Logger() : logLevel_(Logger::A2_DEBUG), - fpp_(0), - stdoutfpp_(new BufferedFile(stdout)), stdoutField_(0) {} Logger::~Logger() { - delete fpp_; - delete stdoutfpp_; } void Logger::openFile(const std::string& filename) { closeFile(); - fpp_ = new BufferedFile(filename, BufferedFile::APPEND); - if(!fpp_) { - throw DL_ABORT_EX(fmt(EX_FILE_OPEN, filename.c_str(), "n/a")); + if(filename == DEV_STDOUT) { + fpp_ = global::cout; + } else { + fpp_.reset(new BufferedFile(filename, BufferedFile::APPEND)); + if(!fpp_) { + throw DL_ABORT_EX(fmt(EX_FILE_OPEN, filename.c_str(), "n/a")); + } } } void Logger::closeFile() { if(fpp_) { - fpp_->close(); - fpp_ = 0; + fpp_.reset(); } } @@ -124,8 +124,9 @@ const std::string& levelToString(Logger::LEVEL level) } // namespace namespace { +template void writeHeader -(BufferedFile& fp, Logger::LEVEL level, const char* sourceFile, int lineNum) +(Output& fp, Logger::LEVEL level, const char* sourceFile, int lineNum) { struct timeval tv; gettimeofday(&tv, 0); @@ -146,9 +147,10 @@ void writeHeader } // namespace namespace { -void writeStackTrace(BufferedFile& fp, const std::string& stackTrace) +template +void writeStackTrace(Output& fp, const std::string& stackTrace) { - fp.write(stackTrace.data(), stackTrace.size()); + fp.write(stackTrace.c_str()); } } // namespace @@ -168,11 +170,11 @@ void Logger::writeLog fpp_->flush(); } if(toConsole) { - stdoutfpp_->write("\n", 1); - writeHeader(*stdoutfpp_, level, 0, 0); - stdoutfpp_->printf("%s\n", msg); - writeStackTrace(*stdoutfpp_, trace); - stdoutfpp_->flush(); + global::cout->printf("\n"); + writeHeader(*global::cout, level, 0, 0); + global::cout->printf("%s\n", msg); + writeStackTrace(*global::cout, trace); + global::cout->flush(); } } diff --git a/src/Logger.h b/src/Logger.h index dc955247..fb7068bc 100644 --- a/src/Logger.h +++ b/src/Logger.h @@ -39,10 +39,12 @@ #include +#include "SharedHandle.h" + namespace aria2 { class Exception; -class BufferedFile; +class OutputFile; class Logger { public: @@ -55,8 +57,7 @@ public: }; private: LEVEL logLevel_; - BufferedFile* fpp_; - BufferedFile* stdoutfpp_; + SharedHandle fpp_; int stdoutField_; // Don't allow copying Logger(const Logger&); diff --git a/src/Makefile.am b/src/Makefile.am index 64ccb3ea..67833bf9 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -225,7 +225,14 @@ SRCS = Socket.h\ StreamPieceSelector.h\ DefaultStreamPieceSelector.cc DefaultStreamPieceSelector.h\ InorderStreamPieceSelector.cc InorderStreamPieceSelector.h\ - MetalinkHttpEntry.cc MetalinkHttpEntry.h + MetalinkHttpEntry.cc MetalinkHttpEntry.h\ + OutputFile.h\ + NullOutputFile.h\ + console.cc console.h + +if MINGW_BUILD +SRCS += WinConsoleFile.cc WinConsoleFile.h +endif # MINGW_BUILD if ENABLE_XML_RPC SRCS += XmlRpcRequestParserController.cc XmlRpcRequestParserController.h\ diff --git a/src/MultiUrlRequestInfo.cc b/src/MultiUrlRequestInfo.cc index 760fe7e5..27659aa4 100644 --- a/src/MultiUrlRequestInfo.cc +++ b/src/MultiUrlRequestInfo.cc @@ -59,6 +59,7 @@ #include "TimeA2.h" #include "fmt.h" #include "SocketCore.h" +#include "OutputFile.h" #ifdef ENABLE_SSL # include "TLSContext.h" #endif // ENABLE_SSL @@ -142,7 +143,7 @@ MultiUrlRequestInfo::MultiUrlRequestInfo (const std::vector >& requestGroups, const SharedHandle