pub use crate::codegen::generators::trait_bridge::find_bridge_param;
use crate::codegen::generators::trait_bridge::{
TraitBridgeGenerator, TraitBridgeSpec, bridge_param_type as param_type, gen_bridge_all, visitor_param_type,
};
use crate::core::config::TraitBridgeConfig;
use crate::core::ir::{ApiSurface, MethodDef, TypeDef, TypeRef};
use std::collections::HashMap;
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 fn gen_trait_bridge(
trait_type: &TypeDef,
bridge_cfg: &TraitBridgeConfig,
core_import: &str,
error_type: &str,
error_constructor: &str,
api: &ApiSurface,
) -> String {
if bridge_cfg.exclude_languages.contains(&"ruby".to_string()) {
return String::new();
}
let trait_path = trait_type.rust_path.replace('-', "_");
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("Rb", bridge_cfg);
let mut out = String::with_capacity(8192);
gen_visitor_bridge(
&mut out,
trait_type,
bridge_cfg,
&struct_name,
&trait_path,
core_import,
&type_paths,
);
out
} else {
let generator = MagnusBridgeGenerator {
core_import: core_import.to_string(),
type_paths: type_paths.clone(),
error_type: error_type.to_string(),
error_constructor: error_constructor.to_string(),
};
let spec = TraitBridgeSpec {
trait_def: trait_type,
bridge_config: bridge_cfg,
core_import,
wrapper_prefix: "Rb",
type_paths,
error_type: error_type.to_string(),
error_constructor: error_constructor.to_string(),
};
let output = gen_bridge_all(&spec, &generator);
let mut prefixed = String::with_capacity(output.imports.len() * 64 + output.code.len());
let imports_to_emit: Vec<_> = output
.imports
.iter()
.filter(|imp| *imp != "magnus::prelude::*")
.collect();
for imp in &imports_to_emit {
prefixed.push_str("#[allow(unused_imports)]\n");
prefixed.push_str("use ");
prefixed.push_str(imp);
prefixed.push_str(" as _;\n");
}
prefixed.push_str(&output.code);
prefixed
}
}
fn gen_visitor_bridge(
out: &mut String,
trait_type: &TypeDef,
bridge_cfg: &TraitBridgeConfig,
struct_name: &str,
trait_path: &str,
core_crate: &str,
type_paths: &std::collections::HashMap<String, String>,
) {
let methods: Vec<String> = trait_type
.methods
.iter()
.filter(|m| m.trait_source.is_none())
.map(|m| gen_visitor_method_magnus(m, bridge_cfg, type_paths))
.collect();
let rendered = crate::backends::magnus::template_env::render(
"visitor_bridge.rs.jinja",
minijinja::context! {
core_crate => core_crate,
struct_name => struct_name,
trait_path => trait_path,
methods => methods,
},
);
let debug_count = rendered.matches("impl std::fmt::Debug").count();
if debug_count != 1 {
eprintln!(
"[ALEF BUG] visitor_bridge.rs.jinja rendered {} Debug impls (expected 1) for struct {}",
debug_count, struct_name
);
eprintln!(
"[ALEF BUG] Rendered output (first 2000 chars):\n{}",
&rendered[..rendered.len().min(2000)]
);
}
out.push_str(&rendered);
out.push('\n');
}
fn gen_visitor_method_magnus(
method: &MethodDef,
bridge_cfg: &TraitBridgeConfig,
type_paths: &std::collections::HashMap<String, String>,
) -> String {
let name = &method.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 has_args = !method.params.is_empty();
let args_tuple = if has_args {
let args_exprs: Vec<String> = method.params.iter().map(|p| build_magnus_arg(p, bridge_cfg)).collect();
if args_exprs.len() == 1 {
format!("({},)", args_exprs[0])
} else {
format!("({})", args_exprs.join(", "))
}
} else {
String::new()
};
crate::backends::magnus::template_env::render(
"visitor_method.rs.jinja",
minijinja::context! {
name => name,
signature => signature,
return_type => return_type,
has_args => has_args,
args_tuple => args_tuple,
},
)
}
fn build_magnus_arg(p: &crate::core::ir::ParamDef, bridge_cfg: &TraitBridgeConfig) -> String {
if let TypeRef::Named(n) = &p.ty {
if Some(n.as_str()) == bridge_cfg.context_type.as_deref() {
return format!("nodecontext_to_rb_hash({}{})", if p.is_ref { "" } else { "&" }, p.name);
}
}
if p.optional && matches!(&p.ty, TypeRef::String) {
return format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; match {} {{ Some(s) => ruby.str_new(s).as_value(), None => ruby.qnil().as_value() }} }}",
p.name
);
}
if matches!(&p.ty, TypeRef::String) && p.is_ref {
return format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; ruby.str_new({}) }}",
p.name
);
}
if matches!(&p.ty, TypeRef::String) {
return format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; ruby.str_new({}.as_str()) }}",
p.name
);
}
if matches!(&p.ty, TypeRef::Vec(_)) {
let ruby = "unsafe { magnus::Ruby::get_unchecked() }";
return format!(
"{{ let arr = {ruby}.ary_new_capa({name}.len()); for item in {name} {{ let _ = arr.push(item.to_string()); }} arr }}",
name = p.name,
);
}
p.name.to_string()
}
struct MagnusBridgeGenerator {
core_import: String,
type_paths: HashMap<String, String>,
error_type: String,
error_constructor: String,
}
impl MagnusBridgeGenerator {
fn error_path(&self) -> String {
if self.error_type.contains("::") || self.error_type.contains('<') {
self.error_type.clone()
} else {
format!("{}::{}", self.core_import, self.error_type)
}
}
fn make_error(&self, msg_expr: &str) -> String {
self.error_constructor.replace("{msg}", msg_expr)
}
}
impl TraitBridgeGenerator for MagnusBridgeGenerator {
fn foreign_object_type(&self) -> &str {
"magnus::value::Opaque<magnus::Value>"
}
fn bridge_imports(&self) -> Vec<String> {
vec![
"magnus::value::InnerValue".to_string(),
"magnus::TryConvert".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 is_unit = matches!(method.return_type, TypeRef::Unit);
let args: Vec<String> = method.params.iter().map(|p| self.ruby_arg_expr(p)).collect();
let call = if args.is_empty() {
format!("value.funcall::<_, _, magnus::Value>(\"{name}\", ())")
} else {
let args_tuple = if args.len() == 1 {
format!("({},)", args[0])
} else {
format!("({})", args.join(", "))
};
format!("value.funcall::<_, _, magnus::Value>(\"{name}\", {args_tuple})")
};
let err_expr = if has_error {
self.make_error(&format!("format!(\"Ruby method '{name}' failed: {{}}\", e)"))
} else {
String::new()
};
let mut body = crate::backends::magnus::template_env::render(
"sync_method_body.rs.jinja",
minijinja::context! {
call => call,
has_error => has_error,
is_unit => is_unit,
err_expr => err_expr,
},
);
if !is_unit {
body.push_str(&self.return_conversion(method, has_error, ""));
}
body
}
fn gen_async_method_body(&self, method: &MethodDef, _spec: &TraitBridgeSpec) -> String {
let name = &method.name;
let has_error = method.error_type.is_some();
let is_unit = matches!(method.return_type, TypeRef::Unit);
let conversions: Vec<String> = method
.params
.iter()
.map(|p| match (&p.ty, p.is_ref) {
(TypeRef::String, true) => format!("let {}_owned = {}.to_string();\n", p.name, p.name),
(TypeRef::Bytes, true) => format!("let {}_owned = {}.to_vec();\n", p.name, p.name),
(TypeRef::Path, true) => format!("let {}_owned = {}.to_path_buf();\n", p.name, p.name),
_ => format!("let {}_owned = {}.clone();\n", p.name, p.name),
})
.collect();
let conversion_bindings = conversions.join("");
let return_type_rust = if is_unit {
"()".to_string()
} else {
self.return_rust_type(&method.return_type)
};
let err_path = self.error_path();
let result_ty = if has_error {
format!("std::result::Result<{return_type_rust}, {err_path}>")
} else {
return_type_rust.clone()
};
let conversions = format!(
"let inner = self.inner;\n\
let cached_name = self.cached_name.clone();\n\
// cached_name is referenced both inside the spawn_blocking closure and after\n\
// the await for the JoinError fallback, so clone once for each consumer.\n\
let cached_name_for_blocking = cached_name.clone();\n\
{conversion_bindings}\n",
conversion_bindings = conversion_bindings,
);
let args: Vec<String> = method
.params
.iter()
.map(|p| {
let param_name = if matches!(&p.ty, TypeRef::String) && p.is_ref {
format!("{}_owned.as_str()", p.name)
} else {
format!("{}_owned", p.name)
};
self.ruby_arg_expr_custom(&p.ty, ¶m_name)
})
.collect();
let call = if args.is_empty() {
format!("value.funcall::<_, _, magnus::Value>(\"{name}\", ())")
} else {
let args_tuple = if args.len() == 1 {
format!("({},)", args[0])
} else {
format!("({})", args.join(", "))
};
format!("value.funcall::<_, _, magnus::Value>(\"{name}\", {args_tuple})")
};
let err_expr_call = self.make_error(&format!(
"format!(\"Plugin '{{}}' method '{name}' failed: {{}}\", cached_name_for_blocking, e)"
));
let err_expr_join = if has_error {
self.make_error("format!(\"spawn_blocking failed for '{}': {}\", cached_name, e)")
} else {
String::new()
};
crate::backends::magnus::template_env::render(
"trait_bridge_async_method_body.rs.jinja",
minijinja::context! {
conversions => conversions,
call => call,
has_error => has_error,
is_unit => is_unit,
result_ty => result_ty,
err_expr_call => err_expr_call,
err_expr_join => err_expr_join,
return_conversion => self.return_conversion(method, has_error, " "),
},
)
}
fn gen_constructor(&self, spec: &TraitBridgeSpec) -> String {
let wrapper = spec.wrapper_name();
let required_methods: Vec<_> = spec.required_methods().iter().map(|m| m.name.as_str()).collect();
crate::backends::magnus::template_env::render(
"trait_bridge_constructor.rs.jinja",
minijinja::context! {
wrapper => wrapper,
required_methods => required_methods,
},
)
}
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 = crate::codegen::generators::trait_bridge::host_function_path(spec, unregister_fn);
let func = format!(
"pub fn {unregister_fn}(name: String) -> Result<(), magnus::Error> {{\n\
{host_path}(&name).map_err(|e| {{\n\
let ruby = unsafe {{ magnus::Ruby::get_unchecked() }};\n\
magnus::Error::new(ruby.exception_runtime_error(), format!(\"{{}}\", e))\n\
}})\n\
}}\n"
);
func
}
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 = crate::codegen::generators::trait_bridge::host_function_path(spec, clear_fn);
let mut out = String::with_capacity(512);
let func = format!(
"pub fn {clear_fn}() -> Result<(), magnus::Error> {{\n\
{host_path}().map_err(|e| {{\n\
let ruby = unsafe {{ magnus::Ruby::get_unchecked() }};\n\
magnus::Error::new(ruby.exception_runtime_error(), format!(\"{{}}\", e))\n\
}})\n\
}}\n"
);
out.push_str(&func);
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 required_methods: Vec<_> = spec
.required_methods()
.iter()
.map(|m| format!("\"{}\"", m.name))
.collect();
let required_methods = required_methods.join(", ");
let register_extra_args = spec
.bridge_config
.register_extra_args
.as_deref()
.map(|a| format!(", {a}"))
.unwrap_or_default();
crate::backends::magnus::template_env::render(
"trait_bridge_registration_fn.rs.jinja",
minijinja::context! {
register_fn => register_fn,
registry_getter => registry_getter,
wrapper => wrapper,
trait_path => trait_path,
required_methods => required_methods,
register_extra_args => register_extra_args,
},
)
}
}
impl MagnusBridgeGenerator {
fn return_rust_type(&self, ty: &TypeRef) -> String {
match ty {
TypeRef::Primitive(p) => {
use crate::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",
}
.to_string()
}
TypeRef::String => "String".to_string(),
TypeRef::Bytes => "Vec<u8>".to_string(),
TypeRef::Vec(inner) => format!("Vec<{}>", self.return_rust_type(inner)),
TypeRef::Optional(inner) => format!("Option<{}>", self.return_rust_type(inner)),
TypeRef::Named(name) => self
.type_paths
.get(name.as_str())
.cloned()
.unwrap_or_else(|| format!("{}::{}", self.core_import, name)),
TypeRef::Unit => "()".to_string(),
TypeRef::Map(k, v) => format!(
"std::collections::HashMap<{}, {}>",
self.return_rust_type(k),
self.return_rust_type(v)
),
TypeRef::Json => "serde_json::Value".to_string(),
TypeRef::Duration => "std::time::Duration".to_string(),
TypeRef::Char => "char".to_string(),
TypeRef::Path => "std::path::PathBuf".to_string(),
}
}
fn needs_json_marshalling(&self, ty: &TypeRef) -> bool {
match ty {
TypeRef::Named(_) | TypeRef::Json => true,
TypeRef::Vec(inner) | TypeRef::Optional(inner) => self.needs_json_marshalling(inner),
TypeRef::Map(k, v) => self.needs_json_marshalling(k) || self.needs_json_marshalling(v),
_ => false,
}
}
fn return_conversion(&self, method: &MethodDef, has_error: bool, indent: &str) -> String {
let rust_ty = self.return_rust_type(&method.return_type);
let err_non_json = if has_error {
self.make_error(&format!(
"format!(\"Ruby method '{}' returned non-JSON value: {{}}\", e)",
method.name
))
} else {
String::new()
};
let err_deserialize = if has_error {
self.make_error(&format!(
"format!(\"Failed to deserialize Ruby '{}' return value: {{}}\", e)",
method.name
))
} else {
String::new()
};
let err_convert = if has_error {
self.make_error(&format!(
"format!(\"Failed to convert Ruby '{}' return value: {{}}\", e)",
method.name
))
} else {
String::new()
};
crate::backends::magnus::template_env::render(
"trait_bridge_return_conversion.rs.jinja",
minijinja::context! {
has_error => has_error,
needs_json => self.needs_json_marshalling(&method.return_type),
indent => indent,
rust_ty => rust_ty,
err_non_json => err_non_json,
err_deserialize => err_deserialize,
err_convert => err_convert,
},
)
}
fn ruby_arg_expr(&self, p: &crate::core::ir::ParamDef) -> String {
self.ruby_arg_expr_custom(&p.ty, &p.name)
}
fn ruby_arg_expr_custom(&self, ty: &TypeRef, var: &str) -> String {
match ty {
TypeRef::String => format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; ruby.str_new(AsRef::<str>::as_ref(&{var})).as_value() }}"
),
TypeRef::Bytes => format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; ruby.str_new(String::from_utf8_lossy(AsRef::<[u8]>::as_ref(&{var})).as_ref()).as_value() }}"
),
TypeRef::Named(_) | TypeRef::Json => format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; serde_json::to_string(&{var}).ok().map(|s| ruby.str_new(s.as_str()).as_value()).unwrap_or_else(|| ruby.qnil().as_value()) }}"
),
TypeRef::Vec(_) | TypeRef::Map(_, _) | TypeRef::Optional(_) => format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; serde_json::to_string(&{var}).ok().map(|s| ruby.str_new(s.as_str()).as_value()).unwrap_or_else(|| ruby.qnil().as_value()) }}"
),
TypeRef::Path => format!(
"{{ let ruby = unsafe {{ magnus::Ruby::get_unchecked() }}; ruby.str_new(<_ as AsRef<std::path::Path>>::as_ref(&{var}).to_string_lossy().as_ref()).as_value() }}"
),
_ => var.to_string(),
}
}
}
#[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,
opaque_types: &ahash::AHashSet<String>,
default_types: &std::collections::HashSet<&str>,
core_import: &str,
) -> String {
use crate::core::ir::TypeRef;
let struct_name = crate::codegen::generators::trait_bridge::bridge_wrapper_name("Rb", 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 mut sig_parts = Vec::new();
for (idx, p) in func.params.iter().enumerate() {
if idx == bridge_param_idx {
if is_optional {
sig_parts.push(format!("{}: Option<magnus::Value>", p.name));
} else {
sig_parts.push(format!("{}: magnus::Value", 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 has_error = func.error_type.is_some();
let ret = mapper.wrap_return(&return_type, has_error);
let err_conv = ".map_err(|e| magnus::Error::new(unsafe { magnus::Ruby::get_unchecked() }.exception_runtime_error(), e.to_string()))";
let bridge_wrap = if is_optional {
format!(
"let {param_name}: Option<{handle_path}> = match {param_name} {{\n \
Some(v) if !v.is_nil() => {{\n \
let bridge = {struct_name}::new(v);\n \
Some(std::sync::Arc::new(std::sync::Mutex::new(bridge)) as {handle_path})\n \
}},\n \
_ => None,\n \
}};"
)
} else {
format!(
"let {param_name} = {{\n \
let bridge = {struct_name}::new({param_name});\n \
std::sync::Arc::new(std::sync::Mutex::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 named_type = 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 core_path = format!("{core_import}::{named_type}");
let is_dt = default_types.contains(named_type.as_str());
if is_dt {
if p.optional || matches!(&p.ty, TypeRef::Optional(_)) {
format!("let {name}_core: Option<{core_path}> = {name}.map(Into::into);\n ")
} else {
format!("let {name}_core: {core_path} = {name}.into();\n ")
}
} else if p.optional || matches!(&p.ty, TypeRef::Optional(_)) {
format!(
"let {name}_core: Option<{core_path}> = {name}.as_deref().filter(|s| *s != \"nil\").map(|s| serde_json::from_str(s){err_conv}).transpose()?;\n "
)
} else {
format!(
"let {name}_core: {core_path} = serde_json::from_str(&{name}){err_conv}?;\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 func_name = &func.name;
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("#[allow(unused_variables)]\n");
let sig = format!("pub fn {func_name}({params_str}) -> {ret} {{\n {body}\n}}\n");
out.push_str(&sig);
out
}
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,
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("Rb", bridge_cfg);
let handle_path = crate::codegen::generators::trait_bridge::bridge_handle_path(api, bridge_cfg, core_import);
let non_option_params: Vec<_> = func
.params
.iter()
.enumerate()
.filter(|(idx, _)| *idx != options_param_idx)
.collect();
let mut sig_parts = Vec::new();
for (_, p) in &non_option_params {
let ty = mapper.map_type(&p.ty);
sig_parts.push(format!("{}: {}", p.name, ty));
}
let bridge_param_name = bridge_cfg.param_name.as_deref().unwrap_or("visitor");
sig_parts.push(format!("{bridge_param_name}: Option<magnus::Value>"));
let _params_str = sig_parts.join(", ");
let return_type = mapper.map_type(&func.return_type);
let has_error = func.error_type.is_some();
let ret = mapper.wrap_return(&return_type, has_error);
let err_conv = ".map_err(|e| magnus::Error::new(unsafe { magnus::Ruby::get_unchecked() }.exception_runtime_error(), e.to_string()))";
let options_name = &func.params[options_param_idx].name;
let Some(options_field) = bridge_cfg.resolved_options_field() else {
return String::new();
};
let Some(options_type) = ({
let raw = &func.params[options_param_idx].ty;
let inner = match raw {
TypeRef::Optional(b) => b.as_ref(),
other => other,
};
if let TypeRef::Named(n) = inner {
Some(n.as_str())
} else {
bridge_cfg.options_type.as_deref()
}
}) else {
return String::new();
};
let visitor_extract = format!(
"let {options_name}_core = match {bridge_param_name} {{\n \
Some(v) if !v.is_nil() => {{\n \
if magnus::RHash::from_value(v).is_some() {{\n \
let json = v.funcall::<_, _, String>(\"to_json\", ()).map_err(|e| {{\n \
magnus::Error::new(\n \
unsafe {{ magnus::Ruby::get_unchecked() }}.exception_runtime_error(),\n \
format!(\"failed to serialize Ruby options to JSON: {{}}\", e),\n \
)\n \
}})?;\n \
serde_json::from_str::<{core_import}::{options_type}>(&json).map_err(|e| {{\n \
magnus::Error::new(\n \
unsafe {{ magnus::Ruby::get_unchecked() }}.exception_runtime_error(),\n \
format!(\"failed to deserialize options JSON: {{}}\", e),\n \
)\n \
}})?\n \
}} else if let Ok(opts_binding) = <&{options_type} as magnus::TryConvert>::try_convert(v) {{\n \
opts_binding.clone().into()\n \
}} else {{\n \
let bridge = {struct_name}::new(v);\n \
let handle = std::sync::Arc::new(std::sync::Mutex::new(bridge)) as {handle_path};\n \
let mut opts = {core_import}::{options_type}::default();\n \
opts.{options_field} = Some(handle);\n \
opts\n \
}}\n \
}},\n \
_ => {core_import}::{options_type}::default(),\n \
}};",
struct_name = struct_name,
handle_path = handle_path,
core_import = core_import,
options_name = options_name,
options_type = options_type,
options_field = options_field,
bridge_param_name = bridge_param_name,
);
let call_args: String = non_option_params
.iter()
.map(|(_, p)| 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(),
})
.chain(std::iter::once(format!("Some({options_name}_core)")))
.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 = if func.error_type.is_some() {
if return_wrap == "val" {
format!("{visitor_extract}\n {core_call}{err_conv}")
} else {
format!("{visitor_extract}\n {core_call}.map(|val| {return_wrap}){err_conv}")
}
} else {
format!("{visitor_extract}\n {core_call}")
};
let func_name = &func.name;
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("#[allow(unused_variables)]\n");
out.push_str("pub fn ");
out.push_str(func_name);
out.push_str("(args: &[magnus::Value]) -> ");
out.push_str(&ret);
out.push_str(" {\n");
out.push_str(" let args = magnus::scan_args::scan_args::<\n");
out.push_str(" (");
for (_, p) in &non_option_params {
out.push_str(&mapper.map_type(&p.ty));
out.push_str(", ");
}
out.push_str("), (Option<magnus::Value>,), (), (), (), ()\n");
out.push_str(" >(args)?;\n");
out.push_str(" let (");
for (i, (_, p)) in non_option_params.iter().enumerate() {
if i > 0 {
out.push_str(", ");
}
out.push_str(&p.name);
}
out.push_str(",) = args.required;\n");
out.push_str(" let (");
out.push_str(bridge_param_name);
out.push_str(",) = args.optional;\n");
out.push_str(" ");
out.push_str(&body);
out.push('\n');
out.push_str("}\n");
out
}