use crate::codegen::generators::trait_bridge::{
BridgeOutput, TraitBridgeGenerator, TraitBridgeSpec, bridge_param_type as param_type, host_function_path,
to_camel_case, visitor_param_type,
};
use crate::core::config::TraitBridgeConfig;
use crate::core::ir::{ApiSurface, MethodDef, TypeDef, TypeRef};
use std::collections::HashMap;
pub use crate::codegen::generators::trait_bridge::find_bridge_param;
pub fn find_options_field_binding<'a>(
func: &crate::core::ir::FunctionDef,
bridges: &'a [TraitBridgeConfig],
) -> Option<(usize, &'a TraitBridgeConfig)> {
for bridge in bridges {
if bridge.bind_via != crate::core::config::BridgeBinding::OptionsField {
continue;
}
if let Some(options_type) = &bridge.options_type {
for (idx, param) in func.params.iter().enumerate() {
let matches = match ¶m.ty {
crate::core::ir::TypeRef::Named(n) => n == options_type,
crate::core::ir::TypeRef::Optional(inner) => {
if let crate::core::ir::TypeRef::Named(n) = inner.as_ref() {
n == options_type
} else {
false
}
}
_ => false,
};
if matches {
return Some((idx, bridge));
}
}
}
}
None
}
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 js_method_name = to_camel_case(name);
let snake_method_name = name.clone();
let has_error = method.error_type.is_some();
let js_args_exprs = build_napi_args(method, spec.bridge_config);
let inner_tuple_ty = unknown_tuple_type(js_args_exprs.len());
let args_tuple_ty = if js_args_exprs.is_empty() {
inner_tuple_ty.clone()
} else {
format!("napi::bindgen_prelude::FnArgs<{inner_tuple_ty}>")
};
let empty_args = js_args_exprs.is_empty();
let tuple_args = if empty_args {
String::new()
} else if js_args_exprs.len() == 1 {
format!("({},)", js_args_exprs[0])
} else {
format!("({})", js_args_exprs.join(", "))
};
let error_lookup =
spec.make_error("format!(\"Method '{}' not found on bridge object: {}\", self.cached_name, e)");
let error_call = spec.make_error(&format!(
"format!(\"Plugin '{{}}' method '{}' failed: {{}}\", self.cached_name, e)",
name
));
let error_coercion = spec.make_error(&format!(
"format!(\"Failed to extract return value from method '{}': {{}}\", e)",
name
));
let error_parse = spec.make_error(&format!(
"format!(\"Plugin '{{}}' failed to parse return value for method '{}'\", self.cached_name)",
name
));
let has_default_impl = method.has_default_impl;
if matches!(method.return_type, TypeRef::Unit) {
crate::backends::napi::template_env::render(
"sync_method_unit_return.jinja",
minijinja::context! {
method_name => &js_method_name,
snake_case_method_name => &snake_method_name,
args_tuple_ty => args_tuple_ty,
has_error => has_error,
has_default_impl => has_default_impl,
empty_args => empty_args,
tuple_args => tuple_args,
error_lookup => error_lookup,
error_call => error_call,
},
)
} else {
crate::backends::napi::template_env::render(
"sync_method_non_unit_return.jinja",
minijinja::context! {
method_name => &js_method_name,
snake_case_method_name => &snake_method_name,
args_tuple_ty => args_tuple_ty,
has_error => has_error,
has_default_impl => has_default_impl,
empty_args => empty_args,
tuple_args => tuple_args,
error_lookup => error_lookup,
error_call => error_call,
error_coercion => error_coercion,
error_parse => error_parse,
},
)
}
}
fn gen_async_method_body(&self, method: &MethodDef, spec: &TraitBridgeSpec) -> String {
let name = &method.name;
let js_method_name = to_camel_case(name);
let snake_method_name = name.clone();
let js_args_exprs = build_napi_args(method, spec.bridge_config);
let inner_tuple_ty = unknown_tuple_type(js_args_exprs.len());
let args_tuple_ty = if js_args_exprs.is_empty() {
inner_tuple_ty.clone()
} else {
format!("napi::bindgen_prelude::FnArgs<{inner_tuple_ty}>")
};
let empty_args = js_args_exprs.is_empty();
let tuple_args = if empty_args {
String::new()
} else if js_args_exprs.len() == 1 {
format!("({},)", js_args_exprs[0])
} else {
format!("({})", js_args_exprs.join(", "))
};
let error_lookup = spec.make_error("format!(\"Method '{}' not found on bridge object: {}\", cached_name, e)");
let error_call = spec.make_error(&format!(
"format!(\"Plugin '{{}}' method '{}' failed: {{}}\", cached_name, e)",
name
));
let error_coercion = spec.make_error(&format!(
"format!(\"Failed to extract return value from method '{}': {{}}\", e)",
name
));
let error_parse = spec.make_error(&format!(
"\"Failed to parse return value for method '{}'\".to_string()",
name
));
if matches!(method.return_type, TypeRef::Unit) {
crate::backends::napi::template_env::render(
"async_method_unit_return.jinja",
minijinja::context! {
method_name => &js_method_name,
snake_case_method_name => &snake_method_name,
args_tuple_ty => args_tuple_ty,
empty_args => empty_args,
tuple_args => tuple_args,
error_lookup => error_lookup,
error_call => error_call,
},
)
} else {
crate::backends::napi::template_env::render(
"async_method_non_unit_return.jinja",
minijinja::context! {
method_name => &js_method_name,
snake_case_method_name => &snake_method_name,
args_tuple_ty => args_tuple_ty,
empty_args => empty_args,
tuple_args => tuple_args,
error_lookup => error_lookup,
error_call => error_call,
error_coercion => error_coercion,
error_parse => error_parse,
},
)
}
}
fn gen_constructor(&self, spec: &TraitBridgeSpec) -> String {
let wrapper = spec.wrapper_name();
let required_methods = spec
.required_methods()
.iter()
.map(|m| {
let js_name = to_camel_case(&m.name);
let snake_name = m.name.clone();
minijinja::context! {
name => js_name,
snake_case_name => snake_name,
}
})
.collect::<Vec<_>>();
crate::backends::napi::template_env::render(
"trait_bridge_constructor.jinja",
minijinja::context! {
wrapper_name => wrapper,
required_methods => required_methods,
requires_plugin_name => spec.bridge_config.super_trait.is_some(),
},
)
}
fn gen_unregistration_fn(&self, spec: &TraitBridgeSpec) -> String {
let Some(unregister_fn) = spec.bridge_config.unregister_fn.as_deref() else {
return String::new();
};
let host_path = host_function_path(spec, unregister_fn);
let camel = to_camel_case(unregister_fn);
crate::backends::napi::template_env::render(
"unregistration_fn.jinja",
minijinja::context! {
unregister_fn => unregister_fn,
camel_fn_name => camel,
host_path => host_path,
},
)
}
fn gen_clear_fn(&self, spec: &TraitBridgeSpec) -> String {
let Some(clear_fn) = spec.bridge_config.clear_fn.as_deref() else {
return String::new();
};
let host_path = host_function_path(spec, clear_fn);
let camel = to_camel_case(clear_fn);
crate::backends::napi::template_env::render(
"clear_fn.jinja",
minijinja::context! {
clear_fn => clear_fn,
camel_fn_name => camel,
host_path => host_path,
},
)
}
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 extra = spec
.bridge_config
.register_extra_args
.as_deref()
.map(|a| format!(", {a}"))
.unwrap_or_default();
crate::backends::napi::template_env::render(
"registration_fn.jinja",
minijinja::context! {
register_fn => register_fn,
wrapper => wrapper,
trait_path => trait_path,
registry_getter => registry_getter,
extra_args => extra,
},
)
}
}
pub fn gen_trait_bridge(
trait_type: &TypeDef,
bridge_cfg: &TraitBridgeConfig,
core_import: &str,
error_type: &str,
error_constructor: &str,
api: &ApiSurface,
) -> anyhow::Result<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('-', "_"))),
)
.chain(
api.excluded_type_paths
.iter()
.map(|(name, path)| (name.clone(), 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 = crate::codegen::generators::trait_bridge::bridge_wrapper_name("Js", bridge_cfg);
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,
api,
)?;
Ok(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 lifetime_type_names: std::collections::HashSet<String> = api
.types
.iter()
.filter(|t| t.has_lifetime_params)
.map(|t| t.name.clone())
.collect();
let spec = TraitBridgeSpec {
trait_def: trait_type,
bridge_config: bridge_cfg,
core_import,
wrapper_prefix: "Js",
type_paths,
lifetime_type_names,
error_type: error_type.to_string(),
error_constructor: error_constructor.to_string(),
};
let imports = generator.bridge_imports();
let mut code = String::with_capacity(4096);
let wrapper_name = spec.wrapper_name();
code.push_str(&crate::backends::napi::template_env::render(
"napi_bridge_struct.jinja",
minijinja::context! {
wrapper_name => wrapper_name,
},
));
code.push_str("\n\n");
code.push_str(&crate::codegen::generators::trait_bridge::gen_bridge_debug_impl(&spec));
code.push_str("\n\n");
code.push_str(&generator.gen_constructor(&spec));
code.push_str("\n\n");
if let Some(plugin_impl) = crate::codegen::generators::trait_bridge::gen_bridge_plugin_impl(&spec, &generator) {
code.push_str(&plugin_impl);
code.push_str("\n\n");
}
code.push_str(&crate::codegen::generators::trait_bridge::gen_bridge_trait_impl(
&spec, &generator,
));
if let Some(reg_fn_code) =
crate::codegen::generators::trait_bridge::gen_bridge_registration_fn(&spec, &generator)
{
code.push_str("\n\n");
code.push_str(®_fn_code);
}
if let Some(unreg_fn_code) =
crate::codegen::generators::trait_bridge::gen_bridge_unregistration_fn(&spec, &generator)
{
code.push_str("\n\n");
code.push_str(&unreg_fn_code);
}
if let Some(clear_fn_code) = crate::codegen::generators::trait_bridge::gen_bridge_clear_fn(&spec, &generator) {
code.push_str("\n\n");
code.push_str(&clear_fn_code);
}
Ok(BridgeOutput { imports, code })
}
}
fn gen_visitor_bridge(
trait_type: &TypeDef,
bridge_cfg: &TraitBridgeConfig,
struct_name: &str,
trait_path: &str,
core_crate: &str,
type_paths: &HashMap<String, String>,
api: &ApiSurface,
) -> anyhow::Result<String> {
let result_metadata = crate::codegen::visitor_result::required_visitor_result_metadata(api, bridge_cfg)?;
let context_helper = crate::codegen::visitor_context::visitor_context_helper(
api,
bridge_cfg,
core_crate,
crate::codegen::visitor_context::VisitorContextBackend::Napi,
)?;
let mut method_impls = String::with_capacity(4096);
for method in crate::codegen::generators::trait_bridge::visitor_callback_methods(trait_type, bridge_cfg) {
gen_visitor_method_napi(
&mut method_impls,
method,
trait_path,
core_crate,
bridge_cfg,
type_paths,
&result_metadata,
);
}
Ok(crate::backends::napi::template_env::render(
"visitor_bridge.jinja",
minijinja::context! {
core_crate => core_crate,
context_type_path => context_helper.type_path,
context_field_lines => context_helper.field_lines,
struct_name => struct_name,
trait_path => trait_path,
method_impls => method_impls,
},
))
}
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,
bridge_cfg: &TraitBridgeConfig,
type_paths: &HashMap<String, String>,
result_metadata: &crate::codegen::visitor_result::VisitorResultMetadata,
) {
let name = &method.name;
let js_method_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 signature = sig_parts.join(", ");
let return_type = 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),
};
let arg_count = method.params.len();
let empty_args = arg_count == 0;
let inner_tuple_ty = unknown_tuple_type(arg_count);
let args_tuple_ty = if empty_args {
inner_tuple_ty
} else {
format!("napi::bindgen_prelude::FnArgs<{inner_tuple_ty}>")
};
let js_args_exprs = build_napi_args(method, bridge_cfg);
let arg_exprs: Vec<String> = js_args_exprs
.iter()
.map(|expr| expr.replace("self.env()", "__env"))
.collect();
let tuple_args = if arg_count == 1 {
"(arg_0,)".to_string()
} else if arg_count > 0 {
let arg_names: Vec<String> = (0..arg_count).map(|i| format!("arg_{i}")).collect();
format!("({})", arg_names.join(", "))
} else {
String::new()
};
out.push_str(&crate::backends::napi::template_env::render(
"visitor_method.jinja",
minijinja::context! {
method_name => name,
js_method_name => js_method_name,
signature => signature,
return_type => return_type,
default_result_expr => crate::codegen::visitor_result::default_result_expr(&return_type, result_metadata),
unknown_string_result_expr => crate::codegen::visitor_result::unknown_string_result_expr(
&return_type,
result_metadata,
"s",
),
unit_result_variants => crate::codegen::visitor_result::variant_contexts(&result_metadata.unit_variants),
payload_result_variants => crate::codegen::visitor_result::variant_contexts(
&result_metadata.string_payload_variants,
),
empty_args => empty_args,
arg_exprs => arg_exprs,
tuple_args => tuple_args,
args_tuple_ty => args_tuple_ty,
},
));
}
fn build_napi_args(method: &MethodDef, bridge_cfg: &TraitBridgeConfig) -> Vec<String> {
method
.params
.iter()
.map(|p| {
if let TypeRef::Named(n) = &p.ty {
if Some(n.as_str()) == bridge_cfg.context_type.as_deref() {
return crate::backends::napi::template_env::render(
"visitor_context_arg_expr.jinja",
minijinja::context! { ref_prefix => if p.is_ref { "" } else { "&" }, name => p.name.as_str() },
)
.trim_end()
.to_string();
}
}
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(self.env().raw(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }} \
}}, \
None => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(self.env().raw(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }} \
}}",
name = p.name
);
}
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(self.env().raw(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }} \
}}",
name = p.name
);
}
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(self.env().raw(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }} \
}}",
name = p.name
);
}
if matches!(&p.ty, TypeRef::Primitive(crate::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
);
}
if matches!(&p.ty, TypeRef::Primitive(crate::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(self.env().raw(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }} \
}}",
name = p.name
);
}
if matches!(&p.ty, TypeRef::Primitive(crate::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(self.env().raw(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }} \
}}",
name = p.name
);
}
format!(
"match self.env().create_string(&format!(\"{{:?}}\", {name})) {{ Ok(s) => s.to_unknown(), Err(_) => unsafe {{ \
let r = napi::bindgen_prelude::ToNapiValue::to_napi_value(self.env().raw(), napi::bindgen_prelude::Null).unwrap_or(std::ptr::null_mut()); \
napi::bindgen_prelude::Unknown::from_raw_unchecked(self.env().raw(), r) }} \
}}",
name = p.name
)
})
.collect()
}
#[allow(clippy::too_many_arguments)]
pub fn gen_bridge_function(
api: &ApiSurface,
func: &crate::core::ir::FunctionDef,
bridge_param_idx: usize,
bridge_cfg: &TraitBridgeConfig,
mapper: &dyn crate::codegen::type_mapper::TypeMapper,
_cfg: &crate::codegen::generators::RustBindingConfig<'_>,
_adapter_bodies: &crate::codegen::generators::AdapterBodies,
opaque_types: &ahash::AHashSet<String>,
core_import: &str,
) -> String {
use crate::core::ir::TypeRef;
let struct_name = crate::codegen::generators::trait_bridge::bridge_wrapper_name("Js", bridge_cfg);
let handle_path = crate::codegen::generators::trait_bridge::bridge_handle_path(api, bridge_cfg, core_import);
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 is_options_field_binding = matches!(bridge_cfg.bind_via, crate::core::config::BridgeBinding::OptionsField);
let options_param_idx = if is_options_field_binding {
func.params.iter().enumerate().find(|(_, p)| {
matches!(&p.ty, TypeRef::Named(n) if bridge_cfg.options_type.as_ref().is_some_and(|opt_type| n == opt_type))
}).map(|(i, _)| i)
} else {
None
};
let mut sig_parts = vec![];
for (idx, p) in func.params.iter().enumerate() {
if is_options_field_binding && Some(idx) == options_param_idx {
let ty = if p.optional || (idx > 0 && func.params[..idx].iter().any(|pp| pp.optional)) {
format!("Option<{}>", mapper.map_type(&p.ty))
} else {
mapper.map_type(&p.ty)
};
sig_parts.push(format!("{}: {}", p.name, ty));
} else 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 {
crate::backends::napi::template_env::render(
"bridge_optional_wrap.jinja",
minijinja::context! {
param_name => param_name,
struct_name => struct_name,
handle_path => handle_path,
},
)
} else {
crate::backends::napi::template_env::render(
"bridge_required_wrap.jinja",
minijinja::context! {
param_name => param_name,
struct_name => struct_name,
handle_path => handle_path,
},
)
};
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(),
}
);
let template_name = if p.optional || matches!(&p.ty, TypeRef::Optional(_)) {
"named_core_binding_optional.jinja"
} else {
"named_core_binding_required.jinja"
};
crate::backends::napi::template_env::render(
template_name,
minijinja::context! {
name => name,
core_path => core_path,
},
)
})
.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 = render_bridge_function_body(
func.error_type.is_some(),
&return_wrap,
&bridge_wrap,
&serde_bindings,
&core_call,
err_conv,
);
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 func_name = &func.name;
crate::backends::napi::template_env::render(
"bridge_function.jinja",
minijinja::context! {
has_error => func.error_type.is_some(),
js_name_attr => js_name_attr,
func_name => func_name,
params_str => params_str,
ret => ret,
body => body,
},
)
}
#[allow(clippy::too_many_arguments)]
pub fn gen_options_field_bridge_function(
api: &ApiSurface,
func: &crate::core::ir::FunctionDef,
options_param_idx: usize,
bridge_cfg: &TraitBridgeConfig,
mapper: &dyn crate::codegen::type_mapper::TypeMapper,
_cfg: &crate::codegen::generators::RustBindingConfig<'_>,
opaque_types: &ahash::AHashSet<String>,
core_import: &str,
) -> String {
use crate::core::ir::TypeRef;
let struct_name = crate::codegen::generators::trait_bridge::bridge_wrapper_name("Js", bridge_cfg);
let handle_path = crate::codegen::generators::trait_bridge::bridge_handle_path(api, bridge_cfg, core_import);
let options_param = &func.params[options_param_idx];
let options_name = &options_param.name;
let ir_param_optional = matches!(&options_param.ty, TypeRef::Optional(_));
let visitor_kwarg = bridge_cfg.param_name.as_deref().unwrap_or("visitor");
let field_name = bridge_cfg.resolved_options_field().unwrap_or(visitor_kwarg);
let options_type = bridge_cfg
.options_type
.as_deref()
.unwrap_or_else(|| match &options_param.ty {
TypeRef::Named(name) => name.as_str(),
TypeRef::Optional(inner) => match inner.as_ref() {
TypeRef::Named(name) => name.as_str(),
_ => "Options",
},
_ => "Options",
});
let options_path = format!("{core_import}::{options_type}");
let params_str = {
let mut sig_parts = vec![];
for (i, p) in func.params.iter().enumerate() {
let ty = mapper.map_type(&p.ty);
if i == options_param_idx && !ir_param_optional {
sig_parts.push(format!("{}: Option<{ty}>", p.name));
} else {
sig_parts.push(format!("{}: {ty}", p.name));
}
}
sig_parts.push(format!("{visitor_kwarg}: Option<napi::bindgen_prelude::Object>"));
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 call_args: String = func
.params
.iter()
.enumerate()
.map(|(idx, p)| {
if idx == options_param_idx {
format!("{options_name}_core")
} else {
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!("{}.into()", 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!("{}.map(Into::into)", 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::<Vec<_>>()
.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})");
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 = crate::backends::napi::template_env::render(
"options_field_bridge_body.jinja",
minijinja::context! {
has_error => func.error_type.is_some(),
maps_return => return_wrap != "val",
visitor_kwarg => visitor_kwarg,
handle_path => handle_path,
struct_name => struct_name,
options_name => options_name,
options_path => options_path,
field_name => field_name,
core_call => core_call,
err_conv => err_conv,
return_wrap => return_wrap,
},
);
let mut out = String::with_capacity(1024);
if func.error_type.is_some() {
out.push_str("#[allow(clippy::missing_errors_doc)]\n");
}
out.push_str("#[napi]\n");
let func_name = &func.name;
out.push_str(&crate::backends::napi::template_env::render(
"trait_bridge_fn_wrapper.jinja",
minijinja::context! {
func_name => func_name,
params_str => params_str,
return_type => ret,
body => body,
},
));
out
}
fn render_bridge_function_body(
has_error: bool,
return_wrap: &str,
bridge_wrap: &str,
serde_bindings: &str,
core_call: &str,
err_conv: &str,
) -> String {
let template_name = match (has_error, return_wrap == "val") {
(true, true) => "bridge_function_body_error.jinja",
(true, false) => "bridge_function_body_error_mapped.jinja",
(false, _) => "bridge_function_body_plain.jinja",
};
crate::backends::napi::template_env::render(
template_name,
minijinja::context! {
bridge_wrap => bridge_wrap,
serde_bindings => serde_bindings,
core_call => core_call,
err_conv => err_conv,
return_wrap => return_wrap,
},
)
}
#[cfg(test)]
mod tests {
#[test]
fn visitor_bridge_uses_configured_context_and_result_metadata() {
let (api, trait_type, bridge) = crate::codegen::visitor_context::test_support::neutral_visitor_fixture();
let output = super::gen_trait_bridge(
&trait_type,
&bridge,
"sample_core",
"SampleError",
"SampleError::Message { message: {msg} }",
&api,
)
.expect("visitor bridge should generate");
crate::codegen::visitor_context::test_support::assert_neutral_visitor_output(&output.code);
assert!(output.code.contains("displayName"));
}
}