Skip to content

Commit

Permalink
Add DefinedMacros handler
Browse files Browse the repository at this point in the history
commit-id:7587de32
  • Loading branch information
Draggu committed Nov 7, 2024
1 parent 6d08ec4 commit 996a4f0
Show file tree
Hide file tree
Showing 7 changed files with 135 additions and 7 deletions.
9 changes: 9 additions & 0 deletions scarb/src/compiler/plugin/proc_macro/ffi.rs
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,15 @@ impl ProcMacroInstance {
.collect()
}

pub fn inline_macros(&self) -> Vec<String> {
self.get_expansions()
.iter()
.filter(|e| e.kind == ExpansionKind::Inline)
.map(|e| e.name.clone())
.map(Into::into)
.collect()
}

/// Apply expansion to token stream.
///
/// This function implements the actual calls to functions from the dynamic library.
Expand Down
4 changes: 4 additions & 0 deletions scarb/src/compiler/plugin/proc_macro/host.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1101,4 +1101,8 @@ impl ProcMacroHost {
pub fn into_plugin(self) -> Result<ProcMacroHostPlugin> {
ProcMacroHostPlugin::try_new(self.macros)
}

pub fn macros(&self) -> &[Arc<ProcMacroInstance>] {
&self.macros
}
}
10 changes: 10 additions & 0 deletions scarb/src/ops/proc_macro_server/json_rpc.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
use std::sync::Arc;

use anyhow::Result;
use scarb_proc_macro_server_types::methods::Method;

use crate::compiler::plugin::proc_macro::ProcMacroHost;

pub trait Handler: Method {
fn handle(proc_macros: Arc<ProcMacroHost>, params: Self::Params) -> Result<Self::Response>;
}
50 changes: 50 additions & 0 deletions scarb/src/ops/proc_macro_server/methods/defined_macros.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
use std::sync::Arc;

use anyhow::Result;
use scarb_proc_macro_server_types::methods::defined_macros::{
DefinedMacros, DefinedMacrosResponse,
};

use crate::compiler::plugin::proc_macro::ProcMacroHost;
use crate::ops::proc_macro_server::json_rpc::Handler;

impl Handler for DefinedMacros {
fn handle(proc_macros: Arc<ProcMacroHost>, _params: Self::Params) -> Result<Self::Response> {
let mut response = proc_macros
.macros()
.into_iter()
.map(|e| DefinedMacrosResponse {
attributes: e.declared_attributes(),
inline_macros: e.inline_macros(),
derives: e.declared_derives(),
executables: e.executable_attributes(),
})
.reduce(|mut acc, defined_macros| {
acc.attributes.extend(defined_macros.attributes);
acc.inline_macros.extend(defined_macros.inline_macros);
acc.derives.extend(defined_macros.derives);
acc.executables.extend(defined_macros.executables);

acc
})
.unwrap_or_default();

response.attributes.sort();
response.attributes.dedup();

response.inline_macros.sort();
response.inline_macros.dedup();

response.derives.sort();
response.derives.dedup();

response.executables.sort();
response.executables.dedup();

response
.attributes
.retain(|attr| !response.executables.contains(attr));

Ok(response)
}
}
1 change: 1 addition & 0 deletions scarb/src/ops/proc_macro_server/methods/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
pub mod defined_macros;
30 changes: 23 additions & 7 deletions scarb/src/ops/proc_macro_server/mod.rs
Original file line number Diff line number Diff line change
@@ -1,28 +1,36 @@
use std::num::NonZero;
use std::sync::Arc;
use std::thread::available_parallelism;

use anyhow::{anyhow, Result};
use connection::Connection;
use crossbeam_channel::{Receiver, Sender};
use json_rpc::Handler;
use scarb_proc_macro_server_types::jsonrpc::{ResponseError, RpcRequest, RpcResponse};
use scarb_proc_macro_server_types::methods::defined_macros::DefinedMacros;
use scarb_proc_macro_server_types::methods::Method;
use serde_json::Value;

