1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
//! C# enum and tagged union code generation.
use super::{csharp_file_header, is_tuple_field};
use crate::type_map::csharp_type;
use alef_codegen::naming::to_csharp_name;
use alef_core::ir::EnumDef;
use heck::ToPascalCase;
/// Apply a serde `rename_all` strategy to a variant name.
pub(super) fn apply_rename_all(name: &str, rename_all: Option<&str>) -> String {
use heck::{ToLowerCamelCase, ToPascalCase, ToSnakeCase};
match rename_all {
Some("snake_case") => name.to_snake_case(),
Some("camelCase") => name.to_lower_camel_case(),
Some("PascalCase") => name.to_pascal_case(),
Some("SCREAMING_SNAKE_CASE") => name.to_snake_case().to_uppercase(),
Some("lowercase") => name.to_lowercase(),
Some("UPPERCASE") => name.to_uppercase(),
_ => name.to_lowercase(),
}
}
pub(super) fn gen_enum(enum_def: &EnumDef, namespace: &str) -> String {
use crate::template_env::render;
use minijinja::Value;
let has_data_variants = enum_def.variants.iter().any(|v| !v.fields.is_empty());
// Tagged union: enum has a serde tag AND data variants → generate abstract record hierarchy
if enum_def.serde_tag.is_some() && has_data_variants {
return gen_tagged_union(enum_def, namespace);
}
// If any variant has an explicit serde_rename whose value differs from what
// SnakeCaseLower would produce (e.g. "og:image" vs "og_image"), the global
// JsonStringEnumConverter(SnakeCaseLower) in KreuzcrawlLib.JsonOptions would
// ignore [JsonPropertyName] and use the naming policy instead.
// Also, the non-generic JsonStringEnumConverter does NOT support [JsonPropertyName]
// on enum members at all. For these cases we generate a custom JsonConverter<T>
// that explicitly maps each variant name.
let needs_custom_converter = enum_def.variants.iter().any(|v| {
if let Some(ref rename) = v.serde_rename {
let snake = apply_rename_all(&v.name, enum_def.serde_rename_all.as_deref());
rename != &snake
} else {
false
}
});
let enum_pascal = enum_def.name.to_pascal_case();
// Collect variant data with doc lines for template rendering
let variant_list: Vec<(String, String)> = enum_def
.variants
.iter()
.map(|v| {
let json_name = v
.serde_rename
.clone()
.unwrap_or_else(|| apply_rename_all(&v.name, enum_def.serde_rename_all.as_deref()));
let pascal_name = v.name.to_pascal_case();
(json_name, pascal_name)
})
.collect();
let variants: Vec<Value> = enum_def
.variants
.iter()
.map(|v| {
let json_name = v
.serde_rename
.clone()
.unwrap_or_else(|| apply_rename_all(&v.name, enum_def.serde_rename_all.as_deref()));
let pascal_name = v.name.to_pascal_case();
let doc_lines: Vec<String> = if !v.doc.is_empty() {
v.doc.lines().map(|l| l.to_string()).collect()
} else {
vec![]
};
Value::from_serialize(serde_json::json!({
"json_name": json_name,
"pascal_name": pascal_name,
"doc": !v.doc.is_empty(),
"doc_lines": doc_lines,
}))
})
.collect();
let doc_lines: Vec<String> = if !enum_def.doc.is_empty() {
enum_def.doc.lines().map(|l| l.to_string()).collect()
} else {
vec![]
};
let mut out = render(
"enum_header.jinja",
Value::from_serialize(serde_json::json!({
"namespace": namespace,
"enum_pascal": enum_pascal,
"needs_custom_converter": needs_custom_converter,
"doc": !enum_def.doc.is_empty(),
"doc_lines": doc_lines,
"variants": variants,
})),
);
out.push('\n');
// Generate custom converter class after the enum when needed
if needs_custom_converter {
out.push_str(&render(
"enum_custom_converter.jinja",
Value::from_serialize(serde_json::json!({
"enum_pascal": enum_pascal,
"variants": variant_list.iter().map(|(json_name, pascal_name)| {
serde_json::json!({
"json_name": json_name,
"pascal_name": pascal_name,
})
}).collect::<Vec<_>>(),
})),
));
}
out
}
/// Generate a C# abstract record hierarchy for internally tagged enums.
///
/// Maps `#[serde(tag = "type_field", rename_all = "snake_case")]` Rust enums to
/// a C# polymorphic record hierarchy using .NET 7+ `[JsonPolymorphic]` and `[JsonDerivedType]`
/// attributes. These attributes are the idiomatic way to handle JSON polymorphism in modern C#.
fn gen_tagged_union(enum_def: &EnumDef, namespace: &str) -> String {
use crate::template_env::render;
let tag_field = enum_def.serde_tag.as_deref().unwrap_or("type");
let enum_pascal = enum_def.name.to_pascal_case();
// Namespace prefix used to fully-qualify inner types when their short name is shadowed
// by a nested record of the same name (e.g. ContentPart.ImageUrl shadows ImageUrl).
let ns = namespace;
let mut out = csharp_file_header();
out.push_str("using System.Text.Json.Serialization;\n\n");
out.push_str(&render("namespace_decl.jinja", minijinja::context! { namespace }));
out.push('\n');
// Collect all variant pascal names to check for field-name-to-variant-name clashes
let variant_names: std::collections::HashSet<String> =
enum_def.variants.iter().map(|v| v.name.to_pascal_case()).collect();
// Precompute discriminator values for each variant
let discriminators: Vec<(String, String)> = enum_def
.variants
.iter()
.map(|v| {
let pascal = v.name.to_pascal_case();
let disc = v
.serde_rename
.clone()
.unwrap_or_else(|| apply_rename_all(&v.name, enum_def.serde_rename_all.as_deref()));
(pascal, disc)
})
.collect();
// Doc comment
if !enum_def.doc.is_empty() {
out.push_str("/// <summary>\n");
for line in enum_def.doc.lines() {
out.push_str(&render("doc_line.jinja", minijinja::context! { line }));
}
out.push_str("/// </summary>\n");
}
// [JsonPolymorphic] + [JsonDerivedType] attributes must ALL be on the base type,
// not on the nested records — System.Text.Json resolves polymorphism from the base type.
out.push_str(&render(
"json_polymorphic_attr.jinja",
minijinja::context! { tag_field },
));
for (pascal, discriminator) in &discriminators {
out.push_str(&render(
"json_derived_type_attr.jinja",
minijinja::context! { pascal, discriminator },
));
}
out.push_str(&render(
"abstract_record_header.jinja",
minijinja::context! { enum_pascal },
));
out.push_str("{\n");
// Nested sealed records for each variant (no [JsonDerivedType] here — it's on the base)
for variant in &enum_def.variants {
let pascal = variant.name.to_pascal_case();
if !variant.doc.is_empty() {
out.push_str(" /// <summary>\n");
for line in variant.doc.lines() {
out.push_str(&render("doc_line_indented.jinja", minijinja::context! { line }));
}
out.push_str(" /// </summary>\n");
}
if variant.fields.is_empty() {
// Unit variant → sealed record with no fields
out.push_str(&render(
"unit_variant_record.jinja",
minijinja::context! { pascal, enum_pascal },
));
out.push('\n');
} else {
// CS8910: when a single-field variant has a parameter whose TYPE equals the record name
// (e.g., record ImageUrl(ImageUrl Value)), the primary constructor conflicts with the
// synthesized copy constructor. Use a property-based record body instead.
// This applies to both tuple fields and named fields that get renamed to "Value".
let is_copy_ctor_clash = variant.fields.len() == 1 && {
let field_cs_type = csharp_type(&variant.fields[0].ty);
field_cs_type.as_ref() == pascal
};
if is_copy_ctor_clash {
let cs_type = csharp_type(&variant.fields[0].ty);
// Fully qualify the inner type to avoid the nested record shadowing the
// standalone type of the same name (e.g. `ContentPart.ImageUrl` would shadow
// `LiterLlm.ImageUrl` within the `ContentPart` abstract record body).
let qualified_cs_type = format!("global::{ns}.{cs_type}");
out.push_str(&render(
"variant_record_body_header.jinja",
minijinja::context! { pascal, enum_pascal },
));
out.push_str(" {\n");
out.push_str(&render(
"required_value_property.jinja",
minijinja::context! { qualified_cs_type },
));
out.push_str(" }\n\n");
} else {
// Data variant → sealed record with fields as constructor params
out.push_str(&render(
"variant_record_params_header.jinja",
minijinja::context! { pascal },
));
for (i, field) in variant.fields.iter().enumerate() {
let cs_type = csharp_type(&field.ty);
let cs_type = if field.optional && !cs_type.ends_with('?') {
format!("{cs_type}?")
} else {
cs_type.to_string()
};
// Qualify collection types that would be shadowed by a same-named variant
// (e.g. NodeContent.List nested record shadows System.Collections.Generic.List<T>).
let cs_type = if variant_names.iter().any(|vn| cs_type.starts_with(&format!("{vn}<"))) {
cs_type
.replace("List<", "global::System.Collections.Generic.List<")
.replace("Dictionary<", "global::System.Collections.Generic.Dictionary<")
} else {
cs_type
};
let comma = if i < variant.fields.len() - 1 { "," } else { "" };
if is_tuple_field(field) {
out.push_str(&render(
"variant_field_tuple.jinja",
minijinja::context! { cs_type, comma },
));
} else {
let json_name = field.name.trim_start_matches('_');
let cs_name = to_csharp_name(json_name);
// Check if this field name clashes with:
// 1. The variant pascal name (e.g., "Slide" variant with "slide" field → "Slide" param)
// 2. The field type name (e.g., "ImageUrl" type with "url" field → "Url" param matching a nested record)
// 3. Another variant pascal name (e.g., nested "Title" record with "title" field in "Slide" variant)
let clashes = cs_name == pascal || cs_name == cs_type || variant_names.contains(&cs_name);
if clashes {
// Rename to Value with JSON property mapping to preserve the original field name
out.push_str(&render(
"variant_field_json_value.jinja",
minijinja::context! { json_name, cs_type, comma },
));
} else {
out.push_str(&render(
"variant_field_json_named.jinja",
minijinja::context! { json_name, cs_type, cs_name, comma },
));
}
}
}
out.push_str(&render(
"variant_record_close.jinja",
minijinja::context! { enum_pascal },
));
out.push('\n');
}
}
}
// Add accessor properties for data variants
for variant in &enum_def.variants {
// Only generate accessors for variants with exactly one tuple field
if variant.fields.len() != 1 || !is_tuple_field(&variant.fields[0]) {
continue;
}
let pascal = variant.name.to_pascal_case();
let return_type = csharp_type(&variant.fields[0].ty);
let return_type_nullable = format!("{return_type}?");
out.push_str(&render(
"variant_accessor_summary.jinja",
minijinja::context! { pascal },
));
out.push_str(&render(
"variant_accessor_property.jinja",
minijinja::context! { pascal, return_type_nullable },
));
out.push('\n');
}
out.push_str("}\n");
out
}