2019-03-05 16:05:34 -05:00
|
|
|
// Copyright (c) 2014-2019, The Monero Project
|
2014-07-23 09:03:52 -04:00
|
|
|
//
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without modification, are
|
|
|
|
// permitted provided that the following conditions are met:
|
|
|
|
//
|
|
|
|
// 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
|
|
// conditions and the following disclaimer.
|
|
|
|
//
|
|
|
|
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
|
|
|
// of conditions and the following disclaimer in the documentation and/or other
|
|
|
|
// materials provided with the distribution.
|
|
|
|
//
|
|
|
|
// 3. Neither the name of the copyright holder nor the names of its contributors may be
|
|
|
|
// used to endorse or promote products derived from this software without specific
|
|
|
|
// prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
|
|
|
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
|
|
|
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
|
|
|
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
//
|
|
|
|
// Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
|
2014-03-03 17:07:58 -05:00
|
|
|
|
2018-10-16 14:08:36 -04:00
|
|
|
#include <unistd.h>
|
2014-03-03 17:07:58 -05:00
|
|
|
#include <cstdio>
|
2019-05-06 04:44:50 -04:00
|
|
|
#include <wchar.h>
|
2014-03-03 17:07:58 -05:00
|
|
|
|
2017-10-03 05:12:57 -04:00
|
|
|
#ifdef __GLIBC__
|
|
|
|
#include <gnu/libc-version.h>
|
|
|
|
#endif
|
|
|
|
|
2018-03-03 09:01:52 -05:00
|
|
|
#ifdef __GLIBC__
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
2018-07-15 08:20:36 -04:00
|
|
|
#include <sys/resource.h>
|
2018-03-03 09:01:52 -05:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <dirent.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <string>
|
|
|
|
#endif
|
|
|
|
|
2018-08-25 07:16:01 -04:00
|
|
|
//tools::is_hdd
|
|
|
|
#ifdef __GLIBC__
|
|
|
|
#include <sstream>
|
|
|
|
#include <sys/sysmacros.h>
|
|
|
|
#include <fstream>
|
|
|
|
#endif
|
|
|
|
|
2018-01-08 05:57:34 -05:00
|
|
|
#include "unbound.h"
|
|
|
|
|
2014-03-03 17:07:58 -05:00
|
|
|
#include "include_base_utils.h"
|
2017-02-20 15:46:50 -05:00
|
|
|
#include "file_io_utils.h"
|
2017-11-25 09:50:15 -05:00
|
|
|
#include "wipeable_string.h"
|
2018-10-28 09:46:58 -04:00
|
|
|
#include "misc_os_dependent.h"
|
2014-03-03 17:07:58 -05:00
|
|
|
using namespace epee;
|
|
|
|
|
2017-12-21 06:45:01 -05:00
|
|
|
#include "crypto/crypto.h"
|
2014-03-03 17:07:58 -05:00
|
|
|
#include "util.h"
|
2018-04-12 10:17:38 -04:00
|
|
|
#include "stack_trace.h"
|
2017-11-25 09:50:15 -05:00
|
|
|
#include "memwipe.h"
|
2014-03-03 17:07:58 -05:00
|
|
|
#include "cryptonote_config.h"
|
2017-01-28 09:48:45 -05:00
|
|
|
#include "net/http_client.h" // epee::net_utils::...
|
2019-05-06 04:44:50 -04:00
|
|
|
#include "readline_buffer.h"
|
2014-03-03 17:07:58 -05:00
|
|
|
|
|
|
|
#ifdef WIN32
|
2019-04-18 11:00:13 -04:00
|
|
|
#ifndef STRSAFE_NO_DEPRECATE
|
|
|
|
#define STRSAFE_NO_DEPRECATE
|
|
|
|
#endif
|
2017-08-30 00:05:36 -04:00
|
|
|
#include <windows.h>
|
|
|
|
#include <shlobj.h>
|
|
|
|
#include <strsafe.h>
|
2014-03-03 17:07:58 -05:00
|
|
|
#else
|
2017-08-30 00:05:36 -04:00
|
|
|
#include <sys/file.h>
|
|
|
|
#include <sys/utsname.h>
|
|
|
|
#include <sys/stat.h>
|
2014-03-03 17:07:58 -05:00
|
|
|
#endif
|
2015-11-21 05:13:10 -05:00
|
|
|
#include <boost/filesystem.hpp>
|
2017-11-25 17:25:05 -05:00
|
|
|
#include <boost/algorithm/string.hpp>
|
2017-01-28 09:48:45 -05:00
|
|
|
#include <boost/asio.hpp>
|
2019-03-21 06:03:24 -04:00
|
|
|
#include <boost/format.hpp>
|
2017-02-20 15:46:50 -05:00
|
|
|
#include <openssl/sha.h>
|
2014-03-03 17:07:58 -05:00
|
|
|
|
2018-12-05 09:07:45 -05:00
|
|
|
#undef MONERO_DEFAULT_LOG_CATEGORY
|
|
|
|
#define MONERO_DEFAULT_LOG_CATEGORY "util"
|
|
|
|
|
2018-12-05 09:52:43 -05:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
|
|
|
static int flock_exnb(int fd)
|
|
|
|
{
|
|
|
|
struct flock fl;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
memset(&fl, 0, sizeof(fl));
|
|
|
|
fl.l_type = F_WRLCK;
|
|
|
|
fl.l_whence = SEEK_SET;
|
|
|
|
fl.l_start = 0;
|
|
|
|
fl.l_len = 0;
|
|
|
|
ret = fcntl(fd, F_SETLK, &fl);
|
|
|
|
if (ret < 0)
|
|
|
|
MERROR("Error locking fd " << fd << ": " << errno << " (" << strerror(errno) << ")");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-03-03 17:07:58 -05:00
|
|
|
namespace tools
|
|
|
|
{
|
2015-11-29 08:02:01 -05:00
|
|
|
std::function<void(int)> signal_handler::m_handler;
|
2014-03-03 17:07:58 -05:00
|
|
|
|
2017-08-30 00:05:36 -04:00
|
|
|
private_file::private_file() noexcept : m_handle(), m_filename() {}
|
|
|
|
|
|
|
|
private_file::private_file(std::FILE* handle, std::string&& filename) noexcept
|
|
|
|
: m_handle(handle), m_filename(std::move(filename)) {}
|
|
|
|
|
|
|
|
private_file private_file::create(std::string name)
|
2016-12-16 18:08:24 -05:00
|
|
|
{
|
|
|
|
#ifdef WIN32
|
|
|
|
struct close_handle
|
|
|
|
{
|
|
|
|
void operator()(HANDLE handle) const noexcept
|
|
|
|
{
|
|
|
|
CloseHandle(handle);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::unique_ptr<void, close_handle> process = nullptr;
|
|
|
|
{
|
|
|
|
HANDLE temp{};
|
|
|
|
const bool fail = OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, std::addressof(temp)) == 0;
|
|
|
|
process.reset(temp);
|
|
|
|
if (fail)
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
2016-12-16 18:08:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
DWORD sid_size = 0;
|
|
|
|
GetTokenInformation(process.get(), TokenOwner, nullptr, 0, std::addressof(sid_size));
|
|
|
|
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
2016-12-16 18:08:24 -05:00
|
|
|
|
|
|
|
std::unique_ptr<char[]> sid{new char[sid_size]};
|
|
|
|
if (!GetTokenInformation(process.get(), TokenOwner, sid.get(), sid_size, std::addressof(sid_size)))
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
2016-12-16 18:08:24 -05:00
|
|
|
|
|
|
|
const PSID psid = reinterpret_cast<const PTOKEN_OWNER>(sid.get())->Owner;
|
|
|
|
const DWORD daclSize =
|
|
|
|
sizeof(ACL) + sizeof(ACCESS_ALLOWED_ACE) + GetLengthSid(psid) - sizeof(DWORD);
|
|
|
|
|
|
|
|
const std::unique_ptr<char[]> dacl{new char[daclSize]};
|
|
|
|
if (!InitializeAcl(reinterpret_cast<PACL>(dacl.get()), daclSize, ACL_REVISION))
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
2016-12-16 18:08:24 -05:00
|
|
|
|
|
|
|
if (!AddAccessAllowedAce(reinterpret_cast<PACL>(dacl.get()), ACL_REVISION, (READ_CONTROL | FILE_GENERIC_READ | DELETE), psid))
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
2016-12-16 18:08:24 -05:00
|
|
|
|
|
|
|
SECURITY_DESCRIPTOR descriptor{};
|
|
|
|
if (!InitializeSecurityDescriptor(std::addressof(descriptor), SECURITY_DESCRIPTOR_REVISION))
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
2016-12-16 18:08:24 -05:00
|
|
|
|
|
|
|
if (!SetSecurityDescriptorDacl(std::addressof(descriptor), true, reinterpret_cast<PACL>(dacl.get()), false))
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
2016-12-16 18:08:24 -05:00
|
|
|
|
|
|
|
SECURITY_ATTRIBUTES attributes{sizeof(SECURITY_ATTRIBUTES), std::addressof(descriptor), false};
|
|
|
|
std::unique_ptr<void, close_handle> file{
|
|
|
|
CreateFile(
|
|
|
|
name.c_str(),
|
|
|
|
GENERIC_WRITE, FILE_SHARE_READ,
|
|
|
|
std::addressof(attributes),
|
2017-08-30 00:05:36 -04:00
|
|
|
CREATE_NEW, (FILE_ATTRIBUTE_TEMPORARY | FILE_FLAG_DELETE_ON_CLOSE),
|
2016-12-16 18:08:24 -05:00
|
|
|
nullptr
|
|
|
|
)
|
|
|
|
};
|
|
|
|
if (file)
|
|
|
|
{
|
|
|
|
const int fd = _open_osfhandle(reinterpret_cast<intptr_t>(file.get()), 0);
|
|
|
|
if (0 <= fd)
|
|
|
|
{
|
|
|
|
file.release();
|
|
|
|
std::FILE* real_file = _fdopen(fd, "w");
|
|
|
|
if (!real_file)
|
|
|
|
{
|
|
|
|
_close(fd);
|
|
|
|
}
|
2017-08-30 00:05:36 -04:00
|
|
|
return {real_file, std::move(name)};
|
2016-12-16 18:08:24 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2017-08-30 00:05:36 -04:00
|
|
|
const int fdr = open(name.c_str(), (O_RDONLY | O_CREAT), S_IRUSR);
|
|
|
|
if (0 <= fdr)
|
2016-12-16 18:08:24 -05:00
|
|
|
{
|
2017-08-30 00:05:36 -04:00
|
|
|
struct stat rstats = {};
|
|
|
|
if (fstat(fdr, std::addressof(rstats)) != 0)
|
|
|
|
{
|
|
|
|
close(fdr);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
fchmod(fdr, (S_IRUSR | S_IWUSR));
|
|
|
|
const int fdw = open(name.c_str(), O_RDWR);
|
|
|
|
fchmod(fdr, rstats.st_mode);
|
|
|
|
close(fdr);
|
|
|
|
|
|
|
|
if (0 <= fdw)
|
2016-12-16 18:08:24 -05:00
|
|
|
{
|
2017-08-30 00:05:36 -04:00
|
|
|
struct stat wstats = {};
|
|
|
|
if (fstat(fdw, std::addressof(wstats)) == 0 &&
|
|
|
|
rstats.st_dev == wstats.st_dev && rstats.st_ino == wstats.st_ino &&
|
2018-12-05 09:52:43 -05:00
|
|
|
flock_exnb(fdw) == 0 && ftruncate(fdw, 0) == 0)
|
2017-08-30 00:05:36 -04:00
|
|
|
{
|
|
|
|
std::FILE* file = fdopen(fdw, "w");
|
|
|
|
if (file) return {file, std::move(name)};
|
|
|
|
}
|
|
|
|
close(fdw);
|
2016-12-16 18:08:24 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2017-08-30 00:05:36 -04:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
private_file::~private_file() noexcept
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
boost::system::error_code ec{};
|
|
|
|
boost::filesystem::remove(filename(), ec);
|
|
|
|
}
|
|
|
|
catch (...) {}
|
2016-12-16 18:08:24 -05:00
|
|
|
}
|
|
|
|
|
2018-06-21 04:19:16 -04:00
|
|
|
file_locker::file_locker(const std::string &filename)
|
|
|
|
{
|
|
|
|
#ifdef WIN32
|
|
|
|
m_fd = INVALID_HANDLE_VALUE;
|
|
|
|
std::wstring filename_wide;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
filename_wide = string_tools::utf8_to_utf16(filename);
|
|
|
|
}
|
|
|
|
catch (const std::exception &e)
|
|
|
|
{
|
|
|
|
MERROR("Failed to convert path \"" << filename << "\" to UTF-16: " << e.what());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_fd = CreateFileW(filename_wide.c_str(), GENERIC_READ, 0, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
|
|
if (m_fd != INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
OVERLAPPED ov;
|
|
|
|
memset(&ov, 0, sizeof(ov));
|
|
|
|
if (!LockFileEx(m_fd, LOCKFILE_FAIL_IMMEDIATELY | LOCKFILE_EXCLUSIVE_LOCK, 0, 1, 0, &ov))
|
|
|
|
{
|
|
|
|
MERROR("Failed to lock " << filename << ": " << std::error_code(GetLastError(), std::system_category()));
|
|
|
|
CloseHandle(m_fd);
|
|
|
|
m_fd = INVALID_HANDLE_VALUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MERROR("Failed to open " << filename << ": " << std::error_code(GetLastError(), std::system_category()));
|
|
|
|
}
|
|
|
|
#else
|
2018-12-05 09:52:43 -05:00
|
|
|
m_fd = open(filename.c_str(), O_RDWR | O_CREAT | O_CLOEXEC, 0666);
|
2018-06-21 04:19:16 -04:00
|
|
|
if (m_fd != -1)
|
|
|
|
{
|
2018-12-05 09:52:43 -05:00
|
|
|
if (flock_exnb(m_fd) == -1)
|
2018-06-21 04:19:16 -04:00
|
|
|
{
|
2018-06-27 21:39:04 -04:00
|
|
|
MERROR("Failed to lock " << filename << ": " << std::strerror(errno));
|
2018-06-21 04:19:16 -04:00
|
|
|
close(m_fd);
|
|
|
|
m_fd = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-06-27 21:39:04 -04:00
|
|
|
MERROR("Failed to open " << filename << ": " << std::strerror(errno));
|
2018-06-21 04:19:16 -04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
file_locker::~file_locker()
|
|
|
|
{
|
|
|
|
if (locked())
|
|
|
|
{
|
|
|
|
#ifdef WIN32
|
|
|
|
CloseHandle(m_fd);
|
|
|
|
#else
|
|
|
|
close(m_fd);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bool file_locker::locked() const
|
|
|
|
{
|
|
|
|
#ifdef WIN32
|
|
|
|
return m_fd != INVALID_HANDLE_VALUE;
|
|
|
|
#else
|
|
|
|
return m_fd != -1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-03-03 17:07:58 -05:00
|
|
|
#ifdef WIN32
|
|
|
|
std::string get_windows_version_display_string()
|
|
|
|
{
|
|
|
|
typedef void (WINAPI *PGNSI)(LPSYSTEM_INFO);
|
|
|
|
typedef BOOL (WINAPI *PGPI)(DWORD, DWORD, DWORD, DWORD, PDWORD);
|
|
|
|
#define BUFSIZE 10000
|
|
|
|
|
|
|
|
char pszOS[BUFSIZE] = {0};
|
|
|
|
OSVERSIONINFOEX osvi;
|
|
|
|
SYSTEM_INFO si;
|
|
|
|
PGNSI pGNSI;
|
|
|
|
PGPI pGPI;
|
|
|
|
BOOL bOsVersionInfoEx;
|
|
|
|
DWORD dwType;
|
|
|
|
|
|
|
|
ZeroMemory(&si, sizeof(SYSTEM_INFO));
|
|
|
|
ZeroMemory(&osvi, sizeof(OSVERSIONINFOEX));
|
|
|
|
|
|
|
|
osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
|
|
|
|
bOsVersionInfoEx = GetVersionEx((OSVERSIONINFO*) &osvi);
|
|
|
|
|
|
|
|
if(!bOsVersionInfoEx) return pszOS;
|
|
|
|
|
|
|
|
// Call GetNativeSystemInfo if supported or GetSystemInfo otherwise.
|
|
|
|
|
|
|
|
pGNSI = (PGNSI) GetProcAddress(
|
|
|
|
GetModuleHandle(TEXT("kernel32.dll")),
|
|
|
|
"GetNativeSystemInfo");
|
|
|
|
if(NULL != pGNSI)
|
|
|
|
pGNSI(&si);
|
|
|
|
else GetSystemInfo(&si);
|
|
|
|
|
|
|
|
if ( VER_PLATFORM_WIN32_NT==osvi.dwPlatformId &&
|
|
|
|
osvi.dwMajorVersion > 4 )
|
|
|
|
{
|
|
|
|
StringCchCopy(pszOS, BUFSIZE, TEXT("Microsoft "));
|
|
|
|
|
|
|
|
// Test for the specific product.
|
2018-10-09 13:17:56 -04:00
|
|
|
if ( osvi.dwMajorVersion == 10 )
|
|
|
|
{
|
|
|
|
if ( osvi.dwMinorVersion == 0 )
|
|
|
|
{
|
|
|
|
if( osvi.wProductType == VER_NT_WORKSTATION )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Windows 10 "));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT("Windows Server 2016 " ));
|
|
|
|
}
|
|
|
|
}
|
2014-03-03 17:07:58 -05:00
|
|
|
|
|
|
|
if ( osvi.dwMajorVersion == 6 )
|
|
|
|
{
|
2018-10-09 13:17:56 -04:00
|
|
|
if ( osvi.dwMinorVersion == 0 )
|
2014-03-03 17:07:58 -05:00
|
|
|
{
|
|
|
|
if( osvi.wProductType == VER_NT_WORKSTATION )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Windows Vista "));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT("Windows Server 2008 " ));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( osvi.dwMinorVersion == 1 )
|
|
|
|
{
|
|
|
|
if( osvi.wProductType == VER_NT_WORKSTATION )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Windows 7 "));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT("Windows Server 2008 R2 " ));
|
|
|
|
}
|
|
|
|
|
2018-10-09 13:17:56 -04:00
|
|
|
if ( osvi.dwMinorVersion == 2 )
|
|
|
|
{
|
|
|
|
if( osvi.wProductType == VER_NT_WORKSTATION )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Windows 8 "));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT("Windows Server 2012 " ));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( osvi.dwMinorVersion == 3 )
|
|
|
|
{
|
|
|
|
if( osvi.wProductType == VER_NT_WORKSTATION )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Windows 8.1 "));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT("Windows Server 2012 R2 " ));
|
|
|
|
}
|
|
|
|
|
2014-03-03 17:07:58 -05:00
|
|
|
pGPI = (PGPI) GetProcAddress(
|
|
|
|
GetModuleHandle(TEXT("kernel32.dll")),
|
|
|
|
"GetProductInfo");
|
|
|
|
|
|
|
|
pGPI( osvi.dwMajorVersion, osvi.dwMinorVersion, 0, 0, &dwType);
|
|
|
|
|
|
|
|
switch( dwType )
|
|
|
|
{
|
|
|
|
case PRODUCT_ULTIMATE:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Ultimate Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_PROFESSIONAL:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Professional" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_HOME_PREMIUM:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Home Premium Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_HOME_BASIC:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Home Basic Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_ENTERPRISE:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Enterprise Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_BUSINESS:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Business Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_STARTER:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Starter Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_CLUSTER_SERVER:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Cluster Server Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_DATACENTER_SERVER:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Datacenter Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_DATACENTER_SERVER_CORE:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Datacenter Edition (core installation)" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_ENTERPRISE_SERVER:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Enterprise Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_ENTERPRISE_SERVER_CORE:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Enterprise Edition (core installation)" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_ENTERPRISE_SERVER_IA64:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Enterprise Edition for Itanium-based Systems" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_SMALLBUSINESS_SERVER:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Small Business Server" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_SMALLBUSINESS_SERVER_PREMIUM:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Small Business Server Premium Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_STANDARD_SERVER:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Standard Edition" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_STANDARD_SERVER_CORE:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Standard Edition (core installation)" ));
|
|
|
|
break;
|
|
|
|
case PRODUCT_WEB_SERVER:
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Web Server Edition" ));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( osvi.dwMajorVersion == 5 && osvi.dwMinorVersion == 2 )
|
|
|
|
{
|
|
|
|
if( GetSystemMetrics(SM_SERVERR2) )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Windows Server 2003 R2, "));
|
|
|
|
else if ( osvi.wSuiteMask & VER_SUITE_STORAGE_SERVER )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Windows Storage Server 2003"));
|
|
|
|
else if ( osvi.wSuiteMask & VER_SUITE_WH_SERVER )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Windows Home Server"));
|
|
|
|
else if( osvi.wProductType == VER_NT_WORKSTATION &&
|
|
|
|
si.wProcessorArchitecture==PROCESSOR_ARCHITECTURE_AMD64)
|
|
|
|
{
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Windows XP Professional x64 Edition"));
|
|
|
|
}
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT("Windows Server 2003, "));
|
|
|
|
|
|
|
|
// Test for the server type.
|
|
|
|
if ( osvi.wProductType != VER_NT_WORKSTATION )
|
|
|
|
{
|
|
|
|
if ( si.wProcessorArchitecture==PROCESSOR_ARCHITECTURE_IA64 )
|
|
|
|
{
|
|
|
|
if( osvi.wSuiteMask & VER_SUITE_DATACENTER )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Datacenter Edition for Itanium-based Systems" ));
|
|
|
|
else if( osvi.wSuiteMask & VER_SUITE_ENTERPRISE )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Enterprise Edition for Itanium-based Systems" ));
|
|
|
|
}
|
|
|
|
|
|
|
|
else if ( si.wProcessorArchitecture==PROCESSOR_ARCHITECTURE_AMD64 )
|
|
|
|
{
|
|
|
|
if( osvi.wSuiteMask & VER_SUITE_DATACENTER )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Datacenter x64 Edition" ));
|
|
|
|
else if( osvi.wSuiteMask & VER_SUITE_ENTERPRISE )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Enterprise x64 Edition" ));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT( "Standard x64 Edition" ));
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ( osvi.wSuiteMask & VER_SUITE_COMPUTE_SERVER )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Compute Cluster Edition" ));
|
|
|
|
else if( osvi.wSuiteMask & VER_SUITE_DATACENTER )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Datacenter Edition" ));
|
|
|
|
else if( osvi.wSuiteMask & VER_SUITE_ENTERPRISE )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Enterprise Edition" ));
|
|
|
|
else if ( osvi.wSuiteMask & VER_SUITE_BLADE )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Web Edition" ));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT( "Standard Edition" ));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( osvi.dwMajorVersion == 5 && osvi.dwMinorVersion == 1 )
|
|
|
|
{
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Windows XP "));
|
|
|
|
if( osvi.wSuiteMask & VER_SUITE_PERSONAL )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Home Edition" ));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT( "Professional" ));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( osvi.dwMajorVersion == 5 && osvi.dwMinorVersion == 0 )
|
|
|
|
{
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT("Windows 2000 "));
|
|
|
|
|
|
|
|
if ( osvi.wProductType == VER_NT_WORKSTATION )
|
|
|
|
{
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Professional" ));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( osvi.wSuiteMask & VER_SUITE_DATACENTER )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Datacenter Server" ));
|
|
|
|
else if( osvi.wSuiteMask & VER_SUITE_ENTERPRISE )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( "Advanced Server" ));
|
|
|
|
else StringCchCat(pszOS, BUFSIZE, TEXT( "Server" ));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Include service pack (if any) and build number.
|
|
|
|
|
|
|
|
if( strlen(osvi.szCSDVersion) > 0 )
|
|
|
|
{
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT(" ") );
|
|
|
|
StringCchCat(pszOS, BUFSIZE, osvi.szCSDVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
TCHAR buf[80];
|
|
|
|
|
|
|
|
StringCchPrintf( buf, 80, TEXT(" (build %d)"), osvi.dwBuildNumber);
|
|
|
|
StringCchCat(pszOS, BUFSIZE, buf);
|
|
|
|
|
|
|
|
if ( osvi.dwMajorVersion >= 6 )
|
|
|
|
{
|
|
|
|
if ( si.wProcessorArchitecture==PROCESSOR_ARCHITECTURE_AMD64 )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT( ", 64-bit" ));
|
|
|
|
else if (si.wProcessorArchitecture==PROCESSOR_ARCHITECTURE_INTEL )
|
|
|
|
StringCchCat(pszOS, BUFSIZE, TEXT(", 32-bit"));
|
|
|
|
}
|
|
|
|
|
|
|
|
return pszOS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf( "This sample does not support this version of Windows.\n");
|
|
|
|
return pszOS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
std::string get_nix_version_display_string()
|
|
|
|
{
|
2017-10-10 09:05:28 -04:00
|
|
|
struct utsname un;
|
2014-03-03 17:07:58 -05:00
|
|
|
|
|
|
|
if(uname(&un) < 0)
|
|
|
|
return std::string("*nix: failed to get os version");
|
|
|
|
return std::string() + un.sysname + " " + un.version + " " + un.release;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::string get_os_version_string()
|
|
|
|
{
|
|
|
|
#ifdef WIN32
|
|
|
|
return get_windows_version_display_string();
|
|
|
|
#else
|
|
|
|
return get_nix_version_display_string();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
std::string get_special_folder_path(int nfolder, bool iscreate)
|
|
|
|
{
|
2018-02-22 13:52:55 -05:00
|
|
|
WCHAR psz_path[MAX_PATH] = L"";
|
2014-03-03 17:07:58 -05:00
|
|
|
|
2018-02-22 13:52:55 -05:00
|
|
|
if (SHGetSpecialFolderPathW(NULL, psz_path, nfolder, iscreate))
|
2014-03-03 17:07:58 -05:00
|
|
|
{
|
2018-06-24 20:44:07 -04:00
|
|
|
try
|
|
|
|
{
|
|
|
|
return string_tools::utf16_to_utf8(psz_path);
|
|
|
|
}
|
|
|
|
catch (const std::exception &e)
|
|
|
|
{
|
|
|
|
MERROR("utf16_to_utf8 failed: " << e.what());
|
|
|
|
return "";
|
|
|
|
}
|
2014-03-03 17:07:58 -05:00
|
|
|
}
|
|
|
|
|
2018-02-22 13:52:55 -05:00
|
|
|
LOG_ERROR("SHGetSpecialFolderPathW() failed, could not obtain requested path.");
|
2014-03-03 17:07:58 -05:00
|
|
|
return "";
|
|
|
|
}
|
|
|
|
#endif
|
2014-09-09 20:18:23 -04:00
|
|
|
|
2014-03-03 17:07:58 -05:00
|
|
|
std::string get_default_data_dir()
|
|
|
|
{
|
2014-09-09 20:18:23 -04:00
|
|
|
/* Please for the love of god refactor the ifdefs out of this */
|
|
|
|
|
|
|
|
// namespace fs = boost::filesystem;
|
2014-03-03 17:07:58 -05:00
|
|
|
// Windows < Vista: C:\Documents and Settings\Username\Application Data\CRYPTONOTE_NAME
|
|
|
|
// Windows >= Vista: C:\Users\Username\AppData\Roaming\CRYPTONOTE_NAME
|
2018-01-04 04:05:14 -05:00
|
|
|
// Unix & Mac: ~/.CRYPTONOTE_NAME
|
2014-03-03 17:07:58 -05:00
|
|
|
std::string config_folder;
|
2014-09-09 20:18:23 -04:00
|
|
|
|
2014-03-03 17:07:58 -05:00
|
|
|
#ifdef WIN32
|
2015-03-20 16:40:54 -04:00
|
|
|
config_folder = get_special_folder_path(CSIDL_COMMON_APPDATA, true) + "\\" + CRYPTONOTE_NAME;
|
2014-03-03 17:07:58 -05:00
|
|
|
#else
|
|
|
|
std::string pathRet;
|
|
|
|
char* pszHome = getenv("HOME");
|
|
|
|
if (pszHome == NULL || strlen(pszHome) == 0)
|
|
|
|
pathRet = "/";
|
|
|
|
else
|
|
|
|
pathRet = pszHome;
|
|
|
|
config_folder = (pathRet + "/." + CRYPTONOTE_NAME);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return config_folder;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool create_directories_if_necessary(const std::string& path)
|
|
|
|
{
|
|
|
|
namespace fs = boost::filesystem;
|
|
|
|
boost::system::error_code ec;
|
|
|
|
fs::path fs_path(path);
|
|
|
|
if (fs::is_directory(fs_path, ec))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool res = fs::create_directories(fs_path, ec);
|
|
|
|
if (res)
|
|
|
|
{
|
|
|
|
LOG_PRINT_L2("Created directory: " << path);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LOG_PRINT_L2("Can't create directory: " << path << ", err: "<< ec.message());
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2019-03-29 12:30:24 -04:00
|
|
|
std::error_code replace_file(const std::string& old_name, const std::string& new_name)
|
2014-03-03 17:07:58 -05:00
|
|
|
{
|
|
|
|
int code;
|
|
|
|
#if defined(WIN32)
|
|
|
|
// Maximizing chances for success
|
2018-06-24 20:44:07 -04:00
|
|
|
std::wstring wide_replacement_name;
|
2019-03-29 12:30:24 -04:00
|
|
|
try { wide_replacement_name = string_tools::utf8_to_utf16(old_name); }
|
2018-06-24 20:44:07 -04:00
|
|
|
catch (...) { return std::error_code(GetLastError(), std::system_category()); }
|
|
|
|
std::wstring wide_replaced_name;
|
2019-03-29 12:30:24 -04:00
|
|
|
try { wide_replaced_name = string_tools::utf8_to_utf16(new_name); }
|
2018-06-24 20:44:07 -04:00
|
|
|
catch (...) { return std::error_code(GetLastError(), std::system_category()); }
|
|
|
|
|
|
|
|
DWORD attributes = ::GetFileAttributesW(wide_replaced_name.c_str());
|
2014-03-03 17:07:58 -05:00
|
|
|
if (INVALID_FILE_ATTRIBUTES != attributes)
|
|
|
|
{
|
2018-06-24 20:44:07 -04:00
|
|
|
::SetFileAttributesW(wide_replaced_name.c_str(), attributes & (~FILE_ATTRIBUTE_READONLY));
|
2014-03-03 17:07:58 -05:00
|
|
|
}
|
|
|
|
|
2018-06-24 20:44:07 -04:00
|
|
|
bool ok = 0 != ::MoveFileExW(wide_replacement_name.c_str(), wide_replaced_name.c_str(), MOVEFILE_REPLACE_EXISTING);
|
2014-03-03 17:07:58 -05:00
|
|
|
code = ok ? 0 : static_cast<int>(::GetLastError());
|
|
|
|
#else
|
2019-03-29 12:30:24 -04:00
|
|
|
bool ok = 0 == std::rename(old_name.c_str(), new_name.c_str());
|
2014-03-03 17:07:58 -05:00
|
|
|
code = ok ? 0 : errno;
|
|
|
|
#endif
|
|
|
|
return std::error_code(code, std::system_category());
|
|
|
|
}
|
2015-11-21 05:13:10 -05:00
|
|
|
|
2018-01-08 05:57:34 -05:00
|
|
|
static bool unbound_built_with_threads()
|
|
|
|
{
|
|
|
|
ub_ctx *ctx = ub_ctx_create();
|
|
|
|
if (!ctx) return false; // cheat a bit, should not happen unless OOM
|
2018-04-10 18:37:33 -04:00
|
|
|
char *monero = strdup("monero"), *unbound = strdup("unbound");
|
|
|
|
ub_ctx_zone_add(ctx, monero, unbound); // this calls ub_ctx_finalize first, then errors out with UB_SYNTAX
|
|
|
|
free(unbound);
|
|
|
|
free(monero);
|
2018-01-08 05:57:34 -05:00
|
|
|
// if no threads, bails out early with UB_NOERROR, otherwise fails with UB_AFTERFINAL id already finalized
|
|
|
|
bool with_threads = ub_ctx_async(ctx, 1) != 0; // UB_AFTERFINAL is not defined in public headers, check any error
|
|
|
|
ub_ctx_delete(ctx);
|
|
|
|
MINFO("libunbound was built " << (with_threads ? "with" : "without") << " threads");
|
|
|
|
return with_threads;
|
|
|
|
}
|
|
|
|
|
2015-11-21 05:13:10 -05:00
|
|
|
bool sanitize_locale()
|
|
|
|
{
|
|
|
|
// boost::filesystem throws for "invalid" locales, such as en_US.UTF-8, or kjsdkfs,
|
|
|
|
// so reset it here before any calls to it
|
|
|
|
try
|
|
|
|
{
|
|
|
|
boost::filesystem::path p {std::string("test")};
|
|
|
|
p /= std::string("test");
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
2015-11-25 16:37:05 -05:00
|
|
|
#if defined(__MINGW32__) || defined(__MINGW__)
|
|
|
|
putenv("LC_ALL=C");
|
|
|
|
putenv("LANG=C");
|
|
|
|
#else
|
2015-11-21 05:13:10 -05:00
|
|
|
setenv("LC_ALL", "C", 1);
|
|
|
|
setenv("LANG", "C", 1);
|
2015-11-25 16:37:05 -05:00
|
|
|
#endif
|
2015-11-21 05:13:10 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2018-04-12 10:17:38 -04:00
|
|
|
|
|
|
|
#ifdef STACK_TRACE
|
|
|
|
#ifdef _WIN32
|
|
|
|
// https://stackoverflow.com/questions/1992816/how-to-handle-seg-faults-under-windows
|
|
|
|
static LONG WINAPI windows_crash_handler(PEXCEPTION_POINTERS pExceptionInfo)
|
|
|
|
{
|
|
|
|
tools::log_stack_trace("crashing");
|
|
|
|
exit(1);
|
|
|
|
return EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
}
|
|
|
|
static void setup_crash_dump()
|
|
|
|
{
|
|
|
|
SetUnhandledExceptionFilter(windows_crash_handler);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static void posix_crash_handler(int signal)
|
|
|
|
{
|
|
|
|
tools::log_stack_trace(("crashing with fatal signal " + std::to_string(signal)).c_str());
|
|
|
|
#ifdef NDEBUG
|
|
|
|
_exit(1);
|
|
|
|
#else
|
|
|
|
abort();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
static void setup_crash_dump()
|
|
|
|
{
|
|
|
|
signal(SIGSEGV, posix_crash_handler);
|
|
|
|
signal(SIGBUS, posix_crash_handler);
|
|
|
|
signal(SIGILL, posix_crash_handler);
|
|
|
|
signal(SIGFPE, posix_crash_handler);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
static void setup_crash_dump() {}
|
|
|
|
#endif
|
|
|
|
|
2018-07-15 08:20:36 -04:00
|
|
|
bool disable_core_dumps()
|
|
|
|
{
|
|
|
|
#ifdef __GLIBC__
|
|
|
|
// disable core dumps in release mode
|
|
|
|
struct rlimit rlimit;
|
|
|
|
rlimit.rlim_cur = rlimit.rlim_max = 0;
|
|
|
|
if (setrlimit(RLIMIT_CORE, &rlimit))
|
|
|
|
{
|
|
|
|
MWARNING("Failed to disable core dumps");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-11-03 14:32:05 -04:00
|
|
|
ssize_t get_lockable_memory()
|
|
|
|
{
|
|
|
|
#ifdef __GLIBC__
|
|
|
|
struct rlimit rlim;
|
|
|
|
if (getrlimit(RLIMIT_MEMLOCK, &rlim) < 0)
|
|
|
|
{
|
|
|
|
MERROR("Failed to determine the lockable memory limit");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return rlim.rlim_cur;
|
|
|
|
#else
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-10-03 05:12:57 -04:00
|
|
|
bool on_startup()
|
|
|
|
{
|
2017-10-27 06:39:27 -04:00
|
|
|
mlog_configure("", true);
|
|
|
|
|
2018-04-12 10:17:38 -04:00
|
|
|
setup_crash_dump();
|
|
|
|
|
2017-10-03 05:12:57 -04:00
|
|
|
sanitize_locale();
|
|
|
|
|
|
|
|
#ifdef __GLIBC__
|
|
|
|
const char *ver = gnu_get_libc_version();
|
|
|
|
if (!strcmp(ver, "2.25"))
|
|
|
|
MCLOG_RED(el::Level::Warning, "global", "Running with glibc " << ver << ", hangs may occur - change glibc version if possible");
|
|
|
|
#endif
|
2017-10-21 05:09:51 -04:00
|
|
|
|
2017-11-18 13:02:09 -05:00
|
|
|
#if OPENSSL_VERSION_NUMBER < 0x10100000 || defined(LIBRESSL_VERSION_TEXT)
|
2017-10-21 05:09:51 -04:00
|
|
|
SSL_library_init();
|
|
|
|
#else
|
|
|
|
OPENSSL_init_ssl(0, NULL);
|
|
|
|
#endif
|
|
|
|
|
2018-01-08 05:57:34 -05:00
|
|
|
if (!unbound_built_with_threads())
|
|
|
|
MCLOG_RED(el::Level::Warning, "global", "libunbound was not built with threads enabled - crashes may occur");
|
|
|
|
|
2017-10-03 05:12:57 -04:00
|
|
|
return true;
|
|
|
|
}
|
2016-04-06 14:59:34 -04:00
|
|
|
void set_strict_default_file_permissions(bool strict)
|
|
|
|
{
|
|
|
|
#if defined(__MINGW32__) || defined(__MINGW__)
|
|
|
|
// no clue about the odd one out
|
|
|
|
#else
|
|
|
|
mode_t mode = strict ? 077 : 0;
|
|
|
|
umask(mode);
|
|
|
|
#endif
|
|
|
|
}
|
2016-04-28 15:25:33 -04:00
|
|
|
|
2018-08-25 07:16:01 -04:00
|
|
|
boost::optional<bool> is_hdd(const char *file_path)
|
2018-03-03 09:01:52 -05:00
|
|
|
{
|
|
|
|
#ifdef __GLIBC__
|
2018-08-25 07:16:01 -04:00
|
|
|
struct stat st;
|
|
|
|
std::string prefix;
|
|
|
|
if(stat(file_path, &st) == 0)
|
2018-03-03 09:01:52 -05:00
|
|
|
{
|
2018-08-25 07:16:01 -04:00
|
|
|
std::ostringstream s;
|
|
|
|
s << "/sys/dev/block/" << major(st.st_dev) << ":" << minor(st.st_dev);
|
|
|
|
prefix = s.str();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return boost::none;
|
|
|
|
}
|
|
|
|
std::string attr_path = prefix + "/queue/rotational";
|
|
|
|
std::ifstream f(attr_path, std::ios_base::in);
|
|
|
|
if(not f.is_open())
|
|
|
|
{
|
|
|
|
attr_path = prefix + "/../queue/rotational";
|
|
|
|
f.open(attr_path, std::ios_base::in);
|
|
|
|
if(not f.is_open())
|
2018-03-03 09:01:52 -05:00
|
|
|
{
|
2018-08-25 07:16:01 -04:00
|
|
|
return boost::none;
|
2018-03-03 09:01:52 -05:00
|
|
|
}
|
|
|
|
}
|
2018-08-25 07:16:01 -04:00
|
|
|
unsigned short val = 0xdead;
|
|
|
|
f >> val;
|
|
|
|
if(not f.fail())
|
|
|
|
{
|
|
|
|
return (val == 1);
|
|
|
|
}
|
|
|
|
return boost::none;
|
2018-03-03 09:01:52 -05:00
|
|
|
#else
|
2018-08-25 07:16:01 -04:00
|
|
|
return boost::none;
|
2018-03-03 09:01:52 -05:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-04-28 15:25:33 -04:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
boost::mutex max_concurrency_lock;
|
|
|
|
unsigned max_concurrency = boost::thread::hardware_concurrency();
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_max_concurrency(unsigned n)
|
|
|
|
{
|
|
|
|
if (n < 1)
|
|
|
|
n = boost::thread::hardware_concurrency();
|
|
|
|
unsigned hwc = boost::thread::hardware_concurrency();
|
|
|
|
if (n > hwc)
|
|
|
|
n = hwc;
|
|
|
|
boost::lock_guard<boost::mutex> lock(max_concurrency_lock);
|
|
|
|
max_concurrency = n;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned get_max_concurrency()
|
|
|
|
{
|
|
|
|
boost::lock_guard<boost::mutex> lock(max_concurrency_lock);
|
|
|
|
return max_concurrency;
|
|
|
|
}
|
2017-01-28 09:48:45 -05:00
|
|
|
|
|
|
|
bool is_local_address(const std::string &address)
|
|
|
|
{
|
2018-06-05 06:32:35 -04:00
|
|
|
// always assume Tor/I2P addresses to be untrusted by default
|
|
|
|
if (boost::ends_with(address, ".onion") || boost::ends_with(address, ".i2p"))
|
|
|
|
{
|
|
|
|
MDEBUG("Address '" << address << "' is Tor/I2P, non local");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-01-28 09:48:45 -05:00
|
|
|
// extract host
|
|
|
|
epee::net_utils::http::url_content u_c;
|
|
|
|
if (!epee::net_utils::parse_url(address, u_c))
|
|
|
|
{
|
|
|
|
MWARNING("Failed to determine whether address '" << address << "' is local, assuming not");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (u_c.host.empty())
|
|
|
|
{
|
|
|
|
MWARNING("Failed to determine whether address '" << address << "' is local, assuming not");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolve to IP
|
|
|
|
boost::asio::io_service io_service;
|
|
|
|
boost::asio::ip::tcp::resolver resolver(io_service);
|
|
|
|
boost::asio::ip::tcp::resolver::query query(u_c.host, "");
|
|
|
|
boost::asio::ip::tcp::resolver::iterator i = resolver.resolve(query);
|
|
|
|
while (i != boost::asio::ip::tcp::resolver::iterator())
|
|
|
|
{
|
|
|
|
const boost::asio::ip::tcp::endpoint &ep = *i;
|
|
|
|
if (ep.address().is_loopback())
|
|
|
|
{
|
|
|
|
MDEBUG("Address '" << address << "' is local");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
|
|
|
MDEBUG("Address '" << address << "' is not local");
|
|
|
|
return false;
|
|
|
|
}
|
2017-02-16 17:30:01 -05:00
|
|
|
int vercmp(const char *v0, const char *v1)
|
|
|
|
{
|
|
|
|
std::vector<std::string> f0, f1;
|
2017-12-07 05:32:48 -05:00
|
|
|
boost::split(f0, v0, boost::is_any_of(".-"));
|
|
|
|
boost::split(f1, v1, boost::is_any_of(".-"));
|
|
|
|
for (size_t i = 0; i < std::max(f0.size(), f1.size()); ++i) {
|
|
|
|
if (i >= f0.size())
|
|
|
|
return -1;
|
|
|
|
if (i >= f1.size())
|
|
|
|
return 1;
|
2017-02-16 17:30:01 -05:00
|
|
|
int f0i = atoi(f0[i].c_str()), f1i = atoi(f1[i].c_str());
|
|
|
|
int n = f0i - f1i;
|
|
|
|
if (n)
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2017-02-20 15:46:50 -05:00
|
|
|
|
2017-02-23 18:18:10 -05:00
|
|
|
bool sha256sum(const uint8_t *data, size_t len, crypto::hash &hash)
|
|
|
|
{
|
|
|
|
SHA256_CTX ctx;
|
|
|
|
if (!SHA256_Init(&ctx))
|
|
|
|
return false;
|
|
|
|
if (!SHA256_Update(&ctx, data, len))
|
|
|
|
return false;
|
|
|
|
if (!SHA256_Final((unsigned char*)hash.data, &ctx))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-02-20 15:46:50 -05:00
|
|
|
bool sha256sum(const std::string &filename, crypto::hash &hash)
|
|
|
|
{
|
|
|
|
if (!epee::file_io_utils::is_file_exist(filename))
|
|
|
|
return false;
|
|
|
|
std::ifstream f;
|
|
|
|
f.exceptions(std::ifstream::failbit | std::ifstream::badbit);
|
|
|
|
f.open(filename, std::ios_base::binary | std::ios_base::in | std::ios::ate);
|
|
|
|
if (!f)
|
|
|
|
return false;
|
|
|
|
std::ifstream::pos_type file_size = f.tellg();
|
|
|
|
SHA256_CTX ctx;
|
|
|
|
if (!SHA256_Init(&ctx))
|
|
|
|
return false;
|
|
|
|
size_t size_left = file_size;
|
|
|
|
f.seekg(0, std::ios::beg);
|
|
|
|
while (size_left)
|
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
std::ifstream::pos_type read_size = size_left > sizeof(buf) ? sizeof(buf) : size_left;
|
|
|
|
f.read(buf, read_size);
|
|
|
|
if (!f || !f.good())
|
|
|
|
return false;
|
|
|
|
if (!SHA256_Update(&ctx, buf, read_size))
|
|
|
|
return false;
|
|
|
|
size_left -= read_size;
|
|
|
|
}
|
|
|
|
f.close();
|
|
|
|
if (!SHA256_Final((unsigned char*)hash.data, &ctx))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
2018-06-04 03:54:35 -04:00
|
|
|
|
|
|
|
boost::optional<std::pair<uint32_t, uint32_t>> parse_subaddress_lookahead(const std::string& str)
|
|
|
|
{
|
|
|
|
auto pos = str.find(":");
|
|
|
|
bool r = pos != std::string::npos;
|
|
|
|
uint32_t major;
|
|
|
|
r = r && epee::string_tools::get_xtype_from_string(major, str.substr(0, pos));
|
|
|
|
uint32_t minor;
|
|
|
|
r = r && epee::string_tools::get_xtype_from_string(minor, str.substr(pos + 1));
|
|
|
|
if (r)
|
|
|
|
{
|
|
|
|
return std::make_pair(major, minor);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
2018-07-06 14:10:39 -04:00
|
|
|
|
|
|
|
std::string glob_to_regex(const std::string &val)
|
|
|
|
{
|
|
|
|
std::string newval;
|
|
|
|
|
|
|
|
bool escape = false;
|
|
|
|
for (char c: val)
|
|
|
|
{
|
|
|
|
if (c == '*')
|
|
|
|
newval += escape ? "*" : ".*";
|
|
|
|
else if (c == '?')
|
|
|
|
newval += escape ? "?" : ".";
|
|
|
|
else if (c == '\\')
|
|
|
|
newval += '\\', escape = !escape;
|
|
|
|
else
|
|
|
|
newval += c;
|
|
|
|
}
|
|
|
|
return newval;
|
|
|
|
}
|
2018-09-19 08:39:01 -04:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
std::string input_line_win()
|
|
|
|
{
|
|
|
|
HANDLE hConIn = CreateFileW(L"CONIN$", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, nullptr, OPEN_EXISTING, 0, nullptr);
|
|
|
|
DWORD oldMode;
|
|
|
|
|
|
|
|
FlushConsoleInputBuffer(hConIn);
|
|
|
|
GetConsoleMode(hConIn, &oldMode);
|
|
|
|
SetConsoleMode(hConIn, ENABLE_LINE_INPUT | ENABLE_ECHO_INPUT | ENABLE_PROCESSED_INPUT);
|
|
|
|
|
|
|
|
wchar_t buffer[1024];
|
|
|
|
DWORD read;
|
|
|
|
|
|
|
|
ReadConsoleW(hConIn, buffer, sizeof(buffer)/sizeof(wchar_t)-1, &read, nullptr);
|
|
|
|
buffer[read] = 0;
|
|
|
|
|
|
|
|
SetConsoleMode(hConIn, oldMode);
|
|
|
|
CloseHandle(hConIn);
|
|
|
|
|
|
|
|
int size_needed = WideCharToMultiByte(CP_UTF8, 0, buffer, -1, NULL, 0, NULL, NULL);
|
|
|
|
std::string buf(size_needed, '\0');
|
|
|
|
WideCharToMultiByte(CP_UTF8, 0, buffer, -1, &buf[0], size_needed, NULL, NULL);
|
|
|
|
buf.pop_back(); //size_needed includes null that we needed to have space for
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-10-16 14:08:36 -04:00
|
|
|
void closefrom(int fd)
|
|
|
|
{
|
|
|
|
#if defined __FreeBSD__ || defined __OpenBSD__ || defined __NetBSD__ || defined __DragonFly__
|
|
|
|
::closefrom(fd);
|
|
|
|
#else
|
|
|
|
#if defined __GLIBC__
|
|
|
|
const int sc_open_max = sysconf(_SC_OPEN_MAX);
|
|
|
|
const int MAX_FDS = std::min(65536, sc_open_max);
|
|
|
|
#else
|
|
|
|
const int MAX_FDS = 65536;
|
|
|
|
#endif
|
|
|
|
while (fd < MAX_FDS)
|
|
|
|
{
|
|
|
|
close(fd);
|
|
|
|
++fd;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-10-28 09:46:58 -04:00
|
|
|
std::string get_human_readable_timestamp(uint64_t ts)
|
|
|
|
{
|
|
|
|
char buffer[64];
|
|
|
|
if (ts < 1234567890)
|
|
|
|
return "<unknown>";
|
|
|
|
time_t tt = ts;
|
|
|
|
struct tm tm;
|
|
|
|
misc_utils::get_gmt_time(tt, tm);
|
|
|
|
strftime(buffer, sizeof(buffer), "%Y-%m-%d %H:%M:%S", &tm);
|
|
|
|
return std::string(buffer);
|
|
|
|
}
|
2019-03-21 06:03:24 -04:00
|
|
|
|
2019-05-09 06:13:12 -04:00
|
|
|
std::string get_human_readable_timespan(uint64_t seconds)
|
|
|
|
{
|
|
|
|
if (seconds < 60)
|
|
|
|
return std::to_string(seconds) + " seconds";
|
|
|
|
if (seconds < 3600)
|
|
|
|
return std::to_string((uint64_t)(seconds / 60)) + " minutes";
|
|
|
|
if (seconds < 3600 * 24)
|
|
|
|
return std::to_string((uint64_t)(seconds / 3600)) + " hours";
|
|
|
|
if (seconds < 3600 * 24 * 30.5)
|
|
|
|
return std::to_string((uint64_t)(seconds / (3600 * 24))) + " days";
|
|
|
|
if (seconds < 3600 * 24 * 365.25)
|
|
|
|
return std::to_string((uint64_t)(seconds / (3600 * 24 * 30.5))) + " months";
|
|
|
|
if (seconds < 3600 * 24 * 365.25 * 100)
|
|
|
|
return std::to_string((uint64_t)(seconds / (3600 * 24 * 30.5 * 365.25))) + " years";
|
|
|
|
return "a long time";
|
|
|
|
}
|
|
|
|
|
2019-03-21 06:03:24 -04:00
|
|
|
std::string get_human_readable_bytes(uint64_t bytes)
|
|
|
|
{
|
|
|
|
// Use 1024 for "kilo", 1024*1024 for "mega" and so on instead of the more modern and standard-conforming
|
|
|
|
// 1000, 1000*1000 and so on, to be consistent with other Monero code that also uses base 2 units
|
|
|
|
struct byte_map
|
|
|
|
{
|
|
|
|
const char* const format;
|
|
|
|
const std::uint64_t bytes;
|
|
|
|
};
|
|
|
|
|
|
|
|
static constexpr const byte_map sizes[] =
|
|
|
|
{
|
|
|
|
{"%.0f B", 1024},
|
|
|
|
{"%.2f KB", 1024 * 1024},
|
|
|
|
{"%.2f MB", std::uint64_t(1024) * 1024 * 1024},
|
|
|
|
{"%.2f GB", std::uint64_t(1024) * 1024 * 1024 * 1024},
|
|
|
|
{"%.2f TB", std::uint64_t(1024) * 1024 * 1024 * 1024 * 1024}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct bytes_less
|
|
|
|
{
|
|
|
|
bool operator()(const byte_map& lhs, const byte_map& rhs) const noexcept
|
|
|
|
{
|
|
|
|
return lhs.bytes < rhs.bytes;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const auto size = std::upper_bound(
|
|
|
|
std::begin(sizes), std::end(sizes) - 1, byte_map{"", bytes}, bytes_less{}
|
|
|
|
);
|
|
|
|
const std::uint64_t divisor = size->bytes / 1024;
|
|
|
|
return (boost::format(size->format) % (double(bytes) / divisor)).str();
|
|
|
|
}
|
|
|
|
|
2019-05-06 04:44:50 -04:00
|
|
|
void clear_screen()
|
|
|
|
{
|
|
|
|
std::cout << "\033[2K" << std::flush; // clear whole line
|
|
|
|
std::cout << "\033c" << std::flush; // clear current screen and scrollback
|
|
|
|
std::cout << "\033[2J" << std::flush; // clear current screen only, scrollback is still around
|
|
|
|
std::cout << "\033[3J" << std::flush; // does nothing, should clear current screen and scrollback
|
|
|
|
std::cout << "\033[1;1H" << std::flush; // move cursor top/left
|
|
|
|
std::cout << "\r \r" << std::flush; // erase odd chars if the ANSI codes were printed raw
|
|
|
|
#ifdef _WIN32
|
|
|
|
COORD coord{0, 0};
|
|
|
|
CONSOLE_SCREEN_BUFFER_INFO csbi;
|
|
|
|
HANDLE h = GetStdHandle(STD_OUTPUT_HANDLE);
|
|
|
|
if (GetConsoleScreenBufferInfo(h, &csbi))
|
|
|
|
{
|
|
|
|
DWORD cbConSize = csbi.dwSize.X * csbi.dwSize.Y, w;
|
|
|
|
FillConsoleOutputCharacter(h, (TCHAR)' ', cbConSize, coord, &w);
|
|
|
|
if (GetConsoleScreenBufferInfo(h, &csbi))
|
|
|
|
FillConsoleOutputAttribute(h, csbi.wAttributes, cbConSize, coord, &w);
|
|
|
|
SetConsoleCursorPosition(h, coord);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<std::string, size_t> get_string_prefix_by_width(const std::string &s, size_t columns)
|
|
|
|
{
|
|
|
|
std::string sc = "";
|
|
|
|
size_t avail = s.size();
|
|
|
|
const char *ptr = s.data();
|
|
|
|
wint_t cp = 0;
|
|
|
|
int bytes = 1;
|
|
|
|
size_t sw = 0;
|
|
|
|
char wbuf[8], *wptr;
|
|
|
|
while (avail--)
|
|
|
|
{
|
|
|
|
if ((*ptr & 0x80) == 0)
|
|
|
|
{
|
|
|
|
cp = *ptr++;
|
|
|
|
bytes = 1;
|
|
|
|
}
|
|
|
|
else if ((*ptr & 0xe0) == 0xc0)
|
|
|
|
{
|
|
|
|
if (avail < 1)
|
|
|
|
{
|
|
|
|
MERROR("Invalid UTF-8");
|
|
|
|
return std::make_pair(s, s.size());
|
|
|
|
}
|
|
|
|
cp = (*ptr++ & 0x1f) << 6;
|
|
|
|
cp |= *ptr++ & 0x3f;
|
|
|
|
--avail;
|
|
|
|
bytes = 2;
|
|
|
|
}
|
|
|
|
else if ((*ptr & 0xf0) == 0xe0)
|
|
|
|
{
|
|
|
|
if (avail < 2)
|
|
|
|
{
|
|
|
|
MERROR("Invalid UTF-8");
|
|
|
|
return std::make_pair(s, s.size());
|
|
|
|
}
|
|
|
|
cp = (*ptr++ & 0xf) << 12;
|
|
|
|
cp |= (*ptr++ & 0x3f) << 6;
|
|
|
|
cp |= *ptr++ & 0x3f;
|
|
|
|
avail -= 2;
|
|
|
|
bytes = 3;
|
|
|
|
}
|
|
|
|
else if ((*ptr & 0xf8) == 0xf0)
|
|
|
|
{
|
|
|
|
if (avail < 3)
|
|
|
|
{
|
|
|
|
MERROR("Invalid UTF-8");
|
|
|
|
return std::make_pair(s, s.size());
|
|
|
|
}
|
|
|
|
cp = (*ptr++ & 0x7) << 18;
|
|
|
|
cp |= (*ptr++ & 0x3f) << 12;
|
|
|
|
cp |= (*ptr++ & 0x3f) << 6;
|
|
|
|
cp |= *ptr++ & 0x3f;
|
|
|
|
avail -= 3;
|
|
|
|
bytes = 4;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MERROR("Invalid UTF-8");
|
|
|
|
return std::make_pair(s, s.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
wptr = wbuf;
|
|
|
|
switch (bytes)
|
|
|
|
{
|
|
|
|
case 1: *wptr++ = cp; break;
|
|
|
|
case 2: *wptr++ = 0xc0 | (cp >> 6); *wptr++ = 0x80 | (cp & 0x3f); break;
|
|
|
|
case 3: *wptr++ = 0xe0 | (cp >> 12); *wptr++ = 0x80 | ((cp >> 6) & 0x3f); *wptr++ = 0x80 | (cp & 0x3f); break;
|
|
|
|
case 4: *wptr++ = 0xf0 | (cp >> 18); *wptr++ = 0x80 | ((cp >> 12) & 0x3f); *wptr++ = 0x80 | ((cp >> 6) & 0x3f); *wptr++ = 0x80 | (cp & 0x3f); break;
|
|
|
|
default: MERROR("Invalid UTF-8"); return std::make_pair(s, s.size());
|
|
|
|
}
|
|
|
|
*wptr = 0;
|
|
|
|
sc += std::string(wbuf, bytes);
|
|
|
|
#ifdef _WIN32
|
|
|
|
int cpw = 1; // Guess who does not implement wcwidth
|
|
|
|
#else
|
|
|
|
int cpw = wcwidth(cp);
|
|
|
|
#endif
|
|
|
|
if (cpw > 0)
|
|
|
|
{
|
|
|
|
if (cpw > (int)columns)
|
|
|
|
break;
|
|
|
|
columns -= cpw;
|
|
|
|
sw += cpw;
|
|
|
|
}
|
|
|
|
cp = 0;
|
|
|
|
bytes = 1;
|
|
|
|
}
|
|
|
|
return std::make_pair(sc, sw);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t get_string_width(const std::string &s)
|
|
|
|
{
|
|
|
|
return get_string_prefix_by_width(s, 999999999).second;
|
|
|
|
};
|
|
|
|
|
2019-10-29 13:43:09 -04:00
|
|
|
std::vector<std::pair<std::string, size_t>> split_line_by_width(const std::string &s, size_t columns)
|
2019-05-06 04:44:50 -04:00
|
|
|
{
|
|
|
|
std::vector<std::string> words;
|
|
|
|
std::vector<std::pair<std::string, size_t>> lines;
|
|
|
|
boost::split(words, s, boost::is_any_of(" "), boost::token_compress_on);
|
|
|
|
// split large "words"
|
|
|
|
for (size_t i = 0; i < words.size(); ++i)
|
|
|
|
{
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
std::string prefix = get_string_prefix_by_width(words[i], columns).first;
|
|
|
|
if (prefix == words[i])
|
|
|
|
break;
|
|
|
|
words[i] = words[i].substr(prefix.size());
|
|
|
|
words.insert(words.begin() + i, prefix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lines.push_back(std::make_pair("", 0));
|
|
|
|
while (!words.empty())
|
|
|
|
{
|
|
|
|
const size_t word_len = get_string_width(words.front());
|
|
|
|
size_t line_len = get_string_width(lines.back().first);
|
|
|
|
if (line_len > 0 && line_len + 1 + word_len > columns)
|
|
|
|
{
|
|
|
|
lines.push_back(std::make_pair("", 0));
|
|
|
|
line_len = 0;
|
|
|
|
}
|
|
|
|
if (line_len > 0)
|
|
|
|
{
|
|
|
|
lines.back().first += " ";
|
|
|
|
lines.back().second++;
|
|
|
|
}
|
|
|
|
lines.back().first += words.front();
|
|
|
|
lines.back().second += word_len;
|
|
|
|
words.erase(words.begin());
|
|
|
|
}
|
|
|
|
return lines;
|
|
|
|
}
|
|
|
|
|
2019-10-29 13:43:09 -04:00
|
|
|
std::vector<std::pair<std::string, size_t>> split_string_by_width(const std::string &s, size_t columns)
|
|
|
|
{
|
|
|
|
std::vector<std::string> lines;
|
|
|
|
std::vector<std::pair<std::string, size_t>> all_lines;
|
|
|
|
boost::split(lines, s, boost::is_any_of("\n"), boost::token_compress_on);
|
|
|
|
for (const auto &e: lines)
|
|
|
|
{
|
|
|
|
std::vector<std::pair<std::string, size_t>> new_lines = split_line_by_width(e, columns);
|
|
|
|
for (auto &l: new_lines)
|
|
|
|
all_lines.push_back(std::move(l));
|
|
|
|
}
|
|
|
|
return all_lines;
|
|
|
|
}
|
2014-03-03 17:07:58 -05:00
|
|
|
}
|