mirror of
https://github.com/zed-industries/zed.git
synced 2025-01-13 05:42:59 +00:00
Start on in-memory fs
This commit is contained in:
parent
4dae17a4cf
commit
2fa63a3a50
1 changed files with 161 additions and 1 deletions
|
@ -28,10 +28,11 @@ use postage::{
|
||||||
use smol::{
|
use smol::{
|
||||||
channel::Sender,
|
channel::Sender,
|
||||||
io::{AsyncReadExt, AsyncWriteExt},
|
io::{AsyncReadExt, AsyncWriteExt},
|
||||||
|
lock::RwLock,
|
||||||
};
|
};
|
||||||
use std::{
|
use std::{
|
||||||
cmp::{self, Ordering},
|
cmp::{self, Ordering},
|
||||||
collections::HashMap,
|
collections::{BTreeMap, HashMap},
|
||||||
convert::{TryFrom, TryInto},
|
convert::{TryFrom, TryInto},
|
||||||
ffi::{OsStr, OsString},
|
ffi::{OsStr, OsString},
|
||||||
fmt, fs,
|
fmt, fs,
|
||||||
|
@ -141,6 +142,151 @@ impl Fs for OsFs {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct InMemoryEntry {
|
||||||
|
inode: u64,
|
||||||
|
mtime: SystemTime,
|
||||||
|
is_dir: bool,
|
||||||
|
is_symlink: bool,
|
||||||
|
content: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct InMemoryFsState {
|
||||||
|
entries: BTreeMap<PathBuf, InMemoryEntry>,
|
||||||
|
next_inode: u64,
|
||||||
|
events_tx: watch::Sender<()>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InMemoryFsState {
|
||||||
|
fn validate_path(&self, path: &Path) -> Result<()> {
|
||||||
|
if path
|
||||||
|
.parent()
|
||||||
|
.and_then(|path| self.entries.get(path))
|
||||||
|
.map_or(false, |e| e.is_dir)
|
||||||
|
{
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(anyhow!("invalid "))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct InMemoryFs {
|
||||||
|
state: RwLock<InMemoryFsState>,
|
||||||
|
events_rx: watch::Receiver<()>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InMemoryFs {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
let (events_tx, events_rx) = watch::channel();
|
||||||
|
let mut entries = BTreeMap::new();
|
||||||
|
entries.insert(
|
||||||
|
Path::new("/").to_path_buf(),
|
||||||
|
InMemoryEntry {
|
||||||
|
inode: 0,
|
||||||
|
mtime: SystemTime::now(),
|
||||||
|
is_dir: true,
|
||||||
|
is_symlink: false,
|
||||||
|
content: None,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
Self {
|
||||||
|
state: RwLock::new(InMemoryFsState {
|
||||||
|
entries,
|
||||||
|
next_inode: 1,
|
||||||
|
events_tx,
|
||||||
|
}),
|
||||||
|
events_rx,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn insert_dir(&self, path: &Path) -> Result<()> {
|
||||||
|
let mut state = self.state.write().await;
|
||||||
|
state.validate_path(path)?;
|
||||||
|
|
||||||
|
let inode = state.next_inode;
|
||||||
|
state.next_inode += 1;
|
||||||
|
state.entries.insert(
|
||||||
|
path.to_path_buf(),
|
||||||
|
InMemoryEntry {
|
||||||
|
inode,
|
||||||
|
mtime: SystemTime::now(),
|
||||||
|
is_dir: true,
|
||||||
|
is_symlink: false,
|
||||||
|
content: None,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait::async_trait]
|
||||||
|
impl Fs for InMemoryFs {
|
||||||
|
async fn entry(
|
||||||
|
&self,
|
||||||
|
root_char_bag: CharBag,
|
||||||
|
next_entry_id: &AtomicUsize,
|
||||||
|
path: Arc<Path>,
|
||||||
|
abs_path: &Path,
|
||||||
|
) -> Result<Option<Entry>> {
|
||||||
|
let state = self.state.read().await;
|
||||||
|
if let Some(entry) = state.entries.get(abs_path) {
|
||||||
|
Ok(Some(Entry {
|
||||||
|
id: next_entry_id.fetch_add(1, SeqCst),
|
||||||
|
kind: if entry.is_dir {
|
||||||
|
EntryKind::PendingDir
|
||||||
|
} else {
|
||||||
|
EntryKind::File(char_bag_for_path(root_char_bag, &path))
|
||||||
|
},
|
||||||
|
path: Arc::from(path),
|
||||||
|
inode: entry.inode,
|
||||||
|
mtime: entry.mtime,
|
||||||
|
is_symlink: entry.is_symlink,
|
||||||
|
is_ignored: false,
|
||||||
|
}))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn load(&self, path: &Path) -> Result<String> {
|
||||||
|
let state = self.state.read().await;
|
||||||
|
let text = state
|
||||||
|
.entries
|
||||||
|
.get(path)
|
||||||
|
.and_then(|e| e.content.as_ref())
|
||||||
|
.ok_or_else(|| anyhow!("file {:?} does not exist", path))?;
|
||||||
|
Ok(text.clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn save(&self, path: &Path, text: &Rope) -> Result<()> {
|
||||||
|
let mut state = self.state.write().await;
|
||||||
|
state.validate_path(path)?;
|
||||||
|
if let Some(entry) = state.entries.get_mut(path) {
|
||||||
|
if entry.is_dir {
|
||||||
|
Err(anyhow!("cannot overwrite a directory with a file"))
|
||||||
|
} else {
|
||||||
|
entry.content = Some(text.chunks().collect());
|
||||||
|
entry.mtime = SystemTime::now();
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let inode = state.next_inode;
|
||||||
|
state.next_inode += 1;
|
||||||
|
state.entries.insert(
|
||||||
|
path.to_path_buf(),
|
||||||
|
InMemoryEntry {
|
||||||
|
inode,
|
||||||
|
mtime: SystemTime::now(),
|
||||||
|
is_dir: false,
|
||||||
|
is_symlink: false,
|
||||||
|
content: Some(text.chunks().collect()),
|
||||||
|
},
|
||||||
|
);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
enum ScanState {
|
enum ScanState {
|
||||||
Idle,
|
Idle,
|
||||||
|
@ -200,6 +346,20 @@ impl Worktree {
|
||||||
Worktree::Local(tree)
|
Worktree::Local(tree)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(any(test, feature = "test-support"))]
|
||||||
|
pub fn test(
|
||||||
|
path: impl Into<Arc<Path>>,
|
||||||
|
languages: Arc<LanguageRegistry>,
|
||||||
|
fs: Arc<InMemoryFs>,
|
||||||
|
cx: &mut ModelContext<Worktree>,
|
||||||
|
) -> Self {
|
||||||
|
let (tree, scan_states_tx) = LocalWorktree::new(path, languages, fs.clone(), cx);
|
||||||
|
let background_snapshot = tree.background_snapshot.clone();
|
||||||
|
let id = tree.id;
|
||||||
|
cx.background().spawn(async move {}).detach();
|
||||||
|
Worktree::Local(tree)
|
||||||
|
}
|
||||||
|
|
||||||
pub async fn open_remote(
|
pub async fn open_remote(
|
||||||
rpc: rpc::Client,
|
rpc: rpc::Client,
|
||||||
id: u64,
|
id: u64,
|
||||||
|
|
Loading…
Reference in a new issue