mirror of
https://github.com/MaskRay/ccls.git
synced 2025-01-18 03:25:48 +00:00
Support workspaces as symlinks and symlinks within a workspace
Fix #639
This commit is contained in:
parent
5108cfafcb
commit
ca95f48a53
@ -20,12 +20,22 @@ using namespace clang;
|
||||
|
||||
namespace ccls {
|
||||
std::string pathFromFileEntry(const FileEntry &file) {
|
||||
StringRef name = file.tryGetRealPathName();
|
||||
if (name.empty())
|
||||
name = file.getName();
|
||||
std::string ret = normalizePath(name);
|
||||
// Resolve symlinks outside of workspace folders, e.g. /usr/include/c++/7.3.0
|
||||
return normalizeFolder(ret) ? ret : realPath(ret);
|
||||
std::string ret;
|
||||
if (file.getName().startswith("/../")) {
|
||||
// Resolve symlinks outside of working folders. This handles leading path
|
||||
// components, e.g. (/lib -> /usr/lib) in
|
||||
// /../lib/gcc/x86_64-linux-gnu/10/../../../../include/c++/10/utility
|
||||
ret = file.tryGetRealPathName();
|
||||
} else {
|
||||
// If getName() refers to a file within a workspace folder, we prefer it
|
||||
// (which may be a symlink).
|
||||
ret = normalizePath(file.getName());
|
||||
}
|
||||
if (normalizeFolder(ret))
|
||||
return ret;
|
||||
ret = realPath(ret);
|
||||
normalizeFolder(ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
bool isInsideMainFile(const SourceManager &sm, SourceLocation sl) {
|
||||
|
@ -348,17 +348,16 @@ void do_initialize(MessageHandler *m, InitializeParam ¶m,
|
||||
std::string path = wf.uri.getPath();
|
||||
ensureEndsInSlash(path);
|
||||
std::string real = realPath(path) + '/';
|
||||
workspaceFolders.emplace_back(path, path == real ? "" : real);
|
||||
workspaceFolders.emplace_back(path, real);
|
||||
}
|
||||
if (workspaceFolders.empty()) {
|
||||
std::string real = realPath(project_path) + '/';
|
||||
workspaceFolders.emplace_back(project_path,
|
||||
project_path == real ? "" : real);
|
||||
workspaceFolders.emplace_back(project_path, real);
|
||||
}
|
||||
std::sort(workspaceFolders.begin(), workspaceFolders.end(),
|
||||
[](auto &l, auto &r) { return l.first.size() > r.first.size(); });
|
||||
for (auto &[folder, real] : workspaceFolders)
|
||||
if (real.empty())
|
||||
if (real == folder)
|
||||
LOG_S(INFO) << "workspace folder: " << folder;
|
||||
else
|
||||
LOG_S(INFO) << "workspace folder: " << folder << " -> " << real;
|
||||
|
@ -433,7 +433,9 @@ void Project::loadDirectory(const std::string &root, Project::Folder &folder) {
|
||||
// If workspace folder is real/ but entries use symlink/, convert to
|
||||
// real/.
|
||||
entry.directory = realPath(cmd.Directory);
|
||||
entry.directory.push_back('/');
|
||||
normalizeFolder(entry.directory);
|
||||
entry.directory.pop_back();
|
||||
doPathMapping(entry.directory);
|
||||
entry.filename =
|
||||
realPath(resolveIfRelative(entry.directory, cmd.Filename));
|
||||
|
@ -138,11 +138,15 @@ std::string realPath(const std::string &path) {
|
||||
}
|
||||
|
||||
bool normalizeFolder(std::string &path) {
|
||||
for (auto &[root, real] : g_config->workspaceFolders)
|
||||
if (real.size() && llvm::StringRef(path).startswith(real)) {
|
||||
for (auto &[root, real] : g_config->workspaceFolders) {
|
||||
StringRef p(path);
|
||||
if (p.startswith(root))
|
||||
return true;
|
||||
if (p.startswith(real)) {
|
||||
path = root + path.substr(real.size());
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user