//! Client-side Proc-Macro crate //! //! We separate proc-macro expanding logic to an extern program to allow //! different implementations (e.g. wasm or dylib loading). And this crate //! is used for provide basic infra-structure for commnicate between two //! process: Client (RA itself), Server (the external program) use ra_mbe::ExpandError; use ra_tt::Subtree; use std::{ path::{Path, PathBuf}, sync::Arc, }; trait ProcMacroExpander: std::fmt::Debug + Send + Sync + std::panic::RefUnwindSafe { fn custom_derive(&self, subtree: &Subtree, derive_name: &str) -> Result; } #[derive(Debug, Clone, PartialEq, Eq)] pub struct ProcMacroProcessExpander { process_path: PathBuf, } impl ProcMacroExpander for ProcMacroProcessExpander { fn custom_derive( &self, _subtree: &Subtree, _derive_name: &str, ) -> Result { // FIXME: do nothing for now Ok(Subtree::default()) } } #[derive(Debug, Clone)] pub struct ProcMacro { expander: Arc>, name: String, } impl Eq for ProcMacro {} impl PartialEq for ProcMacro { fn eq(&self, other: &ProcMacro) -> bool { self.name == other.name && Arc::ptr_eq(&self.expander, &other.expander) } } impl ProcMacro { pub fn name(&self) -> String { self.name.clone() } pub fn custom_derive(&self, subtree: &Subtree) -> Result { self.expander.custom_derive(subtree, &self.name) } } #[derive(Debug, Clone, PartialEq, Eq)] pub enum ProcMacroClient { Process { expander: Arc }, Dummy, } impl ProcMacroClient { pub fn extern_process(process_path: &Path) -> ProcMacroClient { let expander = ProcMacroProcessExpander { process_path: process_path.into() }; ProcMacroClient::Process { expander: Arc::new(expander) } } pub fn dummy() -> ProcMacroClient { ProcMacroClient::Dummy } pub fn by_dylib_path(&self, _dylib_path: &Path) -> Vec { // FIXME: return empty for now vec![] } }