use super::errors::{
emit_return_marshalling, emit_return_marshalling_indented, emit_return_statement, emit_return_statement_indented,
};
use super::{
csharp_file_header, emit_named_param_setup, emit_named_param_teardown, emit_named_param_teardown_indented,
is_tuple_field, returns_ptr,
};
use crate::type_map::csharp_type;
use alef_codegen::naming::to_csharp_name;
use alef_core::ir::{DefaultValue, MethodDef, PrimitiveType, TypeDef, TypeRef};
use heck::{ToLowerCamelCase, ToPascalCase};
use std::collections::HashSet;
pub(super) fn gen_opaque_handle(
typ: &TypeDef,
namespace: &str,
exception_name: &str,
enum_names: &HashSet<String>,
streaming_methods: &HashSet<String>,
all_opaque_type_names: &HashSet<String>,
) -> String {
use crate::template_env::render;
use minijinja::Value;
let has_methods = typ.methods.iter().any(|m| !streaming_methods.contains(&m.name));
let uses_list = |tr: &TypeRef| -> bool {
matches!(tr, TypeRef::Vec(_))
|| matches!(tr, TypeRef::Optional(inner) if matches!(inner.as_ref(), TypeRef::Vec(_)))
};
let needs_list = has_methods
&& typ
.methods
.iter()
.any(|m| uses_list(&m.return_type) || m.params.iter().any(|p| uses_list(&p.ty)));
let needs_async = has_methods
&& typ
.methods
.iter()
.any(|m| m.is_async && !streaming_methods.contains(&m.name));
let class_name = typ.name.to_pascal_case();
let free_method = format!("{}Free", class_name);
let doc_lines: Vec<String> = if !typ.doc.is_empty() {
typ.doc.lines().map(|l| l.to_string()).collect()
} else {
vec![]
};
let mut out = render(
"opaque_handle_header.jinja",
Value::from_serialize(serde_json::json!({
"namespace": namespace,
"class_name": class_name,
"free_method": free_method,
"has_methods": has_methods,
"needs_list": needs_list,
"needs_async": needs_async,
"doc": !typ.doc.is_empty(),
"doc_lines": doc_lines,
})),
);
out.push('\n');
let true_opaque_types = all_opaque_type_names;
for method in typ.methods.iter().filter(|m| !streaming_methods.contains(&m.name)) {
out.push('\n');
out.push_str(&gen_opaque_method(
method,
&class_name,
exception_name,
enum_names,
true_opaque_types,
));
}
out.push_str("}\n");
out
}
fn gen_opaque_method(
method: &MethodDef,
class_name: &str,
exception_name: &str,
enum_names: &HashSet<String>,
true_opaque_types: &HashSet<String>,
) -> String {
use crate::template_env::render;
let mut out = String::new();
let visible_params: Vec<alef_core::ir::ParamDef> = method.params.clone();
if !method.doc.is_empty() {
out.push_str(" /// <summary>\n");
for line in method.doc.lines() {
out.push_str(&render("doc_line_indented.jinja", minijinja::context! { line }));
}
out.push_str(" /// </summary>\n");
}
let return_type_str = if method.is_async {
if method.return_type == TypeRef::Unit {
"async Task".to_string()
} else {
let return_type = csharp_type(&method.return_type);
render("async_task_return_type.jinja", minijinja::context! { return_type })
.trim_end_matches('\n')
.to_string()
}
} else if method.return_type == TypeRef::Unit {
"void".to_string()
} else {
csharp_type(&method.return_type).to_string()
};
let method_cs_name = to_csharp_name(&method.name);
let is_static = method.is_static || method.receiver.is_none();
let static_kw = if is_static { "static " } else { "" };
out.push_str(
render(
"opaque_method_header.jinja",
minijinja::context! { static_kw, return_type_str, method_cs_name },
)
.trim_end_matches('\n'),
);
for (i, param) in visible_params.iter().enumerate() {
let param_name = param.name.to_lower_camel_case();
let param_type = csharp_type(¶m.ty);
if param.optional && !param_type.ends_with('?') {
out.push_str(
render(
"param_decl_optional.jinja",
minijinja::context! { param_type, param_name },
)
.trim_end_matches('\n'),
);
} else {
out.push_str(
render(
"param_decl_required.jinja",
minijinja::context! { param_type, param_name },
)
.trim_end_matches('\n'),
);
}
if i < visible_params.len() - 1 {
out.push_str(", ");
}
}
out.push_str(")\n {\n");
emit_named_param_setup(&mut out, &visible_params, " ", true_opaque_types);
let cs_native_name = format!("{class_name}{method_cs_name}");
if method.is_async {
if method.return_type == TypeRef::Unit {
out.push_str(" await Task.Run(() =>\n {\n");
} else {
out.push_str(" return await Task.Run(() =>\n {\n");
}
if method.return_type != TypeRef::Unit {
out.push_str(" var nativeResult = ");
} else {
out.push_str(" ");
}
out.push_str(&render(
"native_call_start.jinja",
minijinja::context! { method_name => &cs_native_name },
));
if !is_static {
out.push_str(" Handle");
for param in &visible_params {
let param_name = param.name.to_lower_camel_case();
let arg = super::native_call_arg(¶m.ty, ¶m_name, param.optional, true_opaque_types);
out.push_str(",\n");
out.push_str(render("indented_arg_async.jinja", minijinja::context! { arg }).trim_end_matches('\n'));
}
} else {
for (i, param) in visible_params.iter().enumerate() {
let param_name = param.name.to_lower_camel_case();
let arg = super::native_call_arg(¶m.ty, ¶m_name, param.optional, true_opaque_types);
if i == 0 {
out.push_str(
render("indented_arg_async.jinja", minijinja::context! { arg }).trim_end_matches('\n'),
);
} else {
out.push_str(",\n");
out.push_str(
render("indented_arg_async.jinja", minijinja::context! { arg }).trim_end_matches('\n'),
);
}
}
}
out.push_str("\n );\n");
if method.return_type != TypeRef::Unit && returns_ptr(&method.return_type) {
if matches!(method.return_type, TypeRef::Optional(_)) {
out.push_str(
" if (nativeResult == IntPtr.Zero)\n {\n return null;\n }\n",
);
} else {
out.push_str(&render(
"null_result_throw.jinja",
minijinja::context! { indent => " ", exception_name, cs_native_name },
));
}
}
emit_return_marshalling_indented(
&mut out,
&method.return_type,
" ",
enum_names,
true_opaque_types,
);
emit_named_param_teardown_indented(&mut out, &visible_params, " ", true_opaque_types);
emit_return_statement_indented(&mut out, &method.return_type, " ");
out.push_str(" });\n");
} else {
if method.return_type != TypeRef::Unit {
out.push_str(" var nativeResult = ");
} else {
out.push_str(" ");
}
out.push_str(&render(
"native_call_start.jinja",
minijinja::context! { method_name => &cs_native_name },
));
if !is_static {
out.push_str(" Handle");
for param in &visible_params {
let param_name = param.name.to_lower_camel_case();
let arg = super::native_call_arg(¶m.ty, ¶m_name, param.optional, true_opaque_types);
out.push_str(",\n");
out.push_str(render("indented_arg_sync.jinja", minijinja::context! { arg }).trim_end_matches('\n'));
}
} else {
for (i, param) in visible_params.iter().enumerate() {
let param_name = param.name.to_lower_camel_case();
let arg = super::native_call_arg(¶m.ty, ¶m_name, param.optional, true_opaque_types);
if i == 0 {
out.push_str(render("indented_arg_sync.jinja", minijinja::context! { arg }).trim_end_matches('\n'));
} else {
out.push_str(",\n");
out.push_str(render("indented_arg_sync.jinja", minijinja::context! { arg }).trim_end_matches('\n'));
}
}
}
out.push_str("\n );\n");
if method.return_type != TypeRef::Unit && returns_ptr(&method.return_type) {
if matches!(method.return_type, TypeRef::Optional(_)) {
out.push_str(
" if (nativeResult == IntPtr.Zero)\n {\n return null;\n }\n",
);
} else {
out.push_str(&render(
"null_result_throw.jinja",
minijinja::context! { indent => " ", exception_name, cs_native_name },
));
}
}
emit_return_marshalling(&mut out, &method.return_type, enum_names, true_opaque_types);
emit_named_param_teardown(&mut out, &visible_params, true_opaque_types);
emit_return_statement(&mut out, &method.return_type);
}
out.push_str(" }\n");
out
}
pub(super) fn gen_record_type(
typ: &TypeDef,
namespace: &str,
enum_names: &HashSet<String>,
complex_enums: &HashSet<String>,
custom_converter_enums: &HashSet<String>,
_lang_rename_all: &str,
bridge_type_aliases: &HashSet<String>,
) -> String {
use crate::template_env::render;
let mut out = csharp_file_header();
out.push_str("using System;\n");
out.push_str("using System.Collections.Generic;\n");
out.push_str("using System.Text.Json;\n");
out.push_str("using System.Text.Json.Serialization;\n\n");
out.push_str(&render("namespace_decl.jinja", minijinja::context! { namespace }));
out.push('\n');
if !typ.doc.is_empty() {
out.push_str("/// <summary>\n");
for line in typ.doc.lines() {
out.push_str(&render("doc_line.jinja", minijinja::context! { line }));
}
out.push_str("/// </summary>\n");
}
let class_name = typ.name.to_pascal_case();
out.push_str(&render("record_class_header.jinja", minijinja::context! { class_name }));
out.push_str("{\n");
for field in &typ.fields {
if is_tuple_field(field) {
continue;
}
if !field.doc.is_empty() {
out.push_str(" /// <summary>\n");
for line in field.doc.lines() {
out.push_str(&render("doc_line_indented.jinja", minijinja::context! { line }));
}
out.push_str(" /// </summary>\n");
}
let is_visitor_bridge = match &field.ty {
TypeRef::Named(n) => bridge_type_aliases.contains(n),
TypeRef::Optional(inner) => matches!(inner.as_ref(), TypeRef::Named(n) if bridge_type_aliases.contains(n)),
_ => false,
};
let field_base_type = match &field.ty {
TypeRef::Named(n) => Some(n.to_pascal_case()),
TypeRef::Optional(inner) => match inner.as_ref() {
TypeRef::Named(n) => Some(n.to_pascal_case()),
_ => None,
},
_ => None,
};
if let Some(ref base) = field_base_type {
if custom_converter_enums.contains(base) {
out.push_str(&render("json_converter_attr.jinja", minijinja::context! { base }));
}
}
if is_visitor_bridge {
out.push_str(" [JsonIgnore]\n");
} else {
let json_name = field.name.clone();
out.push_str(&render(
"json_property_name_attr.jinja",
minijinja::context! { json_name },
));
}
let cs_name = to_csharp_name(&field.name);
let is_complex = matches!(&field.ty, TypeRef::Named(n) if complex_enums.contains(&n.to_pascal_case()));
if is_visitor_bridge {
out.push_str(&render(
"visitor_bridge_property.jinja",
minijinja::context! { cs_name },
));
out.push('\n');
continue;
}
if field.optional {
let mapped = if is_complex {
"JsonElement".to_string()
} else {
csharp_type(&field.ty).to_string()
};
let field_type = if mapped.ends_with('?') {
mapped
} else {
format!("{mapped}?")
};
out.push_str(&render(
"property_with_default.jinja",
minijinja::context! { field_type, cs_name, default_val => "null" },
));
} else if typ.has_default || field.default.is_some() {
let base_type = if is_complex {
"JsonElement".to_string()
} else {
csharp_type(&field.ty).to_string()
};
if matches!(&field.ty, TypeRef::Duration) {
let nullable_type = if base_type.ends_with('?') {
base_type.clone()
} else {
format!("{}?", base_type)
};
out.push_str(&render(
"property_with_default.jinja",
minijinja::context! { field_type => nullable_type, cs_name, default_val => "null" },
));
out.push('\n');
continue;
}
let default_val = match &field.typed_default {
Some(DefaultValue::BoolLiteral(b)) => b.to_string(),
Some(DefaultValue::IntLiteral(n)) => n.to_string(),
Some(DefaultValue::FloatLiteral(f)) => {
let s = f.to_string();
let s = if s.contains('.') { s } else { format!("{s}.0") };
match &field.ty {
TypeRef::Primitive(PrimitiveType::F32) => format!("{}f", s),
_ => s,
}
}
Some(DefaultValue::StringLiteral(s)) => {
let escaped = s
.replace('\\', "\\\\")
.replace('"', "\\\"")
.replace('\n', "\\n")
.replace('\r', "\\r")
.replace('\t', "\\t");
format!("\"{}\"", escaped)
}
Some(DefaultValue::EnumVariant(v)) => {
if base_type == "string" || base_type == "string?" {
format!("\"{}\"", v.to_pascal_case())
} else if base_type == "JsonElement" || base_type == "JsonElement?" {
"null".to_string()
} else {
format!("{}.{}", base_type, v.to_pascal_case())
}
}
Some(DefaultValue::None) => "null".to_string(),
Some(DefaultValue::Empty) | None => match &field.ty {
TypeRef::Vec(_) => "[]".to_string(),
TypeRef::Map(k, v) => format!("new Dictionary<{}, {}>()", csharp_type(k), csharp_type(v)),
TypeRef::String | TypeRef::Char | TypeRef::Path => "\"\"".to_string(),
TypeRef::Json => "null".to_string(),
TypeRef::Bytes => "Array.Empty<byte>()".to_string(),
TypeRef::Primitive(p) => match p {
PrimitiveType::Bool => "false".to_string(),
PrimitiveType::F32 => "0.0f".to_string(),
PrimitiveType::F64 => "0.0".to_string(),
_ => "0".to_string(),
},
TypeRef::Named(name) => {
let pascal = name.to_pascal_case();
if complex_enums.contains(&pascal) {
"null".to_string()
} else if enum_names.contains(&pascal) {
"null".to_string()
} else {
"default!".to_string()
}
}
_ => "default!".to_string(),
},
};
let field_type = if (default_val == "null" && !base_type.ends_with('?')) || is_complex {
format!("{}?", base_type)
} else {
base_type
};
out.push_str(&render(
"property_with_default.jinja",
minijinja::context! { field_type, cs_name, default_val },
));
} else {
let field_type = if is_complex {
"JsonElement".to_string()
} else {
csharp_type(&field.ty).to_string()
};
if matches!(&field.ty, TypeRef::Duration) {
out.push_str(&render(
"property_with_default.jinja",
minijinja::context! { field_type, cs_name, default_val => "null" },
));
} else {
let default_val = match &field.ty {
TypeRef::String | TypeRef::Char | TypeRef::Path | TypeRef::Json => "\"\"",
TypeRef::Vec(_) => "[]",
TypeRef::Bytes => "Array.Empty<byte>()",
TypeRef::Primitive(PrimitiveType::Bool) => "false",
TypeRef::Primitive(PrimitiveType::F32) => "0.0f",
TypeRef::Primitive(PrimitiveType::F64) => "0.0",
TypeRef::Primitive(_) => "0",
_ => "default!",
};
out.push_str(&render(
"property_with_default.jinja",
minijinja::context! { field_type, cs_name, default_val },
));
}
}
out.push('\n');
}
out.push_str("}\n");
out
}