use super::errors::{emit_return_marshalling_indented, emit_return_statement, emit_return_statement_indented};
use super::{
StreamingMethodMeta, 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_codegen::shared::binding_fields;
use alef_core::ir::{DefaultValue, MethodDef, PrimitiveType, TypeDef, TypeRef};
use heck::{ToLowerCamelCase, ToPascalCase};
use std::collections::{HashMap, HashSet};
pub(super) fn gen_opaque_handle(
typ: &TypeDef,
namespace: &str,
exception_name: &str,
enum_names: &HashSet<String>,
streaming_methods: &HashSet<String>,
streaming_methods_meta: &HashMap<String, StreamingMethodMeta>,
all_opaque_type_names: &HashSet<String>,
) -> String {
use crate::template_env::render;
use minijinja::Value;
let has_streaming = typ
.methods
.iter()
.any(|m| streaming_methods.contains(&m.name) && streaming_methods_meta.contains_key(&m.name));
let has_methods = has_streaming || 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_streaming
|| (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_streaming
|| (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,
"needs_streaming": has_streaming,
"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 {
if streaming_methods.contains(&method.name) {
if let Some(meta) = streaming_methods_meta.get(&method.name) {
out.push('\n');
out.push_str(&gen_opaque_streaming_method(method, &class_name, exception_name, meta));
}
continue;
}
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_streaming_method(
method: &MethodDef,
class_name: &str,
exception_name: &str,
meta: &StreamingMethodMeta,
) -> String {
use crate::template_env::render;
use minijinja::Value;
let cs_method_name = to_csharp_name(&method.name);
let cs_type_name = class_name.to_string();
let item_pascal = meta.item_type.to_pascal_case();
let req_param = method.params.iter().find(|p| matches!(&p.ty, TypeRef::Named(_)));
let (req_pascal, req_param_name) = match req_param {
Some(p) => match &p.ty {
TypeRef::Named(n) => (n.to_pascal_case(), p.name.to_lower_camel_case()),
_ => (item_pascal.clone(), "req".to_string()),
},
None => (item_pascal.clone(), "req".to_string()),
};
let req_param_type = req_pascal.clone();
let start_native = format!("{cs_type_name}{cs_method_name}Start");
let next_native = format!("{cs_type_name}{cs_method_name}Next");
let free_native = format!("{cs_type_name}{cs_method_name}Free");
let req_from_json = format!("{req_pascal}FromJson");
let req_free = format!("{req_pascal}Free");
let item_to_json = format!("{item_pascal}ToJson");
let item_free = format!("{item_pascal}Free");
let doc_lines: Vec<String> = method.doc.lines().map(ToString::to_string).collect();
render(
"opaque_streaming_method.jinja",
Value::from_serialize(serde_json::json!({
"has_doc": !method.doc.is_empty(),
"doc_lines": doc_lines,
"method_name": cs_method_name,
"item_type": item_pascal,
"request_type": req_param_type,
"request_param": req_param_name,
"request_from_json": req_from_json,
"request_free": req_free,
"start_native": start_native,
"next_native": next_native,
"free_native": free_native,
"item_to_json": item_to_json,
"item_free": item_free,
"exception_name": exception_name,
})),
)
}
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() {
let doc_lines: Vec<String> = method.doc.lines().map(ToString::to_string).collect();
out.push_str(&render(
"doc_comment_block.jinja",
minijinja::context! {
has_doc => true,
indent => " ",
doc_lines => doc_lines,
},
));
}
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, exception_name);
let cs_native_name = format!("{class_name}{method_cs_name}");
if super::functions::is_bytes_result_method(method) {
let mut args_block = String::new();
let arg_indent = if method.is_async {
" "
} else {
" "
};
if !is_static {
args_block.push_str(&render(
"native_arg_line.jinja",
minijinja::context! { indent => arg_indent, arg => "Handle" },
));
}
for param in visible_params.iter() {
let param_name = param.name.to_lower_camel_case();
let arg = super::native_call_arg(¶m.ty, ¶m_name, param.optional, true_opaque_types);
args_block.push_str(&render(
"native_arg_line.jinja",
minijinja::context! { indent => arg_indent, arg },
));
if matches!(param.ty, TypeRef::Bytes) {
args_block.push_str(&render(
"native_bytes_len_arg_line.jinja",
minijinja::context! { indent => arg_indent, param_name },
));
}
}
out.push_str(&render(
"opaque_bytes_result_call.jinja",
minijinja::context! {
is_async => method.is_async,
native_method_name => &cs_native_name,
args_block => &args_block,
exception_name,
},
));
out.push_str(" }\n\n");
return out;
}
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'));
if matches!(param.ty, TypeRef::Bytes) {
out.push_str(",\n");
out.push_str(
render(
"indented_arg_async.jinja",
minijinja::context! { arg => format!("(nuint){param_name}.Length") },
)
.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'),
);
}
if matches!(param.ty, TypeRef::Bytes) {
out.push_str(",\n");
out.push_str(
render(
"indented_arg_async.jinja",
minijinja::context! { arg => format!("(nuint){param_name}.Length") },
)
.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,
&HashSet::new(), );
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'));
if matches!(param.ty, TypeRef::Bytes) {
out.push_str(",\n");
out.push_str(
render(
"indented_arg_sync.jinja",
minijinja::context! { arg => format!("(nuint){param_name}.Length") },
)
.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'));
}
if matches!(param.ty, TypeRef::Bytes) {
out.push_str(",\n");
out.push_str(
render(
"indented_arg_sync.jinja",
minijinja::context! { arg => format!("(nuint){param_name}.Length") },
)
.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,
&HashSet::new(),
);
emit_named_param_teardown(&mut out, &visible_params, true_opaque_types);
emit_return_statement(&mut out, &method.return_type);
}
out.push_str(" }\n");
out
}
#[allow(clippy::too_many_arguments)]
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>,
exception_class: &str,
) -> 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() {
let doc_lines: Vec<String> = typ.doc.lines().map(ToString::to_string).collect();
out.push_str(&render(
"doc_comment_block.jinja",
minijinja::context! {
has_doc => true,
indent => "",
doc_lines => doc_lines,
},
));
}
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 binding_fields(&typ.fields) {
if is_tuple_field(field) {
continue;
}
if !field.doc.is_empty() {
let doc_lines: Vec<String> = field.doc.lines().map(ToString::to_string).collect();
out.push_str(&render(
"doc_comment_block.jinja",
minijinja::context! {
has_doc => true,
indent => " ",
doc_lines => doc_lines,
},
));
}
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 needs_bytes_int_converter = !field.optional && matches!(&field.ty, TypeRef::Bytes);
if needs_bytes_int_converter {
out.push_str(" [JsonConverter(typeof(ByteArrayToIntArrayConverter))]\n");
}
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 }));
}
}
let is_flattened_json = field.serde_flatten && matches!(&field.ty, TypeRef::Json);
if is_flattened_json {
let cs_name = to_csharp_name(&field.name);
out.push_str(" [JsonExtensionData]\n");
out.push_str(&render(
"json_extension_data_property.jinja",
minijinja::context! { cs_name },
));
out.push('\n');
continue;
}
if is_visitor_bridge {
out.push_str(" [JsonIgnore]\n");
} else {
let json_name = field.serde_rename.clone().unwrap_or_else(|| 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 => "[]".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()
};
let should_emit_required = match &field.ty {
TypeRef::String | TypeRef::Char | TypeRef::Path | TypeRef::Json => true,
TypeRef::Named(_) if !is_complex => true,
TypeRef::Vec(_) | TypeRef::Map(_, _) | TypeRef::Bytes => false,
TypeRef::Primitive(_) => false,
TypeRef::Duration => false,
_ => false,
};
if should_emit_required {
out.push_str(&render(
"property_required_init.jinja",
minijinja::context! { field_type, cs_name },
));
} else 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 => "[]",
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 /// <summary>\n");
out.push_str(" /// Parse a <see cref=\"");
out.push_str(&class_name);
out.push_str("\"/> from a JSON string.\n");
out.push_str(" /// </summary>\n");
out.push_str(" /// <exception cref=\"");
out.push_str(exception_class);
out.push_str("\">When the JSON cannot be deserialised.</exception>\n");
out.push_str(" public static ");
out.push_str(&class_name);
out.push_str(" FromJson(string json)\n");
out.push_str(" {\n");
out.push_str(" try\n");
out.push_str(" {\n");
out.push_str(" return JsonSerializer.Deserialize<");
out.push_str(&class_name);
out.push_str(">(json, JsonOptions)\n");
out.push_str(" ?? throw new ");
out.push_str(exception_class);
out.push_str("($\"Failed to parse ");
out.push_str(&class_name);
out.push_str(" from JSON: deserializer returned null\");\n");
out.push_str(" }\n");
out.push_str(" catch (");
out.push_str(exception_class);
out.push_str(")\n");
out.push_str(" {\n");
out.push_str(" throw;\n");
out.push_str(" }\n");
out.push_str(" catch (Exception e)\n");
out.push_str(" {\n");
out.push_str(" throw new ");
out.push_str(exception_class);
out.push_str("($\"Failed to parse ");
out.push_str(&class_name);
out.push_str(" from JSON: {e.Message}\", e);\n");
out.push_str(" }\n");
out.push_str(" }\n");
out.push_str("\n private static readonly JsonSerializerOptions JsonOptions = new()\n");
out.push_str(" {\n");
out.push_str(" DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingDefault,\n");
out.push_str(" Converters = { new JsonStringEnumConverter(JsonNamingPolicy.SnakeCaseLower) },\n");
out.push_str(" };\n");
out.push_str("}\n");
out
}
pub(crate) fn gen_byte_array_to_int_array_converter(namespace: &str) -> 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');
out.push_str("/// <summary>\n");
out.push_str("/// Converts byte arrays to and from JSON integer arrays.\n");
out.push_str("/// </summary>\n");
out.push_str("/// <remarks>\n");
out.push_str("/// System.Text.Json serializes byte[] as base64 strings by default, but Rust's serde\n");
out.push_str("/// for Vec<u8> expects JSON arrays of integers [72, 101, 108, ...].\n");
out.push_str("/// Apply this converter to byte[] fields that are serialized to FFI with\n");
out.push_str("/// [JsonConverter(typeof(ByteArrayToIntArrayConverter))].\n");
out.push_str("/// </remarks>\n");
out.push_str("public sealed class ByteArrayToIntArrayConverter : JsonConverter<byte[]>\n");
out.push_str("{\n");
out.push_str(" /// <summary>\n");
out.push_str(" /// Reads a JSON array of integers and converts it to a byte array.\n");
out.push_str(" /// </summary>\n");
out.push_str(" public override byte[]? Read(\n");
out.push_str(" ref Utf8JsonReader reader,\n");
out.push_str(" Type typeToConvert,\n");
out.push_str(" JsonSerializerOptions options)\n");
out.push_str(" {\n");
out.push_str(" if (reader.TokenType != JsonTokenType.StartArray)\n");
out.push_str(" {\n");
out.push_str(" throw new JsonException(\"Expected JSON array for byte[]\");\n");
out.push_str(" }\n\n");
out.push_str(" var bytes = new List<byte>();\n");
out.push_str(" while (reader.Read())\n");
out.push_str(" {\n");
out.push_str(" if (reader.TokenType == JsonTokenType.EndArray)\n");
out.push_str(" {\n");
out.push_str(" break;\n");
out.push_str(" }\n");
out.push_str(" if (reader.TokenType == JsonTokenType.Number)\n");
out.push_str(" {\n");
out.push_str(" bytes.Add((byte)reader.GetInt32());\n");
out.push_str(" }\n");
out.push_str(" else\n");
out.push_str(" {\n");
out.push_str(" throw new JsonException($\"Unexpected token type: {reader.TokenType}\");\n");
out.push_str(" }\n");
out.push_str(" }\n\n");
out.push_str(" return bytes.ToArray();\n");
out.push_str(" }\n\n");
out.push_str(" /// <summary>\n");
out.push_str(" /// Writes a byte array as a JSON array of integers.\n");
out.push_str(" /// </summary>\n");
out.push_str(" public override void Write(\n");
out.push_str(" Utf8JsonWriter writer,\n");
out.push_str(" byte[] value,\n");
out.push_str(" JsonSerializerOptions options)\n");
out.push_str(" {\n");
out.push_str(" writer.WriteStartArray();\n");
out.push_str(" foreach (var b in value)\n");
out.push_str(" {\n");
out.push_str(" writer.WriteNumberValue(b);\n");
out.push_str(" }\n");
out.push_str(" writer.WriteEndArray();\n");
out.push_str(" }\n");
out.push_str("}\n");
out
}