mirror of
https://github.com/monero-project/monero.git
synced 2024-12-15 12:56:33 +02:00
618f20ce49
Really really finall version of this changes I hope.
805 lines
25 KiB
C++
805 lines
25 KiB
C++
/// @file
|
|
/// @author rfree (current maintainer in monero.cc project)
|
|
/// @brief various general utils taken from (and relate to) otshell project, including loggiang/debug
|
|
|
|
/* See other files here for the LICENCE that applies here. */
|
|
/* See header file .hpp for info */
|
|
|
|
#include <algorithm>
|
|
#include <functional>
|
|
#include <cctype>
|
|
#include <locale>
|
|
#include <fstream>
|
|
#include <iostream>
|
|
#include <iomanip>
|
|
|
|
#include "utils.hpp"
|
|
|
|
#include "ccolor.hpp"
|
|
|
|
#include "lib_common1.hpp"
|
|
|
|
#include "runoptions.hpp"
|
|
|
|
#if defined(_WIN32) || defined(WIN32) || defined(_WIN64) || defined (WIN64)
|
|
#define OS_TYPE_WINDOWS
|
|
#elif defined(__unix__) || defined(__posix) || defined(__linux) || defined(__darwin) || defined(__APPLE__) || defined(__clang__)
|
|
#define OS_TYPE_POSIX
|
|
#else
|
|
#warning "Compiler/OS platform is not recognized. Just assuming it will work as POSIX then"
|
|
#define OS_TYPE_POSIX
|
|
#endif
|
|
|
|
#if defined(OS_TYPE_WINDOWS)
|
|
#include <windows.h>
|
|
#elif defined(OS_TYPE_POSIX)
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#else
|
|
#error "Compiler/OS platform detection failed - not supported"
|
|
#endif
|
|
|
|
|
|
namespace nOT {
|
|
namespace nUtils {
|
|
|
|
INJECT_OT_COMMON_USING_NAMESPACE_COMMON_1 // <=== namespaces
|
|
|
|
// ====================================================================
|
|
|
|
// Numerical values of the debug levels - see hpp
|
|
const int _debug_level_nr_dbg3=20;
|
|
const int _debug_level_nr_dbg2=30;
|
|
const int _debug_level_nr_dbg1=40;
|
|
const int _debug_level_nr_info=50;
|
|
const int _debug_level_nr_note=60;
|
|
const int _debug_level_nr_fact=75;
|
|
const int _debug_level_nr_mark=80;
|
|
const int _debug_level_nr_warn=90;
|
|
const int _debug_level_nr_erro=100;
|
|
|
|
// ====================================================================
|
|
|
|
myexception::myexception(const char * what)
|
|
: std::runtime_error(what)
|
|
{ }
|
|
|
|
myexception::myexception(const std::string &what)
|
|
: std::runtime_error(what)
|
|
{ }
|
|
|
|
void myexception::Report() const {
|
|
_erro("Error: " << what());
|
|
}
|
|
|
|
//myexception::~myexception() { }
|
|
|
|
// ====================================================================
|
|
|
|
// text trimming
|
|
// http://stackoverflow.com/questions/216823/whats-the-best-way-to-trim-stdstring
|
|
std::string & ltrim(std::string &s) {
|
|
s.erase(s.begin(), std::find_if(s.begin(), s.end(), std::not1(std::ptr_fun<int, int>(std::isspace))));
|
|
return s;
|
|
}
|
|
|
|
std::string & rtrim(std::string &s) {
|
|
s.erase(std::find_if(s.rbegin(), s.rend(), std::not1(std::ptr_fun<int, int>(std::isspace))).base(), s.end());
|
|
return s;
|
|
}
|
|
|
|
std::string & trim(std::string &s) {
|
|
return ltrim(rtrim(s));
|
|
}
|
|
|
|
std::string get_current_time() {
|
|
std::chrono::system_clock::time_point now = std::chrono::system_clock::now();
|
|
time_t time_now = std::chrono::system_clock::to_time_t(now);
|
|
std::chrono::high_resolution_clock::duration duration = now.time_since_epoch();
|
|
int64_t micro = std::chrono::duration_cast<std::chrono::microseconds>(duration).count();
|
|
|
|
// std::localtime() - This function may not be thread-safe.
|
|
#ifdef OS_TYPE_WINDOWS
|
|
struct tm * tm_pointer = std::localtime( &time_now ); // thread-safe on mingw-w64 (thread local variable) and on MSVC btw
|
|
// http://stackoverflow.com/questions/18551409/localtime-r-support-on-mingw
|
|
// tm_pointer points to thread-local data, memory is owned/managed by the system/library
|
|
#else
|
|
// linux, freebsd, have this
|
|
struct tm tm_object; // automatic storage duration http://en.cppreference.com/w/cpp/language/storage_duration
|
|
struct tm * tm_pointer = & tm_object; // just point to our data
|
|
auto x = localtime_r( &time_now , tm_pointer ); // modifies our own (this thread) data in tm_object, this is safe http://linux.die.net/man/3/localtime_r
|
|
if (x != tm_pointer) return "(internal error in get_current_time)"; // redundant check in case of broken implementation of localtime_r
|
|
#endif
|
|
// tm_pointer now points to proper time data, and that memory is automatically managed
|
|
if (!tm_pointer) return "(internal error in get_current_time - NULL)"; // redundant check in case of broken implementation of used library methods
|
|
|
|
std::stringstream stream;
|
|
stream << std::setfill('0')
|
|
<< std::setw(2) << tm_pointer->tm_year+1900
|
|
<< '-' << std::setw(2) << tm_pointer->tm_mon+1
|
|
<< '-' << std::setw(2) << tm_pointer->tm_mday
|
|
<< ' ' << std::setw(2) << tm_pointer->tm_hour
|
|
<< ':' << std::setw(2) << tm_pointer->tm_min
|
|
<< ':' << std::setw(2) << tm_pointer->tm_sec
|
|
<< '.' << std::setw(6) << (micro%1000000); // 6 because microseconds
|
|
return stream.str();
|
|
}
|
|
|
|
cNullstream g_nullstream; // extern a stream that does nothing (eats/discards data)
|
|
|
|
std::recursive_mutex gLoggerGuard; // extern
|
|
std::atomic<int> gLoggerGuardDepth; // extern
|
|
|
|
std::atomic<int> & gLoggerGuardDepth_Get() {
|
|
// TODO std::once would be nicer here
|
|
|
|
static bool once=0;
|
|
|
|
if (!once) { // initialize it once
|
|
once=1;
|
|
gLoggerGuardDepth=0;
|
|
}
|
|
|
|
return gLoggerGuardDepth; // global, atomic counter
|
|
}
|
|
|
|
|
|
// ====================================================================
|
|
|
|
namespace nDetail {
|
|
|
|
const char* DbgShortenCodeFileName(const char *s) {
|
|
const char *p = s;
|
|
const char *a = s;
|
|
|
|
bool inc=1;
|
|
while (*p) {
|
|
++p;
|
|
if (inc && ('\0' != * p)) { a=p; inc=false; } // point to the current character (if valid) becasue previous one was slash
|
|
if ((*p)=='/') { a=p; inc=true; } // point at current slash (but set inc to try to point to next character)
|
|
}
|
|
return a;
|
|
}
|
|
|
|
}
|
|
|
|
// a workaround for MSVC compiler; e.g. see https://bugs.webkit.org/show_bug.cgi?format=multiple&id=125795
|
|
#ifndef _MSC_VER
|
|
template<typename T, typename ...Args>
|
|
std::unique_ptr<T> make_unique( Args&& ...args )
|
|
{
|
|
return std::unique_ptr<T>( new T( std::forward<Args>(args)... ) );
|
|
}
|
|
#else
|
|
using std::make_unique;
|
|
#endif
|
|
// ====================================================================
|
|
|
|
char cFilesystemUtils::GetDirSeparatorSys() {
|
|
// TODO nicer os detection?
|
|
#if defined(OS_TYPE_POSIX)
|
|
return '/';
|
|
#elif defined(OS_TYPE_WINDOWS)
|
|
return '\\';
|
|
#else
|
|
#error "Do not know how to compile this for your platform."
|
|
#endif
|
|
}
|
|
|
|
char cFilesystemUtils::GetDirSeparatorInter() {
|
|
return '/';
|
|
}
|
|
|
|
string cFilesystemUtils::FileInternalToSystem(const std::string &name) {
|
|
string ret;
|
|
ret.resize(name.size());
|
|
std::replace_copy(name.begin(), name.end(), ret.begin(),
|
|
GetDirSeparatorInter() , GetDirSeparatorSys());
|
|
return ret;
|
|
}
|
|
|
|
string cFilesystemUtils::FileSystemToInternal(const std::string &name) {
|
|
string ret;
|
|
ret.reserve(name.size());
|
|
std::replace_copy(name.begin(), name.end(), ret.begin(),
|
|
GetDirSeparatorSys() , GetDirSeparatorInter());
|
|
return ret;
|
|
}
|
|
|
|
bool cFilesystemUtils::CreateDirTree(const std::string & dir, bool only_below) {
|
|
const bool dbg=false;
|
|
//struct stat st;
|
|
const char dirchS = cFilesystemUtils::GetDirSeparatorSys();
|
|
const char dirchI = cFilesystemUtils::GetDirSeparatorInter();
|
|
std::istringstream iss(dir);
|
|
string partI; // current par is in internal format (though it should not matter since it doesn't contain any slashes). eg "bar"
|
|
string sofarS=""; // sofarS - the so far created dir part is in SYSTEM format. eg "foo/bar"
|
|
if (dir.size()<1) return false; // illegal name
|
|
// dir[0] is valid from here
|
|
if ( only_below && ((dir[0]==dirchS) || (dir[0]==dirchI))) return false; // no jumping to top (on any os)
|
|
|
|
while (getline(iss,partI,dirchI)) { // get new component eg "bar" into part
|
|
if (dbg) cout << '['<<partI<<']' << endl;
|
|
sofarS += partI;
|
|
if (partI.size()<1) return false; // bad format?
|
|
if ((only_below) && (partI=="..")) return false; // trying to go up
|
|
|
|
if (dbg) cout << "test ["<<sofarS<<"]"<<endl;
|
|
// TODO nicer os detection?
|
|
#if defined(OS_TYPE_POSIX)
|
|
struct stat st;
|
|
bool exists = stat(sofarS.c_str() ,&st) == 0; // *
|
|
if (exists) {
|
|
if (! S_ISDIR(st.st_mode)) {
|
|
// std::cerr << "This exists, but as a file: [" << sofar << "]" << (size_t)st.st_ino << endl;
|
|
return false; // exists but is a file nor dir
|
|
}
|
|
}
|
|
#elif defined(OS_TYPE_WINDOWS)
|
|
DWORD dwAttrib = GetFileAttributesA(sofarS.c_str());
|
|
bool exists = (dwAttrib != INVALID_FILE_ATTRIBUTES && (dwAttrib & FILE_ATTRIBUTE_DIRECTORY));
|
|
#else
|
|
#error "Do not know how to compile this for your platform."
|
|
#endif
|
|
|
|
if (!exists) {
|
|
if (dbg) cout << "mkdir ["<<sofarS<<"]"<<endl;
|
|
#if defined(OS_TYPE_POSIX)
|
|
bool ok = 0== mkdir(sofarS.c_str(), 0700); // ***
|
|
#elif defined(OS_TYPE_WINDOWS)
|
|
bool ok = (bool) CreateDirectoryA(sofarS.c_str(), NULL); // TODO use -W() after conversion to unicode UTF16
|
|
#else
|
|
#error "Do not know how to compile this for your platform."
|
|
#endif
|
|
if (!ok) return false;
|
|
}
|
|
sofarS += dirchS;
|
|
}
|
|
return true;
|
|
}
|
|
// ====================================================================
|
|
|
|
namespace nDetail {
|
|
|
|
struct channel_use_info { ///< feedback information about using (e.g. opening) given debug channel - used internally by logging system
|
|
/// TODO not yet used in code
|
|
/// e.g. used to write into channel net/in/all that given message was a first logged message from never-before-logged thread or PID etc
|
|
bool m_was_interesting; ///< anything interesting happened when using the channel?
|
|
std::vector<std::string> m_extra_msg; ///< any additional messages about this channel use
|
|
};
|
|
|
|
cDebugScopeGuard::cDebugScopeGuard() : mLevel(-1) {
|
|
}
|
|
|
|
cDebugScopeGuard::~cDebugScopeGuard() {
|
|
if (mLevel != -1) {
|
|
gCurrentLogger.write_stream(mLevel,mChan) << mMsg << " ... end" << gCurrentLogger.endline() << std::flush;
|
|
}
|
|
}
|
|
|
|
void cDebugScopeGuard::Assign(const string &chan, const int level, const string &msg) {
|
|
mChan=chan;
|
|
mLevel=level;
|
|
mMsg=msg;
|
|
}
|
|
|
|
} // namespace nDetail
|
|
|
|
// ====================================================================
|
|
|
|
cLogger::cLogger() :
|
|
mStream(NULL),
|
|
mStreamBrokenDebug(NULL),
|
|
mIsBroken(true), // before constructor finishes
|
|
mLevel(_debug_level_nr_warn),
|
|
mThread2Number_Biggest(0), // the CURRENT biggest value (no thread yet in map)
|
|
mPid2Number_Biggest(0)
|
|
{
|
|
mStream = & std::cout;
|
|
mStreamBrokenDebug = & std::cerr; // the backup stream
|
|
*mStreamBrokenDebug << "Creating the logger system" << endl;
|
|
mIsBroken=false; // ok, constr. succeeded, so string is not broken now
|
|
|
|
// this is here, because it could be using logging itself to log creation of first thread/PID etc
|
|
Thread2Number( std::this_thread::get_id() ); // convert current id to short number, useful to reserve a number so that main thread is usually called 1
|
|
Pid2Number( getpid() ); // add this proces ID as first one
|
|
}
|
|
|
|
cLogger::~cLogger() {
|
|
for (auto pair : mChannels) {
|
|
std::ofstream *ptr = pair.second;
|
|
delete ptr;
|
|
pair.second=NULL;
|
|
}
|
|
}
|
|
|
|
void cLogger::SetStreamBroken() {
|
|
SetStreamBroken("(no additional details about this problem)");
|
|
}
|
|
|
|
void cLogger::SetStreamBroken(const std::string &msg) {
|
|
_dbg_dbg("Stream is broken (msg: " << msg << ")");
|
|
if (!mIsBroken) { // if not already marked as broken
|
|
_dbg_dbg("(It was not broken before)");
|
|
std::cerr << OT_CODE_STAMP << "WARNING: due to a problem in the debug/logging system itself ("<<msg<<") - we are switching back to fallback stream (e.g. cerr)" << std::endl;
|
|
if (mStreamBrokenDebug == nullptr) {
|
|
std::cerr << OT_CODE_STAMP << " ERROR: in addition, while reporting this problem, mStreamBrokenDebug stream is NULL: " << mStreamBrokenDebug << std::endl;
|
|
} else {
|
|
(*mStreamBrokenDebug) << OT_CODE_STAMP << "WARNING: due to debug stream problem ("<<msg<<") - switching back to fallback stream (e.g. cerr)" << std::endl;
|
|
}
|
|
mIsBroken = true;
|
|
}
|
|
}
|
|
|
|
std::ostream & cLogger::write_stream(int level) {
|
|
return write_stream(level,"");
|
|
}
|
|
|
|
std::ostream & cLogger::write_stream(int level, const std::string & channel ) {
|
|
_dbg_dbg("level="<<level<<" channel="<<channel);
|
|
if (level >= mLevel) {
|
|
if (mStream) { // TODO now disabling mStream also disables writting to any channel
|
|
_dbg_dbg("Selecting output...");
|
|
ostream & output = SelectOutput(level,channel);
|
|
_dbg_dbg("Selecting output... done, output=" << (void*)(&output));
|
|
#if defined(OS_TYPE_WINDOWS)
|
|
output << windows_stream(level);
|
|
#endif
|
|
output << icon(level) << ' ';
|
|
std::thread::id this_id = std::this_thread::get_id();
|
|
output << "{" << Thread2Number(this_id) << "}";
|
|
auto nicePid = Pid2Number(getpid());
|
|
if (nicePid>0) output << " {p" << nicePid << "}";
|
|
output << ' ';
|
|
return output; // <--- return
|
|
} else _dbg_dbg("Not writting: No mStream");
|
|
} else _dbg_dbg("Not writting: Too low level level="<<level<<" not >= mLevel="<<mLevel);
|
|
return g_nullstream;
|
|
}
|
|
|
|
std::string cLogger::GetLogBaseDir() const {
|
|
return "log";
|
|
}
|
|
|
|
void cLogger::OpenNewChannel(const std::string & channel) noexcept {
|
|
try {
|
|
_dbg_dbg("Openning channel for channel="<<channel);
|
|
OpenNewChannel_(channel);
|
|
}
|
|
catch (const std::exception &except) {
|
|
SetStreamBroken(OT_CODE_STAMP + " Got exception when opening debug channel: " + ToStr(except.what()));
|
|
}
|
|
catch (...) {
|
|
SetStreamBroken(OT_CODE_STAMP + " Got not-standard exception when opening debug channel.");
|
|
}
|
|
}
|
|
|
|
void cLogger::OpenNewChannel_(const std::string & channel) { // channel=="net/sleep"
|
|
_dbg_dbg("Openning channel for channel="<<channel);
|
|
size_t last_split = channel.find_last_of(cFilesystemUtils::GetDirSeparatorInter());
|
|
|
|
string fname_system; // the full file name in system format
|
|
|
|
if (last_split==string::npos) { // The channel name has no directory, eg channel=="test"
|
|
string dir = GetLogBaseDir();
|
|
string basefile = channel + ".log";
|
|
string fname = dir + cFilesystemUtils::GetDirSeparatorInter() + basefile;
|
|
fname_system = cFilesystemUtils::FileInternalToSystem(fname); // <-
|
|
}
|
|
else { // there is a directory eg channel=="net/sleep"
|
|
// net/sleep
|
|
// ^----- last_split
|
|
string dir = GetLogBaseDir() + cFilesystemUtils::GetDirSeparatorInter() + channel.substr(0, last_split);
|
|
string basefile = channel.substr(last_split+1) + ".log";
|
|
string fname = dir + cFilesystemUtils::GetDirSeparatorInter() + basefile;
|
|
fname_system = cFilesystemUtils::FileInternalToSystem(fname); // <-
|
|
bool dirok = cFilesystemUtils::CreateDirTree(dir);
|
|
if (!dirok) { string err = "In logger failed to open directory (" + dir +") for channel (" + channel +")"; throw std::runtime_error(err); }
|
|
}
|
|
|
|
_dbg_dbg("Openning fname_system="<<fname_system);
|
|
std::ofstream * thefile = new std::ofstream( fname_system.c_str() ); // file system
|
|
*thefile << "====== Log opened: " << fname_system << " (in " << ((void*)thefile) << ") ======" << endl;
|
|
// cerr << "====== Log opened: " << fname_system << " (in " << ((void*)thefile) << ") ======" << endl;
|
|
_dbg_dbg( "====== Log opened: " << fname_system << " (in " << ((void*)thefile) << ") ======" );
|
|
mChannels.insert( std::pair<string,std::ofstream*>(channel , thefile ) ); // <- created the channel mapping
|
|
}
|
|
|
|
std::ostream & cLogger::SelectOutput(int level, const std::string & channel) noexcept {
|
|
try {
|
|
if (mIsBroken) {
|
|
_dbg_dbg("The stream is broken mIsBroken="<<mIsBroken<<" so will return backup stream");
|
|
return *mStreamBrokenDebug;
|
|
}
|
|
if (channel=="") {
|
|
_dbg_dbg("No channel given (channel="<<channel<<") so will return main stream");
|
|
return *mStream;
|
|
}
|
|
|
|
auto obj = mChannels.find(channel);
|
|
if (obj == mChannels.end()) { // not found - need to make new channel
|
|
_dbg_dbg("No stream openened for channel="<<channel<<" so will create it now");
|
|
OpenNewChannel(channel); // <- create channel
|
|
obj = mChannels.find(channel); // find again
|
|
if (obj == mChannels.end()) { // still not found! something is wrong
|
|
SetStreamBroken( OT_CODE_STAMP + " WARNING: can not get stream for channel="+ToStr(channel)+" level="+ToStr(channel) );
|
|
return *mStreamBrokenDebug;
|
|
}
|
|
}
|
|
auto the_stream_ptr = obj->second;
|
|
_dbg_dbg("Found the stream file for channel="<<channel<<" as the_stream_ptr="<<the_stream_ptr);
|
|
ASRT(the_stream_ptr);
|
|
return *the_stream_ptr; // <--- RETURN
|
|
}
|
|
catch (std::exception &except) {
|
|
SetStreamBroken( OT_CODE_STAMP + " Got exception: " + ToStr(except.what()) );
|
|
_dbg_dbg("Exception! Returning broken stream");
|
|
return *mStreamBrokenDebug;
|
|
}
|
|
catch (...) {
|
|
SetStreamBroken( OT_CODE_STAMP + " Got not-standard exception.");
|
|
_dbg_dbg("Exception! Returning broken stream");
|
|
return *mStreamBrokenDebug;
|
|
}
|
|
|
|
// dead code
|
|
}
|
|
|
|
void cLogger::setOutStreamFile(const string &fname) { // switch to using this file
|
|
_mark("WILL SWITCH DEBUG NOW to file: " << fname);
|
|
mOutfile = make_unique<std::ofstream>(fname);
|
|
mStream = & (*mOutfile);
|
|
_mark("Started new debug, to file: " << fname);
|
|
}
|
|
|
|
void cLogger::setOutStreamFromGlobalOptions() {
|
|
if ( gRunOptions.getDebug() ) {
|
|
if ( gRunOptions.getDebugSendToFile() ) {
|
|
mOutfile = make_unique<std::ofstream> ("debuglog.txt");
|
|
mStream = & (*mOutfile);
|
|
}
|
|
else if ( gRunOptions.getDebugSendToCerr() ) {
|
|
mStream = & std::cerr;
|
|
}
|
|
else {
|
|
mStream = & g_nullstream;
|
|
}
|
|
}
|
|
else {
|
|
mStream = & g_nullstream;
|
|
}
|
|
}
|
|
|
|
void cLogger::setDebugLevel(int level) {
|
|
bool note_before = (mLevel > level); // report the level change before or after the change? (on higher level)
|
|
if (note_before) _note("Setting debug level to "<<level);
|
|
mLevel = level;
|
|
if (!note_before) _note("Setting debug level to "<<level);
|
|
}
|
|
|
|
std::string cLogger::icon(int level) const {
|
|
// TODO replan to avoid needles converting back and forth char*, string etc
|
|
|
|
using namespace zkr;
|
|
#if defined(OS_TYPE_POSIX)
|
|
if (level >= 100) return cc::back::lightred + ToStr(cc::fore::lightyellow) + ToStr("ERROR ") + ToStr(cc::fore::lightyellow) + " " ;
|
|
if (level >= 90) return cc::back::lightyellow + ToStr(cc::fore::black) + ToStr("Warn ") + ToStr(cc::fore::red)+ " " ;
|
|
if (level >= 80) return cc::back::lightmagenta + ToStr(cc::fore::black) + ToStr("MARK "); //+ zkr::cc::console + ToStr(cc::fore::lightmagenta)+ " ";
|
|
if (level >= 75) return cc::back::lightyellow + ToStr(cc::fore::black) + ToStr("FACT ") + zkr::cc::console + ToStr(cc::fore::lightyellow)+ " ";
|
|
if (level >= 70) return cc::fore::green + ToStr("Note ");
|
|
if (level >= 50) return cc::fore::cyan + ToStr("info ");
|
|
if (level >= 40) return cc::fore::lightwhite + ToStr("dbg ");
|
|
if (level >= 30) return cc::fore::lightblue + ToStr("dbg ");
|
|
if (level >= 20) return cc::fore::blue + ToStr("dbg ");
|
|
|
|
#elif defined(OS_TYPE_WINDOWS)
|
|
if (level >= 100) return ToStr("ERROR ");
|
|
if (level >= 90) return ToStr("Warn ");
|
|
if (level >= 80) return ToStr("MARK ");
|
|
if (level >= 75) return ToStr("FACT ");
|
|
if (level >= 70) return ToStr("Note ");
|
|
if (level >= 50) return ToStr("info ");
|
|
if (level >= 40) return ToStr("dbg ");
|
|
if (level >= 30) return ToStr("dbg ");
|
|
if (level >= 20) return ToStr("dbg ");
|
|
#endif
|
|
|
|
return " ";
|
|
}
|
|
|
|
std::string cLogger::endline() const {
|
|
#if defined(OS_TYPE_POSIX)
|
|
return ToStr("") + zkr::cc::console + ToStr("\n"); // TODO replan to avoid needles converting back and forth char*, string etc
|
|
#elif defined(OS_TYPE_WINDOWS)
|
|
return ToStr("\n");
|
|
#endif
|
|
}
|
|
|
|
int cLogger::Thread2Number(const std::thread::id id) {
|
|
auto found = mThread2Number.find( id );
|
|
if (found == mThread2Number.end()) { // new one
|
|
mThread2Number_Biggest++;
|
|
mThread2Number[id] = mThread2Number_Biggest;
|
|
_info_c("dbg/main", "This is a new thread (used in debug), thread id="<<id); // can cause some recursion
|
|
return mThread2Number_Biggest;
|
|
} else {
|
|
return mThread2Number[id];
|
|
}
|
|
}
|
|
|
|
int cLogger::Pid2Number(const t_anypid id) {
|
|
auto found = mPid2Number.find( id );
|
|
if (found == mPid2Number.end()) { // new one
|
|
mPid2Number_Biggest++;
|
|
mPid2Number[id] = mPid2Number_Biggest;
|
|
_info_c("dbg/main", "This is a new process (used in debug), process pid="<<id); // can cause some recursion
|
|
return mPid2Number_Biggest;
|
|
} else {
|
|
return mPid2Number[id];
|
|
}
|
|
}
|
|
|
|
// ====================================================================
|
|
// object gCurrentLogger is defined later - in global namespace below
|
|
|
|
|
|
// ====================================================================
|
|
// vector debug
|
|
|
|
void DisplayStringEndl(std::ostream & out, const std::string text) {
|
|
out << text;
|
|
out << std::endl;
|
|
}
|
|
|
|
std::string SpaceFromEscape(const std::string &s) {
|
|
std::ostringstream newStr;
|
|
for(size_t i = 0; i < s.length();i++) {
|
|
if(s[i] == '\\' && s[i+1] ==32)
|
|
newStr<<"";
|
|
else
|
|
newStr<<s[i];
|
|
}
|
|
return newStr.str();
|
|
}
|
|
|
|
std::string EscapeFromSpace(const std::string &s) {
|
|
std::ostringstream newStr;
|
|
for(size_t i = 0; i < s.length();i++) {
|
|
if(s[i] == 32)
|
|
newStr << "\\" << " ";
|
|
else
|
|
newStr << s[i];
|
|
}
|
|
return newStr.str();
|
|
}
|
|
|
|
|
|
std::string EscapeString(const std::string &s) {
|
|
std::ostringstream newStr;
|
|
for(size_t i = 0; i < s.length();i++) {
|
|
if(s[i] >=32 && s[i] <= 126)
|
|
newStr<<s[i];
|
|
else
|
|
newStr<<"\\"<< (int) s[i];
|
|
}
|
|
|
|
return newStr.str();
|
|
}
|
|
|
|
|
|
bool CheckIfBegins(const std::string & beggining, const std::string & all) {
|
|
if (all.compare(0, beggining.length(), beggining) == 0) {
|
|
return 1;
|
|
}
|
|
else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
bool CheckIfEnds (std::string const & ending, std::string const & all){
|
|
if (all.length() >= ending.length()) {
|
|
return (0 == all.compare (all.length() - ending.length(), ending.length(), ending));
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
vector<string> WordsThatMatch(const std::string & sofar, const vector<string> & possib) {
|
|
vector<string> ret;
|
|
for ( auto rec : possib) { // check of possibilities
|
|
if (CheckIfBegins(sofar,rec)) {
|
|
rec = EscapeFromSpace(rec);
|
|
ret.push_back(rec); // this record matches
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
char GetLastChar(const std::string & str) { // TODO unicode?
|
|
auto s = str.length();
|
|
if (s==0) throw std::runtime_error("Getting last character of empty string (" + ToStr(s) + ")" + OT_CODE_STAMP);
|
|
return str.at( s - 1);
|
|
}
|
|
|
|
std::string GetLastCharIf(const std::string & str) { // TODO unicode?
|
|
auto s = str.length();
|
|
if (s==0) return ""; // empty string signalizes ther is nothing to be returned
|
|
return std::string( 1 , str.at( s - 1) );
|
|
}
|
|
|
|
// ====================================================================
|
|
|
|
// ASRT - assert. Name like ASSERT() was too long, and ASS() was just... no.
|
|
// Use it like this: ASRT( x>y ); with the semicolon at end, a clever trick forces this syntax :)
|
|
|
|
void Assert(bool result, const std::string &stamp, const std::string &condition) {
|
|
if (!result) {
|
|
_erro("Assert failed at "+stamp+": ASSERT( " << condition << ")");
|
|
throw std::runtime_error("Assert failed at "+stamp+": ASSERT( " + condition + ")");
|
|
}
|
|
}
|
|
|
|
// ====================================================================
|
|
// advanced string
|
|
|
|
const std::string GetMultiline(string endLine) {
|
|
std::string result(""); // Taken from OT_CLI_ReadUntilEOF
|
|
while (true) {
|
|
std::string input_line("");
|
|
if (std::getline(std::cin, input_line, '\n'))
|
|
{
|
|
input_line += "\n";
|
|
if (input_line[0] == '~')
|
|
break;
|
|
result += input_line;
|
|
}
|
|
if (std::cin.eof() )
|
|
{
|
|
std::cin.clear();
|
|
break;
|
|
}
|
|
if (std::cin.fail() )
|
|
{
|
|
std::cin.clear();
|
|
break;
|
|
}
|
|
if (std::cin.bad())
|
|
{
|
|
std::cin.clear();
|
|
break;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
vector<string> SplitString(const string & str){
|
|
std::istringstream iss(str);
|
|
vector<string> vec { std::istream_iterator<string>{iss}, std::istream_iterator<string>{} };
|
|
return vec;
|
|
}
|
|
|
|
bool checkPrefix(const string & str, char prefix) {
|
|
if (str.at(0) == prefix)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
// ====================================================================
|
|
// operation on files
|
|
|
|
|
|
#ifdef __unix
|
|
|
|
void cEnvUtils::GetTmpTextFile() {
|
|
// TODO make this name configurable (depending on project)
|
|
char filename[] = "/tmp/otshellutils_text.XXXXXX";
|
|
fd = mkstemp(filename);
|
|
if (fd == -1) {
|
|
_erro("Can't create the file: " << filename);
|
|
return;
|
|
}
|
|
mFilename = filename;
|
|
}
|
|
|
|
void cEnvUtils::CloseFile() {
|
|
close(fd);
|
|
unlink( mFilename.c_str() );
|
|
}
|
|
|
|
void cEnvUtils::OpenEditor() {
|
|
char* editor = std::getenv("OT_EDITOR"); //TODO Read editor from configuration file
|
|
if (editor == NULL)
|
|
editor = std::getenv("VISUAL");
|
|
if (editor == NULL)
|
|
editor = std::getenv("EDITOR");
|
|
|
|
string command;
|
|
if (editor != NULL)
|
|
command = ToStr(editor) + " " + mFilename;
|
|
else
|
|
command = "/usr/bin/editor " + mFilename;
|
|
_dbg3("Opening editor with command: " << command);
|
|
if ( system( command.c_str() ) == -1 )
|
|
_erro("Cannot execute system command: " << command);
|
|
}
|
|
|
|
const string cEnvUtils::ReadFromTmpFile() {
|
|
std::ifstream ifs(mFilename);
|
|
string msg((std::istreambuf_iterator<char>(ifs)), std::istreambuf_iterator<char>());
|
|
return msg;
|
|
}
|
|
|
|
const string cEnvUtils::Compose() {
|
|
GetTmpTextFile();
|
|
OpenEditor();
|
|
string input = ReadFromTmpFile();
|
|
CloseFile();
|
|
return input;
|
|
}
|
|
|
|
#endif
|
|
|
|
const string cEnvUtils::ReadFromFile(const string path) {
|
|
std::ifstream ifs(path);
|
|
string msg((std::istreambuf_iterator<char>(ifs)), std::istreambuf_iterator<char>());
|
|
return msg;
|
|
}
|
|
|
|
void hintingToTxt(std::fstream & file, string command, vector<string> &commands) {
|
|
if(file.good()) {
|
|
file<<command<<"~"<<endl;
|
|
for (auto a: commands) {
|
|
file <<a<< " ";
|
|
file.flush();
|
|
}
|
|
file<<endl;
|
|
}
|
|
}
|
|
|
|
string stringToColor(const string &hash) {
|
|
// Generete vector with all possible light colors
|
|
vector <string> lightColors;
|
|
using namespace zkr;
|
|
lightColors.push_back(cc::fore::lightblue);
|
|
lightColors.push_back(cc::fore::lightred);
|
|
lightColors.push_back(cc::fore::lightmagenta);
|
|
lightColors.push_back(cc::fore::lightgreen);
|
|
lightColors.push_back(cc::fore::lightcyan);
|
|
lightColors.push_back(cc::fore::lightyellow);
|
|
lightColors.push_back(cc::fore::lightwhite);
|
|
|
|
int sum=0;
|
|
|
|
for (auto ch : hash) sum+=ch;
|
|
auto color = sum%(lightColors.size()-1);
|
|
|
|
return lightColors.at( color );
|
|
}
|
|
|
|
|
|
// ====================================================================
|
|
// algorthms
|
|
|
|
|
|
} // namespace nUtil
|
|
|
|
|
|
} // namespace OT
|
|
|
|
// global namespace
|
|
|
|
const extern int _dbg_ignore = 0; // see description in .hpp
|
|
|
|
std::string GetObjectName() {
|
|
//static std::string * name=nullptr;
|
|
//if (!name) name = new std::string("(global)");
|
|
return "";
|
|
}
|
|
|
|
// ====================================================================
|
|
|
|
nOT::nUtils::cLogger gCurrentLogger;
|
|
|