/* * Copyright (C) 2001-2008 Jacek Sieka, arnetheduck on gmail point com * * 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "stdinc.h" #include "DCPlusPlus.h" #include "Util.h" #include "File.h" #include "SettingsManager.h" #include "StringTokenizer.h" #include "SettingsManager.h" #include "version.h" #include "File.h" #include "SimpleXML.h" #ifndef _WIN32 #include #include #include #include #include #include #endif #include #include "CID.h" #include "FastAlloc.h" namespace dcpp { #ifndef _DEBUG FastCriticalSection FastAllocBase::cs; #endif string Util::emptyString; wstring Util::emptyStringW; tstring Util::emptyStringT; bool Util::away = false; bool Util::manualAway = false; string Util::awayMsg; time_t Util::awayTime; Util::CountryList Util::countries; string Util::configPath; string Util::systemPath; string Util::dataPath; string Util::localePath; static void sgenrand(unsigned long seed); extern "C" void bz_internal_error(int errcode) { dcdebug("bzip2 internal error: %d\n", errcode); } #if defined(_WIN32) && _MSC_VER == 1400 void WINAPI invalidParameterHandler(const wchar_t*, const wchar_t*, const wchar_t*, unsigned int, uintptr_t) { //do nothing, this exist because vs2k5 crt needs it not to crash on errors. } #endif void Util::initialize() { Text::initialize(); sgenrand((unsigned long)time(NULL)); #ifdef _WIN32 TCHAR buf[MAX_PATH+1]; ::GetModuleFileName(NULL, buf, MAX_PATH); // System config path is DC++ executable path... systemPath = Util::getFilePath(Text::fromT(buf)); configPath = systemPath; dataPath = systemPath; localePath = dataPath + "locale\\"; #else systemPath = "/etc/"; char* home = getenv("HOME"); configPath = home ? Text::toUtf8(home) + "/.dc++/" : "/tmp/"; dataPath = configPath; // dataPath in linux is usually prefix + /share/app_name, so we can't represent it here localePath = dataPath; // TODO no good default here either, fix #endif // Load boot settings try { SimpleXML boot; boot.fromXML(File(systemPath + "dcppboot.xml", File::READ, File::OPEN).read()); boot.stepIn(); if(boot.findChild("ConfigPath")) { StringMap params; #ifdef _WIN32 TCHAR path[MAX_PATH]; params["APPDATA"] = Text::fromT((::SHGetFolderPath(NULL, CSIDL_APPDATA, NULL, SHGFP_TYPE_CURRENT, path), path)); params["PERSONAL"] = Text::fromT((::SHGetFolderPath(NULL, CSIDL_PERSONAL, NULL, SHGFP_TYPE_CURRENT, path), path)); configPath = Util::formatParams(boot.getChildData(), params, false); #endif } } catch(const Exception& ) { // Unable to load boot settings... } if(!File::isAbsolute(configPath)) { configPath = systemPath + configPath; } #if _MSC_VER == 1400 _set_invalid_parameter_handler(reinterpret_cast<_invalid_parameter_handler>(invalidParameterHandler)); #endif try { // This product includes GeoIP data created by MaxMind, available from http://maxmind.com/ // Updates at http://www.maxmind.com/app/geoip_country string file = Util::getDataPath() + "GeoIpCountryWhois.csv"; string data = File(file, File::READ, File::OPEN).read(); const char* start = data.c_str(); string::size_type linestart = 0; string::size_type comma1 = 0; string::size_type comma2 = 0; string::size_type comma3 = 0; string::size_type comma4 = 0; string::size_type lineend = 0; CountryIter last = countries.end(); uint32_t startIP = 0; uint32_t endIP = 0, endIPprev = 0; for(;;) { comma1 = data.find(',', linestart); if(comma1 == string::npos) break; comma2 = data.find(',', comma1 + 1); if(comma2 == string::npos) break; comma3 = data.find(',', comma2 + 1); if(comma3 == string::npos) break; comma4 = data.find(',', comma3 + 1); if(comma4 == string::npos) break; lineend = data.find('\n', comma4); if(lineend == string::npos) break; startIP = Util::toUInt32(start + comma2 + 2); endIP = Util::toUInt32(start + comma3 + 2); uint16_t* country = (uint16_t*)(start + comma4 + 2); if((startIP-1) != endIPprev) last = countries.insert(last, make_pair((startIP-1), (uint16_t)16191)); last = countries.insert(last, make_pair(endIP, *country)); endIPprev = endIP; linestart = lineend + 1; } } catch(const FileException&) { } } #ifdef _WIN32 static const char badChars[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, '<', '>', '/', '"', '|', '?', '*', 0 }; #else static const char badChars[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, '<', '>', '\\', '"', '|', '?', '*', 0 }; #endif /** * Replaces all strange characters in a file with '_' * @todo Check for invalid names such as nul and aux... */ string Util::validateFileName(string tmp) { string::size_type i = 0; // First, eliminate forbidden chars while( (i = tmp.find_first_of(badChars, i)) != string::npos) { tmp[i] = '_'; i++; } // Then, eliminate all ':' that are not the second letter ("c:\...") i = 0; while( (i = tmp.find(':', i)) != string::npos) { if(i == 1) { i++; continue; } tmp[i] = '_'; i++; } // Remove the .\ that doesn't serve any purpose i = 0; while( (i = tmp.find("\\.\\", i)) != string::npos) { tmp.erase(i+1, 2); } i = 0; while( (i = tmp.find("/./", i)) != string::npos) { tmp.erase(i+1, 2); } // Remove any double \\ that are not at the beginning of the path... i = 1; while( (i = tmp.find("\\\\", i)) != string::npos) { tmp.erase(i+1, 1); } i = 1; while( (i = tmp.find("//", i)) != string::npos) { tmp.erase(i+1, 1); } // And last, but not least, the infamous ..\! ... i = 0; while( ((i = tmp.find("\\..\\", i)) != string::npos) ) { tmp[i + 1] = '_'; tmp[i + 2] = '_'; tmp[i + 3] = '_'; i += 2; } i = 0; while( ((i = tmp.find("/../", i)) != string::npos) ) { tmp[i + 1] = '_'; tmp[i + 2] = '_'; tmp[i + 3] = '_'; i += 2; } // Dots at the end of path names aren't popular i = 0; while( ((i = tmp.find(".\\", i)) != string::npos) ) { tmp[i] = '_'; i += 1; } i = 0; while( ((i = tmp.find("./", i)) != string::npos) ) { tmp[i] = '_'; i += 1; } return tmp; } string Util::cleanPathChars(string aNick) { string::size_type i = 0; while( (i = aNick.find_first_of("/.\\", i)) != string::npos) { aNick[i] = '_'; } return aNick; } string Util::getShortTimeString(time_t t) { char buf[255]; tm* _tm = localtime(&t); if(_tm == NULL) { strcpy(buf, "xx:xx"); } else { strftime(buf, 254, SETTING(TIME_STAMPS_FORMAT).c_str(), _tm); } return Text::toUtf8(buf); } /** * Decodes a URL the best it can... * Default ports: * http:// -> port 80 * dchub:// -> port 411 */ void Util::decodeUrl(const string& url, string& aServer, uint16_t& aPort, string& aFile) { // First, check for a protocol: xxxx:// string::size_type i = 0, j, k; aServer = emptyString; aFile = emptyString; if( (j=url.find("://", i)) != string::npos) { // Protocol found string protocol = url.substr(0, j); i = j + 3; if(protocol == "http") { aPort = 80; } else if(protocol == "dchub") { aPort = 411; } } if( (j=url.find('/', i)) != string::npos) { // We have a filename... aFile = url.substr(j); } if( (k=url.find(':', i)) != string::npos) { // Port if(j == string::npos) { aPort = static_cast(Util::toInt(url.substr(k+1))); } else if(k < j) { aPort = static_cast(Util::toInt(url.substr(k+1, j-k-1))); } } else { k = j; } if(k == string::npos) { aServer = url.substr(i); if(i==0) aPort = 411; } else aServer = url.substr(i, k-i); } string Util::getAwayMessage() { return (formatTime(awayMsg.empty() ? SETTING(DEFAULT_AWAY_MESSAGE) : awayMsg, awayTime)) + " <" APPNAME " v" VERSIONSTRING ">"; } string Util::formatBytes(int64_t aBytes) { char buf[128]; if(aBytes < 1024) { snprintf(buf, sizeof(buf), _("%d B"), (int)(aBytes&0xffffffff)); } else if(aBytes < 1024*1024) { snprintf(buf, sizeof(buf), _("%.02f KiB"), (double)aBytes/(1024.0)); } else if(aBytes < 1024*1024*1024) { snprintf(buf, sizeof(buf), _("%.02f MiB"), (double)aBytes/(1024.0*1024.0)); } else if(aBytes < (int64_t)1024*1024*1024*1024) { snprintf(buf, sizeof(buf), _("%.02f GiB"), (double)aBytes/(1024.0*1024.0*1024.0)); } else if(aBytes < (int64_t)1024*1024*1024*1024*1024) { snprintf(buf, sizeof(buf), _("%.02f TiB"), (double)aBytes/(1024.0*1024.0*1024.0*1024.0)); } else { snprintf(buf, sizeof(buf), _("%.02f PiB"), (double)aBytes/(1024.0*1024.0*1024.0*1024.0*1024.0)); } return buf; } string Util::formatExactSize(int64_t aBytes) { #ifdef _WIN32 TCHAR tbuf[128]; TCHAR number[64]; NUMBERFMT nf; _sntprintf(number, 64, _T("%I64d"), aBytes); TCHAR Dummy[16]; TCHAR sep[2] = _T(","); /*No need to read these values from the system because they are not used to format the exact size*/ nf.NumDigits = 0; nf.LeadingZero = 0; nf.NegativeOrder = 0; nf.lpDecimalSep = sep; GetLocaleInfo( LOCALE_SYSTEM_DEFAULT, LOCALE_SGROUPING, Dummy, 16 ); nf.Grouping = Util::toInt(Text::fromT(Dummy)); GetLocaleInfo( LOCALE_SYSTEM_DEFAULT, LOCALE_STHOUSAND, Dummy, 16 ); nf.lpThousandSep = Dummy; GetNumberFormat(LOCALE_USER_DEFAULT, 0, number, &nf, tbuf, sizeof(tbuf)/sizeof(tbuf[0])); char buf[128]; _snprintf(buf, sizeof(buf), _("%s B"), Text::fromT(tbuf).c_str()); return buf; #else char buf[128]; snprintf(buf, sizeof(buf), _("%'lld B"), (long long int)aBytes); return string(buf); #endif } string Util::getLocalIp() { string tmp; char buf[256]; gethostname(buf, 255); hostent* he = gethostbyname(buf); if(he == NULL || he->h_addr_list[0] == 0) return Util::emptyString; sockaddr_in dest; int i = 0; // We take the first ip as default, but if we can find a better one, use it instead... memcpy(&(dest.sin_addr), he->h_addr_list[i++], he->h_length); tmp = inet_ntoa(dest.sin_addr); if(Util::isPrivateIp(tmp) || strncmp(tmp.c_str(), "169", 3) == 0) { while(he->h_addr_list[i]) { memcpy(&(dest.sin_addr), he->h_addr_list[i], he->h_length); string tmp2 = inet_ntoa(dest.sin_addr); if(!Util::isPrivateIp(tmp2) && strncmp(tmp2.c_str(), "169", 3) != 0) { tmp = tmp2; } i++; } } return tmp; } bool Util::isPrivateIp(string const& ip) { struct in_addr addr; addr.s_addr = inet_addr(ip.c_str()); if (addr.s_addr != INADDR_NONE) { unsigned long haddr = ntohl(addr.s_addr); return ((haddr & 0xff000000) == 0x0a000000 || // 10.0.0.0/8 (haddr & 0xff000000) == 0x7f000000 || // 127.0.0.0/8 (haddr & 0xfff00000) == 0xac100000 || // 172.16.0.0/12 (haddr & 0xffff0000) == 0xc0a80000); // 192.168.0.0/16 } return false; } typedef const uint8_t* ccp; static wchar_t utf8ToLC(ccp& str) { wchar_t c = 0; if(str[0] & 0x80) { if(str[0] & 0x40) { if(str[0] & 0x20) { if(str[1] == 0 || str[2] == 0 || !((((unsigned char)str[1]) & ~0x3f) == 0x80) || !((((unsigned char)str[2]) & ~0x3f) == 0x80)) { str++; return 0; } c = ((wchar_t)(unsigned char)str[0] & 0xf) << 12 | ((wchar_t)(unsigned char)str[1] & 0x3f) << 6 | ((wchar_t)(unsigned char)str[2] & 0x3f); str += 3; } else { if(str[1] == 0 || !((((unsigned char)str[1]) & ~0x3f) == 0x80)) { str++; return 0; } c = ((wchar_t)(unsigned char)str[0] & 0x1f) << 6 | ((wchar_t)(unsigned char)str[1] & 0x3f); str += 2; } } else { str++; return 0; } } else { wchar_t c = Text::asciiToLower((char)str[0]); str++; return c; } return Text::toLower(c); } string::size_type Util::findSubString(const string& aString, const string& aSubString, string::size_type start) throw() { if(aString.length() < start) return (string::size_type)string::npos; if(aString.length() - start < aSubString.length()) return (string::size_type)string::npos; if(aSubString.empty()) return 0; // Hm, should start measure in characters or in bytes? bytes for now... const uint8_t* tx = (const uint8_t*)aString.c_str() + start; const uint8_t* px = (const uint8_t*)aSubString.c_str(); const uint8_t* end = tx + aString.length() - start - aSubString.length() + 1; wchar_t wp = utf8ToLC(px); while(tx < end) { const uint8_t* otx = tx; if(wp == utf8ToLC(tx)) { const uint8_t* px2 = px; const uint8_t* tx2 = tx; for(;;) { if(*px2 == 0) return otx - (uint8_t*)aString.c_str(); if(utf8ToLC(px2) != utf8ToLC(tx2)) break; } } } return (string::size_type)string::npos; } wstring::size_type Util::findSubString(const wstring& aString, const wstring& aSubString, wstring::size_type pos) throw() { if(aString.length() < pos) return static_cast(wstring::npos); if(aString.length() - pos < aSubString.length()) return static_cast(wstring::npos); if(aSubString.empty()) return 0; wstring::size_type j = 0; wstring::size_type end = aString.length() - aSubString.length() + 1; for(; pos < end; ++pos) { if(Text::toLower(aString[pos]) == Text::toLower(aSubString[j])) { wstring::size_type tmp = pos+1; bool found = true; for(++j; j < aSubString.length(); ++j, ++tmp) { if(Text::toLower(aString[tmp]) != Text::toLower(aSubString[j])) { j = 0; found = false; break; } } if(found) return pos; } } return static_cast(wstring::npos); } int Util::stricmp(const char* a, const char* b) { while(*a) { wchar_t ca = 0, cb = 0; int na = Text::utf8ToWc(a, ca); int nb = Text::utf8ToWc(b, cb); ca = Text::toLower(ca); cb = Text::toLower(cb); if(ca != cb) { return (int)ca - (int)cb; } a += abs(na); b += abs(nb); } wchar_t ca = 0, cb = 0; Text::utf8ToWc(a, ca); Text::utf8ToWc(b, cb); return (int)Text::toLower(ca) - (int)Text::toLower(cb); } int Util::strnicmp(const char* a, const char* b, size_t n) { const char* end = a + n; while(*a && a < end) { wchar_t ca = 0, cb = 0; int na = Text::utf8ToWc(a, ca); int nb = Text::utf8ToWc(b, cb); ca = Text::toLower(ca); cb = Text::toLower(cb); if(ca != cb) { return (int)ca - (int)cb; } a += abs(na); b += abs(nb); } wchar_t ca = 0, cb = 0; Text::utf8ToWc(a, ca); Text::utf8ToWc(b, cb); return (a >= end) ? 0 : ((int)Text::toLower(ca) - (int)Text::toLower(cb)); } string Util::encodeURI(const string& aString, bool reverse) { // reference: rfc2396 string tmp = aString; if(reverse) { string::size_type idx; for(idx = 0; idx < tmp.length(); ++idx) { if(tmp.length() > idx + 2 && tmp[idx] == '%' && isxdigit(tmp[idx+1]) && isxdigit(tmp[idx+2])) { tmp[idx] = fromHexEscape(tmp.substr(idx+1,2)); tmp.erase(idx+1, 2); } else { // reference: rfc1630, magnet-uri draft if(tmp[idx] == '+') tmp[idx] = ' '; } } } else { const string disallowed = ";/?:@&=+$," // reserved "<>#%\" " // delimiters "{}|\\^[]`"; // unwise string::size_type idx, loc; for(idx = 0; idx < tmp.length(); ++idx) { if(tmp[idx] == ' ') { tmp[idx] = '+'; } else { if(tmp[idx] <= 0x1F || tmp[idx] >= 0x7f || (loc = disallowed.find_first_of(tmp[idx])) != string::npos) { tmp.replace(idx, 1, toHexEscape(tmp[idx])); idx+=2; } } } } return tmp; } /** * This function takes a string and a set of parameters and transforms them according to * a simple formatting rule, similar to strftime. In the message, every parameter should be * represented by %[name]. It will then be replaced by the corresponding item in * the params stringmap. After that, the string is passed through strftime with the current * date/time and then finally written to the log file. If the parameter is not present at all, * it is removed from the string completely... */ string Util::formatParams(const string& msg, StringMap& params, bool filter) { string result = msg; string::size_type i, j, k; i = 0; while (( j = result.find("%[", i)) != string::npos) { if( (result.size() < j + 2) || ((k = result.find(']', j + 2)) == string::npos) ) { break; } string name = result.substr(j + 2, k - j - 2); StringMapIter smi = params.find(name); if(smi == params.end()) { result.erase(j, k-j + 1); i = j; } else { if(smi->second.find_first_of("%\\./") != string::npos) { string tmp = smi->second; // replace all % in params with %% for strftime string::size_type m = 0; while(( m = tmp.find('%', m)) != string::npos) { tmp.replace(m, 1, "%%"); m+=2; } if(filter) { // Filter chars that produce bad effects on file systems m = 0; while(( m = tmp.find_first_of("\\./", m)) != string::npos) { tmp[m] = '_'; } } result.replace(j, k-j + 1, tmp); i = j + tmp.size(); } else { result.replace(j, k-j + 1, smi->second); i = j + smi->second.size(); } } } result = formatTime(result, time(NULL)); return result; } /** Fix for wide formatting bug in wcsftime in the ms c lib for multibyte encodings of unicode in singlebyte locales */ string fixedftime(const string& format, struct tm* t) { string ret = format; const char codes[] = "aAbBcdHIjmMpSUwWxXyYzZ%"; char tmp[4]; tmp[0] = '%'; tmp[1] = tmp[2] = tmp[3] = 0; StringMap sm; static const size_t BUF_SIZE = 1024; boost::scoped_array buf(new char[BUF_SIZE]); for(size_t i = 0; i < strlen(codes); ++i) { tmp[1] = codes[i]; tmp[2] = 0; strftime(&buf[0], BUF_SIZE-1, tmp, t); sm[tmp] = &buf[0]; tmp[1] = '#'; tmp[2] = codes[i]; strftime(&buf[0], BUF_SIZE-1, tmp, t); sm[tmp] = &buf[0]; } for(StringMapIter i = sm.begin(); i != sm.end(); ++i) { for(string::size_type j = ret.find(i->first); j != string::npos; j = ret.find(i->first, j)) { ret.replace(j, i->first.length(), i->second); j += i->second.length() - i->first.length(); } } return ret; } string Util::formatTime(const string &msg, const time_t t) { if (!msg.empty()) { size_t bufsize = msg.size() + 256; struct tm* loc = localtime(&t); if(!loc) { return Util::emptyString; } #if _WIN32 tstring buf(bufsize, 0); buf.resize(_tcsftime(&buf[0], buf.size()-1, Text::toT(msg).c_str(), loc)); if(buf.empty()) { return fixedftime(msg, loc); } return Text::fromT(buf); #else // will this give wide representations for %a and %A? // surely win32 can't have a leg up on linux/unixen in this area. - Todd string buf(bufsize, 0); buf.resize(strftime(&buf[0], bufsize-1, msg.c_str(), loc)); while(buf.empty()) { bufsize+=64; buf.resize(bufsize); buf.resize(strftime(&buf[0], bufsize-1, msg.c_str(), loc)); } return Text::toUtf8(buf); #endif } return Util::emptyString; } /* Below is a high-speed random number generator with much better granularity than the CRT one in msvc...(no, I didn't write it...see copyright) */ /* Copyright (C) 1997 Makoto Matsumoto and Takuji Nishimura. Any feedback is very welcome. For any question, comments, see http://www.math.keio.ac.jp/matumoto/emt.html or email matumoto@math.keio.ac.jp */ /* Period parameters */ #define N 624 #define M 397 #define MATRIX_A 0x9908b0df /* constant vector a */ #define UPPER_MASK 0x80000000 /* most significant w-r bits */ #define LOWER_MASK 0x7fffffff /* least significant r bits */ /* Tempering parameters */ #define TEMPERING_MASK_B 0x9d2c5680 #define TEMPERING_MASK_C 0xefc60000 #define TEMPERING_SHIFT_U(y) (y >> 11) #define TEMPERING_SHIFT_S(y) (y << 7) #define TEMPERING_SHIFT_T(y) (y << 15) #define TEMPERING_SHIFT_L(y) (y >> 18) static unsigned long mt[N]; /* the array for the state vector */ static int mti=N+1; /* mti==N+1 means mt[N] is not initialized */ /* initializing the array with a NONZERO seed */ static void sgenrand(unsigned long seed) { /* setting initial seeds to mt[N] using */ /* the generator Line 25 of Table 1 in */ /* [KNUTH 1981, The Art of Computer Programming */ /* Vol. 2 (2nd Ed.), pp102] */ mt[0]= seed & 0xffffffff; for (mti=1; mti= N) { /* generate N words at one time */ int kk; if (mti == N+1) /* if sgenrand() has not been called, */ sgenrand(4357); /* a default initial seed is used */ for (kk=0;kk> 1) ^ mag01[y & 0x1]; } for (;kk> 1) ^ mag01[y & 0x1]; } y = (mt[N-1]&UPPER_MASK)|(mt[0]&LOWER_MASK); mt[N-1] = mt[M-1] ^ (y >> 1) ^ mag01[y & 0x1]; mti = 0; } y = mt[mti++]; y ^= TEMPERING_SHIFT_U(y); y ^= TEMPERING_SHIFT_S(y) & TEMPERING_MASK_B; y ^= TEMPERING_SHIFT_T(y) & TEMPERING_MASK_C; y ^= TEMPERING_SHIFT_L(y); return y; } string Util::getOsVersion() { string os; #ifdef _WIN32 OSVERSIONINFOEX ver; memset(&ver, 0, sizeof(OSVERSIONINFOEX)); ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX); if(!GetVersionEx((OSVERSIONINFO*)&ver)) { ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); if(!GetVersionEx((OSVERSIONINFO*)&ver)) { os = "Windows (version unknown)"; } } if(os.empty()) { if(ver.dwPlatformId != VER_PLATFORM_WIN32_NT) { os = "Win9x/ME/Junk"; } else if(ver.dwMajorVersion == 4) { os = "WinNT4"; } else if(ver.dwMajorVersion == 5) { if(ver.dwMinorVersion == 0) { os = "Win2000"; } else if(ver.dwMinorVersion == 1) { os = "WinXP"; } else if(ver.dwMinorVersion == 2) { os = "Win2003"; } else { os = "Unknown WinNT5"; } if(ver.wProductType & VER_NT_WORKSTATION) os += " Pro"; else if(ver.wProductType & VER_NT_SERVER) os += " Server"; else if(ver.wProductType & VER_NT_DOMAIN_CONTROLLER) os += " DC"; } else if(ver.dwMajorVersion == 6) { os = "WinVista"; } if(ver.wServicePackMajor != 0) { os += "SP"; os += Util::toString(ver.wServicePackMajor); if(ver.wServicePackMinor != 0) { os += '.'; os += Util::toString(ver.wServicePackMinor); } } } #else // _WIN32 struct utsname n; if(uname(&n) != 0) { os = "unix (unknown version)"; } else { os = Text::toUtf8(string(n.sysname) + " " + n.release + " (" + n.machine + ")"); } #endif // _WIN32 return os; } /* getIpCountry This function returns the country(Abbreviation) of an ip for exemple: it returns "PT", whitch standards for "Portugal" more info: http://www.maxmind.com/app/csv */ string Util::getIpCountry (string IP) { if (BOOLSETTING(GET_USER_COUNTRY)) { dcassert(count(IP.begin(), IP.end(), '.') == 3); //e.g IP 23.24.25.26 : w=23, x=24, y=25, z=26 string::size_type a = IP.find('.'); string::size_type b = IP.find('.', a+1); string::size_type c = IP.find('.', b+2); uint32_t ipnum = (Util::toUInt32(IP.c_str()) << 24) | (Util::toUInt32(IP.c_str() + a + 1) << 16) | (Util::toUInt32(IP.c_str() + b + 1) << 8) | (Util::toUInt32(IP.c_str() + c + 1) ); CountryIter i = countries.lower_bound(ipnum); if(i != countries.end()) { return string((char*)&(i->second), 2); } } return Util::emptyString; //if doesn't returned anything already, something is wrong... } string Util::formatMessage(const string& nick, const string& message, bool thirdPerson) { // let's *not* obey the spec here and add a space after the star. :P string tmp = (thirdPerson ? "* " + nick + ' ' : '<' + nick + "> ") + message; // Check all '<' and '[' after newlines as they're probably pasts... size_t i = 0; while( (i = tmp.find('\n', i)) != string::npos) { if(i + 1 < tmp.length()) { if(tmp[i+1] == '[' || tmp[i+1] == '<') { tmp.insert(i+1, "- "); i += 2; } } i++; } return Text::toDOS(tmp); } string Util::getTimeString() { char buf[64]; time_t _tt; time(&_tt); tm* _tm = localtime(&_tt); if(_tm == NULL) { strcpy(buf, "xx:xx:xx"); } else { strftime(buf, 64, "%X", _tm); } return buf; } string Util::toAdcFile(const string& file) { if(file == "files.xml.bz2" || file == "files.xml") return file; string ret; ret.reserve(file.length() + 1); ret += '/'; ret += file; for(string::size_type i = 0; i < ret.length(); ++i) { if(ret[i] == '\\') { ret[i] = '/'; } } return ret; } string Util::toNmdcFile(const string& file) { if(file.empty()) return Util::emptyString; string ret(file.substr(1)); for(string::size_type i = 0; i < ret.length(); ++i) { if(ret[i] == '/') { ret[i] = '\\'; } } return ret; } string Util::translateError(int aError) { #ifdef _WIN32 LPTSTR lpMsgBuf; DWORD chars = FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, aError, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language (LPTSTR) &lpMsgBuf, 0, NULL ); if(chars == 0) { return string(); } string tmp = Text::fromT(lpMsgBuf); // Free the buffer. LocalFree( lpMsgBuf ); string::size_type i = 0; while( (i = tmp.find_first_of("\r\n", i)) != string::npos) { tmp.erase(i, 1); } return tmp; #else // _WIN32 return Text::toUtf8(strerror(aError)); #endif // _WIN32 } } // namespace dcpp