2019-01-06 11:36:22 -06:00
|
|
|
use std::{collections::HashSet, fs};
|
2018-12-18 07:38:05 -06:00
|
|
|
|
2019-01-06 11:36:22 -06:00
|
|
|
use flexi_logger::Logger;
|
2018-12-18 07:38:05 -06:00
|
|
|
use ra_vfs::{Vfs, VfsChange};
|
2019-01-06 11:36:22 -06:00
|
|
|
use tempfile::tempdir;
|
2018-12-18 07:38:05 -06:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_vfs_works() -> std::io::Result<()> {
|
2019-01-06 11:36:22 -06:00
|
|
|
Logger::with_str("debug").start().unwrap();
|
|
|
|
|
2018-12-18 07:38:05 -06:00
|
|
|
let files = [
|
|
|
|
("a/foo.rs", "hello"),
|
|
|
|
("a/bar.rs", "world"),
|
|
|
|
("a/b/baz.rs", "nested hello"),
|
|
|
|
];
|
|
|
|
|
|
|
|
let dir = tempdir()?;
|
|
|
|
for (path, text) in files.iter() {
|
|
|
|
let file_path = dir.path().join(path);
|
|
|
|
fs::create_dir_all(file_path.parent().unwrap())?;
|
|
|
|
fs::write(file_path, text)?
|
|
|
|
}
|
|
|
|
|
|
|
|
let a_root = dir.path().join("a");
|
|
|
|
let b_root = dir.path().join("a/b");
|
|
|
|
|
2018-12-19 06:58:34 -06:00
|
|
|
let (mut vfs, _) = Vfs::new(vec![a_root, b_root]);
|
2018-12-18 07:38:05 -06:00
|
|
|
for _ in 0..2 {
|
|
|
|
let task = vfs.task_receiver().recv().unwrap();
|
|
|
|
vfs.handle_task(task);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
let files = vfs
|
|
|
|
.commit_changes()
|
|
|
|
.into_iter()
|
|
|
|
.flat_map(|change| {
|
|
|
|
let files = match change {
|
|
|
|
VfsChange::AddRoot { files, .. } => files,
|
|
|
|
_ => panic!("unexpected change"),
|
|
|
|
};
|
|
|
|
files.into_iter().map(|(_id, path, text)| {
|
|
|
|
let text: String = (&*text).clone();
|
|
|
|
(format!("{}", path.display()), text)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.collect::<HashSet<_>>();
|
|
|
|
|
|
|
|
let expected_files = [
|
|
|
|
("foo.rs", "hello"),
|
|
|
|
("bar.rs", "world"),
|
|
|
|
("baz.rs", "nested hello"),
|
|
|
|
]
|
|
|
|
.iter()
|
|
|
|
.map(|(path, text)| (path.to_string(), text.to_string()))
|
|
|
|
.collect::<HashSet<_>>();
|
|
|
|
|
|
|
|
assert_eq!(files, expected_files);
|
|
|
|
}
|
|
|
|
|
2019-01-06 11:36:22 -06:00
|
|
|
// on disk change
|
|
|
|
fs::write(&dir.path().join("a/b/baz.rs"), "quux").unwrap();
|
|
|
|
let change = vfs.change_receiver().recv().unwrap();
|
|
|
|
vfs.handle_change(change);
|
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::ChangeFile { text, .. }] => assert_eq!(text.as_str(), "quux"),
|
|
|
|
_ => panic!("unexpected changes"),
|
2018-12-18 07:38:05 -06:00
|
|
|
}
|
|
|
|
|
2019-01-06 11:36:22 -06:00
|
|
|
// in memory change
|
|
|
|
vfs.change_file_overlay(&dir.path().join("a/b/baz.rs"), Some("m".to_string()));
|
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::ChangeFile { text, .. }] => assert_eq!(text.as_str(), "m"),
|
|
|
|
_ => panic!("unexpected changes"),
|
2018-12-18 07:38:05 -06:00
|
|
|
}
|
|
|
|
|
2019-01-06 11:36:22 -06:00
|
|
|
// in memory remove, restores data on disk
|
2018-12-18 07:38:05 -06:00
|
|
|
vfs.remove_file_overlay(&dir.path().join("a/b/baz.rs"));
|
2019-01-06 11:36:22 -06:00
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::ChangeFile { text, .. }] => assert_eq!(text.as_str(), "quux"),
|
|
|
|
_ => panic!("unexpected changes"),
|
2018-12-18 07:38:05 -06:00
|
|
|
}
|
|
|
|
|
2019-01-06 11:36:22 -06:00
|
|
|
// in memory add
|
|
|
|
vfs.add_file_overlay(&dir.path().join("a/b/spam.rs"), Some("spam".to_string()));
|
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::AddFile { text, path, .. }] => {
|
|
|
|
assert_eq!(text.as_str(), "spam");
|
2018-12-18 07:38:05 -06:00
|
|
|
assert_eq!(path, "spam.rs");
|
|
|
|
}
|
2019-01-06 11:36:22 -06:00
|
|
|
_ => panic!("unexpected changes"),
|
2018-12-18 07:38:05 -06:00
|
|
|
}
|
|
|
|
|
2019-01-06 11:36:22 -06:00
|
|
|
// in memory remove
|
2018-12-18 07:38:05 -06:00
|
|
|
vfs.remove_file_overlay(&dir.path().join("a/b/spam.rs"));
|
2019-01-06 11:36:22 -06:00
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::RemoveFile { path, .. }] => assert_eq!(path, "spam.rs"),
|
|
|
|
_ => panic!("unexpected changes"),
|
|
|
|
}
|
|
|
|
|
|
|
|
// on disk add
|
|
|
|
fs::write(&dir.path().join("a/new.rs"), "new hello").unwrap();
|
|
|
|
let change = vfs.change_receiver().recv().unwrap();
|
|
|
|
vfs.handle_change(change);
|
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::AddFile { text, path, .. }] => {
|
|
|
|
assert_eq!(text.as_str(), "new hello");
|
|
|
|
assert_eq!(path, "new.rs");
|
|
|
|
}
|
|
|
|
_ => panic!("unexpected changes"),
|
|
|
|
}
|
|
|
|
|
|
|
|
// on disk rename
|
|
|
|
fs::rename(&dir.path().join("a/new.rs"), &dir.path().join("a/new1.rs")).unwrap();
|
|
|
|
let change = vfs.change_receiver().recv().unwrap();
|
|
|
|
vfs.handle_change(change);
|
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::RemoveFile {
|
|
|
|
path: removed_path, ..
|
|
|
|
}, VfsChange::AddFile {
|
|
|
|
text,
|
|
|
|
path: added_path,
|
|
|
|
..
|
|
|
|
}] => {
|
|
|
|
assert_eq!(removed_path, "new.rs");
|
|
|
|
assert_eq!(added_path, "new1.rs");
|
|
|
|
assert_eq!(text.as_str(), "new hello");
|
|
|
|
}
|
|
|
|
_ => panic!("unexpected changes"),
|
|
|
|
}
|
|
|
|
|
|
|
|
// on disk remove
|
|
|
|
fs::remove_file(&dir.path().join("a/new1.rs")).unwrap();
|
|
|
|
let change = vfs.change_receiver().recv().unwrap();
|
|
|
|
vfs.handle_change(change);
|
|
|
|
match vfs.commit_changes().as_slice() {
|
|
|
|
[VfsChange::RemoveFile { path, .. }] => assert_eq!(path, "new1.rs"),
|
|
|
|
_ => panic!("unexpected changes"),
|
|
|
|
}
|
|
|
|
|
|
|
|
match vfs.change_receiver().try_recv() {
|
|
|
|
Err(crossbeam_channel::TryRecvError::Empty) => (),
|
|
|
|
res => panic!("unexpected {:?}", res),
|
2018-12-18 07:38:05 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
vfs.shutdown().unwrap();
|
|
|
|
Ok(())
|
|
|
|
}
|