| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 | /* <!-- copyright *//* * aria2 - The high speed download utility * * Copyright (C) 2010 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 "Logger.h"#include <cstring>#include <iostream>#include "DlAbortEx.h"#include "fmt.h"#include "message.h"#include "A2STR.h"#include "a2time.h"namespace aria2 {namespace {static const std::string DEBUG_LABEL("DEBUG");static const std::string INFO_LABEL("INFO");static const std::string NOTICE_LABEL("NOTICE");static const std::string WARN_LABEL("WARN");static const std::string ERROR_LABEL("ERROR");} // namespaceLogger::Logger()  : logLevel_(Logger::A2_DEBUG),    stdoutField_(0){}Logger::~Logger() {}void Logger::openFile(const std::string& filename){  file_.open(filename.c_str(), std::ios::app|std::ios::binary);  if(!file_) {    throw DL_ABORT_EX(fmt(EX_FILE_OPEN, filename.c_str(), "n/a"));  }}void Logger::closeFile(){  if(file_.is_open()) {    file_.close();  }}void Logger::setStdoutLogLevel(Logger::LEVEL level, bool enabled){  if(enabled) {    stdoutField_ |= level;  } else {    stdoutField_ &= ~level;  }}bool Logger::levelEnabled(LEVEL level){  return (level >= logLevel_ && file_.is_open()) || stdoutField_&level;}namespace {const std::string& levelToString(Logger::LEVEL level){  switch(level) {  case Logger::A2_DEBUG:    return DEBUG_LABEL;  case Logger::A2_INFO:    return INFO_LABEL;  case Logger::A2_NOTICE:    return NOTICE_LABEL;  case Logger::A2_WARN:    return WARN_LABEL;  case Logger::A2_ERROR:    return ERROR_LABEL;  default:    return A2STR::NIL;  }}} // namespacenamespace {void writeHeader(std::ostream& o, Logger::LEVEL level, const char* sourceFile, int lineNum){  struct timeval tv;  gettimeofday(&tv, 0);  char datestr[27]; // 'YYYY-MM-DD hh:mm:ss.uuuuuu'+'\0' = 27 bytes  struct tm tm;  //tv.tv_sec may not be of type time_t.  time_t timesec = tv.tv_sec;  localtime_r(×ec, &tm);  size_t dateLength =    strftime(datestr, sizeof(datestr), "%Y-%m-%d %H:%M:%S", &tm);  assert(dateLength <= (size_t)20);  snprintf(datestr+dateLength,           sizeof(datestr)-dateLength,           ".%06ld", tv.tv_usec);  o << datestr << " " << levelToString(level) << " - ";  if(sourceFile) {    o << "[" << sourceFile << ":" << lineNum << "] ";  }}} // namespacenamespace {void writeStackTrace(std::ostream& o, const std::string& stackTrace){  o << stackTrace;}} // namespacenamespace {void writeLog(std::ostream& o, Logger::LEVEL level, const char* sourceFile, int lineNum, const char* msg, const std::string& trace, bool toStream, bool toConsole){  if(toStream) {    writeHeader(o, level, sourceFile, lineNum);    o << msg << "\n";    writeStackTrace(o, trace);    o << std::flush;  }  if(toConsole) {    std::cout << "\n";    writeHeader(std::cout, level, 0, 0);    std::cout << msg << "\n";    writeStackTrace(std::cout, trace);    std::cout << std::flush;  }}} // namespacevoid Logger::log(LEVEL level, const char* sourceFile, int lineNum, const char* msg){  writeLog(file_, level, sourceFile, lineNum, msg, A2STR::NIL,           level >= logLevel_ && file_.is_open(),           stdoutField_&level);}void Logger::log(LEVEL level, const char* sourceFile, int lineNum, const std::string& msg){  log(level, sourceFile, lineNum, msg.c_str());}void Logger::log(LEVEL level, const char* sourceFile, int lineNum, const char* msg, const Exception& ex){  writeLog(file_, level, sourceFile, lineNum, msg, ex.stackTrace(),           level >= logLevel_ && file_.is_open(),           stdoutField_&level);}void Logger::log(LEVEL level, const char* sourceFile, int lineNum, const std::string& msg, const Exception& ex){  log(level, sourceFile, lineNum, msg.c_str(), ex);}} // namespace aria2
 |