ccls/src/utils.cc

347 lines
10 KiB
C++
Raw Normal View History

2017-02-17 09:57:44 +00:00
#include "utils.h"
2018-03-31 17:37:03 +00:00
#include "filesystem.hh"
#include "platform.h"
#include <doctest/doctest.h>
2018-01-30 00:27:43 +00:00
#include <siphash.h>
2017-11-18 19:02:09 +00:00
#include <loguru/loguru.hpp>
#include <algorithm>
2017-03-06 07:19:00 +00:00
#include <cassert>
#include <cctype>
#include <cstring>
2017-09-22 01:14:57 +00:00
#include <fstream>
#include <functional>
2018-02-07 05:26:38 +00:00
#include <queue>
#include <sstream>
2018-01-03 12:38:01 +00:00
#include <string>
2017-04-08 22:54:36 +00:00
#include <unordered_map>
2017-02-17 09:57:44 +00:00
2017-12-19 00:35:43 +00:00
// DEFAULT_RESOURCE_DIRECTORY is passed with quotes for non-MSVC compilers, ie,
// foo vs "foo".
#if defined(_MSC_VER)
2017-12-19 00:35:43 +00:00
#define _STRINGIFY(x) #x
#define ENSURE_STRING_MACRO_ARGUMENT(x) _STRINGIFY(x)
#else
2017-12-18 19:47:58 +00:00
#define ENSURE_STRING_MACRO_ARGUMENT(x) x
2017-12-19 00:35:43 +00:00
#endif
2018-03-31 17:37:03 +00:00
void TrimInPlace(std::string& s) {
2017-09-22 01:14:57 +00:00
s.erase(s.begin(),
std::find_if(s.begin(), s.end(),
std::not1(std::ptr_fun<int, int>(std::isspace))));
s.erase(std::find_if(s.rbegin(), s.rend(),
2017-09-22 01:14:57 +00:00
std::not1(std::ptr_fun<int, int>(std::isspace)))
.base(),
s.end());
}
std::string Trim(std::string s) {
TrimInPlace(s);
return s;
}
2018-02-01 09:12:36 +00:00
void RemoveLastCR(std::string& s) {
if (!s.empty() && *s.rbegin() == '\r')
s.pop_back();
}
2018-01-14 21:18:12 +00:00
uint64_t HashUsr(const std::string& s) {
return HashUsr(s.c_str(), s.size());
}
2018-01-14 21:18:12 +00:00
uint64_t HashUsr(const char* s) {
return HashUsr(s, strlen(s));
}
uint64_t HashUsr(const char* s, size_t n) {
union {
uint64_t ret;
uint8_t out[8];
};
2018-01-14 21:18:12 +00:00
// k is an arbitrary key. Don't change it.
const uint8_t k[16] = {0xd0, 0xe5, 0x4d, 0x61, 0x74, 0x63, 0x68, 0x52,
0x61, 0x79, 0xea, 0x70, 0xca, 0x70, 0xf0, 0x0d};
(void)siphash(reinterpret_cast<const uint8_t*>(s), n, k, out, 8);
return ret;
}
2017-03-31 04:15:42 +00:00
// See http://stackoverflow.com/a/2072890
bool EndsWith(std::string_view value, std::string_view ending) {
2017-03-31 04:15:42 +00:00
if (ending.size() > value.size())
return false;
return std::equal(ending.rbegin(), ending.rend(), value.rbegin());
}
bool StartsWith(std::string_view value, std::string_view start) {
2017-03-31 04:15:42 +00:00
if (start.size() > value.size())
return false;
return std::equal(start.begin(), start.end(), value.begin());
}
2017-09-22 01:14:57 +00:00
bool AnyStartsWith(const std::vector<std::string>& values,
const std::string& start) {
return std::any_of(
std::begin(values), std::end(values),
[&start](const std::string& value) { return StartsWith(value, start); });
}
bool StartsWithAny(const std::string& value,
const std::vector<std::string>& startings) {
2017-10-25 01:28:23 +00:00
return std::any_of(std::begin(startings), std::end(startings),
[&value](const std::string& starting) {
return StartsWith(value, starting);
});
}
2017-09-22 01:14:57 +00:00
bool EndsWithAny(const std::string& value,
const std::vector<std::string>& endings) {
return std::any_of(
std::begin(endings), std::end(endings),
[&value](const std::string& ending) { return EndsWith(value, ending); });
}
bool FindAnyPartial(const std::string& value,
const std::vector<std::string>& values) {
2018-01-30 00:27:43 +00:00
return std::any_of(std::begin(values), std::end(values),
[&value](const std::string& v) {
return value.find(v) != std::string::npos;
});
}
std::string GetDirName(std::string path) {
if (path.size() && path.back() == '/')
path.pop_back();
size_t last_slash = path.find_last_of('/');
2018-02-22 07:34:32 +00:00
if (last_slash == std::string::npos)
return ".";
if (last_slash == 0)
return "/";
return path.substr(0, last_slash);
}
std::string GetBaseName(const std::string& path) {
size_t last_slash = path.find_last_of('/');
if (last_slash != std::string::npos && (last_slash + 1) < path.size())
return path.substr(last_slash + 1);
return path;
}
std::string StripFileType(const std::string& path) {
2018-03-31 17:37:03 +00:00
fs::path p(path);
return p.parent_path() / p.stem();
}
2017-09-22 01:14:57 +00:00
std::vector<std::string> SplitString(const std::string& str,
const std::string& delimiter) {
// http://stackoverflow.com/a/13172514
std::vector<std::string> strings;
std::string::size_type pos = 0;
std::string::size_type prev = 0;
while ((pos = str.find(delimiter, prev)) != std::string::npos) {
strings.push_back(str.substr(prev, pos - prev));
prev = pos + 1;
}
// To get the last substring (or only, if delimiter is not found)
strings.push_back(str.substr(prev));
return strings;
}
std::string LowerPathIfCaseInsensitive(const std::string& path) {
std::string result = path;
#if defined(_WIN32)
for (size_t i = 0; i < result.size(); ++i)
result[i] = (char)tolower(result[i]);
#endif
return result;
}
static void GetFilesInFolderHelper(
2017-09-22 01:14:57 +00:00
std::string folder,
bool recursive,
std::string output_prefix,
const std::function<void(const std::string&)>& handler) {
2018-03-31 17:37:03 +00:00
std::queue<std::pair<fs::path, fs::path>> q;
q.push(std::make_pair(fs::path(folder), fs::path(output_prefix)));
2018-01-11 02:43:01 +00:00
while (!q.empty()) {
2018-03-31 17:37:03 +00:00
for (auto it = fs::directory_iterator(q.front().first); it != fs::directory_iterator(); ++it) {
auto path = it->path();
std::string filename = path.filename();
if (filename[0] != '.' || filename == ".ccls") {
fs::file_status status = it->symlink_status();
if (fs::is_regular_file(status))
handler(q.front().second / filename);
else if (fs::is_directory(status) || fs::is_symlink(status)) {
if (recursive) {
2018-03-31 17:37:03 +00:00
std::string child_dir = q.front().second / filename;
if (fs::is_directory(status))
q.push(make_pair(path, child_dir));
}
2017-03-11 02:24:51 +00:00
}
2017-03-06 07:19:00 +00:00
}
2017-02-17 09:57:44 +00:00
}
q.pop();
}
2017-02-17 09:57:44 +00:00
}
2017-09-22 01:14:57 +00:00
std::vector<std::string> GetFilesInFolder(std::string folder,
bool recursive,
bool add_folder_to_path) {
EnsureEndsInSlash(folder);
std::vector<std::string> result;
2017-09-22 01:14:57 +00:00
GetFilesInFolderHelper(
folder, recursive, add_folder_to_path ? folder : "",
[&result](const std::string& path) { result.push_back(path); });
return result;
}
2017-09-22 01:14:57 +00:00
void GetFilesInFolder(std::string folder,
bool recursive,
bool add_folder_to_path,
const std::function<void(const std::string&)>& handler) {
EnsureEndsInSlash(folder);
2017-09-22 01:14:57 +00:00
GetFilesInFolderHelper(folder, recursive, add_folder_to_path ? folder : "",
handler);
2017-03-06 07:19:00 +00:00
}
void EnsureEndsInSlash(std::string& path) {
if (path.empty() || path[path.size() - 1] != '/')
path += '/';
}
2017-12-02 05:07:30 +00:00
std::string EscapeFileName(std::string path) {
if (path.size() && path.back() == '/')
path.pop_back();
std::replace(path.begin(), path.end(), '\\', '@');
std::replace(path.begin(), path.end(), '/', '@');
std::replace(path.begin(), path.end(), ':', '@');
2017-12-02 05:07:30 +00:00
return path;
}
2017-03-09 18:02:55 +00:00
// http://stackoverflow.com/a/6089413
std::istream& SafeGetline(std::istream& is, std::string& t) {
2017-09-22 01:14:57 +00:00
t.clear();
// The characters in the stream are read one-by-one using a std::streambuf.
// That is faster than reading them one-by-one using the std::istream. Code
// that uses streambuf this way must be guarded by a sentry object. The sentry
// object performs various tasks, such as thread synchronization and updating
// the stream state.
2017-09-22 01:14:57 +00:00
std::istream::sentry se(is, true);
std::streambuf* sb = is.rdbuf();
for (;;) {
int c = sb->sbumpc();
if (c == EOF) {
// Also handle the case when the last line has no line ending
if (t.empty())
is.setstate(std::ios::eofbit);
return is;
2017-03-09 18:02:55 +00:00
}
t += (char)c;
if (c == '\n')
return is;
2017-09-22 01:14:57 +00:00
}
2017-03-09 18:02:55 +00:00
}
bool FileExists(const std::string& filename) {
2018-03-31 17:37:03 +00:00
return fs::exists(filename);
}
2018-03-31 03:16:33 +00:00
std::optional<std::string> ReadContent(const std::string& filename) {
LOG_S(INFO) << "Reading " << filename;
std::ifstream cache;
cache.open(filename);
try {
return std::string(std::istreambuf_iterator<char>(cache),
std::istreambuf_iterator<char>());
} catch (std::ios_base::failure&) {
2018-03-31 03:16:33 +00:00
return std::nullopt;
}
}
2018-03-31 17:37:03 +00:00
std::vector<std::string> ReadFileLines(std::string filename) {
2017-02-17 09:57:44 +00:00
std::vector<std::string> result;
2018-03-31 17:37:03 +00:00
std::ifstream fin(filename);
for (std::string line; std::getline(fin, line);)
2017-02-17 09:57:44 +00:00
result.push_back(line);
return result;
}
2018-03-31 17:37:03 +00:00
std::vector<std::string> ToLines(const std::string& content) {
std::vector<std::string> result;
std::istringstream lines(content);
std::string line;
2018-03-31 17:37:03 +00:00
while (getline(lines, line))
result.push_back(line);
return result;
}
std::string TextReplacer::Apply(const std::string& content) {
std::string result = content;
for (const Replacement& replacement : replacements) {
while (true) {
size_t idx = result.find(replacement.from);
if (idx == std::string::npos)
break;
result.replace(result.begin() + idx,
result.begin() + idx + replacement.from.size(),
replacement.to);
}
}
return result;
}
2017-03-01 08:36:11 +00:00
void WriteToFile(const std::string& filename, const std::string& content) {
std::ofstream file(filename,
std::ios::out | std::ios::trunc | std::ios::binary);
if (!file.good()) {
LOG_S(ERROR) << "Cannot write to " << filename;
return;
}
2017-03-01 08:36:11 +00:00
file << content;
2017-03-09 18:02:55 +00:00
}
std::string GetDefaultResourceDirectory() {
std::string result;
std::string resource_directory =
std::string(ENSURE_STRING_MACRO_ARGUMENT(DEFAULT_RESOURCE_DIRECTORY));
2018-01-03 12:38:01 +00:00
// Remove double quoted resource dir if it was passed with quotes
// by the build system.
if (resource_directory.size() >= 2 && resource_directory[0] == '"' &&
resource_directory[resource_directory.size() - 1] == '"') {
resource_directory =
resource_directory.substr(1, resource_directory.size() - 2);
}
if (resource_directory.compare(0, 2, "..") == 0) {
std::string executable_path = GetExecutablePath();
size_t pos = executable_path.find_last_of('/');
result = executable_path.substr(0, pos + 1);
result += resource_directory;
} else {
result = resource_directory;
}
2018-01-04 02:32:15 +00:00
return NormalizePath(result);
2017-12-18 19:47:58 +00:00
}
TEST_SUITE("StripFileType") {
TEST_CASE("all") {
REQUIRE(StripFileType("") == "");
REQUIRE(StripFileType("bar") == "bar");
REQUIRE(StripFileType("bar.cc") == "bar");
REQUIRE(StripFileType("foo/bar.cc") == "foo/bar");
}
}