use alef_codegen::generators::trait_bridge::{BridgeOutput, TraitBridgeGenerator, TraitBridgeSpec, gen_bridge_all};
use alef_core::config::TraitBridgeConfig;
use alef_core::ir::{ApiSurface, MethodDef, TypeDef, TypeRef};
use std::collections::HashMap;
use std::fmt::Write;
pub struct NapiBridgeGenerator {
pub core_import: String,
pub type_paths: HashMap<String, String>,
pub error_type: String,
}
impl TraitBridgeGenerator for NapiBridgeGenerator {
fn foreign_object_type(&self) -> &str {
"napi::bindgen_prelude::Object<'static>"
}
fn bridge_imports(&self) -> Vec<String> {
vec![
"napi::bindgen_prelude::{JsObjectValue, ToNapiValue, Unknown, Object}".to_string(),
"napi::JsValue".to_string(),
"std::sync::Arc".to_string(),
]
}
fn gen_sync_method_body(&self, method: &MethodDef, spec: &TraitBridgeSpec) -> String {
let name = &method.name;
let has_error = method.error_type.is_some();
let mut out = String::with_capacity(512);
let js_args_exprs = build_napi_args(method);
let args_tuple_ty = unknown_tuple_type(js_args_exprs.len());
writeln!(
out,
"let func: napi::bindgen_prelude::Function<{args_tuple_ty}, napi::bindgen_prelude::Unknown> = match self.inner.get_named_property(\"{name}\") {{"
)
.ok();
writeln!(out, " Ok(f) => f,").ok();
if has_error {
let err = spec.make_error("format!(\"Method '{}' not found on bridge object: {}\", self.cached_name, e)");
writeln!(out, " Err(e) => return Err({err}),").ok();
} else {
writeln!(out, " Err(_) => return Default::default(),").ok();
}
writeln!(out, "}};").ok();
if js_args_exprs.is_empty() {
writeln!(out, "let result = func.call(());").ok();
} else {
let tuple_str = if js_args_exprs.len() == 1 {
format!("({},)", js_args_exprs[0])
} else {
format!("({})", js_args_exprs.join(", "))
};
writeln!(out, "let result = func.call({tuple_str});").ok();
}
if has_error {
writeln!(out, "match result {{").ok();
let err = spec.make_error(&format!(
"format!(\"Plugin '{{}}' method '{}' failed: {{}}\", self.cached_name, e)",
name
));
writeln!(out, " Err(e) => Err({err}),").ok();
writeln!(out, " Ok(val) => {{").ok();
if matches!(method.return_type, TypeRef::Unit) {
writeln!(out, " Ok(())").ok();
} else {
writeln!(out, " // Convert JS value to Rust type via string coercion").ok();
writeln!(out, " let s = val.coerce_to_string()").ok();
writeln!(out, " .and_then(|s| s.into_utf8())").ok();
writeln!(out, " .and_then(|s| s.into_owned())").ok();
writeln!(out, " .map_err(|e| {{").ok();
let err = spec.make_error(&format!(
"format!(\"Failed to extract return value from method '{}': {{}}\", e)",
name
));
writeln!(out, " {err}").ok();
writeln!(out, " }})?;").ok();
writeln!(out, " match s.as_str() {{").ok();
writeln!(out, " \"\" | \"null\" => Ok(Default::default()),").ok();
writeln!(out, " _ => {{").ok();
writeln!(out, " serde_json::from_str::<_>(&s).map_err(|_| {{").ok();
let err = spec.make_error(&format!(
"format!(\"Failed to parse return value for method '{}'\", {})",
name, "self.cached_name"
));
writeln!(out, " {err}").ok();
writeln!(out, " }})").ok();
writeln!(out, " }}").ok();
writeln!(out, " }}").ok();
}
writeln!(out, " }}").ok();
writeln!(out, "}}").ok();
} else {
writeln!(out, "match result {{").ok();
writeln!(out, " Err(_) => Default::default(),").ok();
writeln!(out, " Ok(val) => {{").ok();
if matches!(method.return_type, TypeRef::Unit) {
writeln!(out, " ()").ok();
} else {
writeln!(out, " // Convert JS value to Rust type via string coercion").ok();
writeln!(out, " let s = val.coerce_to_string()").ok();
writeln!(out, " .and_then(|s| s.into_utf8())").ok();
writeln!(out, " .and_then(|s| s.into_owned())").ok();
writeln!(out, " .unwrap_or_default();").ok();
writeln!(out, " match s.as_str() {{").ok();
writeln!(out, " \"\" | \"null\" => Default::default(),").ok();
writeln!(
out,
" _ => serde_json::from_str::<_>(&s).unwrap_or_default()"
)
.ok();
writeln!(out, " }}").ok();
}
writeln!(out, " }}").ok();
writeln!(out, "}}").ok();
}
out
}
fn gen_async_method_body(&self, method: &MethodDef, spec: &TraitBridgeSpec) -> String {
let name = &method.name;
let mut out = String::with_capacity(1024);
writeln!(out, "let cached_name = self.cached_name.clone();").ok();
let js_args_exprs = build_napi_args(method);
let args_tuple_ty = unknown_tuple_type(js_args_exprs.len());
writeln!(
out,
"let func: napi::bindgen_prelude::Function<{args_tuple_ty}, napi::bindgen_prelude::Unknown> = match self.inner.get_named_property(\"{name}\") {{"
)
.ok();
writeln!(out, " Ok(f) => f,").ok();
writeln!(out, " Err(e) => {{").ok();
let err = spec.make_error("format!(\"Method '{}' not found on bridge object: {}\", cached_name, e)");
writeln!(out, " return Err({err});").ok();
writeln!(out, " }}").ok();
writeln!(out, "}};").ok();
let tuple_str = if js_args_exprs.is_empty() {
"()".to_string()
} else {
if js_args_exprs.len() == 1 {
format!("({},)", js_args_exprs[0])
} else {
format!("({})", js_args_exprs.join(", "))
}
};
writeln!(out, "let result = func.call({tuple_str});").ok();
writeln!(out, "match result {{").ok();
let err = spec.make_error(&format!(
"format!(\"Plugin '{{}}' method '{}' failed: {{}}\", cached_name, e)",
name
));
writeln!(out, " Err(e) => Err({err}),").ok();
writeln!(out, " Ok(val) => {{").ok();
if matches!(method.return_type, TypeRef::Unit) {
writeln!(out, " Ok(())").ok();
} else {
writeln!(out, " // Convert JS value to Rust type via string coercion").ok();
writeln!(out, " let s = val.coerce_to_string()").ok();
writeln!(out, " .and_then(|s| s.into_utf8())").ok();
writeln!(out, " .and_then(|s| s.into_owned())").ok();
writeln!(out, " .map_err(|e| {{").ok();
let err = spec.make_error(&format!(
"format!(\"Failed to extract return value from method '{}': {{}}\", e)",
name
));
writeln!(out, " {err}").ok();
writeln!(out, " }})?;").ok();
writeln!(out, " match s.as_str() {{").ok();
writeln!(out, " \"\" | \"null\" => Ok(Default::default()),").ok();
writeln!(out, " _ => serde_json::from_str::<_>(&s).map_err(|_| {{").ok();
let err = spec.make_error(&format!(
"\"Failed to parse return value for method '{}'\".to_string()",
name
));
writeln!(out, " {err}").ok();
writeln!(out, " }})").ok();
writeln!(out, " }}").ok();
}
writeln!(out, " }}").ok();
writeln!(out, "}}").ok();
out
}
fn gen_constructor(&self, spec: &TraitBridgeSpec) -> String {
let wrapper = spec.wrapper_name();
let mut out = String::with_capacity(512);
writeln!(out, "impl {wrapper} {{").ok();
writeln!(out, " /// Create a new bridge wrapping a NAPI Object.").ok();
writeln!(out, " ///").ok();
writeln!(out, " /// Validates that the object provides all required methods.").ok();
writeln!(
out,
" pub fn new(js_obj: napi::bindgen_prelude::Object<'_>) -> napi::Result<Self> {{"
)
.ok();
for req_method in spec.required_methods() {
writeln!(
out,
" if !js_obj.has_named_property(\"{}\").unwrap_or(false) {{",
req_method.name
)
.ok();
writeln!(out, " return Err(napi::Error::new(").ok();
writeln!(out, " napi::Status::GenericFailure,").ok();
writeln!(
out,
" format!(\"Object missing required method: {{}}\", \"{}\")",
req_method.name
)
.ok();
writeln!(out, " ));").ok();
writeln!(out, " }}").ok();
}
writeln!(
out,
" // SAFETY: The JS object is owned by the Node.js runtime and lives for"
)
.ok();
writeln!(
out,
" // the duration of the enclosing #[napi] call. The bridge is only used"
)
.ok();
writeln!(
out,
" // synchronously during that same call, so 'static is safe here."
)
.ok();
writeln!(
out,
" let js_obj: napi::bindgen_prelude::Object<'static> = unsafe {{"
)
.ok();
writeln!(out, " std::mem::transmute(js_obj)").ok();
writeln!(out, " }};").ok();
writeln!(
out,
" let cached_name = match js_obj.get_named_property::<String>(\"name\") {{"
)
.ok();
writeln!(out, " Ok(n) => n,").ok();
writeln!(out, " Err(_) => \"unknown\".to_string(),").ok();
writeln!(out, " }};").ok();
writeln!(out, " Ok(Self {{").ok();
writeln!(out, " inner: js_obj,").ok();
writeln!(out, " cached_name,").ok();
writeln!(out, " }})").ok();
writeln!(out, " }}").ok();
writeln!(out).ok();
writeln!(out, " /// Extract napi::Env from the stored Object.").ok();
writeln!(out, " fn env(&self) -> napi::Env {{").ok();
writeln!(
out,
" // SAFETY: Object<'static> is 3 pointer-sized words; first word is napi_env."
)
.ok();
writeln!(
out,
" let raw: [*mut std::ffi::c_void; 3] = unsafe {{ std::mem::transmute_copy(&self.inner) }};"
)
.ok();
writeln!(out, " napi::Env::from_raw(raw[0] as napi::sys::napi_env)").ok();
writeln!(out, " }}").ok();
writeln!(out, "}}").ok();
writeln!(out).ok();
writeln!(
out,
"// SAFETY: The bridge is created from a NAPI Object that is pinned to the"
)
.ok();
writeln!(
out,
"// Node.js event loop thread. All access occurs on that thread. Send+Sync"
)
.ok();
writeln!(
out,
"// are required by the Plugin trait but the bridge is never actually moved"
)
.ok();
writeln!(out, "// across threads.").ok();
writeln!(out, "unsafe impl Send for {wrapper} {{}}").ok();
writeln!(out, "unsafe impl Sync for {wrapper} {{}}").ok();
out
}
fn gen_registration_fn(&self, spec: &TraitBridgeSpec) -> String {
let Some(register_fn) = spec.bridge_config.register_fn.as_deref() else {
return String::new();
};
let Some(registry_getter) = spec.bridge_config.registry_getter.as_deref() else {
return String::new();
};
let wrapper = spec.wrapper_name();
let trait_path = spec.trait_path();
let mut out = String::with_capacity(1024);
writeln!(out, "#[napi]").ok();
writeln!(
out,
"pub fn {register_fn}(obj: napi::bindgen_prelude::Object) -> napi::Result<()> {{"
)
.ok();
writeln!(out, " let bridge = {wrapper}::new(obj)?;").ok();
writeln!(out, " let arc: Arc<dyn {trait_path}> = Arc::new(bridge);").ok();
let extra = spec
.bridge_config
.register_extra_args
.as_deref()
.map(|a| format!(", {a}"))
.unwrap_or_default();
writeln!(out, " let registry = {registry_getter}();").ok();
writeln!(out, " let mut registry = registry.write();").ok();
writeln!(out, " registry.register(arc{extra}).map_err(|e| napi::Error::new(").ok();
writeln!(out, " napi::Status::GenericFailure,").ok();
writeln!(out, " format!(\"Failed to register backend: {{}}\", e)").ok();
writeln!(out, " ))").ok();
writeln!(out, "}}").ok();
out
}
}
pub fn gen_trait_bridge(
trait_type: &TypeDef,
bridge_cfg: &TraitBridgeConfig,
core_import: &str,
error_type: &str,
error_constructor: &str,
api: &ApiSurface,
) -> BridgeOutput {
let type_paths: HashMap<String, String> = api
.types
.iter()
.map(|t| (t.name.clone(), t.rust_path.replace('-', "_")))
.chain(
api.enums
.iter()
.map(|e| (e.name.clone(), e.rust_path.replace('-', "_"))),
)
.collect();
let is_visitor_bridge = bridge_cfg.type_alias.is_some()
&& bridge_cfg.register_fn.is_none()
&& bridge_cfg.super_trait.is_none()
&& trait_type.methods.iter().all(|m| m.has_default_impl);
if is_visitor_bridge {
let struct_name = format!("Js{}Bridge", bridge_cfg.trait_name);
let trait_path = trait_type.rust_path.replace('-', "_");
let code = gen_visitor_bridge(
trait_type,
bridge_cfg,
&struct_name,
&trait_path,
core_import,
&type_paths,
);
BridgeOutput { imports: vec![], code }
} else {
let generator = NapiBridgeGenerator {
core_import: core_import.to_string(),
type_paths: type_paths.clone(),
error_type: error_type.to_string(),
};
let spec = TraitBridgeSpec {
trait_def: trait_type,
bridge_config: bridge_cfg,
core_import,
wrapper_prefix: "Js",
type_paths,
error_type: error_type.to_string(),
error_constructor: error_constructor.to_string(),
};
gen_bridge_all(&spec, &generator)
}
}
fn gen_visitor_bridge(
trait_type: &TypeDef,
_bridge_cfg: &TraitBridgeConfig,
struct_name: &str,
trait_path: &str,
core_crate: &str,
type_paths: &HashMap<String, String>,
) -> String {
let mut out = String::with_capacity(8192);
writeln!(out, "#[allow(unused_imports)]").unwrap();
writeln!(
out,
"use napi::bindgen_prelude::{{JsObjectValue, ToNapiValue, Unknown, Object}};"
)
.unwrap();
writeln!(out, "#[allow(unused_imports)]").unwrap();
writeln!(out, "use napi::JsValue;").unwrap();
writeln!(out).unwrap();
writeln!(out, "fn nodecontext_to_js_object<'e>(").unwrap();
writeln!(out, " env: &'e napi::Env,").unwrap();
writeln!(out, " ctx: &{core_crate}::visitor::NodeContext,").unwrap();
writeln!(out, ") -> napi::Result<napi::bindgen_prelude::Object<'e>> {{").unwrap();
writeln!(out, " let mut obj = napi::bindgen_prelude::Object::new(env)?;").unwrap();
writeln!(
out,
" obj.set_named_property(\"nodeType\", env.create_string(&format!(\"{{:?}}\", ctx.node_type))?)?;"
)
.unwrap();
writeln!(
out,
" obj.set_named_property(\"tagName\", env.create_string(&ctx.tag_name)?)?;"
)
.unwrap();
writeln!(
out,
" obj.set_named_property(\"depth\", env.create_uint32(ctx.depth as u32)?)?;"
)
.unwrap();
writeln!(
out,
" obj.set_named_property(\"indexInParent\", env.create_uint32(ctx.index_in_parent as u32)?)?;"
)
.unwrap();
writeln!(out, " obj.set_named_property(\"isInline\", ctx.is_inline)?;").unwrap();
writeln!(out, " let parent_tag = match &ctx.parent_tag {{").unwrap();
writeln!(out, " Some(s) => env.create_string(s)?.to_unknown(),").unwrap();
writeln!(out, " None => {{").unwrap();
writeln!(
out,
" // SAFETY: napi_get_null returns a valid napi_value for the given env."
)
.unwrap();
writeln!(
out,
" let raw = unsafe {{ napi::bindgen_prelude::ToNapiValue::to_napi_value(env.raw(), napi::bindgen_prelude::Null)? }};"
)
.unwrap();
writeln!(
out,
" unsafe {{ napi::bindgen_prelude::Unknown::from_raw_unchecked(env.raw(), raw) }}"
)
.unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, " }};").unwrap();
writeln!(out, " obj.set_named_property(\"parentTag\", parent_tag)?;").unwrap();
writeln!(out, " let mut attrs = napi::bindgen_prelude::Object::new(env)?;").unwrap();
writeln!(out, " for (k, v) in &ctx.attributes {{").unwrap();
writeln!(out, " attrs.set_named_property(k, env.create_string(v)?)?;").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, " obj.set_named_property(\"attributes\", attrs)?;").unwrap();
writeln!(out, " Ok(obj)").unwrap();
writeln!(out, "}}").unwrap();
writeln!(out).unwrap();
writeln!(out, "pub struct {struct_name} {{").unwrap();
writeln!(out, " obj: napi::bindgen_prelude::Object<'static>,").unwrap();
writeln!(out, "}}").unwrap();
writeln!(out).unwrap();
writeln!(out, "impl std::fmt::Debug for {struct_name} {{").unwrap();
writeln!(
out,
" fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {{"
)
.unwrap();
writeln!(out, " write!(f, \"{struct_name}\")").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, "}}").unwrap();
writeln!(out).unwrap();
writeln!(out, "impl {struct_name} {{").unwrap();
writeln!(
out,
" pub fn new(js_obj: napi::bindgen_prelude::Object<'_>) -> Self {{"
)
.unwrap();
writeln!(
out,
" // SAFETY: The JS object is owned by the Node.js runtime and lives for"
)
.unwrap();
writeln!(
out,
" // the duration of the enclosing #[napi] call. The bridge is only used"
)
.unwrap();
writeln!(
out,
" // synchronously during that same call, so 'static is safe here."
)
.unwrap();
writeln!(
out,
" let obj: napi::bindgen_prelude::Object<'static> = unsafe {{ std::mem::transmute(js_obj) }};"
)
.unwrap();
writeln!(out, " Self {{ obj }}").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out).unwrap();
writeln!(out, " fn env(&self) -> napi::Env {{").unwrap();
writeln!(
out,
" // SAFETY: Object<'static> is 3 pointer-sized words; first word is napi_env."
)
.unwrap();
writeln!(
out,
" let raw: [*mut std::ffi::c_void; 3] = unsafe {{ std::mem::transmute_copy(&self.obj) }};"
)
.unwrap();
writeln!(out, " napi::Env::from_raw(raw[0] as napi::sys::napi_env)").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, "}}").unwrap();
writeln!(out).unwrap();
writeln!(out, "impl {trait_path} for {struct_name} {{").unwrap();
for method in &trait_type.methods {
if method.trait_source.is_some() {
continue;
}
gen_visitor_method_napi(&mut out, method, trait_path, core_crate, type_paths);
}
writeln!(out, "}}").unwrap();
writeln!(out).unwrap();
out
}
fn visitor_param_type(ty: &TypeRef, is_ref: bool, optional: bool, tp: &HashMap<String, String>) -> String {
if optional && matches!(ty, TypeRef::String) && is_ref {
return "Option<&str>".to_string();
}
if is_ref {
if let TypeRef::Vec(inner) = ty {
let inner_str = param_type(inner, "", false, tp);
return format!("&[{inner_str}]");
}
}
param_type(ty, "", is_ref, tp)
}
fn unknown_tuple_type(count: usize) -> String {
if count == 0 {
return "()".to_string();
}
let parts = vec!["napi::bindgen_prelude::Unknown"; count];
format!("({}{})", parts.join(", "), if count == 1 { "," } else { "" })
}
fn gen_visitor_method_napi(
out: &mut String,
method: &MethodDef,
_trait_path: &str,
_core_crate: &str,
type_paths: &HashMap<String, String>,
) {
let name = &method.name;
let js_name = to_camel_case(name);
let mut sig_parts = vec!["&mut self".to_string()];
for p in &method.params {
let ty_str = visitor_param_type(&p.ty, p.is_ref, p.optional, type_paths);
sig_parts.push(format!("{}: {}", p.name, ty_str));
}
let sig = sig_parts.join(", ");
let ret_ty = match &method.return_type {
TypeRef::Named(n) => type_paths
.get(n.as_str())
.map(|p| p.replace('-', "_"))
.unwrap_or_else(|| n.clone()),
other => param_type(other, "", false, type_paths),
};
writeln!(out, " fn {name}({sig}) -> {ret_ty} {{").unwrap();
writeln!(
out,
" let has_method = self.obj.has_named_property(\"{js_name}\").unwrap_or(false);"
)
.unwrap();
writeln!(out, " if !has_method {{").unwrap();
writeln!(out, " return {ret_ty}::Continue;").unwrap();
writeln!(out, " }}").unwrap();
let arg_count = method.params.len();
let args_tuple_ty = unknown_tuple_type(arg_count);
writeln!(
out,
" let func: napi::bindgen_prelude::Function<{args_tuple_ty}, napi::bindgen_prelude::Unknown> = match self.obj.get_named_property(\"{js_name}\") {{"
)
.unwrap();
writeln!(out, " Ok(f) => f,").unwrap();
writeln!(out, " Err(_) => return {ret_ty}::Continue,").unwrap();
writeln!(out, " }};").unwrap();
let js_args_exprs = build_napi_args(method);
if arg_count == 0 {
writeln!(out, " let result = func.call(());").unwrap();
} else {
writeln!(out, " let __env = self.env();").unwrap();
for (i, expr) in js_args_exprs.iter().enumerate() {
let expr = expr.replace("self.env()", "__env");
writeln!(out, " let arg_{i}: napi::bindgen_prelude::Unknown = {expr};").unwrap();
}
let tuple_args: Vec<String> = (0..arg_count).map(|i| format!("arg_{i}")).collect();
let tuple_str = if arg_count == 1 {
format!("({},)", tuple_args[0])
} else {
format!("({})", tuple_args.join(", "))
};
writeln!(out, " let result = func.call({tuple_str});").unwrap();
}
writeln!(out, " match result {{").unwrap();
writeln!(out, " Err(_) => {ret_ty}::Continue,").unwrap();
writeln!(out, " Ok(val) => {{").unwrap();
writeln!(
out,
" if let Ok(s) = val.coerce_to_string().and_then(|s| s.into_utf8()).and_then(|s| s.into_owned()) {{"
)
.unwrap();
writeln!(out, " match s.to_lowercase().as_str() {{").unwrap();
writeln!(out, " \"continue\" => {ret_ty}::Continue,").unwrap();
writeln!(out, " \"skip\" => {ret_ty}::Skip,").unwrap();
writeln!(
out,
" \"preserve_html\" | \"preservehtml\" => {ret_ty}::PreserveHtml,"
)
.unwrap();
writeln!(
out,
" other => {ret_ty}::Custom(other.to_string()),"
)
.unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, " }} else {{").unwrap();
writeln!(out, " {ret_ty}::Continue").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out, " }}").unwrap();
writeln!(out).unwrap();
}
fn build_napi_args(method: &MethodDef) -> Vec<String> {
method
.params
.iter()
.map(|p| {
if let TypeRef::Named(n) = &p.ty {
if n == "NodeContext" {
return format!(
"match nodecontext_to_js_object(&self.env(), {}{}) {{ Ok(o) => o.to_unknown(), Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}",
if p.is_ref { "" } else { "&" },
p.name
);
}
}
// Option<&str>
if p.optional && matches!(&p.ty, TypeRef::String) && p.is_ref {
return format!(
"match {name} {{ \
Some(s) => match self.env().create_string(s) {{ \
Ok(v) => v.to_unknown(), \
Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}, \
None => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}",
name = p.name
);
}
// &str
if matches!(&p.ty, TypeRef::String) && p.is_ref {
return format!(
"match self.env().create_string({name}) {{ \
Ok(s) => s.to_unknown(), \
Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}",
name = p.name
);
}
// String (owned)
if matches!(&p.ty, TypeRef::String) {
return format!(
"match self.env().create_string({name}.as_str()) {{ \
Ok(s) => s.to_unknown(), \
Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}",
name = p.name
);
}
// Bool
if matches!(&p.ty, TypeRef::Primitive(alef_core::ir::PrimitiveType::Bool)) {
return format!(
"unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(self.env().raw(), {name}).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }}",
name = p.name
);
}
// u32 / usize: create_uint32 needs a u32; usize requires the cast but u32 does not.
if matches!(&p.ty, TypeRef::Primitive(alef_core::ir::PrimitiveType::U32)) {
return format!(
"match self.env().create_uint32({name}) {{ Ok(n) => n.to_unknown(), Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}",
name = p.name
);
}
if matches!(&p.ty, TypeRef::Primitive(alef_core::ir::PrimitiveType::Usize)) {
return format!(
"match self.env().create_uint32({name} as u32) {{ Ok(n) => n.to_unknown(), Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}",
name = p.name
);
}
// Vec<String> or &[String] - serialize to JSON string as fallback
// Default: serialize as debug string
format!(
"match self.env().create_string(&format!(\"{{:?}}\", {name})) {{ Ok(s) => s.to_unknown(), Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(std::ptr::null_mut(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(std::ptr::null_mut(), r) }} \
}}",
name = p.name
)
})
.collect()
}
fn to_camel_case(name: &str) -> String {
let mut result = String::with_capacity(name.len());
let mut capitalize_next = false;
for (i, c) in name.chars().enumerate() {
if c == '_' {
capitalize_next = true;
} else if capitalize_next {
result.extend(c.to_uppercase());
capitalize_next = false;
} else if i == 0 {
result.extend(c.to_lowercase());
} else {
result.push(c);
}
}
result
}
fn param_type(ty: &TypeRef, ci: &str, is_ref: bool, tp: &HashMap<String, String>) -> String {
match ty {
TypeRef::Bytes if is_ref => "&[u8]".into(),
TypeRef::Bytes => "Vec<u8>".into(),
TypeRef::String if is_ref => "&str".into(),
TypeRef::String => "String".into(),
TypeRef::Path if is_ref => "&std::path::Path".into(),
TypeRef::Path => "std::path::PathBuf".into(),
TypeRef::Named(n) => {
let qualified = tp.get(n).cloned().unwrap_or_else(|| format!("{ci}::{n}"));
if is_ref { format!("&{qualified}") } else { qualified }
}
TypeRef::Vec(inner) => format!("Vec<{}>", param_type(inner, ci, false, tp)),
TypeRef::Optional(inner) => format!("Option<{}>", param_type(inner, ci, false, tp)),
TypeRef::Primitive(p) => prim(p).into(),
TypeRef::Unit => "()".into(),
TypeRef::Char => "char".into(),
TypeRef::Map(k, v) => format!(
"std::collections::HashMap<{}, {}>",
param_type(k, ci, false, tp),
param_type(v, ci, false, tp)
),
TypeRef::Json => "serde_json::Value".into(),
TypeRef::Duration => "std::time::Duration".into(),
}
}
fn prim(p: &alef_core::ir::PrimitiveType) -> &'static str {
use alef_core::ir::PrimitiveType::*;
match p {
Bool => "bool",
U8 => "u8",
U16 => "u16",
U32 => "u32",
U64 => "u64",
I8 => "i8",
I16 => "i16",
I32 => "i32",
I64 => "i64",
F32 => "f32",
F64 => "f64",
Usize => "usize",
Isize => "isize",
}
}
pub fn find_bridge_param<'a>(
func: &alef_core::ir::FunctionDef,
bridges: &'a [TraitBridgeConfig],
) -> Option<(usize, &'a TraitBridgeConfig)> {
for (idx, param) in func.params.iter().enumerate() {
let named = match ¶m.ty {
TypeRef::Named(n) => Some(n.as_str()),
TypeRef::Optional(inner) => {
if let TypeRef::Named(n) = inner.as_ref() {
Some(n.as_str())
} else {
None
}
}
_ => None,
};
for bridge in bridges {
if let Some(type_name) = named {
if bridge.type_alias.as_deref() == Some(type_name) {
return Some((idx, bridge));
}
}
if bridge.param_name.as_deref() == Some(param.name.as_str()) {
return Some((idx, bridge));
}
}
}
None
}
#[allow(clippy::too_many_arguments)]
pub fn gen_bridge_function(
func: &alef_core::ir::FunctionDef,
bridge_param_idx: usize,
bridge_cfg: &TraitBridgeConfig,
mapper: &dyn alef_codegen::type_mapper::TypeMapper,
_cfg: &alef_codegen::generators::RustBindingConfig<'_>,
_adapter_bodies: &alef_codegen::generators::AdapterBodies,
opaque_types: &ahash::AHashSet<String>,
core_import: &str,
) -> String {
use alef_core::ir::TypeRef;
let struct_name = format!("Js{}Bridge", bridge_cfg.trait_name);
let handle_path = format!("{core_import}::visitor::VisitorHandle");
let param_name = &func.params[bridge_param_idx].name;
let bridge_param = &func.params[bridge_param_idx];
let is_optional = bridge_param.optional || matches!(&bridge_param.ty, TypeRef::Optional(_));
let mut sig_parts = vec![];
for (idx, p) in func.params.iter().enumerate() {
if idx == bridge_param_idx {
if is_optional {
sig_parts.push(format!("{}: Option<napi::bindgen_prelude::Object>", p.name));
} else {
sig_parts.push(format!("{}: napi::bindgen_prelude::Object", p.name));
}
} else {
let promoted = idx > bridge_param_idx || func.params[..idx].iter().any(|pp| pp.optional);
let ty = if p.optional || promoted {
format!("Option<{}>", mapper.map_type(&p.ty))
} else {
mapper.map_type(&p.ty)
};
sig_parts.push(format!("{}: {}", p.name, ty));
}
}
let params_str = sig_parts.join(", ");
let return_type = mapper.map_type(&func.return_type);
let ret = mapper.wrap_return(&return_type, func.error_type.is_some());
let err_conv = ".map_err(|e| napi::Error::new(napi::Status::GenericFailure, e.to_string()))";
let bridge_wrap = if is_optional {
format!(
"let {param_name} = {param_name}.map(|v| {{\n \
let bridge = {struct_name}::new(v);\n \
std::rc::Rc::new(std::cell::RefCell::new(bridge)) as {handle_path}\n \
}});"
)
} else {
format!(
"let {param_name} = {{\n \
let bridge = {struct_name}::new({param_name});\n \
std::rc::Rc::new(std::cell::RefCell::new(bridge)) as {handle_path}\n \
}};"
)
};
let serde_bindings: String = func
.params
.iter()
.enumerate()
.filter(|(idx, p)| {
if *idx == bridge_param_idx {
return false;
}
let named = match &p.ty {
TypeRef::Named(n) => Some(n.as_str()),
TypeRef::Optional(inner) => {
if let TypeRef::Named(n) = inner.as_ref() {
Some(n.as_str())
} else {
None
}
}
_ => None,
};
named.is_some_and(|n| !opaque_types.contains(n))
})
.map(|(_, p)| {
let name = &p.name;
let core_path = format!(
"{core_import}::{}",
match &p.ty {
TypeRef::Named(n) => n.clone(),
TypeRef::Optional(inner) =>
if let TypeRef::Named(n) = inner.as_ref() {
n.clone()
} else {
String::new()
},
_ => String::new(),
}
);
if p.optional || matches!(&p.ty, TypeRef::Optional(_)) {
format!("let {name}_core: Option<{core_path}> = {name}.map(|v| v.into());\n ")
} else {
format!("let {name}_core: {core_path} = {name}.into();\n ")
}
})
.collect();
let call_args: Vec<String> = func
.params
.iter()
.enumerate()
.map(|(idx, p)| {
if idx == bridge_param_idx {
return p.name.clone();
}
match &p.ty {
TypeRef::Named(n) if opaque_types.contains(n.as_str()) => {
if p.optional {
format!("{}.as_ref().map(|v| &v.inner)", p.name)
} else {
format!("&{}.inner", p.name)
}
}
TypeRef::Named(_) => format!("{}_core", p.name),
TypeRef::Optional(inner) => {
if let TypeRef::Named(n) = inner.as_ref() {
if opaque_types.contains(n.as_str()) {
format!("{}.as_ref().map(|v| &v.inner)", p.name)
} else {
format!("{}_core", p.name)
}
} else {
p.name.clone()
}
}
TypeRef::String | TypeRef::Char => {
if p.is_ref {
format!("&{}", p.name)
} else {
p.name.clone()
}
}
_ => p.name.clone(),
}
})
.collect();
let call_args_str = call_args.join(", ");
let core_fn_path = {
let path = func.rust_path.replace('-', "_");
if path.starts_with(core_import) {
path
} else {
format!("{core_import}::{}", func.name)
}
};
let core_call = format!("{core_fn_path}({call_args_str})");
let return_wrap = match &func.return_type {
TypeRef::Named(name) if opaque_types.contains(name.as_str()) => {
format!("{name} {{ inner: std::sync::Arc::new(val) }}")
}
TypeRef::Named(_) => "val.into()".to_string(),
TypeRef::String | TypeRef::Bytes => "val.into()".to_string(),
_ => "val".to_string(),
};
let body = if func.error_type.is_some() {
if return_wrap == "val" {
format!("{bridge_wrap}\n {serde_bindings}{core_call}{err_conv}")
} else {
format!("{bridge_wrap}\n {serde_bindings}{core_call}.map(|val| {return_wrap}){err_conv}")
}
} else {
format!("{bridge_wrap}\n {serde_bindings}{core_call}")
};
let js_name = {
let mut result = String::with_capacity(func.name.len());
let mut capitalize_next = false;
for (i, c) in func.name.chars().enumerate() {
if c == '_' {
capitalize_next = true;
} else if capitalize_next {
result.extend(c.to_uppercase());
capitalize_next = false;
} else if i == 0 {
result.extend(c.to_lowercase());
} else {
result.push(c);
}
}
result
};
let js_name_attr = if js_name != func.name {
format!("(js_name = \"{}\")", js_name)
} else {
String::new()
};
let mut out = String::with_capacity(1024);
if func.error_type.is_some() {
writeln!(out, "#[allow(clippy::missing_errors_doc)]").ok();
}
writeln!(out, "#[napi{js_name_attr}]").ok();
let func_name = &func.name;
writeln!(out, "pub fn {func_name}({params_str}) -> {ret} {{").ok();
writeln!(out, " {body}").ok();
writeln!(out, "}}").ok();
out
}