ccls/src/messages/cquery_freshen_index.cc

97 lines
3.2 KiB
C++
Raw Normal View History

#include "cache_manager.h"
#include "match.h"
2017-12-06 03:32:33 +00:00
#include "message_handler.h"
#include "platform.h"
2017-12-29 16:29:47 +00:00
#include "project.h"
#include "queue_manager.h"
#include "timer.h"
2017-12-29 16:29:47 +00:00
#include "timestamp_manager.h"
#include "working_files.h"
2017-12-06 03:32:33 +00:00
#include <loguru.hpp>
#include <queue>
#include <unordered_set>
2017-12-06 03:32:33 +00:00
namespace {
MethodType kMethodType = "$cquery/freshenIndex";
2018-03-22 05:01:21 +00:00
struct In_CqueryFreshenIndex : public NotificationInMessage {
MethodType GetMethodType() const override { return kMethodType; }
struct Params {
bool dependencies = true;
std::vector<std::string> whitelist;
std::vector<std::string> blacklist;
};
Params params;
2017-12-06 04:39:44 +00:00
};
MAKE_REFLECT_STRUCT(In_CqueryFreshenIndex::Params,
2018-01-30 00:27:43 +00:00
dependencies,
whitelist,
blacklist);
MAKE_REFLECT_STRUCT(In_CqueryFreshenIndex, params);
REGISTER_IN_MESSAGE(In_CqueryFreshenIndex);
2017-12-06 04:39:44 +00:00
struct Handler_CqueryFreshenIndex : BaseMessageHandler<In_CqueryFreshenIndex> {
MethodType GetMethodType() const override { return kMethodType; }
void Run(In_CqueryFreshenIndex* request) override {
2017-12-06 03:32:33 +00:00
LOG_S(INFO) << "Freshening " << project->entries.size() << " files";
// TODO: think about this flow and test it more.
GroupMatch matcher(request->params.whitelist, request->params.blacklist);
2017-12-06 03:32:33 +00:00
// Unmark all files whose timestamp has changed.
std::shared_ptr<ICacheManager> cache_manager = ICacheManager::Make(config);
std::queue<const QueryFile*> q;
// |need_index| stores every filename ever enqueued.
std::unordered_set<std::string> need_index;
// Reverse dependency graph.
std::unordered_map<std::string, std::vector<std::string>> graph;
// filename -> QueryFile mapping for files haven't enqueued.
std::unordered_map<std::string, const QueryFile*> path_to_file;
for (const auto& file : db->files)
if (file.def) {
if (matcher.IsMatch(file.def->path))
q.push(&file);
else
path_to_file[file.def->path] = &file;
for (const std::string& dependency : file.def->dependencies)
graph[dependency].push_back(file.def->path);
}
while (!q.empty()) {
const QueryFile* file = q.front();
q.pop();
need_index.insert(file->def->path);
2017-12-06 03:32:33 +00:00
optional<int64_t> modification_timestamp =
GetLastModificationTime(file->def->path);
2017-12-06 03:32:33 +00:00
if (!modification_timestamp)
continue;
optional<int64_t> cached_modification =
timestamp_manager->GetLastCachedModificationTime(cache_manager.get(),
file->def->path);
2017-12-06 03:32:33 +00:00
if (modification_timestamp != cached_modification)
file_consumer_shared->Reset(file->def->path);
if (request->params.dependencies)
for (const std::string& path : graph[file->def->path]) {
auto it = path_to_file.find(path);
if (it != path_to_file.end()) {
q.push(it->second);
path_to_file.erase(it);
}
}
2017-12-06 03:32:33 +00:00
}
Timer time;
2017-12-06 03:32:33 +00:00
// Send index requests for every file.
project->Index(config, QueueManager::instance(), working_files,
std::monostate());
time.ResetAndPrint("[perf] Dispatched $cquery/freshenIndex index requests");
2017-12-06 03:32:33 +00:00
}
};
REGISTER_MESSAGE_HANDLER(Handler_CqueryFreshenIndex);
} // namespace