use crate::compiler::plugin::proc_macro::ProcMacroHost;

mod connection;
mod json_rpc;
mod methods;

pub fn start_proc_macro_server(_proc_macros: ProcMacroHost) -> Result<()> {
pub fn start_proc_macro_server(proc_macros: ProcMacroHost) -> Result<()> {
let connection = Connection::new();
let available_parallelism = available_parallelism().map(NonZero::get).unwrap_or(4);
let proc_macros = Arc::new(proc_macros);

for i in 0..available_parallelism {
let receiver = connection.receiver.clone();
let sender = connection.sender.clone();
let proc_macros = proc_macros.clone();

std::thread::Builder::new()
.name(format!("proc-macro-server-worker-thread-{i}"))
.spawn(move || {
handle_requests(receiver, sender);
handle_requests(proc_macros, receiver, sender);
})
.expect("failed to spawn thread");
}
Expand All @@ -32,10 +40,14 @@ pub fn start_proc_macro_server(_proc_macros: ProcMacroHost) -> Result<()> {
Ok(())
}

fn handle_requests(receiver: Receiver<RpcRequest>, sender: Sender<RpcResponse>) {
fn handle_requests(
proc_macros: Arc<ProcMacroHost>,
receiver: Receiver<RpcRequest>,
sender: Sender<RpcResponse>,
) {
for request in receiver {
let id = request.id;
let response = route_request(request);
let response = route_request(proc_macros.clone(), request);

let response = match response {
Ok(result) => RpcResponse {
Expand All @@ -56,10 +68,14 @@ fn handle_requests(receiver: Receiver<RpcRequest>, sender: Sender<RpcResponse>)
}
}

fn route_request(request: RpcRequest) -> Result<Value> {
#[allow(clippy::match_single_binding)]
fn route_request(proc_macros: Arc<ProcMacroHost>, request: RpcRequest) -> Result<Value> {
match request.method.as_str() {
//TODO add method handlers
DefinedMacros::METHOD => run_handler::<DefinedMacros>(proc_macros.clone(), request.value),
_ => Err(anyhow!("method not found")),
}
}

fn run_handler<M: Handler>(proc_macros: Arc<ProcMacroHost>, value: Value) -> Result<Value> {
M::handle(proc_macros, serde_json::from_value(value).unwrap())
.map(|res| serde_json::to_value(res).unwrap())
}
38 changes: 38 additions & 0 deletions scarb/tests/proc_macro_server.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
use assert_fs::prelude::PathChild;
use assert_fs::TempDir;
use scarb_proc_macro_server_types::methods::defined_macros::DefinedMacros;
use scarb_proc_macro_server_types::methods::defined_macros::DefinedMacrosParams;
use scarb_test_support::cairo_plugin_project_builder::CairoPluginProjectBuilder;
use scarb_test_support::proc_macro_server::ProcMacroClient;
use scarb_test_support::proc_macro_server::SIMPLE_MACROS;
use scarb_test_support::project_builder::ProjectBuilder;

#[test]
fn defined_macros() {
let t = TempDir::new().unwrap();
let plugin_package = t.child("some");

CairoPluginProjectBuilder::default()
.lib_rs(SIMPLE_MACROS)
.build(&plugin_package);

let project = t.child("test_package");

ProjectBuilder::start()
.name("test_package")
.version("1.0.0")
.lib_cairo("")
.dep("some", plugin_package)
.build(&project);

let mut proc_macro_server = ProcMacroClient::new(&project);

let response = proc_macro_server
.request_and_wait::<DefinedMacros>(DefinedMacrosParams {})
.ok();

assert_eq!(response.attributes, vec!["some".to_string()]);
assert_eq!(response.derives, vec!["some_derive".to_string()]);
assert_eq!(response.inline_macros, vec!["inline_some".to_string()]);
assert_eq!(response.executables, vec!["some_executable".to_string()]);
}

0 comments on commit 996a4f0

Please sign in to comment.