#include "STDInclude.hpp" #define VA_BUFFER_COUNT 4 #define VA_BUFFER_SIZE 65536 namespace Utils { std::string GetMimeType(std::string url) { wchar_t* mimeType = nullptr; FindMimeFromData(NULL, std::wstring(url.begin(), url.end()).data(), NULL, 0, NULL, 0, &mimeType, 0); if (mimeType) { std::wstring wMimeType(mimeType); return std::string(wMimeType.begin(), wMimeType.end()); } return "application/octet-stream"; } const char *VA(const char *fmt, ...) { static char g_vaBuffer[VA_BUFFER_COUNT][VA_BUFFER_SIZE]; static int g_vaNextBufferIndex = 0; va_list ap; va_start(ap, fmt); char* dest = g_vaBuffer[g_vaNextBufferIndex]; vsnprintf(g_vaBuffer[g_vaNextBufferIndex], VA_BUFFER_SIZE, fmt, ap); g_vaNextBufferIndex = (g_vaNextBufferIndex + 1) % VA_BUFFER_COUNT; va_end(ap); return dest; } std::string StrToLower(std::string input) { std::transform(input.begin(), input.end(), input.begin(), ::tolower); return input; } std::string StrToUpper(std::string input) { std::transform(input.begin(), input.end(), input.begin(), ::toupper); return input; } bool EndsWith(std::string haystack, std::string needle) { return (strstr(haystack.data(), needle.data()) == (haystack.data() + haystack.size() - needle.size())); } std::string DumpHex(std::string data, std::string separator) { std::string result; for (unsigned int i = 0; i < data.size(); ++i) { if (i > 0) { result.append(separator); } result.append(Utils::VA("%02X", data[i] & 0xFF)); } return result; } std::string XORString(std::string str, char value) { for (unsigned int i = 0; i < str.size(); ++i) { str[i] ^= value; } return str; } // Complementary function for memset, which checks if a memory is set bool MemIsSet(void* mem, char chr, size_t length) { char* memArr = reinterpret_cast(mem); for (size_t i = 0; i < length; ++i) { if (memArr[i] != chr) { return false; } } return true; } std::vector Explode(const std::string& str, char delim) { std::vector result; std::istringstream iss(str); for (std::string token; std::getline(iss, token, delim);) { std::string _entry = std::move(token); // Remove trailing 0x0 bytes while (_entry.size() && !_entry[_entry.size() - 1]) { _entry = _entry.substr(0, _entry.size() - 1); } result.push_back(_entry); } return result; } void Replace(std::string &string, std::string find, std::string replace) { size_t nPos = 0; while ((nPos = string.find(find, nPos)) != std::string::npos) { string = string.replace(nPos, find.length(), replace); nPos += replace.length(); } } bool StartsWith(std::string haystack, std::string needle) { return (haystack.size() >= needle.size() && !strncmp(needle.data(), haystack.data(), needle.size())); } // trim from start std::string <rim(std::string &s) { s.erase(s.begin(), std::find_if(s.begin(), s.end(), std::not1(std::ptr_fun(std::isspace)))); return s; } // trim from end std::string &RTrim(std::string &s) { s.erase(std::find_if(s.rbegin(), s.rend(), std::not1(std::ptr_fun(std::isspace))).base(), s.end()); return s; } // trim from both ends std::string &Trim(std::string &s) { return LTrim(RTrim(s)); } std::string FormatTimeSpan(int milliseconds) { int secondsTotal = milliseconds / 1000; int seconds = secondsTotal % 60; int minutesTotal = secondsTotal / 60; int minutes = minutesTotal % 60; int hoursTotal = minutesTotal / 60; return Utils::VA("%02d:%02d:%02d", hoursTotal, minutes, seconds); } std::string ParseChallenge(std::string data) { auto pos = data.find_first_of("\n "); if (pos == std::string::npos) return data; return data.substr(0, pos).data(); } // TODO: Use modern file reading methods bool FileExists(std::string file) { FILE* fp; fopen_s(&fp, file.data(), "r"); if (fp) { fclose(fp); return true; } return false; } void WriteFile(std::string file, std::string data) { std::ofstream stream(file, std::ios::binary); if (stream.is_open()) { stream.write(data.data(), data.size()); stream.close(); } } std::string ReadFile(std::string file) { std::string buffer; if (FileExists(file)) { std::streamsize size = 0; std::ifstream stream(file, std::ios::binary); if (!stream.is_open()) return buffer; stream.seekg(0, std::ios::end); size = stream.tellg(); stream.seekg(0, std::ios::beg); if (size > -1) { buffer.clear(); buffer.resize((uint32_t)size); stream.read(const_cast(buffer.data()), size); } stream.close(); } return buffer; } // Infostring class void InfoString::Set(std::string key, std::string value) { this->KeyValuePairs[key] = value; } std::string InfoString::Get(std::string key) { if (this->KeyValuePairs.find(key) != this->KeyValuePairs.end()) { return this->KeyValuePairs[key]; } return ""; } std::string InfoString::Build() { std::string infoString; bool first = true; for (auto i = this->KeyValuePairs.begin(); i != this->KeyValuePairs.end(); ++i) { if (first) first = false; else infoString.append("\\"); infoString.append(i->first); // Key infoString.append("\\"); infoString.append(i->second); // Value } return infoString; } void InfoString::Dump() { for (auto i = this->KeyValuePairs.begin(); i != this->KeyValuePairs.end(); ++i) { OutputDebugStringA(Utils::VA("%s: %s", i->first.data(), i->second.data())); } } json11::Json InfoString::to_json() { return this->KeyValuePairs; } void InfoString::Parse(std::string buffer) { if (buffer[0] == '\\') { buffer = buffer.substr(1); } std::vector KeyValues = Utils::Explode(buffer, '\\'); for (unsigned int i = 0; i < (KeyValues.size() - 1); i+=2) { this->KeyValuePairs[KeyValues[i]] = KeyValues[i + 1]; } } }