mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-27 20:35:09 +00:00
Add support for incremental text synchronization
This commit is contained in:
parent
1cde354c35
commit
1a2d4e2921
4 changed files with 125 additions and 13 deletions
4
Cargo.lock
generated
4
Cargo.lock
generated
|
@ -1193,9 +1193,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ra_vfs"
|
name = "ra_vfs"
|
||||||
version = "0.5.3"
|
version = "0.6.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "58a265769d5e5655345a9fcbd870a1a7c3658558c0d8efaed79e0669358f46b8"
|
checksum = "fcaa5615f420134aea7667253db101d03a5c5f300eac607872dc2a36407b2ac9"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"crossbeam-channel",
|
"crossbeam-channel",
|
||||||
"jod-thread",
|
"jod-thread",
|
||||||
|
|
|
@ -39,7 +39,7 @@ ra_prof = { path = "../ra_prof" }
|
||||||
ra_project_model = { path = "../ra_project_model" }
|
ra_project_model = { path = "../ra_project_model" }
|
||||||
ra_syntax = { path = "../ra_syntax" }
|
ra_syntax = { path = "../ra_syntax" }
|
||||||
ra_text_edit = { path = "../ra_text_edit" }
|
ra_text_edit = { path = "../ra_text_edit" }
|
||||||
ra_vfs = "0.5.2"
|
ra_vfs = "0.6.0"
|
||||||
|
|
||||||
# This should only be used in CLI
|
# This should only be used in CLI
|
||||||
ra_db = { path = "../ra_db" }
|
ra_db = { path = "../ra_db" }
|
||||||
|
|
|
@ -16,7 +16,7 @@ pub fn server_capabilities() -> ServerCapabilities {
|
||||||
ServerCapabilities {
|
ServerCapabilities {
|
||||||
text_document_sync: Some(TextDocumentSyncCapability::Options(TextDocumentSyncOptions {
|
text_document_sync: Some(TextDocumentSyncCapability::Options(TextDocumentSyncOptions {
|
||||||
open_close: Some(true),
|
open_close: Some(true),
|
||||||
change: Some(TextDocumentSyncKind::Full),
|
change: Some(TextDocumentSyncKind::Incremental),
|
||||||
will_save: None,
|
will_save: None,
|
||||||
will_save_wait_until: None,
|
will_save_wait_until: None,
|
||||||
save: Some(SaveOptions::default()),
|
save: Some(SaveOptions::default()),
|
||||||
|
|
|
@ -6,9 +6,12 @@ mod subscriptions;
|
||||||
pub(crate) mod pending_requests;
|
pub(crate) mod pending_requests;
|
||||||
|
|
||||||
use std::{
|
use std::{
|
||||||
|
borrow::Cow,
|
||||||
env,
|
env,
|
||||||
error::Error,
|
error::Error,
|
||||||
fmt, panic,
|
fmt,
|
||||||
|
ops::Range,
|
||||||
|
panic,
|
||||||
path::PathBuf,
|
path::PathBuf,
|
||||||
sync::Arc,
|
sync::Arc,
|
||||||
time::{Duration, Instant},
|
time::{Duration, Instant},
|
||||||
|
@ -18,11 +21,12 @@ use crossbeam_channel::{never, select, unbounded, RecvError, Sender};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use lsp_server::{Connection, ErrorCode, Message, Notification, Request, RequestId, Response};
|
use lsp_server::{Connection, ErrorCode, Message, Notification, Request, RequestId, Response};
|
||||||
use lsp_types::{
|
use lsp_types::{
|
||||||
NumberOrString, WorkDoneProgress, WorkDoneProgressBegin, WorkDoneProgressCreateParams,
|
DidChangeTextDocumentParams, NumberOrString, TextDocumentContentChangeEvent, WorkDoneProgress,
|
||||||
WorkDoneProgressEnd, WorkDoneProgressReport,
|
WorkDoneProgressBegin, WorkDoneProgressCreateParams, WorkDoneProgressEnd,
|
||||||
|
WorkDoneProgressReport,
|
||||||
};
|
};
|
||||||
use ra_flycheck::{url_from_path_with_drive_lowercasing, CheckTask};
|
use ra_flycheck::{url_from_path_with_drive_lowercasing, CheckTask};
|
||||||
use ra_ide::{Canceled, FileId, LibraryData, SourceRootId};
|
use ra_ide::{Canceled, FileId, LibraryData, LineIndex, SourceRootId};
|
||||||
use ra_prof::profile;
|
use ra_prof::profile;
|
||||||
use ra_project_model::{PackageRoot, ProjectWorkspace};
|
use ra_project_model::{PackageRoot, ProjectWorkspace};
|
||||||
use ra_vfs::{VfsFile, VfsTask, Watch};
|
use ra_vfs::{VfsFile, VfsTask, Watch};
|
||||||
|
@ -33,6 +37,7 @@ use threadpool::ThreadPool;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
config::{Config, FilesWatcher},
|
config::{Config, FilesWatcher},
|
||||||
|
conv::{ConvWith, TryConvWith},
|
||||||
diagnostics::DiagnosticTask,
|
diagnostics::DiagnosticTask,
|
||||||
main_loop::{
|
main_loop::{
|
||||||
pending_requests::{PendingRequest, PendingRequests},
|
pending_requests::{PendingRequest, PendingRequests},
|
||||||
|
@ -579,12 +584,16 @@ fn on_notification(
|
||||||
Err(not) => not,
|
Err(not) => not,
|
||||||
};
|
};
|
||||||
let not = match notification_cast::<req::DidChangeTextDocument>(not) {
|
let not = match notification_cast::<req::DidChangeTextDocument>(not) {
|
||||||
Ok(mut params) => {
|
Ok(params) => {
|
||||||
let uri = params.text_document.uri;
|
let DidChangeTextDocumentParams { text_document, content_changes } = params;
|
||||||
|
let world = state.snapshot();
|
||||||
|
let file_id = text_document.try_conv_with(&world)?;
|
||||||
|
let line_index = world.analysis().file_line_index(file_id)?;
|
||||||
|
let uri = text_document.uri;
|
||||||
let path = uri.to_file_path().map_err(|()| format!("invalid uri: {}", uri))?;
|
let path = uri.to_file_path().map_err(|()| format!("invalid uri: {}", uri))?;
|
||||||
let text =
|
state.vfs.write().change_file_overlay(&path, |old_text| {
|
||||||
params.content_changes.pop().ok_or_else(|| "empty changes".to_string())?.text;
|
apply_document_changes(old_text, Cow::Borrowed(&line_index), content_changes);
|
||||||
state.vfs.write().change_file_overlay(path.as_path(), text);
|
});
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
Err(not) => not,
|
Err(not) => not,
|
||||||
|
@ -653,6 +662,48 @@ fn on_notification(
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn apply_document_changes(
|
||||||
|
old_text: &mut String,
|
||||||
|
mut line_index: Cow<'_, LineIndex>,
|
||||||
|
content_changes: Vec<TextDocumentContentChangeEvent>,
|
||||||
|
) {
|
||||||
|
// The changes we got must be applied sequentially, but can cross lines so we
|
||||||
|
// have to keep our line index updated.
|
||||||
|
// Some clients (e.g. Code) sort the ranges in reverse. As an optimization, we
|
||||||
|
// remember the last valid line in the index and only rebuild it if needed.
|
||||||
|
enum IndexValid {
|
||||||
|
All,
|
||||||
|
UpToLine(u64),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl IndexValid {
|
||||||
|
fn covers(&self, line: u64) -> bool {
|
||||||
|
match *self {
|
||||||
|
IndexValid::UpToLine(to) => to >= line,
|
||||||
|
_ => true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut index_valid = IndexValid::All;
|
||||||
|
for change in content_changes {
|
||||||
|
match change.range {
|
||||||
|
Some(range) => {
|
||||||
|
if !index_valid.covers(range.start.line) {
|
||||||
|
line_index = Cow::Owned(LineIndex::new(&old_text));
|
||||||
|
}
|
||||||
|
index_valid = IndexValid::UpToLine(range.start.line);
|
||||||
|
let range = range.conv_with(&line_index);
|
||||||
|
old_text.replace_range(Range::<usize>::from(range), &change.text);
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
*old_text = change.text;
|
||||||
|
index_valid = IndexValid::UpToLine(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn on_check_task(
|
fn on_check_task(
|
||||||
task: CheckTask,
|
task: CheckTask,
|
||||||
world_state: &mut WorldState,
|
world_state: &mut WorldState,
|
||||||
|
@ -958,3 +1009,64 @@ where
|
||||||
{
|
{
|
||||||
Request::new(id, R::METHOD.to_string(), params)
|
Request::new(id, R::METHOD.to_string(), params)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use std::borrow::Cow;
|
||||||
|
|
||||||
|
use lsp_types::{Position, Range, TextDocumentContentChangeEvent};
|
||||||
|
use ra_ide::LineIndex;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn apply_document_changes() {
|
||||||
|
fn run(text: &mut String, changes: Vec<TextDocumentContentChangeEvent>) {
|
||||||
|
let line_index = Cow::Owned(LineIndex::new(&text));
|
||||||
|
super::apply_document_changes(text, line_index, changes);
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! c {
|
||||||
|
[$($sl:expr, $sc:expr; $el:expr, $ec:expr => $text:expr),+] => {
|
||||||
|
vec![$(TextDocumentContentChangeEvent {
|
||||||
|
range: Some(Range {
|
||||||
|
start: Position { line: $sl, character: $sc },
|
||||||
|
end: Position { line: $el, character: $ec },
|
||||||
|
}),
|
||||||
|
range_length: None,
|
||||||
|
text: String::from($text),
|
||||||
|
}),+]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut text = String::new();
|
||||||
|
run(&mut text, vec![]);
|
||||||
|
assert_eq!(text, "");
|
||||||
|
run(
|
||||||
|
&mut text,
|
||||||
|
vec![TextDocumentContentChangeEvent {
|
||||||
|
range: None,
|
||||||
|
range_length: None,
|
||||||
|
text: String::from("the"),
|
||||||
|
}],
|
||||||
|
);
|
||||||
|
assert_eq!(text, "the");
|
||||||
|
run(&mut text, c![0, 3; 0, 3 => " quick"]);
|
||||||
|
assert_eq!(text, "the quick");
|
||||||
|
run(&mut text, c![0, 0; 0, 4 => "", 0, 5; 0, 5 => " foxes"]);
|
||||||
|
assert_eq!(text, "quick foxes");
|
||||||
|
run(&mut text, c![0, 11; 0, 11 => "\ndream"]);
|
||||||
|
assert_eq!(text, "quick foxes\ndream");
|
||||||
|
run(&mut text, c![1, 0; 1, 0 => "have "]);
|
||||||
|
assert_eq!(text, "quick foxes\nhave dream");
|
||||||
|
run(&mut text, c![0, 0; 0, 0 => "the ", 1, 4; 1, 4 => " quiet", 1, 16; 1, 16 => "s\n"]);
|
||||||
|
assert_eq!(text, "the quick foxes\nhave quiet dreams\n");
|
||||||
|
run(&mut text, c![0, 15; 0, 15 => "\n", 2, 17; 2, 17 => "\n"]);
|
||||||
|
assert_eq!(text, "the quick foxes\n\nhave quiet dreams\n\n");
|
||||||
|
run(
|
||||||
|
&mut text,
|
||||||
|
c![1, 0; 1, 0 => "DREAM", 2, 0; 2, 0 => "they ", 3, 0; 3, 0 => "DON'T THEY?"],
|
||||||
|
);
|
||||||
|
assert_eq!(text, "the quick foxes\nDREAM\nthey have quiet dreams\nDON'T THEY?\n");
|
||||||
|
run(&mut text, c![0, 10; 1, 5 => "", 2, 0; 2, 12 => ""]);
|
||||||
|
assert_eq!(text, "the quick \nthey have quiet dreams\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue