Avoid circular model update when sending diagnostics operations
This commit is contained in:
parent
f0353d6aba
commit
61e06487b7
2 changed files with 76 additions and 64 deletions
|
@ -679,7 +679,7 @@ impl Buffer {
|
||||||
version: Option<i32>,
|
version: Option<i32>,
|
||||||
mut diagnostics: Vec<lsp::Diagnostic>,
|
mut diagnostics: Vec<lsp::Diagnostic>,
|
||||||
cx: &mut ModelContext<Self>,
|
cx: &mut ModelContext<Self>,
|
||||||
) -> Result<()> {
|
) -> Result<Operation> {
|
||||||
let version = version.map(|version| version as usize);
|
let version = version.map(|version| version as usize);
|
||||||
let content = if let Some(version) = version {
|
let content = if let Some(version) = version {
|
||||||
let language_server = self.language_server.as_mut().unwrap();
|
let language_server = self.language_server.as_mut().unwrap();
|
||||||
|
@ -768,9 +768,8 @@ impl Buffer {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.diagnostics_update_count += 1;
|
self.diagnostics_update_count += 1;
|
||||||
self.send_operation(Operation::UpdateDiagnostics(self.diagnostics.clone()), cx);
|
|
||||||
cx.notify();
|
cx.notify();
|
||||||
Ok(())
|
Ok(Operation::UpdateDiagnostics(self.diagnostics.clone()))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn diagnostics_in_range<'a, T: 'a + ToOffset>(
|
pub fn diagnostics_in_range<'a, T: 'a + ToOffset>(
|
||||||
|
|
|
@ -649,6 +649,79 @@ impl Worktree {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn update_diagnostics(
|
||||||
|
&mut self,
|
||||||
|
params: lsp::PublishDiagnosticsParams,
|
||||||
|
cx: &mut ModelContext<Worktree>,
|
||||||
|
) -> Result<()> {
|
||||||
|
let this = self.as_local_mut().ok_or_else(|| anyhow!("not local"))?;
|
||||||
|
let file_path = params
|
||||||
|
.uri
|
||||||
|
.to_file_path()
|
||||||
|
.map_err(|_| anyhow!("URI is not a file"))?
|
||||||
|
.strip_prefix(&this.abs_path)
|
||||||
|
.context("path is not within worktree")?
|
||||||
|
.to_owned();
|
||||||
|
|
||||||
|
for buffer in this.open_buffers.values() {
|
||||||
|
if let Some(buffer) = buffer.upgrade(cx) {
|
||||||
|
if buffer
|
||||||
|
.read(cx)
|
||||||
|
.file()
|
||||||
|
.map_or(false, |file| file.path().as_ref() == file_path)
|
||||||
|
{
|
||||||
|
let (remote_id, operation) = buffer.update(cx, |buffer, cx| {
|
||||||
|
(
|
||||||
|
buffer.remote_id(),
|
||||||
|
buffer.update_diagnostics(params.version, params.diagnostics, cx),
|
||||||
|
)
|
||||||
|
});
|
||||||
|
self.send_buffer_update(remote_id, operation?, cx);
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.diagnostics.insert(file_path, params.diagnostics);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn send_buffer_update(
|
||||||
|
&mut self,
|
||||||
|
buffer_id: u64,
|
||||||
|
operation: Operation,
|
||||||
|
cx: &mut ModelContext<Self>,
|
||||||
|
) {
|
||||||
|
if let Some((rpc, remote_id)) = match self {
|
||||||
|
Worktree::Local(worktree) => worktree
|
||||||
|
.remote_id
|
||||||
|
.borrow()
|
||||||
|
.map(|id| (worktree.rpc.clone(), id)),
|
||||||
|
Worktree::Remote(worktree) => Some((worktree.client.clone(), worktree.remote_id)),
|
||||||
|
} {
|
||||||
|
cx.spawn(|worktree, mut cx| async move {
|
||||||
|
if let Err(error) = rpc
|
||||||
|
.request(proto::UpdateBuffer {
|
||||||
|
worktree_id: remote_id,
|
||||||
|
buffer_id,
|
||||||
|
operations: vec![language::proto::serialize_operation(&operation)],
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
worktree.update(&mut cx, |worktree, _| {
|
||||||
|
log::error!("error sending buffer operation: {}", error);
|
||||||
|
match worktree {
|
||||||
|
Worktree::Local(t) => &mut t.queued_operations,
|
||||||
|
Worktree::Remote(t) => &mut t.queued_operations,
|
||||||
|
}
|
||||||
|
.push((buffer_id, operation));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.detach();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Deref for Worktree {
|
impl Deref for Worktree {
|
||||||
|
@ -836,7 +909,6 @@ impl LocalWorktree {
|
||||||
while let Ok(diagnostics) = diagnostics_rx.recv().await {
|
while let Ok(diagnostics) = diagnostics_rx.recv().await {
|
||||||
if let Some(handle) = cx.read(|cx| this.upgrade(cx)) {
|
if let Some(handle) = cx.read(|cx| this.upgrade(cx)) {
|
||||||
handle.update(&mut cx, |this, cx| {
|
handle.update(&mut cx, |this, cx| {
|
||||||
let this = this.as_local_mut().unwrap();
|
|
||||||
this.update_diagnostics(diagnostics, cx).log_err();
|
this.update_diagnostics(diagnostics, cx).log_err();
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
|
@ -1200,38 +1272,6 @@ impl LocalWorktree {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn update_diagnostics(
|
|
||||||
&mut self,
|
|
||||||
params: lsp::PublishDiagnosticsParams,
|
|
||||||
cx: &mut ModelContext<Worktree>,
|
|
||||||
) -> Result<()> {
|
|
||||||
let file_path = params
|
|
||||||
.uri
|
|
||||||
.to_file_path()
|
|
||||||
.map_err(|_| anyhow!("URI is not a file"))?
|
|
||||||
.strip_prefix(&self.abs_path)
|
|
||||||
.context("path is not within worktree")?
|
|
||||||
.to_owned();
|
|
||||||
|
|
||||||
for buffer in self.open_buffers.values() {
|
|
||||||
if let Some(buffer) = buffer.upgrade(cx) {
|
|
||||||
if buffer
|
|
||||||
.read(cx)
|
|
||||||
.file()
|
|
||||||
.map_or(false, |file| file.path().as_ref() == file_path)
|
|
||||||
{
|
|
||||||
buffer.update(cx, |buffer, cx| {
|
|
||||||
buffer.update_diagnostics(params.version, params.diagnostics, cx)
|
|
||||||
})?;
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
self.diagnostics.insert(file_path, params.diagnostics);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build_gitignore(abs_path: &Path, fs: &dyn Fs) -> Result<Gitignore> {
|
fn build_gitignore(abs_path: &Path, fs: &dyn Fs) -> Result<Gitignore> {
|
||||||
|
@ -1932,34 +1972,7 @@ impl language::File for File {
|
||||||
|
|
||||||
fn buffer_updated(&self, buffer_id: u64, operation: Operation, cx: &mut MutableAppContext) {
|
fn buffer_updated(&self, buffer_id: u64, operation: Operation, cx: &mut MutableAppContext) {
|
||||||
self.worktree.update(cx, |worktree, cx| {
|
self.worktree.update(cx, |worktree, cx| {
|
||||||
if let Some((rpc, remote_id)) = match worktree {
|
worktree.send_buffer_update(buffer_id, operation, cx);
|
||||||
Worktree::Local(worktree) => worktree
|
|
||||||
.remote_id
|
|
||||||
.borrow()
|
|
||||||
.map(|id| (worktree.rpc.clone(), id)),
|
|
||||||
Worktree::Remote(worktree) => Some((worktree.client.clone(), worktree.remote_id)),
|
|
||||||
} {
|
|
||||||
cx.spawn(|worktree, mut cx| async move {
|
|
||||||
if let Err(error) = rpc
|
|
||||||
.request(proto::UpdateBuffer {
|
|
||||||
worktree_id: remote_id,
|
|
||||||
buffer_id,
|
|
||||||
operations: vec![language::proto::serialize_operation(&operation)],
|
|
||||||
})
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
worktree.update(&mut cx, |worktree, _| {
|
|
||||||
log::error!("error sending buffer operation: {}", error);
|
|
||||||
match worktree {
|
|
||||||
Worktree::Local(t) => &mut t.queued_operations,
|
|
||||||
Worktree::Remote(t) => &mut t.queued_operations,
|
|
||||||
}
|
|
||||||
.push((buffer_id, operation));
|
|
||||||
});
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue