Unify save and save_as for local worktrees
This fixes state propagation bugs due to missing RPC calls in save_as.
This commit is contained in:
parent
3a7cfc3901
commit
cdf64b6cad
3 changed files with 42 additions and 41 deletions
|
@ -2326,19 +2326,27 @@ async fn test_propagate_saves_and_fs_changes(
|
||||||
assert!(buffer.file().is_none());
|
assert!(buffer.file().is_none());
|
||||||
});
|
});
|
||||||
|
|
||||||
|
new_buffer_a.update(cx_a, |buffer, cx| {
|
||||||
|
buffer.edit([(0..0, "ok")], None, cx);
|
||||||
|
});
|
||||||
project_a
|
project_a
|
||||||
.update(cx_a, |project, cx| {
|
.update(cx_a, |project, cx| {
|
||||||
project.save_buffer_as(new_buffer_a, "/a/file3.rs".into(), cx)
|
project.save_buffer_as(new_buffer_a.clone(), "/a/file3.rs".into(), cx)
|
||||||
})
|
})
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
deterministic.run_until_parked();
|
deterministic.run_until_parked();
|
||||||
new_buffer_b.read_with(cx_b, |buffer, _| {
|
new_buffer_b.read_with(cx_b, |buffer_b, _| {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
buffer.file().unwrap().path().as_ref(),
|
buffer_b.file().unwrap().path().as_ref(),
|
||||||
Path::new("file3.rs")
|
Path::new("file3.rs")
|
||||||
);
|
);
|
||||||
|
|
||||||
|
new_buffer_a.read_with(cx_a, |buffer_a, _| {
|
||||||
|
assert_eq!(buffer_b.saved_mtime(), buffer_a.saved_mtime());
|
||||||
|
assert_eq!(buffer_b.saved_version(), buffer_a.saved_version());
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2909,7 +2917,9 @@ async fn test_buffer_conflict_after_save(
|
||||||
assert!(!buf.has_conflict());
|
assert!(!buf.has_conflict());
|
||||||
});
|
});
|
||||||
|
|
||||||
cx_b.update(|cx| Project::save_buffer(buffer_b.clone(), cx)).await.unwrap();
|
cx_b.update(|cx| Project::save_buffer(buffer_b.clone(), cx))
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
cx_a.foreground().forbid_parking();
|
cx_a.foreground().forbid_parking();
|
||||||
buffer_b.read_with(cx_b, |buffer_b, _| assert!(!buffer_b.is_dirty()));
|
buffer_b.read_with(cx_b, |buffer_b, _| assert!(!buffer_b.is_dirty()));
|
||||||
buffer_b.read_with(cx_b, |buf, _| {
|
buffer_b.read_with(cx_b, |buf, _| {
|
||||||
|
|
|
@ -1451,7 +1451,7 @@ impl Project {
|
||||||
let worktree = file.worktree.clone();
|
let worktree = file.worktree.clone();
|
||||||
let path = file.path.clone();
|
let path = file.path.clone();
|
||||||
worktree.update(cx, |worktree, cx| match worktree {
|
worktree.update(cx, |worktree, cx| match worktree {
|
||||||
Worktree::Local(worktree) => worktree.save_buffer(buffer, path, cx),
|
Worktree::Local(worktree) => worktree.save_buffer(buffer, path, false, cx),
|
||||||
Worktree::Remote(worktree) => worktree.save_buffer(buffer, cx),
|
Worktree::Remote(worktree) => worktree.save_buffer(buffer, cx),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -1474,7 +1474,9 @@ impl Project {
|
||||||
let (worktree, path) = worktree_task.await?;
|
let (worktree, path) = worktree_task.await?;
|
||||||
worktree
|
worktree
|
||||||
.update(&mut cx, |worktree, cx| match worktree {
|
.update(&mut cx, |worktree, cx| match worktree {
|
||||||
Worktree::Local(worktree) => worktree.save_buffer_as(buffer.clone(), path, cx),
|
Worktree::Local(worktree) => {
|
||||||
|
worktree.save_buffer(buffer.clone(), path.into(), true, cx)
|
||||||
|
}
|
||||||
Worktree::Remote(_) => panic!("cannot remote buffers as new files"),
|
Worktree::Remote(_) => panic!("cannot remote buffers as new files"),
|
||||||
})
|
})
|
||||||
.await?;
|
.await?;
|
||||||
|
|
|
@ -728,8 +728,10 @@ impl LocalWorktree {
|
||||||
&self,
|
&self,
|
||||||
buffer_handle: ModelHandle<Buffer>,
|
buffer_handle: ModelHandle<Buffer>,
|
||||||
path: Arc<Path>,
|
path: Arc<Path>,
|
||||||
|
replace_file: bool,
|
||||||
cx: &mut ModelContext<Worktree>,
|
cx: &mut ModelContext<Worktree>,
|
||||||
) -> Task<Result<(clock::Global, RopeFingerprint, SystemTime)>> {
|
) -> Task<Result<(clock::Global, RopeFingerprint, SystemTime)>> {
|
||||||
|
let handle = cx.handle();
|
||||||
let buffer = buffer_handle.read(cx);
|
let buffer = buffer_handle.read(cx);
|
||||||
|
|
||||||
let rpc = self.client.clone();
|
let rpc = self.client.clone();
|
||||||
|
@ -742,53 +744,40 @@ impl LocalWorktree {
|
||||||
let save = self.write_file(path, text, buffer.line_ending(), cx);
|
let save = self.write_file(path, text, buffer.line_ending(), cx);
|
||||||
|
|
||||||
cx.as_mut().spawn(|mut cx| async move {
|
cx.as_mut().spawn(|mut cx| async move {
|
||||||
let mtime = save.await?.mtime;
|
let entry = save.await?;
|
||||||
|
|
||||||
if let Some(project_id) = project_id {
|
if let Some(project_id) = project_id {
|
||||||
rpc.send(proto::BufferSaved {
|
rpc.send(proto::BufferSaved {
|
||||||
project_id,
|
project_id,
|
||||||
buffer_id,
|
buffer_id,
|
||||||
version: serialize_version(&version),
|
version: serialize_version(&version),
|
||||||
mtime: Some(mtime.into()),
|
mtime: Some(entry.mtime.into()),
|
||||||
fingerprint: serialize_fingerprint(fingerprint),
|
fingerprint: serialize_fingerprint(fingerprint),
|
||||||
})?;
|
})?;
|
||||||
}
|
}
|
||||||
buffer_handle.update(&mut cx, |buffer, cx| {
|
|
||||||
buffer.did_save(version.clone(), fingerprint, mtime, None, cx);
|
|
||||||
});
|
|
||||||
anyhow::Ok((version, fingerprint, mtime))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn save_buffer_as(
|
|
||||||
&self,
|
|
||||||
buffer_handle: ModelHandle<Buffer>,
|
|
||||||
path: impl Into<Arc<Path>>,
|
|
||||||
cx: &mut ModelContext<Worktree>,
|
|
||||||
) -> Task<Result<()>> {
|
|
||||||
let handle = cx.handle();
|
|
||||||
let buffer = buffer_handle.read(cx);
|
|
||||||
|
|
||||||
let text = buffer.as_rope().clone();
|
|
||||||
let fingerprint = text.fingerprint();
|
|
||||||
let version = buffer.version();
|
|
||||||
let save = self.write_file(path, text, buffer.line_ending(), cx);
|
|
||||||
|
|
||||||
cx.as_mut().spawn(|mut cx| async move {
|
|
||||||
let entry = save.await?;
|
|
||||||
let file = File {
|
|
||||||
entry_id: entry.id,
|
|
||||||
worktree: handle,
|
|
||||||
path: entry.path,
|
|
||||||
mtime: entry.mtime,
|
|
||||||
is_local: true,
|
|
||||||
is_deleted: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
buffer_handle.update(&mut cx, |buffer, cx| {
|
buffer_handle.update(&mut cx, |buffer, cx| {
|
||||||
buffer.did_save(version, fingerprint, file.mtime, Some(Arc::new(file)), cx);
|
buffer.did_save(
|
||||||
|
version.clone(),
|
||||||
|
fingerprint,
|
||||||
|
entry.mtime,
|
||||||
|
if replace_file {
|
||||||
|
Some(Arc::new(File {
|
||||||
|
entry_id: entry.id,
|
||||||
|
worktree: handle,
|
||||||
|
path: entry.path,
|
||||||
|
mtime: entry.mtime,
|
||||||
|
is_local: true,
|
||||||
|
is_deleted: false,
|
||||||
|
}))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
},
|
||||||
|
cx,
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
Ok(())
|
Ok((version, fingerprint, entry.mtime))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue