Use global::cout to output string to stdout.

In MinGW32 build, global::cout is WinConsoleFile. When outputing to
console, it uses WriteConsoleW, so Unicode characters are written.
Fixed memory leak in WChar<->MultiByte conversion functions.  The
portion of code which uses std::cout is rewritten to use global::cout.
This commit is contained in:
Tatsuhiro Tsujikawa 2011-08-09 23:33:55 +09:00
parent 02307cee0a
commit a10cda2f17
25 changed files with 614 additions and 185 deletions

View file

@ -435,6 +435,7 @@ AM_CONDITIONAL([HAVE_POLL], [test "x$have_poll" = "xyes"])
case "$target" in case "$target" in
*mingw*) *mingw*)
AM_CONDITIONAL([MINGW_BUILD], true)
dnl defined in ws2tcpip.h, but only if _WIN32_WINNT >= 0x0501 dnl defined in ws2tcpip.h, but only if _WIN32_WINNT >= 0x0501
AM_CONDITIONAL([HAVE_GETADDRINFO], true) AM_CONDITIONAL([HAVE_GETADDRINFO], true)
dnl defined in ws2tcpip.h, but missing in C:\mingw\lib\libws2_32.a dnl defined in ws2tcpip.h, but missing in C:\mingw\lib\libws2_32.a
@ -444,6 +445,7 @@ case "$target" in
fi fi
;; ;;
*) *)
AM_CONDITIONAL([MINGW_BUILD], false)
AM_CONDITIONAL([HAVE_TIMEGETTIME], false) AM_CONDITIONAL([HAVE_TIMEGETTIME], false)
;; ;;
esac esac

View file

@ -77,6 +77,11 @@ size_t BufferedFile::write(const void* ptr, size_t count)
return fwrite(ptr, 1, count, fp_); 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) char* BufferedFile::gets(char* s, int size)
{ {
return fgets(s, size, fp_); return fgets(s, size, fp_);

View file

@ -35,7 +35,7 @@
#ifndef D_BUFFERED_FILE_H #ifndef D_BUFFERED_FILE_H
#define D_BUFFERED_FILE_H #define D_BUFFERED_FILE_H
#include "common.h" #include "OutputFile.h"
#include <cstdio> #include <cstdio>
#include <string> #include <string>
@ -44,14 +44,14 @@
namespace aria2 { namespace aria2 {
// This is a wrapper class for fopen/fclose/fread/fwrite/fgets. // This is a wrapper class for fopen/fclose/fread/fwrite/fgets.
class BufferedFile { class BufferedFile:public OutputFile {
private: private:
typedef void (BufferedFile::*unspecified_bool_type)() const; typedef void (BufferedFile::*unspecified_bool_type)() const;
void good_state() const {} void good_state() const {}
public: public:
BufferedFile(const std::string& filename, const std::string& mode); BufferedFile(const std::string& filename, const std::string& mode);
BufferedFile(FILE* fp); BufferedFile(FILE* fp);
~BufferedFile(); virtual ~BufferedFile();
// Returns true if file is opened and both ferror and feof returns // Returns true if file is opened and both ferror and feof returns
// 0. Otherwise returns false. // 0. Otherwise returns false.
operator unspecified_bool_type() const; operator unspecified_bool_type() const;
@ -59,6 +59,7 @@ public:
size_t read(void* ptr, size_t count); size_t read(void* ptr, size_t count);
// wrapper for fwrite. Using 1 for 2nd argument of fwrite. // wrapper for fwrite. Using 1 for 2nd argument of fwrite.
size_t write(const void* ptr, size_t count); size_t write(const void* ptr, size_t count);
virtual size_t write(const char* str);
// wrapper for fgets // wrapper for fgets
char* gets(char* s, int size); char* gets(char* s, int size);
// wrapper for fgets, but trailing '\n' is replaced with '\0'. // wrapper for fgets, but trailing '\n' is replaced with '\0'.
@ -71,9 +72,9 @@ public:
// given stream. Returns written size. // given stream. Returns written size.
size_t transfer(std::ostream& out); size_t transfer(std::ostream& out);
// wrapper for fprintf // wrapper for fprintf
int printf(const char* format, ...); virtual int printf(const char* format, ...);
// wrapper for fflush // wrapper for fflush
int flush(); virtual int flush();
// Mode for reading // Mode for reading
static const std::string READ; static const std::string READ;
// Mode for writing // Mode for writing

View file

@ -61,6 +61,7 @@
#include "DownloadContext.h" #include "DownloadContext.h"
#include "wallclock.h" #include "wallclock.h"
#include "FileEntry.h" #include "FileEntry.h"
#include "console.h"
#ifdef ENABLE_BITTORRENT #ifdef ENABLE_BITTORRENT
# include "bittorrent_helper.h" # include "bittorrent_helper.h"
# include "PeerStorage.h" # include "PeerStorage.h"
@ -185,15 +186,17 @@ public:
void operator()(const SharedHandle<RequestGroup>& rg) void operator()(const SharedHandle<RequestGroup>& rg)
{ {
const char SEP_CHAR = '-'; const char SEP_CHAR = '-';
printProgress(std::cout, rg, e_, sizeFormatter_); std::stringstream o;
printProgress(o, rg, e_, sizeFormatter_);
const std::vector<SharedHandle<FileEntry> >& fileEntries = const std::vector<SharedHandle<FileEntry> >& fileEntries =
rg->getDownloadContext()->getFileEntries(); rg->getDownloadContext()->getFileEntries();
std::cout << "\n" o << "\n"
<< "FILE: "; << "FILE: ";
writeFilePath(fileEntries.begin(), fileEntries.end(), writeFilePath(fileEntries.begin(), fileEntries.end(),
std::cout, rg->inMemoryDownload()); o, rg->inMemoryDownload());
std::cout << "\n" o << "\n"
<< std::setfill(SEP_CHAR) << std::setw(cols_) << SEP_CHAR << "\n"; << std::setfill(SEP_CHAR) << std::setw(cols_) << SEP_CHAR << "\n";
global::cout->write(o.str().c_str());
} }
}; };
} // namespace } // namespace
@ -207,7 +210,8 @@ void printProgressSummary
const char SEP_CHAR = '='; const char SEP_CHAR = '=';
time_t now; time_t now;
time(&now); time(&now);
std::cout << " *** Download Progress Summary"; std::stringstream o;
o << " *** Download Progress Summary";
{ {
time_t now; time_t now;
struct tm* staticNowtmPtr; struct tm* staticNowtmPtr;
@ -218,11 +222,12 @@ void printProgressSummary
if(lfptr) { if(lfptr) {
*lfptr = '\0'; *lfptr = '\0';
} }
std::cout << " as of " << buf; o << " as of " << buf;
} }
} }
std::cout << " *** " << "\n" o << " *** " << "\n"
<< std::setfill(SEP_CHAR) << std::setw(cols) << SEP_CHAR << "\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(), std::for_each(groups.begin(), groups.end(),
PrintSummary(cols, e, sizeFormatter)); PrintSummary(cols, e, sizeFormatter));
} }
@ -267,7 +272,8 @@ ConsoleStatCalc::calculateStat(const DownloadEngine* e)
} }
#endif // HAVE_TERMIOS_H #endif // HAVE_TERMIOS_H
#endif // !__MINGW32__ #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; std::ostringstream o;
if(e->getRequestGroupMan()->countRequestGroup() > 0) { if(e->getRequestGroupMan()->countRequestGroup() > 0) {
@ -277,7 +283,8 @@ ConsoleStatCalc::calculateStat(const DownloadEngine* e)
lastSummaryNotified_ = global::wallclock; lastSummaryNotified_ = global::wallclock;
printProgressSummary(e->getRequestGroupMan()->getRequestGroups(), cols, e, printProgressSummary(e->getRequestGroupMan()->getRequestGroups(), cols, e,
sizeFormatter); sizeFormatter);
std::cout << "\n"; global::cout->write("\n");
global::cout->flush();
} }
} }
if(!readoutVisibility_) { if(!readoutVisibility_) {
@ -362,17 +369,14 @@ ConsoleStatCalc::calculateStat(const DownloadEngine* e)
#endif // ENABLE_MESSAGE_DIGEST #endif // ENABLE_MESSAGE_DIGEST
std::string readout = o.str(); std::string readout = o.str();
if(isTTY) { if(isTTY) {
std::string::iterator last = readout.begin();
if(truncate_ && readout.size() > cols) { if(truncate_ && readout.size() > cols) {
std::advance(last, cols); readout[cols] = '\0';
} else {
last = readout.end();
} }
std::copy(readout.begin(), last, std::ostream_iterator<char>(std::cout)); global::cout->write(readout.c_str());
std::cout << std::flush; global::cout->flush();
} else { } else {
std::copy(readout.begin(), readout.end(), std::ostream_iterator<char>(std::cout)); global::cout->write(readout.c_str());
std::cout << std::endl; global::cout->write("\n");
} }
} }

View file

@ -317,9 +317,9 @@ void writeFilePath
} }
} else { } else {
if(memory) { if(memory) {
o << "[MEMORY]" << utf8ToNative(File(e->getPath()).getBasename()); o << "[MEMORY]" << File(e->getPath()).getBasename();
} else { } else {
o << utf8ToNative(e->getPath()); o << e->getPath();
} }
size_t count = countRequestedFileEntry(first, last); size_t count = countRequestedFileEntry(first, last);
if(count > 1) { if(count > 1) {

View file

@ -44,6 +44,7 @@
#include "a2time.h" #include "a2time.h"
#include "BufferedFile.h" #include "BufferedFile.h"
#include "util.h" #include "util.h"
#include "console.h"
namespace aria2 { namespace aria2 {
@ -61,31 +62,30 @@ static const std::string ERROR_LABEL("ERROR");
Logger::Logger() Logger::Logger()
: logLevel_(Logger::A2_DEBUG), : logLevel_(Logger::A2_DEBUG),
fpp_(0),
stdoutfpp_(new BufferedFile(stdout)),
stdoutField_(0) stdoutField_(0)
{} {}
Logger::~Logger() Logger::~Logger()
{ {
delete fpp_;
delete stdoutfpp_;
} }
void Logger::openFile(const std::string& filename) void Logger::openFile(const std::string& filename)
{ {
closeFile(); closeFile();
fpp_ = new BufferedFile(filename, BufferedFile::APPEND); if(filename == DEV_STDOUT) {
if(!fpp_) { fpp_ = global::cout;
throw DL_ABORT_EX(fmt(EX_FILE_OPEN, filename.c_str(), "n/a")); } 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() void Logger::closeFile()
{ {
if(fpp_) { if(fpp_) {
fpp_->close(); fpp_.reset();
fpp_ = 0;
} }
} }
@ -124,8 +124,9 @@ const std::string& levelToString(Logger::LEVEL level)
} // namespace } // namespace
namespace { namespace {
template<typename Output>
void writeHeader 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; struct timeval tv;
gettimeofday(&tv, 0); gettimeofday(&tv, 0);
@ -146,9 +147,10 @@ void writeHeader
} // namespace } // namespace
namespace { namespace {
void writeStackTrace(BufferedFile& fp, const std::string& stackTrace) template<typename Output>
void writeStackTrace(Output& fp, const std::string& stackTrace)
{ {
fp.write(stackTrace.data(), stackTrace.size()); fp.write(stackTrace.c_str());
} }
} // namespace } // namespace
@ -168,11 +170,11 @@ void Logger::writeLog
fpp_->flush(); fpp_->flush();
} }
if(toConsole) { if(toConsole) {
stdoutfpp_->write("\n", 1); global::cout->printf("\n");
writeHeader(*stdoutfpp_, level, 0, 0); writeHeader(*global::cout, level, 0, 0);
stdoutfpp_->printf("%s\n", msg); global::cout->printf("%s\n", msg);
writeStackTrace(*stdoutfpp_, trace); writeStackTrace(*global::cout, trace);
stdoutfpp_->flush(); global::cout->flush();
} }
} }

View file

@ -39,10 +39,12 @@
#include <string> #include <string>
#include "SharedHandle.h"
namespace aria2 { namespace aria2 {
class Exception; class Exception;
class BufferedFile; class OutputFile;
class Logger { class Logger {
public: public:
@ -55,8 +57,7 @@ public:
}; };
private: private:
LEVEL logLevel_; LEVEL logLevel_;
BufferedFile* fpp_; SharedHandle<OutputFile> fpp_;
BufferedFile* stdoutfpp_;
int stdoutField_; int stdoutField_;
// Don't allow copying // Don't allow copying
Logger(const Logger&); Logger(const Logger&);

View file

@ -225,7 +225,14 @@ SRCS = Socket.h\
StreamPieceSelector.h\ StreamPieceSelector.h\
DefaultStreamPieceSelector.cc DefaultStreamPieceSelector.h\ DefaultStreamPieceSelector.cc DefaultStreamPieceSelector.h\
InorderStreamPieceSelector.cc InorderStreamPieceSelector.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 if ENABLE_XML_RPC
SRCS += XmlRpcRequestParserController.cc XmlRpcRequestParserController.h\ SRCS += XmlRpcRequestParserController.cc XmlRpcRequestParserController.h\

View file

@ -59,6 +59,7 @@
#include "TimeA2.h" #include "TimeA2.h"
#include "fmt.h" #include "fmt.h"
#include "SocketCore.h" #include "SocketCore.h"
#include "OutputFile.h"
#ifdef ENABLE_SSL #ifdef ENABLE_SSL
# include "TLSContext.h" # include "TLSContext.h"
#endif // ENABLE_SSL #endif // ENABLE_SSL
@ -142,7 +143,7 @@ MultiUrlRequestInfo::MultiUrlRequestInfo
(const std::vector<SharedHandle<RequestGroup> >& requestGroups, (const std::vector<SharedHandle<RequestGroup> >& requestGroups,
const SharedHandle<Option>& op, const SharedHandle<Option>& op,
const SharedHandle<StatCalc>& statCalc, const SharedHandle<StatCalc>& statCalc,
std::ostream& summaryOut) const SharedHandle<OutputFile>& summaryOut)
: requestGroups_(requestGroups), : requestGroups_(requestGroups),
option_(op), option_(op),
statCalc_(statCalc), statCalc_(statCalc),
@ -153,11 +154,10 @@ MultiUrlRequestInfo::~MultiUrlRequestInfo() {}
void MultiUrlRequestInfo::printMessageForContinue() void MultiUrlRequestInfo::printMessageForContinue()
{ {
summaryOut_ << "\n" summaryOut_->printf
<< _("aria2 will resume download if the transfer is restarted.") ("\n%s\n%s\n",
<< "\n" _("aria2 will resume download if the transfer is restarted."),
<< _("If there are any errors, then see the log file. See '-l' option in help/man page for details.") _("If there are any errors, then see the log file. See '-l' option in help/man page for details."));
<< "\n";
} }
error_code::Value MultiUrlRequestInfo::execute() error_code::Value MultiUrlRequestInfo::execute()
@ -251,8 +251,8 @@ error_code::Value MultiUrlRequestInfo::execute()
if(!serverStatOf.empty()) { if(!serverStatOf.empty()) {
e->getRequestGroupMan()->saveServerStat(serverStatOf); e->getRequestGroupMan()->saveServerStat(serverStatOf);
} }
e->getRequestGroupMan()->showDownloadResults(summaryOut_); e->getRequestGroupMan()->showDownloadResults(*summaryOut_);
summaryOut_ << std::flush; summaryOut_->flush();
RequestGroupMan::DownloadStat s = RequestGroupMan::DownloadStat s =
e->getRequestGroupMan()->getDownloadStat(); e->getRequestGroupMan()->getDownloadStat();

View file

@ -38,7 +38,6 @@
#include "common.h" #include "common.h"
#include <vector> #include <vector>
#include <iosfwd>
#include "SharedHandle.h" #include "SharedHandle.h"
#include "DownloadResult.h" #include "DownloadResult.h"
@ -48,6 +47,7 @@ namespace aria2 {
class RequestGroup; class RequestGroup;
class Option; class Option;
class StatCalc; class StatCalc;
class OutputFile;
class MultiUrlRequestInfo { class MultiUrlRequestInfo {
private: private:
@ -57,7 +57,7 @@ private:
SharedHandle<StatCalc> statCalc_; SharedHandle<StatCalc> statCalc_;
std::ostream& summaryOut_; SharedHandle<OutputFile> summaryOut_;
void printMessageForContinue(); void printMessageForContinue();
public: public:
@ -65,7 +65,7 @@ public:
(const std::vector<SharedHandle<RequestGroup> >& requestGroups, (const std::vector<SharedHandle<RequestGroup> >& requestGroups,
const SharedHandle<Option>& op, const SharedHandle<Option>& op,
const SharedHandle<StatCalc>& statCalc, const SharedHandle<StatCalc>& statCalc,
std::ostream& summaryOut); const SharedHandle<OutputFile>& summaryOut);
virtual ~MultiUrlRequestInfo(); virtual ~MultiUrlRequestInfo();

52
src/NullOutputFile.h Normal file
View file

@ -0,0 +1,52 @@
/* <!-- copyright */
/*
* aria2 - The high speed download utility
*
* Copyright (C) 2011 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 --> */
#ifndef D_NULL_OUTPUT_FILE_H
#define D_NULL_OUTPUT_FILE_H
#include "OutputFile.h"
namespace aria2 {
class NullOutputFile:public OutputFile {
public:
virtual ~NullOutputFile() {}
virtual size_t write(const char* str) { return 0; }
virtual int printf(const char* format, ...) { return 0; }
virtual int flush() { return 0; }
};
} // namespace aria2
#endif // D_NULL_OUTPUT_FILE_H

54
src/OutputFile.h Normal file
View file

@ -0,0 +1,54 @@
/* <!-- copyright */
/*
* aria2 - The high speed download utility
*
* Copyright (C) 2011 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 --> */
#ifndef D_OUTPUT_FILE_H
#define D_OUTPUT_FILE_H
#include "common.h"
#include <cstdlib>
namespace aria2 {
class OutputFile {
public:
virtual ~OutputFile() {}
virtual size_t write(const char* str) = 0;
virtual int printf(const char* format, ...) = 0;
virtual int flush() = 0;
};
} // namespace aria2
#endif // D_OUTPUT_FILE_H

View file

@ -72,6 +72,7 @@
#include "DlAbortEx.h" #include "DlAbortEx.h"
#include "message.h" #include "message.h"
#include "fmt.h" #include "fmt.h"
#include "console.h"
namespace aria2 { namespace aria2 {
@ -94,6 +95,8 @@ bool Platform::setUp()
} }
initialized_ = true; initialized_ = true;
global::initConsole();
#ifdef ENABLE_NLS #ifdef ENABLE_NLS
setlocale (LC_CTYPE, ""); setlocale (LC_CTYPE, "");
setlocale (LC_MESSAGES, ""); setlocale (LC_MESSAGES, "");

View file

@ -38,7 +38,6 @@
#include <cstring> #include <cstring>
#include <iomanip> #include <iomanip>
#include <sstream> #include <sstream>
#include <ostream>
#include <numeric> #include <numeric>
#include <algorithm> #include <algorithm>
#include <utility> #include <utility>
@ -74,6 +73,7 @@
#include "uri.h" #include "uri.h"
#include "Triplet.h" #include "Triplet.h"
#include "Signature.h" #include "Signature.h"
#include "OutputFile.h"
namespace aria2 { namespace aria2 {
@ -588,7 +588,7 @@ RequestGroupMan::DownloadStat RequestGroupMan::getDownloadStat() const
lastError); lastError);
} }
void RequestGroupMan::showDownloadResults(std::ostream& o) const void RequestGroupMan::showDownloadResults(OutputFile& o) const
{ {
static const std::string MARK_OK("OK"); static const std::string MARK_OK("OK");
static const std::string MARK_ERR("ERR"); static const std::string MARK_ERR("ERR");
@ -598,16 +598,17 @@ void RequestGroupMan::showDownloadResults(std::ostream& o) const
// Download Results: // Download Results:
// idx|stat|path/length // idx|stat|path/length
// ===+====+======================================================================= // ===+====+=======================================================================
o << "\n" o.printf("\n%s"
<<_("Download Results:") << "\n" "\ngid|stat|avg speed |path/URI"
<< "gid|stat|avg speed |path/URI" << "\n" "\n===+====+===========+",
<< "===+====+===========+"; _("Download Results:"));
#ifdef __MINGW32__ #ifdef __MINGW32__
int pathRowSize = 58; int pathRowSize = 58;
#else // !__MINGW32__ #else // !__MINGW32__
int pathRowSize = 59; int pathRowSize = 59;
#endif // !__MINGW32__ #endif // !__MINGW32__
o << std::setfill('=') << std::setw(pathRowSize) << '=' << "\n"; std::string line(pathRowSize, '=');
o.printf("%s\n", line.c_str());
int ok = 0; int ok = 0;
int err = 0; int err = 0;
int inpr = 0; int inpr = 0;
@ -632,29 +633,30 @@ void RequestGroupMan::showDownloadResults(std::ostream& o) const
status = MARK_ERR; status = MARK_ERR;
++err; ++err;
} }
o << formatDownloadResult(status, *itr) << "\n"; o.write(formatDownloadResult(status, *itr).c_str());
o.write("\n");
} }
if(ok > 0 || err > 0 || inpr > 0 || rm > 0) { if(ok > 0 || err > 0 || inpr > 0 || rm > 0) {
o << "\n" o.printf("\n%s\n", _("Status Legend:"));
<< _("Status Legend:") << "\n";
if(ok > 0) { if(ok > 0) {
o << " (OK):download completed."; o.write(" (OK):download completed.");
} }
if(err > 0) { if(err > 0) {
o << "(ERR):error occurred."; o.write("(ERR):error occurred.");
} }
if(inpr > 0) { if(inpr > 0) {
o << "(INPR):download in-progress."; o.write("(INPR):download in-progress.");
} }
if(rm > 0) { if(rm > 0) {
o << "(RM):download removed."; o.write("(RM):download removed.");
} }
o << "\n"; o.write("\n");
} }
} }
std::string RequestGroupMan::formatDownloadResult(const std::string& status, const DownloadResultHandle& downloadResult) const std::string RequestGroupMan::formatDownloadResult
(const std::string& status,
const DownloadResultHandle& downloadResult) const
{ {
std::stringstream o; std::stringstream o;
o << std::setw(3) << downloadResult->gid << "|" o << std::setw(3) << downloadResult->gid << "|"

View file

@ -39,7 +39,6 @@
#include <string> #include <string>
#include <deque> #include <deque>
#include <iosfwd>
#include <vector> #include <vector>
#include "SharedHandle.h" #include "SharedHandle.h"
@ -55,6 +54,7 @@ struct DownloadResult;
class ServerStatMan; class ServerStatMan;
class ServerStat; class ServerStat;
class Option; class Option;
class OutputFile;
class RequestGroupMan { class RequestGroupMan {
private: private:
@ -85,9 +85,9 @@ private:
size_t maxDownloadResult_; size_t maxDownloadResult_;
std::string std::string formatDownloadResult
formatDownloadResult(const std::string& status, (const std::string& status,
const SharedHandle<DownloadResult>& downloadResult) const; const DownloadResultHandle& downloadResult) const;
void configureRequestGroup void configureRequestGroup
(const SharedHandle<RequestGroup>& requestGroup) const; (const SharedHandle<RequestGroup>& requestGroup) const;
@ -160,7 +160,7 @@ public:
bool removeReservedGroup(a2_gid_t gid); bool removeReservedGroup(a2_gid_t gid);
void showDownloadResults(std::ostream& o) const; void showDownloadResults(OutputFile& o) const;
bool isSameFileBeingDownloaded(RequestGroup* requestGroup) const; bool isSameFileBeingDownloaded(RequestGroup* requestGroup) const;

105
src/WinConsoleFile.cc Normal file
View file

@ -0,0 +1,105 @@
/* <!-- copyright */
/*
* aria2 - The high speed download utility
*
* Copyright (C) 2011 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 "WinConsoleFile.h"
#include <cstring>
#include <cstdio>
#include <cstdarg>
#include <string>
#ifdef __MINGW32__
# define WIN32_LEAN_AND_MEAN
# include <windows.h>
#endif // __MINGW32__
#include "a2io.h"
#include "util.h"
namespace aria2 {
WinConsoleFile::WinConsoleFile() {}
WinConsoleFile::~WinConsoleFile() {}
namespace {
bool console()
{
DWORD mode;
return GetConsoleMode(GetStdHandle(STD_OUTPUT_HANDLE), &mode);
}
} // namespace
size_t WinConsoleFile::write(const char* str)
{
DWORD written;
if(console()) {
std::wstring msg = utf8ToWChar(str);
WriteConsoleW(GetStdHandle(STD_OUTPUT_HANDLE),
msg.c_str(), msg.size(), &written, 0);
} else {
WriteFile(GetStdHandle(STD_OUTPUT_HANDLE),
str, strlen(str), &written, 0);
}
return written;
}
int WinConsoleFile::printf(const char* format, ...)
{
char buf[1024];
va_list ap;
va_start(ap, format);
int r = vsnprintf(buf, sizeof(buf), format, ap);
va_end(ap);
if(r <= 0) {
return 0;
}
DWORD written;
if(console()) {
std::wstring msg = utf8ToWChar(buf);
WriteConsoleW(GetStdHandle(STD_OUTPUT_HANDLE),
msg.c_str(), msg.size(), &written, 0);
} else {
WriteFile(GetStdHandle(STD_OUTPUT_HANDLE),
buf, r, &written, 0);
}
return written;
}
int WinConsoleFile::flush()
{
return 0;
}
} // namespace aria2

58
src/WinConsoleFile.h Normal file
View file

@ -0,0 +1,58 @@
/* <!-- copyright */
/*
* aria2 - The high speed download utility
*
* Copyright (C) 2011 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 --> */
#ifndef D_WIN_CONSOLE_FILE_H
#define D_WIN_CONSOLE_FILE_H
#include "OutputFile.h"
namespace aria2 {
// This is a wrapper class for WriteConsoleW
class WinConsoleFile:public OutputFile {
public:
WinConsoleFile();
virtual ~WinConsoleFile();
virtual size_t write(const char* str);
virtual int printf(const char* format, ...);
virtual int flush();
private:
// Don't allow copying
WinConsoleFile(const WinConsoleFile&);
WinConsoleFile& operator=(const WinConsoleFile&);
};
} // namespace aria2
#endif // D_WIN_CONSOLE_FILE_H

View file

@ -611,54 +611,6 @@ getInfoHashString(const SharedHandle<DownloadContext>& dctx)
return util::toHex(getTorrentAttrs(dctx)->infoHash); return util::toHex(getTorrentAttrs(dctx)->infoHash);
} }
void print(std::ostream& o, const SharedHandle<DownloadContext>& dctx)
{
SharedHandle<TorrentAttribute> torrentAttrs = getTorrentAttrs(dctx);
o << "*** BitTorrent File Information ***" << "\n";
if(!torrentAttrs->comment.empty()) {
o << "Comment: " << torrentAttrs->comment << "\n";
}
if(torrentAttrs->creationDate) {
o << "Creation Date: " << Time(torrentAttrs->creationDate).toHTTPDate()
<< std::endl;
}
if(!torrentAttrs->createdBy.empty()) {
o << "Created By: " << torrentAttrs->createdBy << "\n";
}
o << "Mode: " << torrentAttrs->mode << "\n";
o << "Announce:" << "\n";
for(std::vector<std::vector<std::string> >::const_iterator tierIter =
torrentAttrs->announceList.begin(),
eoi = torrentAttrs->announceList.end(); tierIter != eoi; ++tierIter) {
for(std::vector<std::string>::const_iterator i = (*tierIter).begin(),
eoi2 = (*tierIter).end(); i != eoi2; ++i) {
o << " " << *i;
}
o << "\n";
}
o << "Info Hash: "
<< util::toHex(torrentAttrs->infoHash) << "\n"
<< "Piece Length: "
<< util::abbrevSize(dctx->getPieceLength()) << "B\n"
<< "The Number of Pieces: "
<< dctx->getNumPieces() << "\n"
<< "Total Length: "
<< util::abbrevSize(dctx->getTotalLength()) << "B ("
<< util::uitos(dctx->getTotalLength(), true) << ")\n";
if(!torrentAttrs->urlList.empty()) {
o << "URL List: " << "\n";
for(std::vector<std::string>::const_iterator i =
torrentAttrs->urlList.begin(),
eoi = torrentAttrs->urlList.end(); i != eoi; ++i) {
o << " " << *i << "\n";
}
}
o << "Name: " << torrentAttrs->name << "\n"
<< "Magnet URI: " << torrent2Magnet(torrentAttrs) << "\n";
util::toStream
(dctx->getFileEntries().begin(), dctx->getFileEntries().end(), o);
}
void computeFastSet void computeFastSet
(std::vector<size_t>& fastSet, const std::string& ipaddr, (std::vector<size_t>& fastSet, const std::string& ipaddr,
size_t numPieces, const unsigned char* infoHash, size_t fastSetSize) size_t numPieces, const unsigned char* infoHash, size_t fastSetSize)

View file

@ -47,6 +47,9 @@
#include "a2netcompat.h" #include "a2netcompat.h"
#include "Peer.h" #include "Peer.h"
#include "ValueBase.h" #include "ValueBase.h"
#include "util.h"
#include "DownloadContext.h"
#include "TimeA2.h"
namespace aria2 { namespace aria2 {
@ -142,9 +145,6 @@ void computeFastSet
(std::vector<size_t>& fastSet, const std::string& ipaddr, (std::vector<size_t>& fastSet, const std::string& ipaddr,
size_t numPieces, const unsigned char* infoHash, size_t fastSetSize); size_t numPieces, const unsigned char* infoHash, size_t fastSetSize);
// Writes the detailed information about torrent loaded in dctx.
void print(std::ostream& o, const SharedHandle<DownloadContext>& dctx);
SharedHandle<TorrentAttribute> getTorrentAttrs SharedHandle<TorrentAttribute> getTorrentAttrs
(const SharedHandle<DownloadContext>& dctx); (const SharedHandle<DownloadContext>& dctx);
@ -323,6 +323,55 @@ void extractPeer
int getCompactLength(int family); int getCompactLength(int family);
// Writes the detailed information about torrent loaded in dctx.
template<typename Output>
void print(Output& o, const SharedHandle<DownloadContext>& dctx)
{
SharedHandle<TorrentAttribute> torrentAttrs = getTorrentAttrs(dctx);
o.write("*** BitTorrent File Information ***\n");
if(!torrentAttrs->comment.empty()) {
o.printf("Comment: %s\n", torrentAttrs->comment.c_str());
}
if(torrentAttrs->creationDate) {
o.printf("Creation Date: %s\n",
Time(torrentAttrs->creationDate).toHTTPDate().c_str());
}
if(!torrentAttrs->createdBy.empty()) {
o.printf("Created By: %s\n", torrentAttrs->createdBy.c_str());
}
o.printf("Mode: %s\n", torrentAttrs->mode.c_str());
o.write("Announce:\n");
for(std::vector<std::vector<std::string> >::const_iterator tierIter =
torrentAttrs->announceList.begin(),
eoi = torrentAttrs->announceList.end(); tierIter != eoi; ++tierIter) {
for(std::vector<std::string>::const_iterator i = (*tierIter).begin(),
eoi2 = (*tierIter).end(); i != eoi2; ++i) {
o.printf(" %s", (*i).c_str());
}
o.write("\n");
}
o.printf("Info Hash: %s\n", util::toHex(torrentAttrs->infoHash).c_str());
o.printf("Piece Length: %sB\n",
util::abbrevSize(dctx->getPieceLength()).c_str());
o.printf("The Number of Pieces: %lu\n",
static_cast<unsigned long>(dctx->getNumPieces()));
o.printf("Total Length: %sB (%s)\n",
util::abbrevSize(dctx->getTotalLength()).c_str(),
util::uitos(dctx->getTotalLength(), true).c_str());
if(!torrentAttrs->urlList.empty()) {
o.write("URL List:\n");
for(std::vector<std::string>::const_iterator i =
torrentAttrs->urlList.begin(),
eoi = torrentAttrs->urlList.end(); i != eoi; ++i) {
o.printf(" %s\n", (*i).c_str());
}
}
o.printf("Name: %s\n", torrentAttrs->name.c_str());
o.printf("Magnet URI: %s\n", torrent2Magnet(torrentAttrs).c_str());
util::toStream
(dctx->getFileEntries().begin(), dctx->getFileEntries().end(), o);
}
} // namespace bittorrent } // namespace bittorrent
} // namespace aria2 } // namespace aria2

58
src/console.cc Normal file
View file

@ -0,0 +1,58 @@
/* <!-- copyright */
/*
* aria2 - The high speed download utility
*
* Copyright (C) 2011 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 "console.h"
namespace aria2 {
namespace global {
#ifdef __MINGW32__
SharedHandle<WinConsoleFile> cout;
#else // !__MINGW32__
SharedHandle<BufferedFile> cout;
#endif // !__MINGW32__
void initConsole()
{
#ifdef __MINGW32__
cout.reset(new WinConsoleFile());
#else // !__MINGW32__
cout.reset(new BufferedFile(stdout));
#endif // !__MINGW32__
}
} // namespace global
} // namespace aria2

62
src/console.h Normal file
View file

@ -0,0 +1,62 @@
/* <!-- copyright */
/*
* aria2 - The high speed download utility
*
* Copyright (C) 2011 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 --> */
#ifndef D_CONSOLE_H
#define D_CONSOLE_H
#include "common.h"
#include "SharedHandle.h"
#ifdef __MINGW32__
# include "WinConsoleFile.h"
#else // !__MINGW32__
# include "BufferedFile.h"
#endif // !__MINGW32__
namespace aria2 {
namespace global {
#ifdef __MINGW32__
extern SharedHandle<WinConsoleFile> cout;
#else // !__MINGW32__
extern SharedHandle<BufferedFile> cout;
#endif // !__MINGW32__
void initConsole();
} // namespace global
} // namespace aria2
#endif // D_CONSOLE_H

View file

@ -38,10 +38,9 @@
#include <unistd.h> #include <unistd.h>
#include <getopt.h> #include <getopt.h>
#include <fstream>
#include <iostream>
#include <numeric> #include <numeric>
#include <vector> #include <vector>
#include <iostream>
#include "SharedHandle.h" #include "SharedHandle.h"
#include "LogFactory.h" #include "LogFactory.h"
@ -69,6 +68,8 @@
#include "SocketCore.h" #include "SocketCore.h"
#include "DownloadContext.h" #include "DownloadContext.h"
#include "fmt.h" #include "fmt.h"
#include "NullOutputFile.h"
#include "console.h"
#ifdef ENABLE_BITTORRENT #ifdef ENABLE_BITTORRENT
# include "bittorrent_helper.h" # include "bittorrent_helper.h"
#endif // ENABLE_BITTORRENT #endif // ENABLE_BITTORRENT
@ -85,9 +86,6 @@ extern int optind, opterr, optopt;
namespace aria2 { namespace aria2 {
// output stream to /dev/null
std::ofstream nullout(DEV_NULL);
SharedHandle<StatCalc> getStatCalc(const SharedHandle<Option>& op) SharedHandle<StatCalc> getStatCalc(const SharedHandle<Option>& op)
{ {
SharedHandle<StatCalc> statCalc; SharedHandle<StatCalc> statCalc;
@ -104,12 +102,12 @@ SharedHandle<StatCalc> getStatCalc(const SharedHandle<Option>& op)
return statCalc; return statCalc;
} }
std::ostream& getSummaryOut(const SharedHandle<Option>& op) SharedHandle<OutputFile> getSummaryOut(const SharedHandle<Option>& op)
{ {
if(op->getAsBool(PREF_QUIET)) { if(op->getAsBool(PREF_QUIET)) {
return nullout; return SharedHandle<OutputFile>(new NullOutputFile());
} else { } else {
return std::cout; return global::cout;
} }
} }
@ -120,7 +118,7 @@ void showTorrentFile(const std::string& uri)
SharedHandle<Option> op(new Option()); SharedHandle<Option> op(new Option());
SharedHandle<DownloadContext> dctx(new DownloadContext()); SharedHandle<DownloadContext> dctx(new DownloadContext());
bittorrent::load(uri, dctx, op); bittorrent::load(uri, dctx, op);
bittorrent::print(std::cout, dctx); bittorrent::print(*global::cout, dctx);
} }
} // namespace } // namespace
#endif // ENABLE_BITTORRENT #endif // ENABLE_BITTORRENT
@ -135,8 +133,9 @@ void showMetalinkFile
op->get(PREF_METALINK_BASE_URI)); op->get(PREF_METALINK_BASE_URI));
std::vector<SharedHandle<FileEntry> > fileEntries; std::vector<SharedHandle<FileEntry> > fileEntries;
MetalinkEntry::toFileEntry(fileEntries, metalinkEntries); MetalinkEntry::toFileEntry(fileEntries, metalinkEntries);
util::toStream(fileEntries.begin(), fileEntries.end(), std::cout); util::toStream(fileEntries.begin(), fileEntries.end(), *global::cout);
std::cout << std::endl; global::cout->write("\n");
global::cout->flush();
} }
} // namespace } // namespace
#endif // ENABLE_METALINK #endif // ENABLE_METALINK
@ -168,7 +167,7 @@ void showFiles
printf("\n\n"); printf("\n\n");
} }
} catch(RecoverableException& e) { } catch(RecoverableException& e) {
std::cout << e.stackTrace() << std::endl; global::cout->printf("%s\n", e.stackTrace().c_str());
} }
} }
} }
@ -269,7 +268,7 @@ error_code::Value main(int argc, char* argv[])
op->remove(PREF_SELECT_FILE); op->remove(PREF_SELECT_FILE);
op->remove(PREF_PAUSE); op->remove(PREF_PAUSE);
if(!op->getAsBool(PREF_ENABLE_RPC) && requestGroups.empty()) { if(!op->getAsBool(PREF_ENABLE_RPC) && requestGroups.empty()) {
std::cout << MSG_NO_FILES_TO_DOWNLOAD << std::endl; global::cout->printf("%s\n", MSG_NO_FILES_TO_DOWNLOAD);
} else { } else {
exitStatus = MultiUrlRequestInfo(requestGroups, op, getStatCalc(op), exitStatus = MultiUrlRequestInfo(requestGroups, op, getStatCalc(op),
getSummaryOut(op)).execute(); getSummaryOut(op)).execute();

View file

@ -97,95 +97,99 @@ namespace aria2 {
#ifdef __MINGW32__ #ifdef __MINGW32__
namespace { namespace {
int utf8ToWChar(wchar_t* out, size_t outLength, const std::string& src) int utf8ToWChar(wchar_t* out, size_t outLength, const char* src)
{ {
return MultiByteToWideChar(CP_UTF8, 0, src.c_str(), -1, return MultiByteToWideChar(CP_UTF8, 0, src, -1, out, outLength);
out, outLength);
} }
} // namespace } // namespace
namespace { namespace {
int ansiToWChar(wchar_t* out, size_t outLength, const std::string& src) int ansiToWChar(wchar_t* out, size_t outLength, const char* src)
{ {
return MultiByteToWideChar(CP_ACP, 0, src.c_str(), -1, return MultiByteToWideChar(CP_ACP, 0, src, -1, out, outLength);
out, outLength);
} }
} // namespace } // namespace
namespace { namespace {
int wCharToUtf8(char* out, size_t outLength, const std::wstring& src) int wCharToUtf8(char* out, size_t outLength, const wchar_t* src)
{ {
return WideCharToMultiByte(CP_UTF8, 0, src.c_str(), -1, return WideCharToMultiByte(CP_UTF8, 0, src, -1, out, outLength, 0, 0);
out, outLength, 0, 0);
} }
} // namespace } // namespace
namespace { namespace {
int wCharToAnsi(char* out, size_t outLength, const std::wstring& src) int wCharToAnsi(char* out, size_t outLength, const wchar_t* src)
{ {
return WideCharToMultiByte(CP_ACP, 0, src.c_str(), -1, return WideCharToMultiByte(CP_ACP, 0, src, -1, out, outLength, 0, 0);
out, outLength, 0, 0);
} }
} // namespace } // namespace
std::wstring utf8ToWChar(const std::string& src) std::wstring utf8ToWChar(const char* src)
{ {
int len = utf8ToWChar(0, 0, src); int len = utf8ToWChar(0, 0, src);
if(len == 0) { if(len == 0) {
abort(); abort();
} }
wchar_t* buf = new wchar_t[len]; array_ptr<wchar_t> buf(new wchar_t[len]);
len = utf8ToWChar(buf, len, src); len = utf8ToWChar(buf, len, src);
if(len == 0) { if(len == 0) {
abort(); abort();
} else { } else {
return buf; std::wstring dest(buf);
return dest;
} }
} }
std::string const std::string& src std::wstring utf8ToWChar(const std::string& src)
{
return utf8ToWChar(src.c_str());
}
std::string utf8ToNative(const std::string& src)
{ {
std::wstring wsrc = utf8ToWChar(src); std::wstring wsrc = utf8ToWChar(src);
int len = wCharToAnsi(0, 0, wsrc); int len = wCharToAnsi(0, 0, wsrc.c_str());
if(len == 0) { if(len == 0) {
abort(); abort();
} }
char* buf = new char[len]; array_ptr<char> buf(new char[len]);
len = wCharToAnsi(buf, len, wsrc); len = wCharToAnsi(buf, len, wsrc.c_str());
if(len == 0) { if(len == 0) {
abort(); abort();
} else { } else {
return buf; std::string dest(buf);
return dest;
} }
} }
std::string wCharToUtf8(const std::wstring& wsrc) std::string wCharToUtf8(const std::wstring& wsrc)
{ {
int len = wCharToUtf8(0, 0, wsrc); int len = wCharToUtf8(0, 0, wsrc.c_str());
if(len == 0) { if(len == 0) {
abort(); abort();
} }
char* buf = new char[len]; array_ptr<char> buf(new char[len]);
len = wCharToUtf8(buf, len, wsrc); len = wCharToUtf8(buf, len, wsrc.c_str());
if(len == 0) { if(len == 0) {
abort(); abort();
} else { } else {
return buf; std::string dest(buf);
return dest;
} }
} }
std::string nativeToUtf8(const std::string& src) std::string nativeToUtf8(const std::string& src)
{ {
int len = ansiToWChar(0, 0, src); int len = ansiToWChar(0, 0, src.c_str());
if(len == 0) { if(len == 0) {
abort(); abort();
} }
wchar_t* buf = new wchar_t[len]; array_ptr<wchar_t> buf(new wchar_t[len]);
len = ansiToWChar(buf, len, src); len = ansiToWChar(buf, len, src.c_str());
if(len == 0) { if(len == 0) {
abort(); abort();
} else { } else {
return wCharToUtf8(buf); return wCharToUtf8(std::wstring(buf));
} }
} }
#endif // __MINGW32__ #endif // __MINGW32__

View file

@ -89,6 +89,8 @@ inline uint64_t hton64(uint64_t x) { return byteswap64(x); }
#ifdef __MINGW32__ #ifdef __MINGW32__
std::wstring utf8ToWChar(const std::string& src); std::wstring utf8ToWChar(const std::string& src);
std::wstring utf8ToWChar(const char* str);
std::string utf8ToNative(const std::string& src); std::string utf8ToNative(const std::string& src);
std::string wCharToUtf8(const std::wstring& wsrc); std::string wCharToUtf8(const std::wstring& wsrc);
@ -261,20 +263,22 @@ int64_t getRealSize(const std::string& sizeWithUnit);
std::string abbrevSize(int64_t size); std::string abbrevSize(int64_t size);
template<typename InputIterator> template<typename InputIterator, typename Output>
void toStream void toStream
(InputIterator first, InputIterator last, std::ostream& os) (InputIterator first, InputIterator last, Output& os)
{ {
os << _("Files:") << "\n"; os.printf("%s\n"
os << "idx|path/length" << "\n"; "idx|path/length\n"
os << "===+===========================================================================" << "\n"; "===+===========================================================================\n", _("Files:"));
int32_t count = 1; int32_t count = 1;
for(; first != last; ++first, ++count) { for(; first != last; ++first, ++count) {
os << std::setw(3) << count << "|" os.printf("%3d|%s\n"
<< utf8ToNative((*first)->getPath()) << "\n"; " |%sB (%s)\n"
os << " |" << util::abbrevSize((*first)->getLength()) << "B (" "---+---------------------------------------------------------------------------\n",
<< util::uitos((*first)->getLength(), true) << ")\n"; count,
os << "---+---------------------------------------------------------------------------" << "\n"; (*first)->getPath().c_str(),
util::abbrevSize((*first)->getLength()).c_str(),
util::uitos((*first)->getLength(), true).c_str());
} }
} }

View file

@ -13,6 +13,8 @@
#include "FileEntry.h" #include "FileEntry.h"
#include "File.h" #include "File.h"
#include "array_fun.h" #include "array_fun.h"
#include "BufferedFile.h"
#include "TestUtil.h"
namespace aria2 { namespace aria2 {
@ -589,7 +591,10 @@ void UtilTest::testToStream()
std::deque<SharedHandle<FileEntry> > entries; std::deque<SharedHandle<FileEntry> > entries;
entries.push_back(f1); entries.push_back(f1);
entries.push_back(f2); entries.push_back(f2);
util::toStream(entries.begin(), entries.end(), os); std::string filename = A2_TEST_OUT_DIR"/aria2_UtilTest_testToStream";
BufferedFile fp(filename, BufferedFile::WRITE);
util::toStream(entries.begin(), entries.end(), fp);
fp.close();
CPPUNIT_ASSERT_EQUAL( CPPUNIT_ASSERT_EQUAL(
std::string("Files:\n" std::string("Files:\n"
"idx|path/length\n" "idx|path/length\n"
@ -600,7 +605,7 @@ void UtilTest::testToStream()
" 2|aria2.txt\n" " 2|aria2.txt\n"
" |556B (556)\n" " |556B (556)\n"
"---+---------------------------------------------------------------------------\n"), "---+---------------------------------------------------------------------------\n"),
os.str()); readFile(filename));
} }
void UtilTest::testIsNumber() void UtilTest::testIsNumber()