2015-04-13 04:09:02 -05:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
|
2015-03-17 18:56:15 -05:00
|
|
|
/*
|
|
|
|
* This file is part of the LibreOffice project.
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
loolwsd: include cleanup and organization
A source file (.cpp) must include its own header first.
This insures that the header is self-contained and
doesn't depend on arbitrary (and accidental) includes
before it to compile.
Furthermore, system headers should go next, followed by
C then C++ headers, then libraries (Poco, etc) and, finally,
project headers come last.
This makes sure that headers and included in the same dependency
order to avoid side-effects. For example, Poco should never rely on
anything from our project in the same way that a C header should
never rely on anything in C++, Poco, or project headers.
Also, includes ought to be sorted where possible, to improve
readability and avoid accidental duplicates (of which there
were a few).
Change-Id: I62cc1343e4a091d69195e37ed659dba20cfcb1ef
Reviewed-on: https://gerrit.libreoffice.org/25262
Reviewed-by: Ashod Nakashian <ashnakash@gmail.com>
Tested-by: Ashod Nakashian <ashnakash@gmail.com>
2016-05-21 09:23:07 -05:00
|
|
|
#include "Util.hpp"
|
2016-04-15 09:07:24 -05:00
|
|
|
#include "config.h"
|
|
|
|
|
2016-04-08 07:22:22 -05:00
|
|
|
#include <execinfo.h>
|
2016-08-30 02:06:39 -05:00
|
|
|
#include <csignal>
|
2015-12-13 11:04:45 -06:00
|
|
|
#include <sys/poll.h>
|
2015-12-30 10:41:41 -06:00
|
|
|
#include <sys/prctl.h>
|
2016-09-29 09:47:28 -05:00
|
|
|
#include <sys/stat.h>
|
2016-04-08 07:22:22 -05:00
|
|
|
#include <sys/uio.h>
|
2016-09-29 09:47:28 -05:00
|
|
|
#include <sys/vfs.h>
|
2016-04-20 08:43:00 -05:00
|
|
|
#include <unistd.h>
|
2015-12-13 11:04:45 -06:00
|
|
|
|
2016-04-20 08:43:00 -05:00
|
|
|
#include <atomic>
|
2016-03-08 01:31:29 -06:00
|
|
|
#include <cassert>
|
2016-09-27 07:48:32 -05:00
|
|
|
#include <chrono>
|
|
|
|
#include <cstdio>
|
2015-03-28 06:53:44 -05:00
|
|
|
#include <cstdlib>
|
2015-04-09 17:25:48 -05:00
|
|
|
#include <cstring>
|
2016-09-27 07:48:32 -05:00
|
|
|
#include <fstream>
|
2015-11-06 04:46:31 -06:00
|
|
|
#include <iomanip>
|
loolwsd: include cleanup and organization
A source file (.cpp) must include its own header first.
This insures that the header is self-contained and
doesn't depend on arbitrary (and accidental) includes
before it to compile.
Furthermore, system headers should go next, followed by
C then C++ headers, then libraries (Poco, etc) and, finally,
project headers come last.
This makes sure that headers and included in the same dependency
order to avoid side-effects. For example, Poco should never rely on
anything from our project in the same way that a C header should
never rely on anything in C++, Poco, or project headers.
Also, includes ought to be sorted where possible, to improve
readability and avoid accidental duplicates (of which there
were a few).
Change-Id: I62cc1343e4a091d69195e37ed659dba20cfcb1ef
Reviewed-on: https://gerrit.libreoffice.org/25262
Reviewed-by: Ashod Nakashian <ashnakash@gmail.com>
Tested-by: Ashod Nakashian <ashnakash@gmail.com>
2016-05-21 09:23:07 -05:00
|
|
|
#include <iostream>
|
2016-03-08 01:31:29 -06:00
|
|
|
#include <mutex>
|
|
|
|
#include <random>
|
2015-11-06 04:46:31 -06:00
|
|
|
#include <sstream>
|
2015-03-17 18:56:15 -05:00
|
|
|
#include <string>
|
2016-11-13 15:12:01 -06:00
|
|
|
#include <thread>
|
2015-03-17 18:56:15 -05:00
|
|
|
|
2016-05-30 19:42:44 -05:00
|
|
|
#include <Poco/Base64Encoder.h>
|
2016-03-08 01:31:29 -06:00
|
|
|
#include <Poco/ConsoleChannel.h>
|
2015-04-22 03:14:11 -05:00
|
|
|
#include <Poco/Exception.h>
|
2015-04-27 06:16:37 -05:00
|
|
|
#include <Poco/Format.h>
|
2015-04-22 03:14:11 -05:00
|
|
|
#include <Poco/Net/WebSocket.h>
|
2015-03-17 18:56:15 -05:00
|
|
|
#include <Poco/Process.h>
|
2016-05-30 19:42:44 -05:00
|
|
|
#include <Poco/RandomStream.h>
|
loolwsd: include cleanup and organization
A source file (.cpp) must include its own header first.
This insures that the header is self-contained and
doesn't depend on arbitrary (and accidental) includes
before it to compile.
Furthermore, system headers should go next, followed by
C then C++ headers, then libraries (Poco, etc) and, finally,
project headers come last.
This makes sure that headers and included in the same dependency
order to avoid side-effects. For example, Poco should never rely on
anything from our project in the same way that a C header should
never rely on anything in C++, Poco, or project headers.
Also, includes ought to be sorted where possible, to improve
readability and avoid accidental duplicates (of which there
were a few).
Change-Id: I62cc1343e4a091d69195e37ed659dba20cfcb1ef
Reviewed-on: https://gerrit.libreoffice.org/25262
Reviewed-by: Ashod Nakashian <ashnakash@gmail.com>
Tested-by: Ashod Nakashian <ashnakash@gmail.com>
2016-05-21 09:23:07 -05:00
|
|
|
#include <Poco/TemporaryFile.h>
|
2015-03-17 18:56:15 -05:00
|
|
|
#include <Poco/Thread.h>
|
2016-03-08 01:31:29 -06:00
|
|
|
#include <Poco/Timestamp.h>
|
2015-04-22 03:14:11 -05:00
|
|
|
#include <Poco/Util/Application.h>
|
2015-03-17 18:56:15 -05:00
|
|
|
|
2016-03-08 01:31:29 -06:00
|
|
|
#include "Common.hpp"
|
loolwsd: include cleanup and organization
A source file (.cpp) must include its own header first.
This insures that the header is self-contained and
doesn't depend on arbitrary (and accidental) includes
before it to compile.
Furthermore, system headers should go next, followed by
C then C++ headers, then libraries (Poco, etc) and, finally,
project headers come last.
This makes sure that headers and included in the same dependency
order to avoid side-effects. For example, Poco should never rely on
anything from our project in the same way that a C header should
never rely on anything in C++, Poco, or project headers.
Also, includes ought to be sorted where possible, to improve
readability and avoid accidental duplicates (of which there
were a few).
Change-Id: I62cc1343e4a091d69195e37ed659dba20cfcb1ef
Reviewed-on: https://gerrit.libreoffice.org/25262
Reviewed-by: Ashod Nakashian <ashnakash@gmail.com>
Tested-by: Ashod Nakashian <ashnakash@gmail.com>
2016-05-21 09:23:07 -05:00
|
|
|
#include "Log.hpp"
|
2016-09-28 14:07:07 -05:00
|
|
|
#include "Util.hpp"
|
2015-03-28 06:53:44 -05:00
|
|
|
|
2015-12-19 08:16:44 -06:00
|
|
|
namespace Util
|
|
|
|
{
|
|
|
|
namespace rng
|
|
|
|
{
|
|
|
|
static std::random_device _rd;
|
|
|
|
static std::mutex _rngMutex;
|
2016-05-30 19:42:44 -05:00
|
|
|
static Poco::RandomBuf _randBuf;
|
2016-03-12 09:15:45 -06:00
|
|
|
|
|
|
|
// Create the prng with a random-device for seed.
|
|
|
|
// If we don't have a hardware random-device, we will get the same seed.
|
|
|
|
// In that case we are better off with an arbitrary, but changing, seed.
|
|
|
|
static std::mt19937_64 _rng = std::mt19937_64(_rd.entropy()
|
|
|
|
? _rd()
|
|
|
|
: (clock() + getpid()));
|
|
|
|
|
|
|
|
// A new seed is used to shuffle the sequence.
|
|
|
|
// N.B. Always reseed after getting forked!
|
|
|
|
void reseed()
|
|
|
|
{
|
|
|
|
_rng.seed(_rd.entropy() ? _rd() : (clock() + getpid()));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a new random number.
|
2015-12-19 08:16:44 -06:00
|
|
|
unsigned getNext()
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(_rngMutex);
|
|
|
|
return _rng();
|
|
|
|
}
|
2016-05-30 19:42:44 -05:00
|
|
|
|
|
|
|
std::vector<char> getBytes(const size_t length)
|
|
|
|
{
|
|
|
|
std::vector<char> v(length);
|
|
|
|
_randBuf.readFromDevice(v.data(), v.size());
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
2016-05-30 20:43:18 -05:00
|
|
|
/// Generates a random string in Base64.
|
|
|
|
/// Note: May contain '/' characters.
|
|
|
|
std::string getB64String(const size_t length)
|
2016-05-30 19:42:44 -05:00
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
Poco::Base64Encoder b64(ss);
|
2016-09-22 06:35:16 -05:00
|
|
|
b64.write(getBytes(length).data(), length);
|
2016-05-30 19:42:44 -05:00
|
|
|
return ss.str().substr(0, length);
|
|
|
|
}
|
2016-05-30 20:43:18 -05:00
|
|
|
|
|
|
|
std::string getFilename(const size_t length)
|
|
|
|
{
|
2016-06-05 19:34:24 -05:00
|
|
|
std::string s = getB64String(length);
|
2016-05-30 20:43:18 -05:00
|
|
|
std::replace(s.begin(), s.end(), '/', '_');
|
|
|
|
return s.substr(0, length);
|
|
|
|
}
|
2015-12-19 08:16:44 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-26 10:07:52 -05:00
|
|
|
namespace Util
|
2015-03-17 18:56:15 -05:00
|
|
|
{
|
2015-12-27 21:47:39 -06:00
|
|
|
std::string encodeId(const unsigned number, const int padding)
|
|
|
|
{
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << std::hex << std::setw(padding) << std::setfill('0') << number;
|
|
|
|
return oss.str();
|
|
|
|
}
|
|
|
|
|
2016-01-09 14:46:08 -06:00
|
|
|
unsigned decodeId(const std::string& str)
|
|
|
|
{
|
|
|
|
unsigned id = 0;
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << std::hex << str;
|
|
|
|
ss >> id;
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2015-03-28 06:53:44 -05:00
|
|
|
bool windowingAvailable()
|
|
|
|
{
|
2015-12-29 19:34:53 -06:00
|
|
|
return std::getenv("DISPLAY") != nullptr;
|
2015-03-28 06:53:44 -05:00
|
|
|
}
|
|
|
|
|
2016-09-30 04:05:36 -05:00
|
|
|
} // namespace Util
|
|
|
|
|
|
|
|
namespace Util
|
|
|
|
{
|
2016-04-06 07:43:44 -05:00
|
|
|
int getMemoryUsage(const Poco::Process::PID nPid)
|
2016-03-04 12:49:01 -06:00
|
|
|
{
|
2016-11-09 20:12:09 -06:00
|
|
|
try
|
2016-03-04 12:49:01 -06:00
|
|
|
{
|
2016-11-09 20:12:09 -06:00
|
|
|
//TODO: Instead of RSS, return PSS
|
|
|
|
const auto cmd = "ps o rss= -p " + std::to_string(nPid);
|
|
|
|
FILE* fp = popen(cmd.c_str(), "r");
|
|
|
|
if (fp == nullptr)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2016-03-04 12:49:01 -06:00
|
|
|
|
2016-11-09 20:12:09 -06:00
|
|
|
std::string sResponse;
|
|
|
|
char cmdBuffer[1024];
|
|
|
|
while (fgets(cmdBuffer, sizeof(cmdBuffer) - 1, fp) != nullptr)
|
|
|
|
{
|
|
|
|
sResponse += cmdBuffer;
|
|
|
|
}
|
|
|
|
pclose(fp);
|
2016-03-04 12:49:01 -06:00
|
|
|
|
2016-11-09 20:12:09 -06:00
|
|
|
return std::stoi(sResponse);
|
2016-03-04 12:49:01 -06:00
|
|
|
}
|
2016-12-19 17:28:26 -06:00
|
|
|
catch (const std::exception&)
|
2016-03-04 12:49:01 -06:00
|
|
|
{
|
2016-12-19 17:28:26 -06:00
|
|
|
LOG_WRN("Trying to find memory of invalid/dead PID " << nPid);
|
2016-03-04 12:49:01 -06:00
|
|
|
}
|
|
|
|
|
2016-11-09 20:12:09 -06:00
|
|
|
return -1;
|
2016-03-04 12:49:01 -06:00
|
|
|
}
|
2016-03-31 03:01:21 -05:00
|
|
|
|
|
|
|
std::string replace(const std::string& s, const std::string& a, const std::string& b)
|
|
|
|
{
|
|
|
|
std::string result = s;
|
|
|
|
std::string::size_type pos;
|
|
|
|
while ((pos = result.find(a)) != std::string::npos)
|
|
|
|
{
|
|
|
|
result = result.replace(pos, a.size(), b);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string formatLinesForLog(const std::string& s)
|
|
|
|
{
|
|
|
|
std::string r;
|
|
|
|
std::string::size_type n = s.size();
|
|
|
|
if (n > 0 && s.back() == '\n')
|
|
|
|
r = s.substr(0, n-1);
|
|
|
|
else
|
|
|
|
r = s;
|
|
|
|
return replace(r, "\n", " / ");
|
|
|
|
}
|
2016-04-07 02:55:57 -05:00
|
|
|
|
|
|
|
void setThreadName(const std::string& s)
|
|
|
|
{
|
|
|
|
if (prctl(PR_SET_NAME, reinterpret_cast<unsigned long>(s.c_str()), 0, 0, 0) != 0)
|
2016-11-13 10:59:34 -06:00
|
|
|
{
|
|
|
|
LOG_SYS("Cannot set thread name to " << s << ".");
|
|
|
|
}
|
2016-04-07 02:55:57 -05:00
|
|
|
}
|
2016-04-15 09:07:24 -05:00
|
|
|
|
2016-06-20 04:51:35 -05:00
|
|
|
void getVersionInfo(std::string& version, std::string& hash)
|
2016-04-15 09:07:24 -05:00
|
|
|
{
|
2016-06-20 04:51:35 -05:00
|
|
|
version = std::string(LOOLWSD_VERSION);
|
|
|
|
hash = std::string(LOOLWSD_VERSION_HASH);
|
2016-04-15 09:07:24 -05:00
|
|
|
hash.resize(std::min(8, (int)hash.length()));
|
|
|
|
}
|
2016-04-20 08:43:00 -05:00
|
|
|
|
|
|
|
std::string UniqueId()
|
|
|
|
{
|
|
|
|
static std::atomic_int counter(0);
|
|
|
|
return std::to_string(Poco::Process::id()) + "/" + std::to_string(counter++);
|
|
|
|
}
|
2015-03-17 18:56:15 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|