mirror of
https://github.com/MaskRay/ccls.git
synced 2025-04-03 07:22:08 +00:00
Add DiagnosticsEngine and add config->diagnostics.frequencyMs to allow reducing textDocument/publishDiagnostics frequency
This commit is contained in:
parent
df95cd4780
commit
df55e79e4e
@ -383,7 +383,7 @@ void TryEnsureDocumentParsed(ClangCompleteManager* manager,
|
|||||||
unsaved, Flags());
|
unsaved, Flags());
|
||||||
|
|
||||||
// Build diagnostics.
|
// Build diagnostics.
|
||||||
if (manager->config_->diagnosticsOnParse && *tu) {
|
if (manager->config_->diagnostics.onParse && *tu) {
|
||||||
// If we're emitting diagnostics, do an immediate reparse, otherwise we will
|
// If we're emitting diagnostics, do an immediate reparse, otherwise we will
|
||||||
// emit stale/bad diagnostics.
|
// emit stale/bad diagnostics.
|
||||||
*tu = ClangTranslationUnit::Reparse(std::move(*tu), unsaved);
|
*tu = ClangTranslationUnit::Reparse(std::move(*tu), unsaved);
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
#include "cache_manager.h"
|
#include "cache_manager.h"
|
||||||
#include "clang_complete.h"
|
#include "clang_complete.h"
|
||||||
#include "code_complete_cache.h"
|
#include "code_complete_cache.h"
|
||||||
|
#include "diagnostics_engine.h"
|
||||||
#include "file_consumer.h"
|
#include "file_consumer.h"
|
||||||
#include "import_manager.h"
|
#include "import_manager.h"
|
||||||
#include "import_pipeline.h"
|
#include "import_pipeline.h"
|
||||||
@ -177,11 +178,12 @@ void RunQueryDbThread(const std::string& bin_name,
|
|||||||
SemanticHighlightSymbolCache semantic_cache;
|
SemanticHighlightSymbolCache semantic_cache;
|
||||||
WorkingFiles working_files;
|
WorkingFiles working_files;
|
||||||
FileConsumerSharedState file_consumer_shared;
|
FileConsumerSharedState file_consumer_shared;
|
||||||
|
DiagnosticsEngine diag_engine(config);
|
||||||
|
|
||||||
ClangCompleteManager clang_complete(
|
ClangCompleteManager clang_complete(
|
||||||
config, &project, &working_files,
|
config, &project, &working_files,
|
||||||
[&](std::string path, std::vector<lsDiagnostic> diagnostics) {
|
[&](std::string path, std::vector<lsDiagnostic> diagnostics) {
|
||||||
EmitDiagnostics(&working_files, path, diagnostics);
|
diag_engine.Publish(&working_files, path, diagnostics);
|
||||||
},
|
},
|
||||||
[&](ClangTranslationUnit* tu, const std::vector<CXUnsavedFile>& unsaved,
|
[&](ClangTranslationUnit* tu, const std::vector<CXUnsavedFile>& unsaved,
|
||||||
const std::string& path, const std::vector<std::string>& args) {
|
const std::string& path, const std::vector<std::string>& args) {
|
||||||
@ -214,6 +216,7 @@ void RunQueryDbThread(const std::string& bin_name,
|
|||||||
handler->db = &db;
|
handler->db = &db;
|
||||||
handler->waiter = indexer_waiter;
|
handler->waiter = indexer_waiter;
|
||||||
handler->project = &project;
|
handler->project = &project;
|
||||||
|
handler->diag_engine = &diag_engine;
|
||||||
handler->file_consumer_shared = &file_consumer_shared;
|
handler->file_consumer_shared = &file_consumer_shared;
|
||||||
handler->import_manager = &import_manager;
|
handler->import_manager = &import_manager;
|
||||||
handler->import_pipeline_status = &import_pipeline_status;
|
handler->import_pipeline_status = &import_pipeline_status;
|
||||||
|
28
src/config.h
28
src/config.h
@ -72,9 +72,6 @@ struct Config {
|
|||||||
// If true, document links are reported for #include directives.
|
// If true, document links are reported for #include directives.
|
||||||
bool showDocumentLinksOnIncludes = true;
|
bool showDocumentLinksOnIncludes = true;
|
||||||
|
|
||||||
// If true, diagnostics from a full document parse will be reported.
|
|
||||||
bool diagnosticsOnParse = true;
|
|
||||||
|
|
||||||
// Version of the client. If undefined the version check is skipped. Used to
|
// Version of the client. If undefined the version check is skipped. Used to
|
||||||
// inform users their vscode client is too old and needs to be updated.
|
// inform users their vscode client is too old and needs to be updated.
|
||||||
optional<int> clientVersion;
|
optional<int> clientVersion;
|
||||||
@ -144,6 +141,23 @@ struct Config {
|
|||||||
std::vector<std::string> includeWhitelist;
|
std::vector<std::string> includeWhitelist;
|
||||||
} completion;
|
} completion;
|
||||||
|
|
||||||
|
struct Diagnostics {
|
||||||
|
// Like index.{whitelist,blacklist}, don't publish diagnostics to
|
||||||
|
// blacklisted files.
|
||||||
|
std::vector<std::string> blacklist;
|
||||||
|
|
||||||
|
// How often should cquery publish diagnostics in completion?
|
||||||
|
// -1: never
|
||||||
|
// 0: as often as possible
|
||||||
|
// xxx: at most every xxx milliseconds
|
||||||
|
int frequencyMs = 0;
|
||||||
|
|
||||||
|
// If true, diagnostics from a full document parse will be reported.
|
||||||
|
bool onParse = true;
|
||||||
|
|
||||||
|
std::vector<std::string> whitelist;
|
||||||
|
} diagnostics;
|
||||||
|
|
||||||
struct Index {
|
struct Index {
|
||||||
// Attempt to convert calls of make* functions to constructors based on
|
// Attempt to convert calls of make* functions to constructors based on
|
||||||
// hueristics.
|
// hueristics.
|
||||||
@ -210,6 +224,11 @@ MAKE_REFLECT_STRUCT(Config::Completion,
|
|||||||
includeMaxPathSize,
|
includeMaxPathSize,
|
||||||
includeSuffixWhitelist,
|
includeSuffixWhitelist,
|
||||||
includeWhitelist);
|
includeWhitelist);
|
||||||
|
MAKE_REFLECT_STRUCT(Config::Diagnostics,
|
||||||
|
blacklist,
|
||||||
|
frequencyMs,
|
||||||
|
onParse,
|
||||||
|
whitelist)
|
||||||
MAKE_REFLECT_STRUCT(Config::Index,
|
MAKE_REFLECT_STRUCT(Config::Index,
|
||||||
attributeMakeCallsToCtor,
|
attributeMakeCallsToCtor,
|
||||||
blacklist,
|
blacklist,
|
||||||
@ -233,13 +252,12 @@ MAKE_REFLECT_STRUCT(Config,
|
|||||||
|
|
||||||
showDocumentLinksOnIncludes,
|
showDocumentLinksOnIncludes,
|
||||||
|
|
||||||
diagnosticsOnParse,
|
|
||||||
|
|
||||||
clientVersion,
|
clientVersion,
|
||||||
|
|
||||||
client,
|
client,
|
||||||
codeLens,
|
codeLens,
|
||||||
completion,
|
completion,
|
||||||
|
diagnostics,
|
||||||
index,
|
index,
|
||||||
workspaceSymbol,
|
workspaceSymbol,
|
||||||
xref,
|
xref,
|
||||||
|
30
src/diagnostics_engine.cc
Normal file
30
src/diagnostics_engine.cc
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
#include "diagnostics_engine.h"
|
||||||
|
|
||||||
|
#include "queue_manager.h"
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
|
||||||
|
DiagnosticsEngine::DiagnosticsEngine(Config* config)
|
||||||
|
: match_(config->diagnostics.whitelist, config->diagnostics.blacklist) {}
|
||||||
|
|
||||||
|
void DiagnosticsEngine::Publish(WorkingFiles* working_files,
|
||||||
|
std::string path,
|
||||||
|
std::vector<lsDiagnostic> diagnostics) {
|
||||||
|
// Cache diagnostics so we can show fixits.
|
||||||
|
working_files->DoActionOnFile(path, [&](WorkingFile* working_file) {
|
||||||
|
if (working_file)
|
||||||
|
working_file->diagnostics_ = diagnostics;
|
||||||
|
});
|
||||||
|
|
||||||
|
int64_t now = std::chrono::duration_cast<std::chrono::milliseconds>(
|
||||||
|
std::chrono::high_resolution_clock::now().time_since_epoch()).count();
|
||||||
|
if (frequencyMs_ >= 0 && (nextPublish_ <= now || diagnostics.empty()) &&
|
||||||
|
match_.IsMatch(path)) {
|
||||||
|
nextPublish_ = now + frequencyMs_;
|
||||||
|
|
||||||
|
Out_TextDocumentPublishDiagnostics out;
|
||||||
|
out.params.uri = lsDocumentUri::FromPath(path);
|
||||||
|
out.params.diagnostics = diagnostics;
|
||||||
|
QueueManager::WriteStdout(IpcId::TextDocumentPublishDiagnostics, out);
|
||||||
|
}
|
||||||
|
}
|
17
src/diagnostics_engine.h
Normal file
17
src/diagnostics_engine.h
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
#include "lsp_diagnostic.h"
|
||||||
|
|
||||||
|
#include "match.h"
|
||||||
|
#include "working_files.h"
|
||||||
|
|
||||||
|
class DiagnosticsEngine {
|
||||||
|
int frequencyMs_ = 0;
|
||||||
|
GroupMatch match_;
|
||||||
|
int64_t nextPublish_ = 0;
|
||||||
|
|
||||||
|
public:
|
||||||
|
DiagnosticsEngine(Config* config);
|
||||||
|
void SetFrequencyMs(int ms) { frequencyMs_ = ms; }
|
||||||
|
void Publish(WorkingFiles* working_files,
|
||||||
|
std::string path,
|
||||||
|
std::vector<lsDiagnostic> diagnostics);
|
||||||
|
};
|
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
#include "cache_manager.h"
|
#include "cache_manager.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
#include "diagnostics_engine.h"
|
||||||
#include "iindexer.h"
|
#include "iindexer.h"
|
||||||
#include "import_manager.h"
|
#include "import_manager.h"
|
||||||
#include "lsp.h"
|
#include "lsp.h"
|
||||||
@ -340,6 +341,7 @@ std::vector<FileContents> PreloadFileContents(
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ParseFile(Config* config,
|
void ParseFile(Config* config,
|
||||||
|
DiagnosticsEngine* diag_engine,
|
||||||
WorkingFiles* working_files,
|
WorkingFiles* working_files,
|
||||||
FileConsumerSharedState* file_consumer_shared,
|
FileConsumerSharedState* file_consumer_shared,
|
||||||
TimestampManager* timestamp_manager,
|
TimestampManager* timestamp_manager,
|
||||||
@ -394,7 +396,8 @@ void ParseFile(Config* config,
|
|||||||
// to identify indexing problems. For interactive sessions, diagnostics are
|
// to identify indexing problems. For interactive sessions, diagnostics are
|
||||||
// handled by code completion.
|
// handled by code completion.
|
||||||
if (!request.is_interactive)
|
if (!request.is_interactive)
|
||||||
EmitDiagnostics(working_files, new_index->path, new_index->diagnostics_);
|
diag_engine->Publish(working_files, new_index->path,
|
||||||
|
new_index->diagnostics_);
|
||||||
|
|
||||||
// When main thread does IdMap request it will request the previous index if
|
// When main thread does IdMap request it will request the previous index if
|
||||||
// needed.
|
// needed.
|
||||||
@ -410,6 +413,7 @@ void ParseFile(Config* config,
|
|||||||
|
|
||||||
bool IndexMain_DoParse(
|
bool IndexMain_DoParse(
|
||||||
Config* config,
|
Config* config,
|
||||||
|
DiagnosticsEngine* diag_engine,
|
||||||
WorkingFiles* working_files,
|
WorkingFiles* working_files,
|
||||||
FileConsumerSharedState* file_consumer_shared,
|
FileConsumerSharedState* file_consumer_shared,
|
||||||
TimestampManager* timestamp_manager,
|
TimestampManager* timestamp_manager,
|
||||||
@ -424,9 +428,9 @@ bool IndexMain_DoParse(
|
|||||||
Project::Entry entry;
|
Project::Entry entry;
|
||||||
entry.filename = request->path;
|
entry.filename = request->path;
|
||||||
entry.args = request->args;
|
entry.args = request->args;
|
||||||
ParseFile(config, working_files, file_consumer_shared, timestamp_manager,
|
ParseFile(config, diag_engine, working_files, file_consumer_shared,
|
||||||
modification_timestamp_fetcher, import_manager, indexer,
|
timestamp_manager, modification_timestamp_fetcher, import_manager,
|
||||||
request.value(), entry);
|
indexer, request.value(), entry);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -582,6 +586,7 @@ void IndexWithTuFromCodeCompletion(
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Indexer_Main(Config* config,
|
void Indexer_Main(Config* config,
|
||||||
|
DiagnosticsEngine* diag_engine,
|
||||||
FileConsumerSharedState* file_consumer_shared,
|
FileConsumerSharedState* file_consumer_shared,
|
||||||
TimestampManager* timestamp_manager,
|
TimestampManager* timestamp_manager,
|
||||||
ImportManager* import_manager,
|
ImportManager* import_manager,
|
||||||
@ -609,9 +614,9 @@ void Indexer_Main(Config* config,
|
|||||||
// IndexMain_DoCreateIndexUpdate so we don't starve querydb from doing any
|
// IndexMain_DoCreateIndexUpdate so we don't starve querydb from doing any
|
||||||
// work. Running both also lets the user query the partially constructed
|
// work. Running both also lets the user query the partially constructed
|
||||||
// index.
|
// index.
|
||||||
did_work =
|
did_work = IndexMain_DoParse(config, diag_engine, working_files,
|
||||||
IndexMain_DoParse(config, working_files, file_consumer_shared,
|
file_consumer_shared, timestamp_manager,
|
||||||
timestamp_manager, &modification_timestamp_fetcher,
|
&modification_timestamp_fetcher,
|
||||||
import_manager, indexer.get()) ||
|
import_manager, indexer.get()) ||
|
||||||
did_work;
|
did_work;
|
||||||
|
|
||||||
@ -767,9 +772,10 @@ TEST_SUITE("ImportPipeline") {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool PumpOnce() {
|
bool PumpOnce() {
|
||||||
return IndexMain_DoParse(
|
return IndexMain_DoParse(&config, &diag_engine, &working_files,
|
||||||
&config, &working_files, &file_consumer_shared, ×tamp_manager,
|
&file_consumer_shared, ×tamp_manager,
|
||||||
&modification_timestamp_fetcher, &import_manager, indexer.get());
|
&modification_timestamp_fetcher, &import_manager,
|
||||||
|
indexer.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
void MakeRequest(const std::string& path,
|
void MakeRequest(const std::string& path,
|
||||||
@ -786,6 +792,7 @@ TEST_SUITE("ImportPipeline") {
|
|||||||
|
|
||||||
QueueManager* queue = nullptr;
|
QueueManager* queue = nullptr;
|
||||||
Config config;
|
Config config;
|
||||||
|
DiagnosticsEngine diag_engine{&config};
|
||||||
WorkingFiles working_files;
|
WorkingFiles working_files;
|
||||||
FileConsumerSharedState file_consumer_shared;
|
FileConsumerSharedState file_consumer_shared;
|
||||||
TimestampManager timestamp_manager;
|
TimestampManager timestamp_manager;
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
|
|
||||||
struct ClangTranslationUnit;
|
struct ClangTranslationUnit;
|
||||||
struct Config;
|
struct Config;
|
||||||
|
class DiagnosticsEngine;
|
||||||
struct FileConsumerSharedState;
|
struct FileConsumerSharedState;
|
||||||
struct ImportManager;
|
struct ImportManager;
|
||||||
struct MultiQueueWaiter;
|
struct MultiQueueWaiter;
|
||||||
@ -35,6 +36,7 @@ void IndexWithTuFromCodeCompletion(
|
|||||||
const std::vector<std::string>& args);
|
const std::vector<std::string>& args);
|
||||||
|
|
||||||
void Indexer_Main(Config* config,
|
void Indexer_Main(Config* config,
|
||||||
|
DiagnosticsEngine* diag_engine,
|
||||||
FileConsumerSharedState* file_consumer_shared,
|
FileConsumerSharedState* file_consumer_shared,
|
||||||
TimestampManager* timestamp_manager,
|
TimestampManager* timestamp_manager,
|
||||||
ImportManager* import_manager,
|
ImportManager* import_manager,
|
||||||
|
@ -1,20 +1,5 @@
|
|||||||
#include "lsp_diagnostic.h"
|
#include "lsp_diagnostic.h"
|
||||||
|
|
||||||
|
#include "match.h"
|
||||||
#include "queue_manager.h"
|
#include "queue_manager.h"
|
||||||
#include "working_files.h"
|
#include "working_files.h"
|
||||||
|
|
||||||
void EmitDiagnostics(WorkingFiles* working_files,
|
|
||||||
std::string path,
|
|
||||||
std::vector<lsDiagnostic> diagnostics) {
|
|
||||||
// Emit diagnostics.
|
|
||||||
Out_TextDocumentPublishDiagnostics out;
|
|
||||||
out.params.uri = lsDocumentUri::FromPath(path);
|
|
||||||
out.params.diagnostics = diagnostics;
|
|
||||||
QueueManager::WriteStdout(IpcId::TextDocumentPublishDiagnostics, out);
|
|
||||||
|
|
||||||
// Cache diagnostics so we can show fixits.
|
|
||||||
working_files->DoActionOnFile(path, [&](WorkingFile* working_file) {
|
|
||||||
if (working_file)
|
|
||||||
working_file->diagnostics_ = diagnostics;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
@ -99,8 +99,3 @@ void Reflect(TVisitor& visitor, Out_TextDocumentPublishDiagnostics& value) {
|
|||||||
MAKE_REFLECT_STRUCT(Out_TextDocumentPublishDiagnostics::Params,
|
MAKE_REFLECT_STRUCT(Out_TextDocumentPublishDiagnostics::Params,
|
||||||
uri,
|
uri,
|
||||||
diagnostics);
|
diagnostics);
|
||||||
|
|
||||||
struct WorkingFiles;
|
|
||||||
void EmitDiagnostics(WorkingFiles* working_files,
|
|
||||||
std::string path,
|
|
||||||
std::vector<lsDiagnostic> diagnostics);
|
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
struct ClangCompleteManager;
|
struct ClangCompleteManager;
|
||||||
struct CodeCompleteCache;
|
struct CodeCompleteCache;
|
||||||
struct Config;
|
struct Config;
|
||||||
|
class DiagnosticsEngine;
|
||||||
struct FileConsumerSharedState;
|
struct FileConsumerSharedState;
|
||||||
struct ImportManager;
|
struct ImportManager;
|
||||||
struct ImportPipelineStatus;
|
struct ImportPipelineStatus;
|
||||||
@ -72,6 +73,7 @@ struct MessageHandler {
|
|||||||
QueryDatabase* db = nullptr;
|
QueryDatabase* db = nullptr;
|
||||||
MultiQueueWaiter* waiter = nullptr;
|
MultiQueueWaiter* waiter = nullptr;
|
||||||
Project* project = nullptr;
|
Project* project = nullptr;
|
||||||
|
DiagnosticsEngine* diag_engine = nullptr;
|
||||||
FileConsumerSharedState* file_consumer_shared = nullptr;
|
FileConsumerSharedState* file_consumer_shared = nullptr;
|
||||||
ImportManager* import_manager = nullptr;
|
ImportManager* import_manager = nullptr;
|
||||||
ImportPipelineStatus* import_pipeline_status = nullptr;
|
ImportPipelineStatus* import_pipeline_status = nullptr;
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
#include "cache_manager.h"
|
#include "cache_manager.h"
|
||||||
|
#include "diagnostics_engine.h"
|
||||||
#include "import_pipeline.h"
|
#include "import_pipeline.h"
|
||||||
#include "include_complete.h"
|
#include "include_complete.h"
|
||||||
#include "message_handler.h"
|
#include "message_handler.h"
|
||||||
@ -581,6 +582,7 @@ struct InitializeHandler : BaseMessageHandler<Ipc_InitializeRequest> {
|
|||||||
EscapeFileName(config->projectRoot));
|
EscapeFileName(config->projectRoot));
|
||||||
|
|
||||||
Timer time;
|
Timer time;
|
||||||
|
diag_engine->SetFrequencyMs(config->diagnostics.frequencyMs);
|
||||||
|
|
||||||
// Open up / load the project.
|
// Open up / load the project.
|
||||||
project->Load(config, config->extraClangArguments,
|
project->Load(config, config->extraClangArguments,
|
||||||
@ -604,9 +606,9 @@ struct InitializeHandler : BaseMessageHandler<Ipc_InitializeRequest> {
|
|||||||
LOG_S(INFO) << "Starting " << config->index.threads << " indexers";
|
LOG_S(INFO) << "Starting " << config->index.threads << " indexers";
|
||||||
for (int i = 0; i < config->index.threads; ++i) {
|
for (int i = 0; i < config->index.threads; ++i) {
|
||||||
WorkThread::StartThread("indexer" + std::to_string(i), [=]() {
|
WorkThread::StartThread("indexer" + std::to_string(i), [=]() {
|
||||||
Indexer_Main(config, file_consumer_shared, timestamp_manager,
|
Indexer_Main(config, diag_engine, file_consumer_shared,
|
||||||
import_manager, import_pipeline_status, project,
|
timestamp_manager, import_manager,
|
||||||
working_files, waiter);
|
import_pipeline_status, project, working_files, waiter);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user