use anyhow::{Result, anyhow};
use fieldwork::Fieldwork;
use mcplease::session::SessionStore;
use serde::{Deserialize, Serialize};
use std::{fmt::Debug, fs, path::PathBuf};
#[derive(Debug, Clone, Serialize, Deserialize, Default, Eq, PartialEq)]
pub(crate) struct SharedContextData {
context_path: Option<PathBuf>,
}
#[derive(Fieldwork, Debug)]
#[fieldwork(get)]
pub(crate) struct RustdocTools {
shared_context_store: SessionStore<SharedContextData>,
#[field(set, with)]
default_session_id: &'static str,
}
impl RustdocTools {
pub(crate) fn new(storage_path: Option<PathBuf>) -> Result<Self> {
let shared_context_store = SessionStore::new(storage_path)?;
Ok(Self {
shared_context_store,
default_session_id: "default",
})
}
pub(crate) fn get_context(&mut self, session_id: Option<&str>) -> Result<Option<PathBuf>> {
let session_id = session_id.unwrap_or_else(|| self.default_session_id());
let shared_data = self.shared_context_store.get_or_create(session_id)?;
Ok(shared_data.context_path.clone())
}
pub(crate) fn set_working_directory(
&mut self,
path: PathBuf,
session_id: Option<&str>,
) -> Result<()> {
let session_id = session_id.unwrap_or_else(|| self.default_session_id());
self.shared_context_store.update(session_id, |data| {
data.context_path = Some(path);
})
}
pub(crate) fn resolve_path(
&mut self,
path_str: &str,
session_id: Option<&str>,
) -> Result<PathBuf> {
let path = PathBuf::from(&*shellexpand::tilde(path_str));
if path.is_absolute() {
return fs::canonicalize(path).map_err(Into::into);
}
self.working_directory(session_id).map(|x| x.join(path_str))
}
pub(crate) fn working_directory(&mut self, session_id: Option<&str>) -> Result<PathBuf> {
match self.get_context(session_id)? {
Some(context) => fs::canonicalize(context).map_err(Into::into),
None => Err(anyhow!(
"Use set_working_directory first or provide an absolute path.",
)),
}
}
}