// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdmisc.h" #include "nel/misc/types_nl.h" #include "nel/misc/common.h" #ifdef NL_OS_WINDOWS # include # include # include #define popen _popen #define pclose _pclose #elif defined NL_OS_MAC # include #elif defined NL_OS_UNIX # include # include # include # include #endif #define MAX_LINE_WIDTH 256 #include "nel/misc/command.h" #include "nel/misc/path.h" #include "nel/misc/i18n.h" using namespace std; #ifndef NL_COMP_MINGW #ifdef NL_OS_WINDOWS # pragma message( " " ) # if FINAL_VERSION # pragma message( "************************" ) # pragma message( "**** FINAL_VERSION *****" ) # pragma message( "************************" ) # else # pragma message( "Not using FINAL_VERSION") # endif // FINAL_VERSION # ifdef ASSERT_THROW_EXCEPTION # pragma message( "nlassert throws exceptions" ) # else # pragma message( "nlassert does not throw exceptions" ) # endif // ASSERT_THROW_EXCEPTION # ifdef _STLPORT_VERSION # pragma message( "Using STLport" ) # else # pragma message( "Using standard STL" ) # endif // _STLPORT_VERSION # pragma message( " " ) # if (_MSC_VER >= 1200) && (_MSC_VER < 1400) && (WINVER < 0x0500) //Using VC7 and later lib, need this to compile on VC6 extern "C" long _ftol2( double dblSource ) { return _ftol( dblSource ); } # endif #endif // NL_OS_WINDOWS #endif // !NL_COMP_MINGW #if defined(NL_HAS_SSE2) && !defined(NL_CPU_X86_64) #ifdef NL_NO_EXCEPTION_SPECS void *operator new(size_t size) { void *p = aligned_malloc(size, NL_DEFAULT_MEMORY_ALIGNMENT); if (p == NULL) throw std::bad_alloc(); return p; } void *operator new[](size_t size) { void *p = aligned_malloc(size, NL_DEFAULT_MEMORY_ALIGNMENT); if (p == NULL) throw std::bad_alloc(); return p; } void operator delete(void *p) noexcept { aligned_free(p); } void operator delete[](void *p) noexcept { aligned_free(p); } #else void *operator new(size_t size) throw(std::bad_alloc) { void *p = aligned_malloc(size, NL_DEFAULT_MEMORY_ALIGNMENT); if (p == NULL) throw std::bad_alloc(); return p; } void *operator new[](size_t size) throw(std::bad_alloc) { void *p = aligned_malloc(size, NL_DEFAULT_MEMORY_ALIGNMENT); if (p == NULL) throw std::bad_alloc(); return p; } void operator delete(void *p) throw() { aligned_free(p); } void operator delete[](void *p) throw() { aligned_free(p); } #endif #endif /* NL_HAS_SSE2 */ #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NLMISC { /* * Portable Sleep() function that suspends the execution of the calling thread for a number of milliseconds. * Note: the resolution of the timer is system-dependant and may be more than 1 millisecond. */ void nlSleep( uint32 ms ) { #ifdef NL_OS_WINDOWS #ifdef NL_DEBUG // a Sleep(0) "block" the other thread in DEBUG/_CONSOLE, so we clamp ms = max(ms, (uint32)1); #endif Sleep( ms ); #elif defined NL_OS_UNIX //usleep( ms*1000 ); // resolution: 20 ms! timespec ts; ts.tv_sec = ms/1000; ts.tv_nsec = (ms%1000)*1000000; int res; do { res = nanosleep( &ts, &ts ); // resolution: 10 ms (with common scheduling policy) } while ( (res != 0) && (errno==EINTR) ); #endif } /* * Returns Thread Id (note: on Linux, Process Id is the same as the Thread Id) */ size_t getThreadId() { #ifdef NL_OS_WINDOWS return GetCurrentThreadId(); #elif defined NL_OS_UNIX return size_t(pthread_self()); // doesnt work on linux kernel 2.6 return getpid(); #endif } /* * Returns a readable string from a vector of bytes. '\0' are replaced by ' ' */ string stringFromVector( const vector& v, bool limited ) { string s; if (!v.empty()) { int size = (int)v.size (); if (limited && size > 1000) { string middle = "......"; s.resize (1000 + middle.size()); memcpy (&*s.begin(), &*v.begin(), 500); memcpy (&*s.begin()+500, &*middle.begin(), middle.size()); memcpy (&*s.begin()+500+middle.size(), &*v.begin()+size-500, 500); } else { s.resize (size); memcpy( &*s.begin(), &*v.begin(), v.size() ); } // Replace '\0' characters string::iterator is; for ( is=s.begin(); is!=s.end(); ++is ) { // remplace non printable char and % with '?' chat if ( ! isprint((uint8)(*is)) || (*is) == '%') { (*is) = '?'; } } } /* if ( ! v.empty() ) { // Copy contents s.resize( v.size() ); memcpy( &*s.begin(), &*v.begin(), v.size() ); // Replace '\0' characters string::iterator is; for ( is=s.begin(); is!=s.end(); ++is ) { // remplace non printable char and % with '?' chat if ( ! isprint((*is)) || (*is) == '%') { (*is) = '?'; } } } */ return s; } sint smprintf( char *buffer, size_t count, const char *format, ... ) { sint ret; va_list args; va_start( args, format ); ret = vsnprintf( buffer, count, format, args ); if ( ret == -1 ) { buffer[count-1] = '\0'; } va_end( args ); return( ret ); } sint64 atoiInt64 (const char *ident, sint64 base) { sint64 number = 0; bool neg = false; // NULL string nlassert (ident != NULL); // empty string if (*ident == '\0') goto end; // + sign if (*ident == '+') ident++; // - sign if (*ident == '-') { neg = true; ident++; } while (*ident != '\0') { if (isdigit((unsigned char)*ident)) { number *= base; number += (*ident)-'0'; } else if (base > 10 && islower((unsigned char)*ident)) { number *= base; number += (*ident)-'a'+10; } else if (base > 10 && isupper((unsigned char)*ident)) { number *= base; number += (*ident)-'A'+10; } else { goto end; } ident++; } end: if (neg) number = -number; return number; } void itoaInt64 (sint64 number, char *str, sint64 base) { str[0] = '\0'; char b[256]; if(!number) { str[0] = '0'; str[1] = '\0'; return; } memset(b,'\0',255); memset(b,'0',64); sint n; sint64 x = number; if (x < 0) x = -x; char baseTable[] = "0123456789abcdefghijklmnopqrstuvwyz"; for(n = 0; n < 64; n ++) { sint num = (sint)(x % base); b[64 - n] = baseTable[num]; if(!x) { int k; int j = 0; if (number < 0) { str[j++] = '-'; } for(k = 64 - n + 1; k <= 64; k++) { str[j ++] = b[k]; } str[j] = '\0'; break; } x /= base; } } uint raiseToNextPowerOf2(uint v) { uint res=1; while(res>=1; if(v) return false; } else v>>=1; } return true; } string bytesToHumanReadable (const std::string &bytes) { return bytesToHumanReadable (atoiInt64(bytes.c_str())); } string bytesToHumanReadable (uint64 bytes) { static const char *divTable[]= { "B", "KiB", "MiB", "GiB", "TiB" }; uint div = 0; uint64 res = bytes; uint64 newres = res; for(;;) { newres /= 1024; if(newres < 8 || div > 3) break; div++; res = newres; } return toString ("%" NL_I64 "u %s", res, divTable[div]); } std::string bytesToHumanReadableUnits (uint64 bytes, const std::vector &units) { if (units.empty()) return ""; uint div = 0; uint last = units.size()-1; uint64 res = bytes; uint64 newres = res; for(;;) { newres /= 1024; if(newres < 8 || div > 3 || div == last) break; ++div; res = newres; } return toString ("%" NL_I64 "u %s", res, units[div].c_str()); } uint32 humanReadableToBytes (const string &str) { uint32 res; if(str.empty()) return 0; // not a number if(str[0]<'0' || str[0]>'9') return 0; if (!fromString(str, res)) return 0; if(str[str.size()-1] == 'B') { if (str.size()<3) return res; // there's no break and it's **normal** switch (str[str.size()-2]) { // kB/KB, MB, GB and TB are 1000 multiples case 'T': res *= 1000; case 'G': res *= 1000; case 'M': res *= 1000; case 'k': res *= 1000; break; // kilo symbol should be a lowercase K case 'K': res *= 1000; break; case 'i': { // KiB, MiB, GiB and TiB are 1024 multiples if (str.size()<4) return res; switch (str[str.size()-3]) { case 'T': res *= 1024; case 'G': res *= 1024; case 'M': res *= 1024; case 'K': res *= 1024; default: ; } } default: ; } } return res; } NLMISC_CATEGORISED_COMMAND(nel,btohr, "Convert a bytes number into an human readable number", "") { nlunreferenced(rawCommandString); nlunreferenced(quiet); nlunreferenced(human); if (args.size() != 1) return false; log.displayNL("%s -> %s", args[0].c_str(), bytesToHumanReadable(args[0]).c_str()); return true; } NLMISC_CATEGORISED_COMMAND(nel,hrtob, "Convert a human readable number into a bytes number", "
") { nlunreferenced(rawCommandString); nlunreferenced(quiet); nlunreferenced(human); if (args.size() != 1) return false; log.displayNL("%s -> %u", args[0].c_str(), humanReadableToBytes(args[0])); return true; } string secondsToHumanReadable (uint32 time) { static const char *divTable[] = { "s", "mn", "h", "d" }; static uint divCoef[] = { 60, 60, 24 }; uint div = 0; uint32 res = time; uint32 newres = res; for(;;) { if(div > 2) break; newres /= divCoef[div]; if(newres < 3) break; div++; res = newres; } return toString ("%u%s", res, divTable[div]); } std::string timestampToHumanReadable(uint32 timestamp) { char buffer[30]; time_t dtime = timestamp; tm *tms = localtime(&dtime); if (tms) { strftime(buffer, 30, "%Y-%m-%d %H:%M:%S", tms); return std::string(buffer); } return ""; } uint32 fromHumanReadable (const std::string &str) { if (str.empty()) return 0; uint32 val; fromString(str, val); switch (str[str.size()-1]) { case 's': return val; // second case 'n': return val*60; // minutes (mn) case 'h': return val*60*60; // hour case 'd': return val*60*60*24; // day case 'b': // bytes switch (str[str.size()-2]) { case 'k': return val*1024; case 'm': return val*1024*1024; case 'g': return val*1024*1024*1024; default : return val; } default: return val; } return 0; } NLMISC_CATEGORISED_COMMAND(nel,stohr, "Convert a second number into an human readable time", "") { nlunreferenced(rawCommandString); nlunreferenced(quiet); nlunreferenced(human); if (args.size() != 1) return false; uint32 seconds; fromString(args[0], seconds); log.displayNL("%s -> %s", args[0].c_str(), secondsToHumanReadable(seconds).c_str()); return true; } std::string toLower(const std::string &str) { string res; res.reserve(str.size()); for(uint i = 0; i < str.size(); i++) { if( (str[i] >= 'A') && (str[i] <= 'Z') ) res += str[i] - 'A' + 'a'; else res += str[i]; } return res; } char toLower(const char ch) { if( (ch >= 'A') && (ch <= 'Z') ) { return ch - 'A' + 'a'; } else { return ch; } } void toLower(char *str) { if (str == 0) return; while(*str != '\0') { if( (*str >= 'A') && (*str <= 'Z') ) { *str = *str - 'A' + 'a'; } str++; } } std::string toUpper(const std::string &str) { string res; res.reserve(str.size()); for(uint i = 0; i < str.size(); i++) { if( (str[i] >= 'a') && (str[i] <= 'z') ) res += str[i] - 'a' + 'A'; else res += str[i]; } return res; } void toUpper(char *str) { if (str == 0) return; while(*str != '\0') { if( (*str >= 'a') && (*str <= 'z') ) { *str = *str - 'a' + 'A'; } str++; } } std::string toHexa(const uint8 &b) { return toString("%02hhx", b); } std::string toHexa(const uint8 *data, uint size) { std::string res; // hexadecimal string will be always twice the original size res.reserve(size * 2); // process each byte for (uint i = 0; i < size; ++i) { res += toHexa(data[i]); } return res; } std::string toHexa(const std::string &str) { return toHexa((uint8*)str.c_str(), (uint)str.length()); } std::string toHexa(const char *str) { return toHexa((uint8*)str, (uint)strlen(str)); } bool fromHexa(const std::string &hexa, uint8 &b) { return fromHexa(hexa.c_str(), b); } bool fromHexa(const std::string &hexa, uint8 *data) { return fromHexa(hexa.c_str(), data); } bool fromHexa(const std::string &hexa, std::string &str) { return fromHexa(hexa.c_str(), str); } bool fromHexa(const char *hexa, uint8 &b) { char c1 = *hexa; char c2 = *(hexa+1); uint8 x1, x2; if (!fromHexa(c1, x1)) return false; if (!fromHexa(c2, x2)) return false; b = (x1 << 4) | x2; return true; } bool fromHexa(const char *hexa, uint8 *data) { // length of the string uint len = strlen(hexa); // process each byte for (uint i = 0; i < len; i += 2) { if (!fromHexa(hexa + i, *(data++))) return false; } return true; } bool fromHexa(const char *hexa, std::string &str) { str.resize(strlen(hexa) * 2); return fromHexa(hexa, (uint8*)str.c_str()); } bool fromHexa(const char hexa, uint8 &b) { if (hexa >= '0' && hexa <= '9') { b = hexa - '0'; return true; } if (hexa >= 'A' && hexa <= 'F') { b = hexa - 'A' + 10; return true; } if (hexa >= 'a' && hexa <= 'f') { b = hexa - 'a' + 10; return true; } return false; } std::string formatThousands(const std::string& s) { sint i, k; sint remaining = (sint)s.length() - 1; static std::string separator = NLMISC::CI18N::get("uiThousandsSeparator").toUtf8(); // Don't add separator if the number is < 10k if (remaining < 4) return s; std::string ns; do { for (i = remaining, k = 0; i >= 0 && k < 3; --i, ++k ) { ns = s[i] + ns; // New char is added to front of ns if ( i > 0 && k == 2) ns = separator + ns; // j > 0 means still more digits } remaining -= 3; } while (remaining >= 0); return ns; } // // Exceptions // Exception::Exception() : _Reason("Unknown Exception") { // nlinfo("Exception will be launched: %s", _Reason.c_str()); } Exception::Exception(const std::string &reason) : _Reason(reason) { nlinfo("Exception will be launched: %s", _Reason.c_str()); } Exception::Exception(const char *format, ...) { NLMISC_CONVERT_VARGS (_Reason, format, NLMISC::MaxCStringSize); nlinfo("Exception will be launched: %s", _Reason.c_str()); } const char *Exception::what() const throw() { return _Reason.c_str(); } bool killProgram(uint32 pid) { #ifdef NL_OS_UNIX int res = kill(pid, SIGKILL); if(res == -1) { char *err = strerror (errno); nlwarning("Failed to kill '%d' err %d: '%s'", pid, errno, err); } return res == 0; /*#elif defined(NL_OS_WINDOWS) // it doesn't work because pid != handle and i don't know how to kill a pid or know the real handle of another service (not -1) int res = TerminateProcess((HANDLE)pid, 888); nlwarning("Failed to kill '%d' err %d: '%s'", pid, GetLastError (), lpMsgBuf); return res != 0; */ #else nlwarning("kill not implemented on this OS"); return false; #endif } bool abortProgram(uint32 pid) { #ifdef NL_OS_UNIX int res = kill(pid, SIGABRT); if(res == -1) { char *err = strerror (errno); nlwarning("Failed to abort '%d' err %d: '%s'", pid, errno, err); } return res == 0; #else nlwarning("abort not implemented on this OS"); return false; #endif } #ifdef NL_OS_WINDOWS static bool createProcess(const std::string &programName, const std::string &arguments, bool log, PROCESS_INFORMATION &pi) { STARTUPINFOW si; memset(&si, 0, sizeof(si)); memset(&pi, 0, sizeof(pi)); si.cb = sizeof(si); // Enable nlassert/nlstop to display the error reason & callstack const char *SE_TRANSLATOR_IN_MAIN_MODULE = "NEL_SE_TRANS"; char envBuf[2]; if (GetEnvironmentVariableA(SE_TRANSLATOR_IN_MAIN_MODULE, envBuf, 2) != 0) { SetEnvironmentVariableA(SE_TRANSLATOR_IN_MAIN_MODULE, NULL); } wchar_t *sProgramName = NULL; std::string args; // a .bat file must have first parameter to NULL and use 2nd parameter to pass filename if (CFile::getExtension(programName) == "bat") { args = "\"" + programName + "\" " + arguments; } else { ucstring ucProgramName; ucProgramName.fromUtf8(programName); sProgramName = new wchar_t[MAX_PATH]; wcscpy(sProgramName, (wchar_t*)ucProgramName.c_str()); // important! we need to specify the executable full path as first argument args = toString("\"%s\" ", programName.c_str()) + arguments; } // or 0 for a window BOOL res = CreateProcessW(sProgramName, utf8ToWide(args), NULL, NULL, FALSE, CREATE_DEFAULT_ERROR_MODE | CREATE_NO_WINDOW, NULL, NULL /* current dir */, &si, &pi); if (sProgramName) { delete [] sProgramName; sProgramName = NULL; } if (!res) { if (log) { sint lastError = getLastError(); nlwarning("LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s'", programName.c_str(), arguments.c_str(), lastError, formatErrorMessage(lastError).c_str()); } CloseHandle( pi.hProcess ); CloseHandle( pi.hThread ); return false; } return true; } #endif bool launchProgram(const std::string &programName, const std::string &arguments, bool log) { #ifdef NL_OS_WINDOWS PROCESS_INFORMATION pi; if (!createProcess(programName, arguments, log, pi)) return false; //nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str()); CloseHandle( pi.hProcess ); CloseHandle( pi.hThread ); return true; #else #ifdef NL_OS_MAC // special OS X case with bundles if (toLower(CFile::getExtension(programName)) == ".app") { // we need to open bundles with "open" command std::string command = NLMISC::toString("open \"%s\"", programName.c_str()); // append arguments if any if (!arguments.empty()) { command += NLMISC::toString(" --args %s", arguments.c_str()); } int res = system(command.c_str()); if (!res) return true; if (log) { nlwarning ("LAUNCH: Failed launched '%s' with arg '%s' return code %d", programName.c_str(), arguments.c_str(), res); } return false; } #endif static bool firstLaunchProgram = true; if (firstLaunchProgram) { // The aim of this is to avoid defunct process. // // From "man signal": //------ // According to POSIX (3.3.1.3) it is unspecified what happens when SIGCHLD is set to SIG_IGN. Here // the BSD and SYSV behaviours differ, causing BSD software that sets the action for SIGCHLD to // SIG_IGN to fail on Linux. //------ // // But it works fine on my GNU/Linux so I do this because it's easier :) and I don't know exactly // what to do to be portable. signal(SIGCHLD, SIG_IGN); firstLaunchProgram = false; } // convert one arg into several args vector args; explodeArguments(arguments, args); // Store the size of each arg vector argv(args.size()+2); uint i = 0; argv[i] = (char *)programName.c_str(); for (; i < args.size(); i++) { argv[i+1] = (char *) args[i].c_str(); } argv[i+1] = NULL; int status = vfork (); ///////////////////////////////////////////////////////// // WARNING : NO MORE INSTRUCTION AFTER VFORK ! // READ VFORK manual ///////////////////////////////////////////////////////// if (status == -1) { char *err = strerror (errno); if (log) nlwarning("LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s'", programName.c_str(), arguments.c_str(), errno, err); } else if (status == 0) { // Exec (the only allowed instruction after vfork) status = execvp(programName.c_str(), &argv.front()); if (status == -1) { perror("Failed launched"); _exit(EXIT_FAILURE); } } else { //nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str()); return true; } #endif return false; } bool launchProgramArray (const std::string &programName, const std::vector &arguments, bool log) { #ifdef NL_OS_WINDOWS PROCESS_INFORMATION pi; std::string argumentsJoined = joinArguments(arguments); if (!createProcess(programName, argumentsJoined, log, pi)) return false; //nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str()); CloseHandle( pi.hProcess ); CloseHandle( pi.hThread ); return true; #else #ifdef NL_OS_MAC // special OS X case with bundles if (toLower(CFile::getExtension(programName)) == "app") { // we need to open bundles with "open" command std::string command = NLMISC::toString("open \"%s\"", programName.c_str()); std::string argumentsJoined = joinArguments(arguments); // append arguments if any if (!argumentsJoined.empty()) { command += NLMISC::toString(" --args %s", argumentsJoined.c_str()); } int res = system(command.c_str()); if (!res) return true; if (log) { nlwarning ("LAUNCH: Failed launched '%s' with arg '%s' return code %d", programName.c_str(), argumentsJoined.c_str(), res); } return false; } #endif static bool firstLaunchProgram = true; if (firstLaunchProgram) { // The aim of this is to avoid defunct process. // // From "man signal": //------ // According to POSIX (3.3.1.3) it is unspecified what happens when SIGCHLD is set to SIG_IGN. Here // the BSD and SYSV behaviours differ, causing BSD software that sets the action for SIGCHLD to // SIG_IGN to fail on Linux. //------ // // But it works fine on my GNU/Linux so I do this because it's easier :) and I don't know exactly // what to do to be portable. signal(SIGCHLD, SIG_IGN); firstLaunchProgram = false; } // Store the size of each arg vector argv(arguments.size()+2); uint i = 0; argv[i] = (char *)programName.c_str(); for (; i < arguments.size(); i++) { argv[i+1] = (char *) arguments[i].c_str(); } argv[i+1] = NULL; int status = vfork (); ///////////////////////////////////////////////////////// // WARNING : NO MORE INSTRUCTION AFTER VFORK ! // READ VFORK manual ///////////////////////////////////////////////////////// if (status == -1) { char *err = strerror (errno); if (log) nlwarning("LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s'", programName.c_str(), joinArguments(arguments).c_str(), errno, err); } else if (status == 0) { // Exec (the only allowed instruction after vfork) status = execvp(programName.c_str(), &argv.front()); if (status == -1) { perror("Failed launched"); _exit(EXIT_FAILURE); } } else { //nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str()); return true; } #endif return false; } sint launchProgramAndWaitForResult(const std::string &programName, const std::string &arguments, bool log) { #ifdef NL_OS_WINDOWS PROCESS_INFORMATION pi; if (!createProcess(programName, arguments, log, pi)) return -1; // Successfully created the process. Wait for it to finish. DWORD ret = WaitForSingleObject(pi.hProcess, INFINITE); if (ret == WAIT_OBJECT_0) { // Get the exit code. DWORD exitCode = 0; BOOL ok = GetExitCodeProcess(pi.hProcess, &exitCode); //nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str()); CloseHandle(pi.hProcess); CloseHandle(pi.hThread); if (ok) return (sint)exitCode; } if (log) { std::string error = toString((uint)ret); if (ret == WAIT_FAILED) { error += "(" + formatErrorMessage(getLastError()) +")"; } nlwarning("LAUNCH: Failed launched '%s' with arg '%s' and error: %s", programName.c_str(), arguments.c_str(), error.c_str()); } return -1; #else // program name is the only required string std::string command = programName; // only appends arguments if any if (!arguments.empty()) command += " " + arguments; // execute the command sint res = system(command.c_str()); if (res && log) nlwarning ("LAUNCH: Failed launched '%s' with arg '%s' return code %d", programName.c_str(), arguments.c_str(), res); return res; #endif } std::string getCommandOutput(const std::string &command) { FILE *pipe = popen(command.c_str(), "r"); if (!pipe) return ""; char buffer[MAX_LINE_WIDTH]; std::string result; while (!feof(pipe)) { if (fgets(buffer, MAX_LINE_WIDTH, pipe) != NULL) result += buffer; } pclose(pipe); return result; } std::string expandEnvironmentVariables(const std::string &s) { size_t len = s.length(); std::string ret; std::string::size_type pos1 = 0, pos2 = 0; // look for environement variables delimiters while(pos2 < len && (pos1 = s.find_first_of("%$", pos2)) != std::string::npos) { // copy string unprocessed part ret += s.substr(pos2, pos1-pos2); // extract a valid variable name (a-zA-Z0-9_) pos2 = pos1+1; while(pos2 < len && (isalnum(s[pos2]) || s[pos2] == '_')) ++pos2; // check if variable name is empty bool found = pos2 > pos1+1; std::string name; if (found) { // found at least 1 character name = s.substr(pos1+1, pos2-pos1-1); } // Windows format needs a trailing % delimiter if (found && s[pos1] == '%') { if (pos2 >= len || s[pos2] != '%') { // not a variable name, because no trailing % found = false; } else { // found a trailing %, next character to check ++pos2; } } // get variable value if name found if (found) { const char *value = getenv(name.c_str()); if (value) { // value found ret += std::string(value); } else { // value not found found = false; nlwarning("Environment variable '%s' not found, won't be replaced", name.c_str()); } } if (!found) { // variable or value not found, don't evaluate variable ret += s.substr(pos1, pos2-pos1); } } // copy last unprocessed part ret += s.substr(pos2); return ret; } bool explodeArguments(const std::string &str, std::vector &args) { if (str.empty()) return false; std::string::size_type pos1 = 0, pos2 = 0; do { // Look for the first non space character pos1 = str.find_first_not_of (" ", pos2); if (pos1 == std::string::npos) break; // Look for the first space or " pos2 = str.find_first_of (" \"", pos1); if (pos2 != std::string::npos) { // " ? if (str[pos2] == '"') { // Look for the final \" pos2 = str.find_first_of ("\"", pos2+1); if (pos2 != std::string::npos) { // Look for the first space pos2 = str.find_first_of (" ", pos2+1); } } } // Compute the size of the string to extract std::string::difference_type length = (pos2 != std::string::npos) ? pos2-pos1 : std::string::npos; std::string tmp = str.substr (pos1, length); // remove escape " from argument if (tmp.length() > 1 && tmp[0] == '"' && tmp[tmp.length()-1] == '"') tmp = tmp.substr(1, tmp.length()-2); args.push_back (tmp); } while(pos2 != std::string::npos); return true; } std::string joinArguments(const std::vector &args) { std::string res; for(uint i = 0, len = (uint)args.size(); i < len; ++i) { const std::string &arg = args[i]; // prepend space if (!res.empty()) res += " "; // escape only if spaces or empty argument if (arg.empty() || arg.find(' ') != std::string::npos) { res += "\"" + arg + "\""; } else { res += arg; } } return res; } std::string escapeArgument(const std::string &arg) { #ifdef NL_OS_WINDOWS // we can't escape %VARIABLE% on command-line under Windows return arg; #else // characters to escapce, only " and $ (to prevent a $something replaced by an environment variable) static const char s_charsToEscape[] = "\"$"; std::string res; std::string::size_type pos = 0, lastPos = 0; // to avoid reallocations res.reserve(arg.size() * 2); while ((pos = arg.find_first_of(s_charsToEscape, lastPos)) != std::string::npos) { // add previous part res += arg.substr(lastPos, pos - lastPos); // not already escaped if (!pos || arg[pos - 1] != '\\') res += '\\'; // add escaped character res += arg[pos]; lastPos = pos+1; } res += arg.substr(lastPos); return res; #endif } /* * Display the bits (with 0 and 1) composing a byte (from right to left) */ void displayByteBits( uint8 b, uint nbits, sint beginpos, bool displayBegin, NLMISC::CLog *log ) { string s1, s2; sint i; for ( i=nbits-1; i!=-1; --i ) { s1 += ( (b >> i) & 1 ) ? '1' : '0'; } log->displayRawNL( "%s", s1.c_str() ); if ( displayBegin ) { for ( i=nbits; i>beginpos+1; --i ) { s2 += " "; } s2 += "^"; log->displayRawNL( "%s beginpos=%u", s2.c_str(), beginpos ); } } //#define displayDwordBits(a,b,c) /* * Display the bits (with 0 and 1) composing a number (uint32) (from right to left) */ void displayDwordBits( uint32 b, uint nbits, sint beginpos, bool displayBegin, NLMISC::CLog *log ) { string s1, s2; sint i; for ( i=nbits-1; i!=-1; --i ) { s1 += ( (b >> i) & 1 ) ? '1' : '0'; } log->displayRawNL( "%s", s1.c_str() ); if ( displayBegin ) { for ( i=nbits; i>beginpos+1; --i ) { s2 += " "; } s2 += "^"; log->displayRawNL( "%s beginpos=%u", s2.c_str(), beginpos ); } } FILE* nlfopen(const std::string &filename, const std::string &mode) { #ifdef NL_OS_WINDOWS return _wfopen(utf8ToWide(filename), utf8ToWide(mode)); #else return fopen(filename.c_str(), mode.c_str()); #endif } int nlfseek64( FILE *stream, sint64 offset, int origin ) { #ifdef NL_OS_WINDOWS // fpos_t pos64 = 0; switch (origin) { case SEEK_CUR: if (fgetpos(stream, &pos64) != 0) return -1; case SEEK_END: pos64 = _filelengthi64(_fileno(stream)); if (pos64 == -1L) return -1; }; // Seek pos64 += offset; // Set the final position return fsetpos (stream, &pos64); #else // NL_OS_WINDOWS // TODO: to fix for Linux and Mac OS X // This code doesn't work under windows : fseek() implementation uses a signed 32 bits offset. What ever we do, it can't seek more than 2 Go. // For the moment, i don't know if it works under linux for seek of more than 2 Go. nlassert ((offset < SINT64_CONSTANT(2147483647)) && (offset > SINT64_CONSTANT(-2147483648))); bool first = true; do { // Get the size of the next fseek sint nextSeek; if (offset > 0) nextSeek = (sint)std::min ((sint64)SINT64_CONSTANT(2147483647), offset); else nextSeek = (sint)std::max ((sint64)-SINT64_CONSTANT(2147483648), offset); // Make a seek int result = fseek ( stream, nextSeek, first?origin:SEEK_CUR ); if (result != 0) return result; // Remaining offset -= nextSeek; first = false; } while (offset); return 0; #endif // NL_OS_WINDOWS } sint64 nlftell64(FILE *stream) { #ifdef NL_OS_WINDOWS fpos_t pos64 = 0; if (fgetpos(stream, &pos64) == 0) { return (sint64) pos64; } else return -1; #else nlunreferenced(stream); // TODO: implement for Linux and Mac OS X nlerror("Not implemented"); return -1; #endif } ////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////// /// Commands ////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////// NLMISC_CATEGORISED_COMMAND(nel, sleep, "Freeze the service for N seconds (for debug purpose)", "") { nlunreferenced(rawCommandString); nlunreferenced(quiet); nlunreferenced(human); if(args.size() != 1) return false; sint32 n; fromString(args[0], n); log.displayNL ("Sleeping during %d seconds", n); nlSleep(n * 1000); return true; } NLMISC_CATEGORISED_COMMAND(nel, system, "Execute the command line using system() function call (wait until the end of the command)", "") { nlunreferenced(rawCommandString); nlunreferenced(quiet); nlunreferenced(human); if(args.size() != 1) return false; string cmd = args[0]; log.displayNL ("Executing '%s'", cmd.c_str()); sint error = system(cmd.c_str()); if (error) { log.displayNL ("Execution of '%s' failed with error code %d", cmd.c_str(), error); } else { log.displayNL ("End of Execution of '%s'", cmd.c_str()); } return true; } NLMISC_CATEGORISED_COMMAND(nel, launchProgram, "Execute the command line using launcProgram() function call (launch in background task without waiting the end of the execution)", " ") { nlunreferenced(rawCommandString); nlunreferenced(quiet); nlunreferenced(human); if(args.size() != 2) return false; string cmd = args[0]; string arg = args[1]; log.displayNL ("Executing '%s' with argument '%s'", cmd.c_str(), arg.c_str()); launchProgram(cmd, arg); log.displayNL ("End of Execution of '%s' with argument '%s'", cmd.c_str(), arg.c_str()); return true; } NLMISC_CATEGORISED_COMMAND(nel, killProgram, "kill a program given the pid", "") { nlunreferenced(rawCommandString); nlunreferenced(quiet); nlunreferenced(human); if(args.size() != 1) return false; uint32 pid; fromString(args[0], pid); killProgram(pid); return true; } #ifdef NL_OS_WINDOWS LONG GetRegKey(HKEY key, LPCWSTR subkey, LPWSTR retdata) { HKEY hkey; LONG retval = RegOpenKeyExW(key, subkey, 0, KEY_QUERY_VALUE, &hkey); if (retval == ERROR_SUCCESS) { long datasize = MAX_PATH; wchar_t data[MAX_PATH]; RegQueryValueW(hkey, NULL, data, &datasize); lstrcpyW(retdata, data); RegCloseKey(hkey); } return retval; } #endif // NL_OS_WINDOWS static bool openDocWithExtension (const std::string &document, const std::string &ext) { #ifdef NL_OS_WINDOWS // First try ShellExecute() HINSTANCE result = ShellExecuteW(NULL, L"open", utf8ToWide(document), NULL, NULL, SW_SHOWDEFAULT); // If it failed, get the .htm regkey and lookup the program if ((uintptr_t)result <= HINSTANCE_ERROR) { wchar_t key[MAX_PATH + MAX_PATH]; // get the type of the extension if (GetRegKey(HKEY_CLASSES_ROOT, utf8ToWide("." + ext), key) == ERROR_SUCCESS) { lstrcatW(key, L"\\shell\\open\\command"); // get the command used to open a file with this extension if (GetRegKey(HKEY_CLASSES_ROOT, key, key) == ERROR_SUCCESS) { std::string program = wideToUtf8(key); // empty program if (program.empty()) return false; if (program[0] == '"') { // program is quoted std::string::size_type pos = program.find('"', 1); if (pos != std::string::npos) { // take part before next quote program = program.substr(1, pos - 1); } } else { // program has a parameter std::string::size_type pos = program.find(' ', 1); if (pos != std::string::npos) { // take part before first space program = program.substr(0, pos); } } // create process PROCESS_INFORMATION pi; return createProcess(program, document, false, pi); } } } else { return true; } #elif defined(NL_OS_MAC) CFURLRef url = CFURLCreateWithBytes(NULL, (const UInt8 *)document.c_str(), document.length(), kCFStringEncodingUTF8, NULL); if (url) { OSStatus res = LSOpenCFURLRef(url, 0); CFRelease(url); if (res != 0) { nlwarning("LSOpenCFURLRef %s returned %d", document.c_str(), (sint)res); return false; } return true; } else { nlwarning("Unable to create URL from %s", document.c_str()); return false; } #else std::string command = "/usr/bin/xdg-open"; if (!CFile::fileExists(command)) { if (ext == "htm") { command = "/etc/alternatives/x-www-browser"; if (!CFile::fileExists(command)) { command.clear(); } } else { command.clear(); } } if (command.empty()) { nlwarning("Unable to open %s", document.c_str()); return false; } // save LD_LIBRARY_PATH const char *previousEnv = getenv("LD_LIBRARY_PATH"); // clear LD_LIBRARY_PATH to avoid problems with Steam Runtime setenv("LD_LIBRARY_PATH", "", 1); bool res = launchProgram(command, document); // restore previous LD_LIBRARY_PATH setenv("LD_LIBRARY_PATH", previousEnv, 1); return res; #endif // NL_OS_WINDOWS return false; } bool openURL(const std::string &url) { return openDocWithExtension(url, "htm"); } bool openDoc(const std::string &document) { // get extension from document fullpath string ext = CFile::getExtension(document); // try to open document return openDocWithExtension(document, ext); } } // NLMISC