1use crate::config::E2eConfig;
7use crate::escape::{escape_java, sanitize_filename};
8use crate::field_access::FieldResolver;
9use crate::fixture::{Assertion, CallbackAction, Fixture, FixtureGroup, HttpFixture};
10use alef_core::backend::GeneratedFile;
11use alef_core::config::ResolvedCrateConfig;
12use alef_core::hash::{self, CommentStyle};
13use alef_core::template_versions as tv;
14use anyhow::Result;
15use heck::{ToLowerCamelCase, ToUpperCamelCase};
16use std::path::PathBuf;
17
18use super::E2eCodegen;
19use super::client;
20
21pub struct JavaCodegen;
23
24impl E2eCodegen for JavaCodegen {
25 fn generate(
26 &self,
27 groups: &[FixtureGroup],
28 e2e_config: &E2eConfig,
29 config: &ResolvedCrateConfig,
30 _type_defs: &[alef_core::ir::TypeDef],
31 ) -> Result<Vec<GeneratedFile>> {
32 let lang = self.language_name();
33 let output_base = PathBuf::from(e2e_config.effective_output()).join(lang);
34
35 let mut files = Vec::new();
36
37 let call = &e2e_config.call;
39 let overrides = call.overrides.get(lang);
40 let _module_path = overrides
41 .and_then(|o| o.module.as_ref())
42 .cloned()
43 .unwrap_or_else(|| call.module.clone());
44 let function_name = overrides
45 .and_then(|o| o.function.as_ref())
46 .cloned()
47 .unwrap_or_else(|| call.function.clone());
48 let class_name = overrides
49 .and_then(|o| o.class.as_ref())
50 .cloned()
51 .unwrap_or_else(|| config.name.to_upper_camel_case());
52 let result_is_simple = overrides.is_some_and(|o| o.result_is_simple);
53 let result_var = &call.result_var;
54
55 let java_pkg = e2e_config.resolve_package("java");
57 let pkg_name = java_pkg
58 .as_ref()
59 .and_then(|p| p.name.as_ref())
60 .cloned()
61 .unwrap_or_else(|| config.name.clone());
62
63 let java_group_id = config.java_group_id();
65 let binding_pkg = config.java_package();
66 let pkg_version = config.resolved_version().unwrap_or_else(|| "0.1.0".to_string());
67
68 files.push(GeneratedFile {
70 path: output_base.join("pom.xml"),
71 content: render_pom_xml(
72 &pkg_name,
73 &java_group_id,
74 &pkg_version,
75 e2e_config.dep_mode,
76 &e2e_config.test_documents_relative_from(0),
77 ),
78 generated_header: false,
79 });
80
81 let needs_mock_server = groups
89 .iter()
90 .flat_map(|g| g.fixtures.iter())
91 .any(|f| f.needs_mock_server());
92
93 let mut test_base = output_base.join("src").join("test").join("java");
97 for segment in java_group_id.split('.') {
98 test_base = test_base.join(segment);
99 }
100 let test_base = test_base.join("e2e");
101
102 if needs_mock_server {
103 files.push(GeneratedFile {
104 path: test_base.join("MockServerListener.java"),
105 content: render_mock_server_listener(&java_group_id),
106 generated_header: true,
107 });
108 files.push(GeneratedFile {
109 path: output_base
110 .join("src")
111 .join("test")
112 .join("resources")
113 .join("META-INF")
114 .join("services")
115 .join("org.junit.platform.launcher.LauncherSessionListener"),
116 content: format!("{java_group_id}.e2e.MockServerListener\n"),
117 generated_header: false,
118 });
119 }
120
121 let options_type = overrides.and_then(|o| o.options_type.clone());
123
124 static EMPTY_ENUM_FIELDS: std::sync::LazyLock<std::collections::HashMap<String, String>> =
126 std::sync::LazyLock::new(std::collections::HashMap::new);
127 let _enum_fields = overrides.map(|o| &o.enum_fields).unwrap_or(&EMPTY_ENUM_FIELDS);
128
129 let mut effective_nested_types = default_java_nested_types();
131 if let Some(overrides_map) = overrides.map(|o| &o.nested_types) {
132 effective_nested_types.extend(overrides_map.clone());
133 }
134
135 let nested_types_optional = overrides.map(|o| o.nested_types_optional).unwrap_or(true);
137
138 let field_resolver = FieldResolver::new(
139 &e2e_config.fields,
140 &e2e_config.fields_optional,
141 &e2e_config.result_fields,
142 &e2e_config.fields_array,
143 &std::collections::HashSet::new(),
144 );
145
146 for group in groups {
147 let active: Vec<&Fixture> = group
148 .fixtures
149 .iter()
150 .filter(|f| super::should_include_fixture(f, lang, e2e_config))
151 .collect();
152
153 if active.is_empty() {
154 continue;
155 }
156
157 let class_file_name = format!("{}Test.java", sanitize_filename(&group.category).to_upper_camel_case());
158 let content = render_test_file(
159 &group.category,
160 &active,
161 &class_name,
162 &function_name,
163 &java_group_id,
164 &binding_pkg,
165 result_var,
166 &e2e_config.call.args,
167 options_type.as_deref(),
168 &field_resolver,
169 result_is_simple,
170 &e2e_config.fields_enum,
171 e2e_config,
172 &effective_nested_types,
173 nested_types_optional,
174 );
175 files.push(GeneratedFile {
176 path: test_base.join(class_file_name),
177 content,
178 generated_header: true,
179 });
180 }
181
182 Ok(files)
183 }
184
185 fn language_name(&self) -> &'static str {
186 "java"
187 }
188}
189
190fn render_pom_xml(
195 pkg_name: &str,
196 java_group_id: &str,
197 pkg_version: &str,
198 dep_mode: crate::config::DependencyMode,
199 test_documents_path: &str,
200) -> String {
201 let (dep_group_id, dep_artifact_id) = if let Some((g, a)) = pkg_name.split_once(':') {
203 (g, a)
204 } else {
205 (java_group_id, pkg_name)
206 };
207 let artifact_id = format!("{dep_artifact_id}-e2e-java");
208 let dep_block = match dep_mode {
209 crate::config::DependencyMode::Registry => {
210 format!(
211 r#" <dependency>
212 <groupId>{dep_group_id}</groupId>
213 <artifactId>{dep_artifact_id}</artifactId>
214 <version>{pkg_version}</version>
215 </dependency>"#
216 )
217 }
218 crate::config::DependencyMode::Local => {
219 format!(
220 r#" <dependency>
221 <groupId>{dep_group_id}</groupId>
222 <artifactId>{dep_artifact_id}</artifactId>
223 <version>{pkg_version}</version>
224 <scope>system</scope>
225 <systemPath>${{project.basedir}}/../../packages/java/target/{dep_artifact_id}-{pkg_version}.jar</systemPath>
226 </dependency>"#
227 )
228 }
229 };
230 crate::template_env::render(
231 "java/pom.xml.jinja",
232 minijinja::context! {
233 artifact_id => artifact_id,
234 java_group_id => java_group_id,
235 dep_block => dep_block,
236 junit_version => tv::maven::JUNIT,
237 jackson_version => tv::maven::JACKSON_E2E,
238 build_helper_version => tv::maven::BUILD_HELPER_MAVEN_PLUGIN,
239 maven_surefire_version => tv::maven::MAVEN_SUREFIRE_PLUGIN_E2E,
240 test_documents_path => test_documents_path,
241 },
242 )
243}
244
245fn render_mock_server_listener(java_group_id: &str) -> String {
254 let header = hash::header(CommentStyle::DoubleSlash);
255 let mut out = header;
256 out.push_str(&format!("package {java_group_id}.e2e;\n\n"));
257 out.push_str("import java.io.BufferedReader;\n");
258 out.push_str("import java.io.File;\n");
259 out.push_str("import java.io.IOException;\n");
260 out.push_str("import java.io.InputStreamReader;\n");
261 out.push_str("import java.nio.charset.StandardCharsets;\n");
262 out.push_str("import java.nio.file.Path;\n");
263 out.push_str("import java.nio.file.Paths;\n");
264 out.push_str("import java.util.regex.Matcher;\n");
265 out.push_str("import java.util.regex.Pattern;\n");
266 out.push_str("import org.junit.platform.launcher.LauncherSession;\n");
267 out.push_str("import org.junit.platform.launcher.LauncherSessionListener;\n");
268 out.push('\n');
269 out.push_str("/**\n");
270 out.push_str(" * Spawns the mock-server binary once per JUnit launcher session and\n");
271 out.push_str(" * exposes its URL as the `mockServerUrl` system property. Generated\n");
272 out.push_str(" * test bodies read the property (with `MOCK_SERVER_URL` env-var\n");
273 out.push_str(" * fallback) so tests can run via plain `mvn test` without any external\n");
274 out.push_str(" * mock-server orchestration. Mirrors the Ruby spec_helper / Python\n");
275 out.push_str(" * conftest spawn pattern. Honors a pre-set MOCK_SERVER_URL by\n");
276 out.push_str(" * skipping the spawn entirely.\n");
277 out.push_str(" */\n");
278 out.push_str("public class MockServerListener implements LauncherSessionListener {\n");
279 out.push_str(" private Process mockServer;\n");
280 out.push('\n');
281 out.push_str(" @Override\n");
282 out.push_str(" public void launcherSessionOpened(LauncherSession session) {\n");
283 out.push_str(" String preset = System.getenv(\"MOCK_SERVER_URL\");\n");
284 out.push_str(" if (preset != null && !preset.isEmpty()) {\n");
285 out.push_str(" System.setProperty(\"mockServerUrl\", preset);\n");
286 out.push_str(" return;\n");
287 out.push_str(" }\n");
288 out.push_str(" Path repoRoot = locateRepoRoot();\n");
289 out.push_str(" if (repoRoot == null) {\n");
290 out.push_str(" throw new IllegalStateException(\"MockServerListener: could not locate repo root (looked for fixtures/ in ancestors of \" + System.getProperty(\"user.dir\") + \")\");\n");
291 out.push_str(" }\n");
292 out.push_str(" String binName = System.getProperty(\"os.name\", \"\").toLowerCase().contains(\"win\") ? \"mock-server.exe\" : \"mock-server\";\n");
293 out.push_str(" File bin = repoRoot.resolve(\"e2e\").resolve(\"rust\").resolve(\"target\").resolve(\"release\").resolve(binName).toFile();\n");
294 out.push_str(" File fixturesDir = repoRoot.resolve(\"fixtures\").toFile();\n");
295 out.push_str(" if (!bin.exists()) {\n");
296 out.push_str(" throw new IllegalStateException(\"MockServerListener: mock-server binary not found at \" + bin + \" — run: cargo build --manifest-path e2e/rust/Cargo.toml --bin mock-server --release\");\n");
297 out.push_str(" }\n");
298 out.push_str(
299 " ProcessBuilder pb = new ProcessBuilder(bin.getAbsolutePath(), fixturesDir.getAbsolutePath())\n",
300 );
301 out.push_str(" .redirectErrorStream(false);\n");
302 out.push_str(" try {\n");
303 out.push_str(" mockServer = pb.start();\n");
304 out.push_str(" } catch (IOException e) {\n");
305 out.push_str(
306 " throw new IllegalStateException(\"MockServerListener: failed to start mock-server\", e);\n",
307 );
308 out.push_str(" }\n");
309 out.push_str(" // Read until we see MOCK_SERVER_URL= and optionally MOCK_SERVERS=.\n");
310 out.push_str(" // Cap the loop so a misbehaving mock-server cannot block indefinitely.\n");
311 out.push_str(" BufferedReader stdout = new BufferedReader(new InputStreamReader(mockServer.getInputStream(), StandardCharsets.UTF_8));\n");
312 out.push_str(" String url = null;\n");
313 out.push_str(" try {\n");
314 out.push_str(" for (int i = 0; i < 16; i++) {\n");
315 out.push_str(" String line = stdout.readLine();\n");
316 out.push_str(" if (line == null) break;\n");
317 out.push_str(" if (line.startsWith(\"MOCK_SERVER_URL=\")) {\n");
318 out.push_str(" url = line.substring(\"MOCK_SERVER_URL=\".length()).trim();\n");
319 out.push_str(" } else if (line.startsWith(\"MOCK_SERVERS=\")) {\n");
320 out.push_str(" String jsonVal = line.substring(\"MOCK_SERVERS=\".length()).trim();\n");
321 out.push_str(" System.setProperty(\"mockServers\", jsonVal);\n");
322 out.push_str(" // Parse JSON map of fixture_id -> url and expose as system properties.\n");
323 out.push_str(" Pattern p = Pattern.compile(\"\\\"([^\\\"]+)\\\":\\\"([^\\\"]+)\\\"\");\n");
324 out.push_str(" Matcher matcher = p.matcher(jsonVal);\n");
325 out.push_str(" while (matcher.find()) {\n");
326 out.push_str(" String fid = matcher.group(1);\n");
327 out.push_str(" String furl = matcher.group(2);\n");
328 out.push_str(" System.setProperty(\"mockServer.\" + fid, furl);\n");
329 out.push_str(" }\n");
330 out.push_str(" break;\n");
331 out.push_str(" } else if (url != null) {\n");
332 out.push_str(" break;\n");
333 out.push_str(" }\n");
334 out.push_str(" }\n");
335 out.push_str(" } catch (IOException e) {\n");
336 out.push_str(" mockServer.destroyForcibly();\n");
337 out.push_str(
338 " throw new IllegalStateException(\"MockServerListener: failed to read mock-server stdout\", e);\n",
339 );
340 out.push_str(" }\n");
341 out.push_str(" if (url == null || url.isEmpty()) {\n");
342 out.push_str(" mockServer.destroyForcibly();\n");
343 out.push_str(" throw new IllegalStateException(\"MockServerListener: mock-server did not emit MOCK_SERVER_URL\");\n");
344 out.push_str(" }\n");
345 out.push_str(" // TCP-readiness probe: ensure axum::serve is accepting before tests start.\n");
346 out.push_str(" // The mock-server binds the TcpListener synchronously then prints the URL\n");
347 out.push_str(" // before tokio::spawn(axum::serve(...)) is polled, so under Surefire\n");
348 out.push_str(" // parallel mode tests can race startup. Poll-connect (max 5s, 50ms backoff)\n");
349 out.push_str(" // until success.\n");
350 out.push_str(" java.net.URI healthUri = java.net.URI.create(url);\n");
351 out.push_str(" String host = healthUri.getHost();\n");
352 out.push_str(" int port = healthUri.getPort();\n");
353 out.push_str(" long deadline = System.nanoTime() + 5_000_000_000L;\n");
354 out.push_str(" while (System.nanoTime() < deadline) {\n");
355 out.push_str(" try (java.net.Socket s = new java.net.Socket()) {\n");
356 out.push_str(" s.connect(new java.net.InetSocketAddress(host, port), 100);\n");
357 out.push_str(" break;\n");
358 out.push_str(" } catch (java.io.IOException ignored) {\n");
359 out.push_str(" try { Thread.sleep(50); } catch (InterruptedException ie) { Thread.currentThread().interrupt(); break; }\n");
360 out.push_str(" }\n");
361 out.push_str(" }\n");
362 out.push_str(" System.setProperty(\"mockServerUrl\", url);\n");
363 out.push_str(" // Drain remaining stdout/stderr in daemon threads so a full pipe\n");
364 out.push_str(" // does not block the child.\n");
365 out.push_str(" Process server = mockServer;\n");
366 out.push_str(" Thread drainOut = new Thread(() -> drain(stdout));\n");
367 out.push_str(" drainOut.setDaemon(true);\n");
368 out.push_str(" drainOut.start();\n");
369 out.push_str(" Thread drainErr = new Thread(() -> drain(new BufferedReader(new InputStreamReader(server.getErrorStream(), StandardCharsets.UTF_8))));\n");
370 out.push_str(" drainErr.setDaemon(true);\n");
371 out.push_str(" drainErr.start();\n");
372 out.push_str(" }\n");
373 out.push('\n');
374 out.push_str(" @Override\n");
375 out.push_str(" public void launcherSessionClosed(LauncherSession session) {\n");
376 out.push_str(" if (mockServer == null) return;\n");
377 out.push_str(" try { mockServer.getOutputStream().close(); } catch (IOException ignored) {}\n");
378 out.push_str(" try {\n");
379 out.push_str(" if (!mockServer.waitFor(2, java.util.concurrent.TimeUnit.SECONDS)) {\n");
380 out.push_str(" mockServer.destroyForcibly();\n");
381 out.push_str(" }\n");
382 out.push_str(" } catch (InterruptedException ignored) {\n");
383 out.push_str(" Thread.currentThread().interrupt();\n");
384 out.push_str(" mockServer.destroyForcibly();\n");
385 out.push_str(" }\n");
386 out.push_str(" }\n");
387 out.push('\n');
388 out.push_str(" private static Path locateRepoRoot() {\n");
389 out.push_str(" Path dir = Paths.get(\"\").toAbsolutePath();\n");
390 out.push_str(" while (dir != null) {\n");
391 out.push_str(" if (dir.resolve(\"fixtures\").toFile().isDirectory()\n");
392 out.push_str(" && dir.resolve(\"e2e\").toFile().isDirectory()) {\n");
393 out.push_str(" return dir;\n");
394 out.push_str(" }\n");
395 out.push_str(" dir = dir.getParent();\n");
396 out.push_str(" }\n");
397 out.push_str(" return null;\n");
398 out.push_str(" }\n");
399 out.push('\n');
400 out.push_str(" private static void drain(BufferedReader reader) {\n");
401 out.push_str(" try {\n");
402 out.push_str(" char[] buf = new char[1024];\n");
403 out.push_str(" while (reader.read(buf) >= 0) { /* drain */ }\n");
404 out.push_str(" } catch (IOException ignored) {}\n");
405 out.push_str(" }\n");
406 out.push_str("}\n");
407 out
408}
409
410#[allow(clippy::too_many_arguments)]
411fn render_test_file(
412 category: &str,
413 fixtures: &[&Fixture],
414 class_name: &str,
415 function_name: &str,
416 java_group_id: &str,
417 binding_pkg: &str,
418 result_var: &str,
419 args: &[crate::config::ArgMapping],
420 options_type: Option<&str>,
421 field_resolver: &FieldResolver,
422 result_is_simple: bool,
423 enum_fields: &std::collections::HashSet<String>,
424 e2e_config: &E2eConfig,
425 nested_types: &std::collections::HashMap<String, String>,
426 nested_types_optional: bool,
427) -> String {
428 let header = hash::header(CommentStyle::DoubleSlash);
429 let test_class_name = format!("{}Test", sanitize_filename(category).to_upper_camel_case());
430
431 let (import_path, simple_class) = if class_name.contains('.') {
434 let simple = class_name.rsplit('.').next().unwrap_or(class_name);
435 (class_name, simple)
436 } else {
437 ("", class_name)
438 };
439
440 let lang_for_om = "java";
442 let needs_object_mapper_for_handle = fixtures.iter().any(|f| {
443 args.iter().filter(|a| a.arg_type == "handle").any(|a| {
444 let v = f.input.get(&a.field).unwrap_or(&serde_json::Value::Null);
445 !(v.is_null() || v.is_object() && v.as_object().is_some_and(|o| o.is_empty()))
446 })
447 });
448 let has_http_fixtures = fixtures.iter().any(|f| f.http.is_some());
450 let needs_object_mapper = needs_object_mapper_for_handle || has_http_fixtures;
451
452 let mut all_options_types: std::collections::BTreeSet<String> = std::collections::BTreeSet::new();
454 if let Some(t) = options_type {
455 all_options_types.insert(t.to_string());
456 }
457 for f in fixtures.iter() {
458 let call_cfg = e2e_config.resolve_call_for_fixture(f.call.as_deref(), &f.input);
459 if let Some(ov) = call_cfg.overrides.get(lang_for_om) {
460 if let Some(t) = &ov.options_type {
461 all_options_types.insert(t.clone());
462 }
463 }
464 let java_has_type = call_cfg
470 .overrides
471 .get(lang_for_om)
472 .and_then(|o| o.options_type.as_deref())
473 .is_some();
474 if !java_has_type {
475 for cand in ["csharp", "c", "go", "php", "python"] {
476 if let Some(o) = call_cfg.overrides.get(cand) {
477 if let Some(t) = &o.options_type {
478 all_options_types.insert(t.clone());
479 break;
480 }
481 }
482 }
483 }
484 for arg in &call_cfg.args {
486 if let Some(elem_type) = &arg.element_type {
487 if elem_type == "BatchBytesItem" || elem_type == "BatchFileItem" {
488 all_options_types.insert(elem_type.clone());
489 }
490 }
491 }
492 }
493
494 let mut nested_types_used: std::collections::BTreeSet<String> = std::collections::BTreeSet::new();
497 for f in fixtures.iter() {
498 let call_cfg = e2e_config.resolve_call_for_fixture(f.call.as_deref(), &f.input);
499 for arg in &call_cfg.args {
500 if arg.arg_type == "json_object" {
501 let field = arg.field.strip_prefix("input.").unwrap_or(&arg.field);
502 if let Some(val) = f.input.get(field) {
503 if !val.is_null() && !val.is_array() {
504 if let Some(obj) = val.as_object() {
505 collect_nested_type_names(obj, nested_types, &mut nested_types_used);
506 }
507 }
508 }
509 }
510 }
511 }
512
513 let binding_pkg_for_imports: String = if !binding_pkg.is_empty() {
518 binding_pkg.to_string()
519 } else if !import_path.is_empty() {
520 import_path
521 .rsplit_once('.')
522 .map(|(p, _)| p.to_string())
523 .unwrap_or_default()
524 } else {
525 String::new()
526 };
527
528 let mut imports: Vec<String> = Vec::new();
530 imports.push("import org.junit.jupiter.api.Test;".to_string());
531 imports.push("import static org.junit.jupiter.api.Assertions.*;".to_string());
532
533 if !import_path.is_empty() {
536 imports.push(format!("import {import_path};"));
537 } else if !binding_pkg_for_imports.is_empty() && !class_name.is_empty() {
538 imports.push(format!("import {binding_pkg_for_imports}.{class_name};"));
539 }
540
541 if needs_object_mapper {
542 imports.push("import com.fasterxml.jackson.databind.ObjectMapper;".to_string());
543 imports.push("import com.fasterxml.jackson.datatype.jdk8.Jdk8Module;".to_string());
544 }
545
546 if !all_options_types.is_empty() {
548 for opts_type in &all_options_types {
549 let qualified = if binding_pkg_for_imports.is_empty() {
550 opts_type.clone()
551 } else {
552 format!("{binding_pkg_for_imports}.{opts_type}")
553 };
554 imports.push(format!("import {qualified};"));
555 }
556 }
557
558 if !nested_types_used.is_empty() && !binding_pkg_for_imports.is_empty() {
560 for type_name in &nested_types_used {
561 imports.push(format!("import {binding_pkg_for_imports}.{type_name};"));
562 }
563 }
564
565 if needs_object_mapper_for_handle && !binding_pkg_for_imports.is_empty() {
567 imports.push(format!("import {binding_pkg_for_imports}.CrawlConfig;"));
568 }
569
570 let has_visitor_fixtures = fixtures.iter().any(|f| f.visitor.is_some());
572 if has_visitor_fixtures && !binding_pkg_for_imports.is_empty() {
573 imports.push(format!("import {binding_pkg_for_imports}.Visitor;"));
574 imports.push(format!("import {binding_pkg_for_imports}.NodeContext;"));
575 imports.push(format!("import {binding_pkg_for_imports}.VisitResult;"));
576 }
577
578 if !all_options_types.is_empty() {
580 imports.push("import java.util.Optional;".to_string());
581 }
582
583 let mut fixtures_body = String::new();
585 for (i, fixture) in fixtures.iter().enumerate() {
586 render_test_method(
587 &mut fixtures_body,
588 fixture,
589 simple_class,
590 function_name,
591 result_var,
592 args,
593 options_type,
594 field_resolver,
595 result_is_simple,
596 enum_fields,
597 e2e_config,
598 nested_types,
599 nested_types_optional,
600 );
601 if i + 1 < fixtures.len() {
602 fixtures_body.push('\n');
603 }
604 }
605
606 crate::template_env::render(
608 "java/test_file.jinja",
609 minijinja::context! {
610 header => header,
611 java_group_id => java_group_id,
612 test_class_name => test_class_name,
613 category => category,
614 imports => imports,
615 needs_object_mapper => needs_object_mapper,
616 fixtures_body => fixtures_body,
617 },
618 )
619}
620
621struct JavaTestClientRenderer;
629
630impl client::TestClientRenderer for JavaTestClientRenderer {
631 fn language_name(&self) -> &'static str {
632 "java"
633 }
634
635 fn sanitize_test_name(&self, id: &str) -> String {
639 id.to_upper_camel_case()
640 }
641
642 fn render_test_open(&self, out: &mut String, fn_name: &str, description: &str, skip_reason: Option<&str>) {
648 let escaped_reason = skip_reason.map(escape_java);
649 let rendered = crate::template_env::render(
650 "java/http_test_open.jinja",
651 minijinja::context! {
652 fn_name => fn_name,
653 description => description,
654 skip_reason => escaped_reason,
655 },
656 );
657 out.push_str(&rendered);
658 }
659
660 fn render_test_close(&self, out: &mut String) {
662 let rendered = crate::template_env::render("java/http_test_close.jinja", minijinja::context! {});
663 out.push_str(&rendered);
664 }
665
666 fn render_call(&self, out: &mut String, ctx: &client::CallCtx<'_>) {
672 const JAVA_RESTRICTED_HEADERS: &[&str] = &["connection", "content-length", "expect", "host", "upgrade"];
674
675 let method = ctx.method.to_uppercase();
676
677 let path = if ctx.query_params.is_empty() {
679 ctx.path.to_string()
680 } else {
681 let pairs: Vec<String> = ctx
682 .query_params
683 .iter()
684 .map(|(k, v)| {
685 let val_str = match v {
686 serde_json::Value::String(s) => s.clone(),
687 other => other.to_string(),
688 };
689 format!("{}={}", k, escape_java(&val_str))
690 })
691 .collect();
692 format!("{}?{}", ctx.path, pairs.join("&"))
693 };
694
695 let body_publisher = if let Some(body) = ctx.body {
696 let json = serde_json::to_string(body).unwrap_or_default();
697 let escaped = escape_java(&json);
698 format!("java.net.http.HttpRequest.BodyPublishers.ofString(\"{escaped}\")")
699 } else {
700 "java.net.http.HttpRequest.BodyPublishers.noBody()".to_string()
701 };
702
703 let content_type = if ctx.body.is_some() {
705 let ct = ctx.content_type.unwrap_or("application/json");
706 if !ctx.headers.keys().any(|k| k.to_lowercase() == "content-type") {
708 Some(ct.to_string())
709 } else {
710 None
711 }
712 } else {
713 None
714 };
715
716 let mut headers_lines: Vec<String> = Vec::new();
718 for (name, value) in ctx.headers {
719 if JAVA_RESTRICTED_HEADERS.contains(&name.to_lowercase().as_str()) {
720 continue;
721 }
722 let escaped_name = escape_java(name);
723 let escaped_value = escape_java(value);
724 headers_lines.push(format!(
725 "builder = builder.header(\"{escaped_name}\", \"{escaped_value}\");"
726 ));
727 }
728
729 let cookies_line = if !ctx.cookies.is_empty() {
731 let cookie_str: Vec<String> = ctx.cookies.iter().map(|(k, v)| format!("{k}={v}")).collect();
732 let cookie_header = escape_java(&cookie_str.join("; "));
733 Some(format!("builder = builder.header(\"Cookie\", \"{cookie_header}\");"))
734 } else {
735 None
736 };
737
738 let rendered = crate::template_env::render(
739 "java/http_request.jinja",
740 minijinja::context! {
741 method => method,
742 path => path,
743 body_publisher => body_publisher,
744 content_type => content_type,
745 headers_lines => headers_lines,
746 cookies_line => cookies_line,
747 response_var => ctx.response_var,
748 },
749 );
750 out.push_str(&rendered);
751 }
752
753 fn render_assert_status(&self, out: &mut String, response_var: &str, status: u16) {
755 let rendered = crate::template_env::render(
756 "java/http_assertions.jinja",
757 minijinja::context! {
758 response_var => response_var,
759 status_code => status,
760 headers => Vec::<std::collections::HashMap<&str, String>>::new(),
761 body_assertion => String::new(),
762 partial_body => Vec::<std::collections::HashMap<&str, String>>::new(),
763 validation_errors => Vec::<std::collections::HashMap<&str, String>>::new(),
764 },
765 );
766 out.push_str(&rendered);
767 }
768
769 fn render_assert_header(&self, out: &mut String, response_var: &str, name: &str, expected: &str) {
773 let escaped_name = escape_java(name);
774 let assertion_code = match expected {
775 "<<present>>" => {
776 format!(
777 "assertTrue({response_var}.headers().firstValue(\"{escaped_name}\").isPresent(), \"header {escaped_name} should be present\");"
778 )
779 }
780 "<<absent>>" => {
781 format!(
782 "assertTrue({response_var}.headers().firstValue(\"{escaped_name}\").isEmpty(), \"header {escaped_name} should be absent\");"
783 )
784 }
785 "<<uuid>>" => {
786 format!(
787 "assertTrue({response_var}.headers().firstValue(\"{escaped_name}\").orElse(\"\").matches(\"[0-9a-fA-F]{{8}}-[0-9a-fA-F]{{4}}-[0-9a-fA-F]{{4}}-[0-9a-fA-F]{{4}}-[0-9a-fA-F]{{12}}\"), \"header {escaped_name} should be a UUID\");"
788 )
789 }
790 literal => {
791 let escaped_value = escape_java(literal);
792 format!(
793 "assertTrue({response_var}.headers().firstValue(\"{escaped_name}\").orElse(\"\").contains(\"{escaped_value}\"), \"header {escaped_name} mismatch\");"
794 )
795 }
796 };
797
798 let mut headers = vec![std::collections::HashMap::new()];
799 headers[0].insert("assertion_code", assertion_code);
800
801 let rendered = crate::template_env::render(
802 "java/http_assertions.jinja",
803 minijinja::context! {
804 response_var => response_var,
805 status_code => 0u16,
806 headers => headers,
807 body_assertion => String::new(),
808 partial_body => Vec::<std::collections::HashMap<&str, String>>::new(),
809 validation_errors => Vec::<std::collections::HashMap<&str, String>>::new(),
810 },
811 );
812 out.push_str(&rendered);
813 }
814
815 fn render_assert_json_body(&self, out: &mut String, response_var: &str, expected: &serde_json::Value) {
817 let body_assertion = match expected {
818 serde_json::Value::Object(_) | serde_json::Value::Array(_) => {
819 let json_str = serde_json::to_string(expected).unwrap_or_default();
820 let escaped = escape_java(&json_str);
821 format!(
822 "var bodyJson = MAPPER.readTree({response_var}.body());\n var expectedJson = MAPPER.readTree(\"{escaped}\");\n assertEquals(expectedJson, bodyJson, \"body mismatch\");"
823 )
824 }
825 serde_json::Value::String(s) => {
826 let escaped = escape_java(s);
827 format!("assertEquals(\"{escaped}\", {response_var}.body().trim(), \"body mismatch\");")
828 }
829 other => {
830 let escaped = escape_java(&other.to_string());
831 format!("assertEquals(\"{escaped}\", {response_var}.body().trim(), \"body mismatch\");")
832 }
833 };
834
835 let rendered = crate::template_env::render(
836 "java/http_assertions.jinja",
837 minijinja::context! {
838 response_var => response_var,
839 status_code => 0u16,
840 headers => Vec::<std::collections::HashMap<&str, String>>::new(),
841 body_assertion => body_assertion,
842 partial_body => Vec::<std::collections::HashMap<&str, String>>::new(),
843 validation_errors => Vec::<std::collections::HashMap<&str, String>>::new(),
844 },
845 );
846 out.push_str(&rendered);
847 }
848
849 fn render_assert_partial_body(&self, out: &mut String, response_var: &str, expected: &serde_json::Value) {
851 if let Some(obj) = expected.as_object() {
852 let mut partial_body: Vec<std::collections::HashMap<&str, String>> = Vec::new();
853 for (key, val) in obj {
854 let escaped_key = escape_java(key);
855 let json_str = serde_json::to_string(val).unwrap_or_default();
856 let escaped_val = escape_java(&json_str);
857 let assertion_code = format!(
858 "assertEquals(MAPPER.readTree(\"{escaped_val}\"), partialJson.get(\"{escaped_key}\"), \"body field '{escaped_key}' mismatch\");"
859 );
860 let mut entry = std::collections::HashMap::new();
861 entry.insert("assertion_code", assertion_code);
862 partial_body.push(entry);
863 }
864
865 let rendered = crate::template_env::render(
866 "java/http_assertions.jinja",
867 minijinja::context! {
868 response_var => response_var,
869 status_code => 0u16,
870 headers => Vec::<std::collections::HashMap<&str, String>>::new(),
871 body_assertion => String::new(),
872 partial_body => partial_body,
873 validation_errors => Vec::<std::collections::HashMap<&str, String>>::new(),
874 },
875 );
876 out.push_str(&rendered);
877 }
878 }
879
880 fn render_assert_validation_errors(
882 &self,
883 out: &mut String,
884 response_var: &str,
885 errors: &[crate::fixture::ValidationErrorExpectation],
886 ) {
887 let mut validation_errors: Vec<std::collections::HashMap<&str, String>> = Vec::new();
888 for err in errors {
889 let escaped_msg = escape_java(&err.msg);
890 let assertion_code = format!(
891 "assertTrue(veBody.contains(\"{escaped_msg}\"), \"expected validation error message: {escaped_msg}\");"
892 );
893 let mut entry = std::collections::HashMap::new();
894 entry.insert("assertion_code", assertion_code);
895 validation_errors.push(entry);
896 }
897
898 let rendered = crate::template_env::render(
899 "java/http_assertions.jinja",
900 minijinja::context! {
901 response_var => response_var,
902 status_code => 0u16,
903 headers => Vec::<std::collections::HashMap<&str, String>>::new(),
904 body_assertion => String::new(),
905 partial_body => Vec::<std::collections::HashMap<&str, String>>::new(),
906 validation_errors => validation_errors,
907 },
908 );
909 out.push_str(&rendered);
910 }
911}
912
913fn render_http_test_method(out: &mut String, fixture: &Fixture, http: &HttpFixture) {
920 if http.expected_response.status_code == 101 {
923 let method_name = fixture.id.to_upper_camel_case();
924 let description = &fixture.description;
925 out.push_str(&crate::template_env::render(
926 "java/http_test_skip_101.jinja",
927 minijinja::context! {
928 method_name => method_name,
929 description => description,
930 },
931 ));
932 return;
933 }
934
935 client::http_call::render_http_test(out, &JavaTestClientRenderer, fixture);
936}
937
938#[allow(clippy::too_many_arguments)]
939fn render_test_method(
940 out: &mut String,
941 fixture: &Fixture,
942 class_name: &str,
943 _function_name: &str,
944 _result_var: &str,
945 _args: &[crate::config::ArgMapping],
946 options_type: Option<&str>,
947 field_resolver: &FieldResolver,
948 result_is_simple: bool,
949 enum_fields: &std::collections::HashSet<String>,
950 e2e_config: &E2eConfig,
951 nested_types: &std::collections::HashMap<String, String>,
952 nested_types_optional: bool,
953) {
954 if let Some(http) = &fixture.http {
956 render_http_test_method(out, fixture, http);
957 return;
958 }
959
960 let call_config = e2e_config.resolve_call_for_fixture(fixture.call.as_deref(), &fixture.input);
963 let lang = "java";
964 let call_overrides = call_config.overrides.get(lang);
965 let effective_function_name = call_overrides
966 .and_then(|o| o.function.as_ref())
967 .cloned()
968 .unwrap_or_else(|| call_config.function.to_lower_camel_case());
969 let effective_result_var = &call_config.result_var;
970 let effective_args = &call_config.args;
971 let function_name = effective_function_name.as_str();
972 let result_var = effective_result_var.as_str();
973 let args: &[crate::config::ArgMapping] = effective_args.as_slice();
974
975 let method_name = fixture.id.to_upper_camel_case();
976 let description = &fixture.description;
977 let expects_error = fixture.assertions.iter().any(|a| a.assertion_type == "error");
978
979 let effective_options_type: Option<String> = call_overrides
985 .and_then(|o| o.options_type.clone())
986 .or_else(|| options_type.map(|s| s.to_string()))
987 .or_else(|| {
988 for cand in ["csharp", "c", "go", "php", "python"] {
992 if let Some(o) = call_config.overrides.get(cand) {
993 if let Some(t) = &o.options_type {
994 return Some(t.clone());
995 }
996 }
997 }
998 None
999 });
1000 let effective_options_type = effective_options_type.as_deref();
1001 let auto_from_json = effective_options_type.is_some()
1006 && call_overrides.and_then(|o| o.options_via.as_deref()).is_none()
1007 && e2e_config
1008 .call
1009 .overrides
1010 .get(lang)
1011 .and_then(|o| o.options_via.as_deref())
1012 .is_none();
1013
1014 let client_factory: Option<String> = call_overrides.and_then(|o| o.client_factory.clone()).or_else(|| {
1016 e2e_config
1017 .call
1018 .overrides
1019 .get(lang)
1020 .and_then(|o| o.client_factory.clone())
1021 });
1022
1023 let options_via: String = call_overrides
1028 .and_then(|o| o.options_via.clone())
1029 .or_else(|| e2e_config.call.overrides.get(lang).and_then(|o| o.options_via.clone()))
1030 .unwrap_or_else(|| {
1031 if auto_from_json {
1032 "from_json".to_string()
1033 } else {
1034 "kwargs".to_string()
1035 }
1036 });
1037
1038 let effective_result_is_simple =
1040 call_overrides.is_some_and(|o| o.result_is_simple) || call_config.result_is_simple || result_is_simple;
1041 let effective_result_is_bytes = call_overrides.is_some_and(|o| o.result_is_bytes);
1042
1043 let needs_deser = effective_options_type.is_some()
1045 && args.iter().any(|arg| {
1046 if arg.arg_type != "json_object" {
1047 return false;
1048 }
1049 let val = super::resolve_field(&fixture.input, &arg.field);
1050 !val.is_null() && !val.is_array()
1051 });
1052
1053 let mut builder_expressions = String::new();
1055 if let (true, Some(opts_type)) = (needs_deser, effective_options_type) {
1056 for arg in args {
1057 if arg.arg_type == "json_object" {
1058 let val = super::resolve_field(&fixture.input, &arg.field);
1059 if !val.is_null() && !val.is_array() {
1060 if options_via == "from_json" {
1061 let json_str = serde_json::to_string(val).unwrap_or_default();
1063 let escaped = escape_java(&json_str);
1064 let var_name = &arg.name;
1065 builder_expressions.push_str(&format!(
1066 " var {var_name} = {opts_type}.fromJson(\"{escaped}\");\n",
1067 ));
1068 } else if let Some(obj) = val.as_object() {
1069 let empty_path_fields: Vec<String> = Vec::new();
1071 let path_fields = call_overrides.map(|o| &o.path_fields).unwrap_or(&empty_path_fields);
1072 let builder_expr = java_builder_expression(
1073 obj,
1074 opts_type,
1075 enum_fields,
1076 nested_types,
1077 nested_types_optional,
1078 path_fields,
1079 );
1080 let var_name = &arg.name;
1081 builder_expressions.push_str(&format!(" var {} = {};\n", var_name, builder_expr));
1082 }
1083 }
1084 }
1085 }
1086 }
1087
1088 let (mut setup_lines, args_str) =
1089 build_args_and_setup(&fixture.input, args, class_name, effective_options_type, fixture);
1090
1091 let extra_args_slice: &[String] = call_overrides.map_or(&[], |o| o.extra_args.as_slice());
1096
1097 let mut visitor_var = String::new();
1099 let mut has_visitor_fixture = false;
1100 if let Some(visitor_spec) = &fixture.visitor {
1101 visitor_var = build_java_visitor(&mut setup_lines, visitor_spec, class_name);
1102 has_visitor_fixture = true;
1103 }
1104
1105 let mut final_args = if has_visitor_fixture {
1107 if args_str.is_empty() {
1108 format!("new ConversionOptions().withVisitor({})", visitor_var)
1109 } else if args_str.contains("new ConversionOptions")
1110 || args_str.contains("ConversionOptionsBuilder")
1111 || args_str.contains(".builder()")
1112 {
1113 if args_str.contains(".build()") {
1116 let idx = args_str.rfind(".build()").unwrap();
1117 format!("{}.withVisitor({}){}", &args_str[..idx], visitor_var, &args_str[idx..])
1118 } else {
1119 format!("{}.withVisitor({})", args_str, visitor_var)
1120 }
1121 } else if args_str.ends_with(", null") {
1122 let base = &args_str[..args_str.len() - 6];
1123 format!("{}, new ConversionOptions().withVisitor({})", base, visitor_var)
1124 } else {
1125 format!("{}, new ConversionOptions().withVisitor({})", args_str, visitor_var)
1126 }
1127 } else {
1128 args_str
1129 };
1130
1131 if !extra_args_slice.is_empty() {
1132 let extra_str = extra_args_slice.join(", ");
1133 final_args = if final_args.is_empty() {
1134 extra_str
1135 } else {
1136 format!("{final_args}, {extra_str}")
1137 };
1138 }
1139
1140 let mut assertions_body = String::new();
1142
1143 let needs_source_var = fixture
1145 .assertions
1146 .iter()
1147 .any(|a| a.assertion_type == "method_result" && a.method.as_deref() == Some("run_query"));
1148 if needs_source_var {
1149 if let Some(source_arg) = args.iter().find(|a| a.field == "source_code") {
1150 let field = source_arg.field.strip_prefix("input.").unwrap_or(&source_arg.field);
1151 if let Some(val) = fixture.input.get(field) {
1152 let java_val = json_to_java(val);
1153 assertions_body.push_str(&format!(" var source = {}.getBytes();\n", java_val));
1154 }
1155 }
1156 }
1157
1158 let mut effective_enum_fields: std::collections::HashSet<String> = enum_fields.clone();
1164 if let Some(co) = call_overrides {
1165 for k in co.enum_fields.keys() {
1166 effective_enum_fields.insert(k.clone());
1167 }
1168 }
1169
1170 for assertion in &fixture.assertions {
1171 render_assertion(
1172 &mut assertions_body,
1173 assertion,
1174 result_var,
1175 class_name,
1176 field_resolver,
1177 effective_result_is_simple,
1178 effective_result_is_bytes,
1179 &effective_enum_fields,
1180 );
1181 }
1182
1183 let throws_clause = " throws Exception";
1184
1185 let (client_setup_lines, call_target) = if let Some(factory) = client_factory.as_deref() {
1188 let factory_name = factory.to_lower_camel_case();
1189 let fixture_id = &fixture.id;
1190 let mut setup: Vec<String> = Vec::new();
1191 let has_mock = fixture.mock_response.is_some() || fixture.http.is_some();
1192 let api_key_var = fixture.env.as_ref().and_then(|e| e.api_key_var.as_deref());
1193 if let Some(var) = api_key_var.filter(|_| has_mock) {
1194 setup.push(format!("String apiKey = System.getenv(\"{var}\");"));
1195 setup.push(format!(
1196 "String baseUrl = (apiKey != null && !apiKey.isEmpty()) ? null : System.getProperty(\"mockServerUrl\", System.getenv(\"MOCK_SERVER_URL\")) + \"/fixtures/{fixture_id}\";"
1197 ));
1198 setup.push(format!(
1199 "System.out.println(\"{fixture_id}: \" + (baseUrl == null ? \"using real API ({var} is set)\" : \"using mock server ({var} not set)\"));"
1200 ));
1201 setup.push(format!(
1202 "var client = {class_name}.{factory_name}(baseUrl == null ? apiKey : \"test-key\", baseUrl, null, null, null);"
1203 ));
1204 } else if has_mock {
1205 if fixture.has_host_root_route() {
1206 setup.push(format!(
1207 "String mockUrl = System.getProperty(\"mockServer.{fixture_id}\", System.getProperty(\"mockServerUrl\", System.getenv(\"MOCK_SERVER_URL\")) + \"/fixtures/{fixture_id}\");"
1208 ));
1209 } else {
1210 setup.push(format!(
1211 "String mockUrl = System.getProperty(\"mockServerUrl\", System.getenv(\"MOCK_SERVER_URL\")) + \"/fixtures/{fixture_id}\";"
1212 ));
1213 }
1214 setup.push(format!(
1215 "var client = {class_name}.{factory_name}(\"test-key\", mockUrl, null, null, null);"
1216 ));
1217 } else if let Some(api_key_var) = api_key_var {
1218 setup.push(format!("String apiKey = System.getenv(\"{api_key_var}\");"));
1219 setup.push(format!(
1220 "org.junit.jupiter.api.Assumptions.assumeTrue(apiKey != null && !apiKey.isEmpty(), \"{api_key_var} not set\");"
1221 ));
1222 setup.push(format!("var client = {class_name}.{factory_name}(apiKey);"));
1223 } else {
1224 setup.push(format!("var client = {class_name}.{factory_name}(\"test-key\");"));
1225 }
1226 (setup, "client".to_string())
1227 } else {
1228 (Vec::new(), class_name.to_string())
1229 };
1230
1231 let combined_setup: Vec<String> = client_setup_lines.into_iter().chain(setup_lines).collect();
1233
1234 let call_expr = format!("{call_target}.{function_name}({final_args})");
1235
1236 let is_streaming = fixture.is_streaming_mock();
1238 let collect_snippet = if is_streaming {
1239 crate::codegen::streaming_assertions::StreamingFieldResolver::collect_snippet("java", result_var, "chunks")
1240 .unwrap_or_default()
1241 } else {
1242 String::new()
1243 };
1244
1245 let rendered = crate::template_env::render(
1246 "java/test_method.jinja",
1247 minijinja::context! {
1248 method_name => method_name,
1249 description => description,
1250 builder_expressions => builder_expressions,
1251 setup_lines => combined_setup,
1252 throws_clause => throws_clause,
1253 expects_error => expects_error,
1254 call_expr => call_expr,
1255 result_var => result_var,
1256 collect_snippet => collect_snippet,
1257 assertions_body => assertions_body,
1258 },
1259 );
1260 out.push_str(&rendered);
1261}
1262
1263fn build_args_and_setup(
1267 input: &serde_json::Value,
1268 args: &[crate::config::ArgMapping],
1269 class_name: &str,
1270 options_type: Option<&str>,
1271 fixture: &crate::fixture::Fixture,
1272) -> (Vec<String>, String) {
1273 let fixture_id = &fixture.id;
1274 if args.is_empty() {
1275 return (Vec::new(), String::new());
1276 }
1277
1278 let mut setup_lines: Vec<String> = Vec::new();
1279 let mut parts: Vec<String> = Vec::new();
1280
1281 for arg in args {
1282 if arg.arg_type == "mock_url" {
1283 if fixture.has_host_root_route() {
1284 setup_lines.push(format!(
1285 "String {} = System.getProperty(\"mockServer.{fixture_id}\", System.getProperty(\"mockServerUrl\", System.getenv(\"MOCK_SERVER_URL\")) + \"/fixtures/{fixture_id}\");",
1286 arg.name,
1287 ));
1288 } else {
1289 setup_lines.push(format!(
1290 "String {} = System.getProperty(\"mockServerUrl\", System.getenv(\"MOCK_SERVER_URL\")) + \"/fixtures/{fixture_id}\";",
1291 arg.name,
1292 ));
1293 }
1294 parts.push(arg.name.clone());
1295 continue;
1296 }
1297
1298 if arg.arg_type == "handle" {
1299 let constructor_name = format!("create{}", arg.name.to_upper_camel_case());
1301 let field = arg.field.strip_prefix("input.").unwrap_or(&arg.field);
1302 let config_value = input.get(field).unwrap_or(&serde_json::Value::Null);
1303 if config_value.is_null()
1304 || config_value.is_object() && config_value.as_object().is_some_and(|o| o.is_empty())
1305 {
1306 setup_lines.push(format!("var {} = {class_name}.{constructor_name}(null);", arg.name,));
1307 } else {
1308 let json_str = serde_json::to_string(config_value).unwrap_or_default();
1309 let name = &arg.name;
1310 setup_lines.push(format!(
1311 "var {name}Config = MAPPER.readValue(\"{}\", CrawlConfig.class);",
1312 escape_java(&json_str),
1313 ));
1314 setup_lines.push(format!(
1315 "var {} = {class_name}.{constructor_name}({name}Config);",
1316 arg.name,
1317 name = name,
1318 ));
1319 }
1320 parts.push(arg.name.clone());
1321 continue;
1322 }
1323
1324 let resolved = super::resolve_field(input, &arg.field);
1325 let val = if resolved.is_null() { None } else { Some(resolved) };
1326 match val {
1327 None | Some(serde_json::Value::Null) if arg.optional => {
1328 if arg.arg_type == "json_object" {
1332 if let Some(opts_type) = options_type {
1333 parts.push(format!("{opts_type}.builder().build()"));
1334 } else {
1335 parts.push("null".to_string());
1336 }
1337 } else {
1338 parts.push("null".to_string());
1339 }
1340 }
1341 None | Some(serde_json::Value::Null) => {
1342 let default_val = match arg.arg_type.as_str() {
1344 "string" | "file_path" => "\"\"".to_string(),
1345 "int" | "integer" => "0".to_string(),
1346 "float" | "number" => "0.0d".to_string(),
1347 "bool" | "boolean" => "false".to_string(),
1348 _ => "null".to_string(),
1349 };
1350 parts.push(default_val);
1351 }
1352 Some(v) => {
1353 if arg.arg_type == "json_object" {
1354 if v.is_array() {
1357 if let Some(elem_type) = &arg.element_type {
1358 if elem_type == "BatchBytesItem" || elem_type == "BatchFileItem" {
1359 parts.push(emit_java_batch_item_array(v, elem_type));
1360 continue;
1361 }
1362 }
1363 let elem_type = arg.element_type.as_deref();
1365 parts.push(json_to_java_typed(v, elem_type));
1366 continue;
1367 }
1368 if options_type.is_some() {
1370 parts.push(arg.name.clone());
1371 continue;
1372 }
1373 parts.push(json_to_java(v));
1374 continue;
1375 }
1376 if arg.arg_type == "bytes" {
1378 let val = json_to_java(v);
1379 parts.push(format!("{val}.getBytes()"));
1380 continue;
1381 }
1382 if arg.arg_type == "file_path" {
1384 let val = json_to_java(v);
1385 parts.push(format!("java.nio.file.Path.of({val})"));
1386 continue;
1387 }
1388 parts.push(json_to_java(v));
1389 }
1390 }
1391 }
1392
1393 (setup_lines, parts.join(", "))
1394}
1395
1396#[allow(clippy::too_many_arguments)]
1397fn render_assertion(
1398 out: &mut String,
1399 assertion: &Assertion,
1400 result_var: &str,
1401 class_name: &str,
1402 field_resolver: &FieldResolver,
1403 result_is_simple: bool,
1404 result_is_bytes: bool,
1405 enum_fields: &std::collections::HashSet<String>,
1406) {
1407 if result_is_bytes {
1412 match assertion.assertion_type.as_str() {
1413 "not_empty" => {
1414 out.push_str(&format!(
1415 " assertTrue({result_var}.length > 0, \"expected non-empty value\");\n"
1416 ));
1417 return;
1418 }
1419 "is_empty" => {
1420 out.push_str(&format!(
1421 " assertEquals(0, {result_var}.length, \"expected empty value\");\n"
1422 ));
1423 return;
1424 }
1425 "count_equals" | "length_equals" => {
1426 if let Some(n) = assertion.value.as_ref().and_then(|v| v.as_u64()) {
1427 out.push_str(&format!(" assertEquals({n}, {result_var}.length);\n"));
1428 }
1429 return;
1430 }
1431 "count_min" | "length_min" => {
1432 if let Some(n) = assertion.value.as_ref().and_then(|v| v.as_u64()) {
1433 out.push_str(&format!(
1434 " assertTrue({result_var}.length >= {n}, \"expected length >= {n}\");\n"
1435 ));
1436 }
1437 return;
1438 }
1439 "not_error" => {
1440 out.push_str(&format!(
1441 " Assertions.assertNotNull({result_var}, \"expected non-null byte[] response\");\n"
1442 ));
1443 return;
1444 }
1445 _ => {
1446 out.push_str(&format!(
1447 " // skipped: assertion type '{}' not supported on byte[] result\n",
1448 assertion.assertion_type
1449 ));
1450 return;
1451 }
1452 }
1453 }
1454
1455 if let Some(f) = &assertion.field {
1457 match f.as_str() {
1458 "chunks_have_content" => {
1460 let pred = format!(
1461 "{result_var}.chunks().orElse(java.util.List.of()).stream().allMatch(c -> c.content() != null && !c.content().isBlank())"
1462 );
1463 out.push_str(&crate::template_env::render(
1464 "java/synthetic_assertion.jinja",
1465 minijinja::context! {
1466 assertion_kind => "chunks_content",
1467 assertion_type => assertion.assertion_type.as_str(),
1468 pred => pred,
1469 field_name => f,
1470 },
1471 ));
1472 return;
1473 }
1474 "chunks_have_heading_context" => {
1475 let pred = format!(
1476 "{result_var}.chunks().orElse(java.util.List.of()).stream().allMatch(c -> c.metadata().headingContext().isPresent())"
1477 );
1478 out.push_str(&crate::template_env::render(
1479 "java/synthetic_assertion.jinja",
1480 minijinja::context! {
1481 assertion_kind => "chunks_heading_context",
1482 assertion_type => assertion.assertion_type.as_str(),
1483 pred => pred,
1484 field_name => f,
1485 },
1486 ));
1487 return;
1488 }
1489 "chunks_have_embeddings" => {
1490 let pred = format!(
1491 "{result_var}.chunks().orElse(java.util.List.of()).stream().allMatch(c -> c.embedding() != null && !c.embedding().isEmpty())"
1492 );
1493 out.push_str(&crate::template_env::render(
1494 "java/synthetic_assertion.jinja",
1495 minijinja::context! {
1496 assertion_kind => "chunks_embeddings",
1497 assertion_type => assertion.assertion_type.as_str(),
1498 pred => pred,
1499 field_name => f,
1500 },
1501 ));
1502 return;
1503 }
1504 "first_chunk_starts_with_heading" => {
1505 let pred = format!(
1506 "{result_var}.chunks().orElse(java.util.List.of()).stream().findFirst().map(c -> c.metadata().headingContext().isPresent()).orElse(false)"
1507 );
1508 out.push_str(&crate::template_env::render(
1509 "java/synthetic_assertion.jinja",
1510 minijinja::context! {
1511 assertion_kind => "first_chunk_heading",
1512 assertion_type => assertion.assertion_type.as_str(),
1513 pred => pred,
1514 field_name => f,
1515 },
1516 ));
1517 return;
1518 }
1519 "embedding_dimensions" => {
1523 let embed_list = if result_is_simple {
1525 result_var.to_string()
1526 } else {
1527 format!("{result_var}.embeddings()")
1528 };
1529 let expr = format!("({embed_list}.isEmpty() ? 0 : {embed_list}.get(0).size())");
1530 let java_val = assertion.value.as_ref().map(json_to_java).unwrap_or_default();
1531 out.push_str(&crate::template_env::render(
1532 "java/synthetic_assertion.jinja",
1533 minijinja::context! {
1534 assertion_kind => "embedding_dimensions",
1535 assertion_type => assertion.assertion_type.as_str(),
1536 expr => expr,
1537 java_val => java_val,
1538 field_name => f,
1539 },
1540 ));
1541 return;
1542 }
1543 "embeddings_valid" | "embeddings_finite" | "embeddings_non_zero" | "embeddings_normalized" => {
1544 let embed_list = if result_is_simple {
1546 result_var.to_string()
1547 } else {
1548 format!("{result_var}.embeddings()")
1549 };
1550 let pred = match f.as_str() {
1551 "embeddings_valid" => {
1552 format!("{embed_list}.stream().allMatch(e -> e != null && !e.isEmpty())")
1553 }
1554 "embeddings_finite" => {
1555 format!("{embed_list}.stream().flatMap(java.util.Collection::stream).allMatch(Float::isFinite)")
1556 }
1557 "embeddings_non_zero" => {
1558 format!("{embed_list}.stream().allMatch(e -> e.stream().anyMatch(v -> v != 0.0f))")
1559 }
1560 "embeddings_normalized" => format!(
1561 "{embed_list}.stream().allMatch(e -> {{ double n = e.stream().mapToDouble(v -> v * v).sum(); return Math.abs(n - 1.0) < 1e-3; }})"
1562 ),
1563 _ => unreachable!(),
1564 };
1565 let assertion_kind = format!("embeddings_{}", f.strip_prefix("embeddings_").unwrap_or(f));
1566 out.push_str(&crate::template_env::render(
1567 "java/synthetic_assertion.jinja",
1568 minijinja::context! {
1569 assertion_kind => assertion_kind,
1570 assertion_type => assertion.assertion_type.as_str(),
1571 pred => pred,
1572 field_name => f,
1573 },
1574 ));
1575 return;
1576 }
1577 "keywords" | "keywords_count" => {
1579 out.push_str(&crate::template_env::render(
1580 "java/synthetic_assertion.jinja",
1581 minijinja::context! {
1582 assertion_kind => "keywords",
1583 field_name => f,
1584 },
1585 ));
1586 return;
1587 }
1588 "metadata" => {
1591 match assertion.assertion_type.as_str() {
1592 "not_empty" | "is_empty" => {
1593 out.push_str(&crate::template_env::render(
1594 "java/synthetic_assertion.jinja",
1595 minijinja::context! {
1596 assertion_kind => "metadata",
1597 assertion_type => assertion.assertion_type.as_str(),
1598 result_var => result_var,
1599 },
1600 ));
1601 return;
1602 }
1603 _ => {} }
1605 }
1606 _ => {}
1607 }
1608 }
1609
1610 if let Some(f) = &assertion.field {
1613 if !f.is_empty() && crate::codegen::streaming_assertions::is_streaming_virtual_field(f) {
1614 if let Some(expr) =
1615 crate::codegen::streaming_assertions::StreamingFieldResolver::accessor(f, "java", "chunks")
1616 {
1617 let line = match assertion.assertion_type.as_str() {
1618 "count_min" => {
1619 if let Some(n) = assertion.value.as_ref().and_then(|v| v.as_u64()) {
1620 format!(" assertTrue({expr}.size() >= {n}, \"expected >= {n} chunks\");\n")
1621 } else {
1622 String::new()
1623 }
1624 }
1625 "count_equals" => {
1626 if let Some(n) = assertion.value.as_ref().and_then(|v| v.as_u64()) {
1627 format!(" assertEquals({n}, {expr}.size());\n")
1628 } else {
1629 String::new()
1630 }
1631 }
1632 "equals" => {
1633 if let Some(serde_json::Value::String(s)) = &assertion.value {
1634 let escaped = crate::escape::escape_java(s);
1635 format!(" assertEquals(\"{escaped}\", {expr});\n")
1636 } else if let Some(n) = assertion.value.as_ref().and_then(|v| v.as_u64()) {
1637 format!(" assertEquals({n}, {expr});\n")
1638 } else {
1639 String::new()
1640 }
1641 }
1642 "not_empty" => format!(" assertFalse({expr}.isEmpty(), \"expected non-empty\");\n"),
1643 "is_empty" => format!(" assertTrue({expr}.isEmpty(), \"expected empty\");\n"),
1644 "is_true" => format!(" assertTrue({expr}, \"expected true\");\n"),
1645 "is_false" => format!(" assertFalse({expr}, \"expected false\");\n"),
1646 "greater_than" => {
1647 if let Some(n) = assertion.value.as_ref().and_then(|v| v.as_u64()) {
1648 format!(" assertTrue({expr} > {n}, \"expected > {n}\");\n")
1649 } else {
1650 String::new()
1651 }
1652 }
1653 "greater_than_or_equal" => {
1654 if let Some(n) = assertion.value.as_ref().and_then(|v| v.as_u64()) {
1655 format!(" assertTrue({expr} >= {n}, \"expected >= {n}\");\n")
1656 } else {
1657 String::new()
1658 }
1659 }
1660 "contains" => {
1661 if let Some(serde_json::Value::String(s)) = &assertion.value {
1662 let escaped = crate::escape::escape_java(s);
1663 format!(
1664 " assertTrue({expr}.contains(\"{escaped}\"), \"expected to contain: {escaped}\");\n"
1665 )
1666 } else {
1667 String::new()
1668 }
1669 }
1670 _ => format!(
1671 " // streaming field '{f}': assertion type '{}' not rendered\n",
1672 assertion.assertion_type
1673 ),
1674 };
1675 if !line.is_empty() {
1676 out.push_str(&line);
1677 }
1678 }
1679 return;
1680 }
1681 }
1682
1683 if let Some(f) = &assertion.field {
1685 if !f.is_empty() && !field_resolver.is_valid_for_result(f) {
1686 out.push_str(&crate::template_env::render(
1687 "java/synthetic_assertion.jinja",
1688 minijinja::context! {
1689 assertion_kind => "skipped",
1690 field_name => f,
1691 },
1692 ));
1693 return;
1694 }
1695 }
1696
1697 let field_is_enum = assertion
1702 .field
1703 .as_deref()
1704 .is_some_and(|f| enum_fields.contains(f) || enum_fields.contains(field_resolver.resolve(f)));
1705
1706 let field_is_array = assertion
1710 .field
1711 .as_deref()
1712 .is_some_and(|f| field_resolver.is_array(field_resolver.resolve(f)));
1713
1714 let field_expr = if result_is_simple {
1715 result_var.to_string()
1716 } else {
1717 match &assertion.field {
1718 Some(f) if !f.is_empty() => {
1719 let accessor = field_resolver.accessor(f, "java", result_var);
1720 let resolved = field_resolver.resolve(f);
1721 if field_resolver.is_optional(resolved) && !field_resolver.has_map_access(f) {
1728 let optional_expr = format!("java.util.Optional.ofNullable({accessor})");
1731 if field_is_enum {
1735 match assertion.assertion_type.as_str() {
1736 "not_empty" | "is_empty" => optional_expr,
1737 _ => format!("{optional_expr}.map(v -> v.getValue()).orElse(\"\")"),
1738 }
1739 } else {
1740 match assertion.assertion_type.as_str() {
1741 "not_empty" | "is_empty" => optional_expr,
1744 "count_min" | "count_equals" => {
1746 format!("{optional_expr}.orElse(java.util.List.of())")
1747 }
1748 "greater_than" | "less_than" | "greater_than_or_equal" | "less_than_or_equal" => {
1750 if field_resolver.is_array(resolved) {
1751 format!("{optional_expr}.orElse(java.util.List.of())")
1752 } else {
1753 format!("{optional_expr}.orElse(0L)")
1754 }
1755 }
1756 "equals" => {
1759 if let Some(expected) = &assertion.value {
1760 if expected.is_number() {
1761 format!("{optional_expr}.orElse(0L)")
1762 } else {
1763 format!("{optional_expr}.orElse(\"\")")
1764 }
1765 } else {
1766 format!("{optional_expr}.orElse(\"\")")
1767 }
1768 }
1769 _ if field_resolver.is_array(resolved) => {
1770 format!("{optional_expr}.orElse(java.util.List.of())")
1771 }
1772 _ => format!("{optional_expr}.orElse(\"\")"),
1773 }
1774 }
1775 } else {
1776 accessor
1777 }
1778 }
1779 _ => result_var.to_string(),
1780 }
1781 };
1782
1783 let string_expr = if field_is_enum && !field_expr.contains(".map(v -> v.getValue())") {
1790 format!("{field_expr}.getValue()")
1791 } else {
1792 field_expr.clone()
1793 };
1794
1795 let assertion_type = assertion.assertion_type.as_str();
1797 let java_val = assertion.value.as_ref().map(json_to_java).unwrap_or_default();
1798 let is_string_val = assertion.value.as_ref().is_some_and(|v| v.is_string());
1799 let is_numeric_val = assertion.value.as_ref().is_some_and(|v| v.is_number());
1800
1801 let values_java: Vec<String> = assertion
1805 .values
1806 .as_ref()
1807 .map(|values| values.iter().map(json_to_java).collect::<Vec<_>>())
1808 .or_else(|| assertion.value.as_ref().map(|v| vec![json_to_java(v)]))
1809 .unwrap_or_default();
1810
1811 let contains_any_expr = if !values_java.is_empty() {
1812 values_java
1813 .iter()
1814 .map(|v| format!("{string_expr}.contains({v})"))
1815 .collect::<Vec<_>>()
1816 .join(" || ")
1817 } else {
1818 String::new()
1819 };
1820
1821 let length_expr = if result_is_bytes {
1822 format!("{field_expr}.length")
1823 } else {
1824 format!("{field_expr}.length()")
1825 };
1826
1827 let n = assertion.value.as_ref().and_then(|v| v.as_u64()).unwrap_or(0);
1828
1829 let call_expr = if let Some(method_name) = &assertion.method {
1830 build_java_method_call(result_var, method_name, assertion.args.as_ref(), class_name)
1831 } else {
1832 String::new()
1833 };
1834
1835 let check = assertion.check.as_deref().unwrap_or("is_true");
1836
1837 let java_check_val = assertion.value.as_ref().map(json_to_java).unwrap_or_default();
1838
1839 let check_n = assertion.value.as_ref().and_then(|v| v.as_u64()).unwrap_or(0);
1840
1841 let is_bool_val = assertion.value.as_ref().is_some_and(|v| v.is_boolean());
1842 let bool_is_true = assertion.value.as_ref().is_some_and(|v| v.as_bool() == Some(true));
1843
1844 let method_returns_collection = assertion
1845 .method
1846 .as_ref()
1847 .is_some_and(|m| matches!(m.as_str(), "find_nodes_by_type" | "findNodesByType"));
1848
1849 let rendered = crate::template_env::render(
1850 "java/assertion.jinja",
1851 minijinja::context! {
1852 assertion_type,
1853 java_val,
1854 string_expr,
1855 field_expr,
1856 field_is_enum,
1857 field_is_array,
1858 is_string_val,
1859 is_numeric_val,
1860 values_java => values_java,
1861 contains_any_expr,
1862 length_expr,
1863 n,
1864 call_expr,
1865 check,
1866 java_check_val,
1867 check_n,
1868 is_bool_val,
1869 bool_is_true,
1870 method_returns_collection,
1871 },
1872 );
1873 out.push_str(&rendered);
1874}
1875
1876fn build_java_method_call(
1880 result_var: &str,
1881 method_name: &str,
1882 args: Option<&serde_json::Value>,
1883 class_name: &str,
1884) -> String {
1885 match method_name {
1886 "root_child_count" => format!("{result_var}.rootNode().childCount()"),
1887 "root_node_type" => format!("{result_var}.rootNode().kind()"),
1888 "named_children_count" => format!("{result_var}.rootNode().namedChildCount()"),
1889 "has_error_nodes" => format!("{class_name}.treeHasErrorNodes({result_var})"),
1890 "error_count" | "tree_error_count" => format!("{class_name}.treeErrorCount({result_var})"),
1891 "tree_to_sexp" => format!("{class_name}.treeToSexp({result_var})"),
1892 "contains_node_type" => {
1893 let node_type = args
1894 .and_then(|a| a.get("node_type"))
1895 .and_then(|v| v.as_str())
1896 .unwrap_or("");
1897 format!("{class_name}.treeContainsNodeType({result_var}, \"{node_type}\")")
1898 }
1899 "find_nodes_by_type" => {
1900 let node_type = args
1901 .and_then(|a| a.get("node_type"))
1902 .and_then(|v| v.as_str())
1903 .unwrap_or("");
1904 format!("{class_name}.findNodesByType({result_var}, \"{node_type}\")")
1905 }
1906 "run_query" => {
1907 let query_source = args
1908 .and_then(|a| a.get("query_source"))
1909 .and_then(|v| v.as_str())
1910 .unwrap_or("");
1911 let language = args
1912 .and_then(|a| a.get("language"))
1913 .and_then(|v| v.as_str())
1914 .unwrap_or("");
1915 let escaped_query = escape_java(query_source);
1916 format!("{class_name}.runQuery({result_var}, \"{language}\", \"{escaped_query}\", source)")
1917 }
1918 _ => {
1919 format!("{result_var}.{}()", method_name.to_lower_camel_case())
1920 }
1921 }
1922}
1923
1924fn json_to_java(value: &serde_json::Value) -> String {
1926 json_to_java_typed(value, None)
1927}
1928
1929fn emit_java_batch_item_array(arr: &serde_json::Value, elem_type: &str) -> String {
1933 if let Some(items) = arr.as_array() {
1934 let item_strs: Vec<String> = items
1935 .iter()
1936 .filter_map(|item| {
1937 if let Some(obj) = item.as_object() {
1938 match elem_type {
1939 "BatchBytesItem" => {
1940 let content = obj.get("content").and_then(|v| v.as_array());
1941 let mime_type = obj.get("mime_type").and_then(|v| v.as_str()).unwrap_or("text/plain");
1942 let content_code = if let Some(arr) = content {
1943 let bytes: Vec<String> = arr
1944 .iter()
1945 .filter_map(|v| v.as_u64().map(|n| format!("(byte) {}", n)))
1946 .collect();
1947 format!("new byte[] {{{}}}", bytes.join(", "))
1948 } else {
1949 "new byte[] {}".to_string()
1950 };
1951 Some(format!("new {}({}, \"{}\", null)", elem_type, content_code, mime_type))
1952 }
1953 "BatchFileItem" => {
1954 let path = obj.get("path").and_then(|v| v.as_str()).unwrap_or("");
1955 Some(format!(
1956 "new {}(java.nio.file.Paths.get(\"{}\"), null)",
1957 elem_type, path
1958 ))
1959 }
1960 _ => None,
1961 }
1962 } else {
1963 None
1964 }
1965 })
1966 .collect();
1967 format!("java.util.Arrays.asList({})", item_strs.join(", "))
1968 } else {
1969 "java.util.List.of()".to_string()
1970 }
1971}
1972
1973fn json_to_java_typed(value: &serde_json::Value, element_type: Option<&str>) -> String {
1974 match value {
1975 serde_json::Value::String(s) => format!("\"{}\"", escape_java(s)),
1976 serde_json::Value::Bool(b) => b.to_string(),
1977 serde_json::Value::Number(n) => {
1978 if n.is_f64() {
1979 match element_type {
1980 Some("f32" | "float" | "Float") => format!("{}f", n),
1981 _ => format!("{}d", n),
1982 }
1983 } else {
1984 n.to_string()
1985 }
1986 }
1987 serde_json::Value::Null => "null".to_string(),
1988 serde_json::Value::Array(arr) => {
1989 let items: Vec<String> = arr.iter().map(|v| json_to_java_typed(v, element_type)).collect();
1990 format!("java.util.List.of({})", items.join(", "))
1991 }
1992 serde_json::Value::Object(_) => {
1993 let json_str = serde_json::to_string(value).unwrap_or_default();
1994 format!("\"{}\"", escape_java(&json_str))
1995 }
1996 }
1997}
1998
1999fn java_builder_expression(
2010 obj: &serde_json::Map<String, serde_json::Value>,
2011 type_name: &str,
2012 enum_fields: &std::collections::HashSet<String>,
2013 nested_types: &std::collections::HashMap<String, String>,
2014 nested_types_optional: bool,
2015 path_fields: &[String],
2016) -> String {
2017 let mut expr = format!("{}.builder()", type_name);
2018 for (key, val) in obj {
2019 let camel_key = key.to_lower_camel_case();
2021 let method_name = format!("with{}", camel_key.to_upper_camel_case());
2022
2023 let java_val = match val {
2024 serde_json::Value::String(s) => {
2025 if enum_fields.contains(&camel_key) {
2028 let enum_type_name = camel_key.to_upper_camel_case();
2030 let variant_name = s.to_upper_camel_case();
2031 format!("{}.{}", enum_type_name, variant_name)
2032 } else if camel_key == "preset" && type_name == "PreprocessingOptions" {
2033 let variant_name = s.to_upper_camel_case();
2035 format!("PreprocessingPreset.{}", variant_name)
2036 } else if path_fields.contains(key) {
2037 format!("Optional.of(java.nio.file.Path.of(\"{}\"))", escape_java(s))
2039 } else {
2040 format!("\"{}\"", escape_java(s))
2042 }
2043 }
2044 serde_json::Value::Bool(b) => b.to_string(),
2045 serde_json::Value::Null => "null".to_string(),
2046 serde_json::Value::Number(n) => {
2047 let camel_key = key.to_lower_camel_case();
2055 let is_plain_field = matches!(camel_key.as_str(), "listIndentWidth" | "wrapWidth");
2056 let is_primitive_builder = matches!(type_name, "SecurityLimits" | "SecurityLimitsBuilder");
2059
2060 if is_plain_field || is_primitive_builder {
2061 if n.is_f64() {
2063 format!("{}d", n)
2064 } else {
2065 format!("{}L", n)
2066 }
2067 } else {
2068 if n.is_f64() {
2070 format!("Optional.of({}d)", n)
2071 } else {
2072 format!("Optional.of({}L)", n)
2073 }
2074 }
2075 }
2076 serde_json::Value::Array(arr) => {
2077 let items: Vec<String> = arr.iter().map(|v| json_to_java_typed(v, None)).collect();
2078 format!("java.util.List.of({})", items.join(", "))
2079 }
2080 serde_json::Value::Object(nested) => {
2081 let nested_type = nested_types
2083 .get(key.as_str())
2084 .cloned()
2085 .unwrap_or_else(|| format!("{}Options", key.to_upper_camel_case()));
2086 let inner = java_builder_expression(
2087 nested,
2088 &nested_type,
2089 enum_fields,
2090 nested_types,
2091 nested_types_optional,
2092 &[],
2093 );
2094 let is_primitive_builder = matches!(type_name, "SecurityLimits" | "SecurityLimitsBuilder");
2098 if is_primitive_builder || !nested_types_optional {
2099 inner
2100 } else {
2101 format!("Optional.of({inner})")
2102 }
2103 }
2104 };
2105 expr.push_str(&format!(".{}({})", method_name, java_val));
2106 }
2107 expr.push_str(".build()");
2108 expr
2109}
2110
2111fn default_java_nested_types() -> std::collections::HashMap<String, String> {
2118 [
2119 ("chunking", "ChunkingConfig"),
2120 ("ocr", "OcrConfig"),
2121 ("images", "ImageExtractionConfig"),
2122 ("html_output", "HtmlOutputConfig"),
2123 ("language_detection", "LanguageDetectionConfig"),
2124 ("postprocessor", "PostProcessorConfig"),
2125 ("acceleration", "AccelerationConfig"),
2126 ("email", "EmailConfig"),
2127 ("pages", "PageConfig"),
2128 ("pdf_options", "PdfConfig"),
2129 ("layout", "LayoutDetectionConfig"),
2130 ("tree_sitter", "TreeSitterConfig"),
2131 ("structured_extraction", "StructuredExtractionConfig"),
2132 ("content_filter", "ContentFilterConfig"),
2133 ("token_reduction", "TokenReductionOptions"),
2134 ("security_limits", "SecurityLimits"),
2135 ]
2136 .iter()
2137 .map(|(k, v)| (k.to_string(), v.to_string()))
2138 .collect()
2139}
2140
2141#[allow(dead_code)]
2148fn collect_enum_and_nested_types(
2149 obj: &serde_json::Map<String, serde_json::Value>,
2150 enum_fields: &std::collections::HashMap<String, String>,
2151 types_out: &mut std::collections::BTreeSet<String>,
2152) {
2153 for (key, val) in obj {
2154 let camel_key = key.to_lower_camel_case();
2156 if let Some(enum_type) = enum_fields.get(&camel_key) {
2157 types_out.insert(enum_type.clone());
2159 } else if camel_key == "preset" {
2160 types_out.insert("PreprocessingPreset".to_string());
2162 }
2163 if let Some(nested) = val.as_object() {
2165 collect_enum_and_nested_types(nested, enum_fields, types_out);
2166 }
2167 }
2168}
2169
2170fn collect_nested_type_names(
2171 obj: &serde_json::Map<String, serde_json::Value>,
2172 nested_types: &std::collections::HashMap<String, String>,
2173 types_out: &mut std::collections::BTreeSet<String>,
2174) {
2175 for (key, val) in obj {
2176 if let Some(type_name) = nested_types.get(key.as_str()) {
2177 types_out.insert(type_name.clone());
2178 }
2179 if let Some(nested) = val.as_object() {
2180 collect_nested_type_names(nested, nested_types, types_out);
2181 }
2182 }
2183}
2184
2185fn build_java_visitor(
2191 setup_lines: &mut Vec<String>,
2192 visitor_spec: &crate::fixture::VisitorSpec,
2193 class_name: &str,
2194) -> String {
2195 setup_lines.push("class _TestVisitor implements Visitor {".to_string());
2196 for (method_name, action) in &visitor_spec.callbacks {
2197 emit_java_visitor_method(setup_lines, method_name, action, class_name);
2198 }
2199 setup_lines.push("}".to_string());
2200 setup_lines.push("var visitor = new _TestVisitor();".to_string());
2201 "visitor".to_string()
2202}
2203
2204fn emit_java_visitor_method(
2206 setup_lines: &mut Vec<String>,
2207 method_name: &str,
2208 action: &CallbackAction,
2209 _class_name: &str,
2210) {
2211 let camel_method = method_to_camel(method_name);
2212 let params = match method_name {
2213 "visit_link" => "NodeContext ctx, String href, String text, String title",
2214 "visit_image" => "NodeContext ctx, String src, String alt, String title",
2215 "visit_heading" => "NodeContext ctx, int level, String text, String id",
2216 "visit_code_block" => "NodeContext ctx, String lang, String code",
2217 "visit_code_inline"
2218 | "visit_strong"
2219 | "visit_emphasis"
2220 | "visit_strikethrough"
2221 | "visit_underline"
2222 | "visit_subscript"
2223 | "visit_superscript"
2224 | "visit_mark"
2225 | "visit_button"
2226 | "visit_summary"
2227 | "visit_figcaption"
2228 | "visit_definition_term"
2229 | "visit_definition_description" => "NodeContext ctx, String text",
2230 "visit_text" => "NodeContext ctx, String text",
2231 "visit_list_item" => "NodeContext ctx, boolean ordered, String marker, String text",
2232 "visit_blockquote" => "NodeContext ctx, String content, long depth",
2233 "visit_table_row" => "NodeContext ctx, java.util.List<String> cells, boolean isHeader",
2234 "visit_custom_element" => "NodeContext ctx, String tagName, String html",
2235 "visit_form" => "NodeContext ctx, String actionUrl, String method",
2236 "visit_input" => "NodeContext ctx, String inputType, String name, String value",
2237 "visit_audio" | "visit_video" | "visit_iframe" => "NodeContext ctx, String src",
2238 "visit_details" => "NodeContext ctx, boolean isOpen",
2239 "visit_element_end" | "visit_table_end" | "visit_definition_list_end" | "visit_figure_end" => {
2240 "NodeContext ctx, String output"
2241 }
2242 "visit_list_start" => "NodeContext ctx, boolean ordered",
2243 "visit_list_end" => "NodeContext ctx, boolean ordered, String output",
2244 _ => "NodeContext ctx",
2245 };
2246
2247 let (action_type, action_value, format_args) = match action {
2249 CallbackAction::Skip => ("skip", String::new(), Vec::new()),
2250 CallbackAction::Continue => ("continue", String::new(), Vec::new()),
2251 CallbackAction::PreserveHtml => ("preserve_html", String::new(), Vec::new()),
2252 CallbackAction::Custom { output } => ("custom_literal", escape_java(output), Vec::new()),
2253 CallbackAction::CustomTemplate { template, .. } => {
2254 let mut format_str = String::with_capacity(template.len());
2256 let mut format_args: Vec<String> = Vec::new();
2257 let mut chars = template.chars().peekable();
2258 while let Some(ch) = chars.next() {
2259 if ch == '{' {
2260 let mut name = String::new();
2262 let mut closed = false;
2263 for inner in chars.by_ref() {
2264 if inner == '}' {
2265 closed = true;
2266 break;
2267 }
2268 name.push(inner);
2269 }
2270 if closed && !name.is_empty() && name.chars().all(|c| c.is_alphanumeric() || c == '_') {
2271 let camel_name = name.as_str().to_lower_camel_case();
2272 format_args.push(camel_name);
2273 format_str.push_str("%s");
2274 } else {
2275 format_str.push('{');
2277 format_str.push_str(&name);
2278 if closed {
2279 format_str.push('}');
2280 }
2281 }
2282 } else {
2283 format_str.push(ch);
2284 }
2285 }
2286 let escaped = escape_java(&format_str);
2287 if format_args.is_empty() {
2288 ("custom_literal", escaped, Vec::new())
2289 } else {
2290 ("custom_formatted", escaped, format_args)
2291 }
2292 }
2293 };
2294
2295 let params = params.to_string();
2296
2297 let rendered = crate::template_env::render(
2298 "java/visitor_method.jinja",
2299 minijinja::context! {
2300 camel_method,
2301 params,
2302 action_type,
2303 action_value,
2304 format_args => format_args,
2305 },
2306 );
2307 setup_lines.push(rendered);
2308}
2309
2310fn method_to_camel(snake: &str) -> String {
2312 snake.to_lower_camel_case()
2313}
2314
2315#[cfg(test)]
2316mod tests {
2317 use crate::config::{CallConfig, E2eConfig, SelectWhen};
2318 use crate::fixture::Fixture;
2319 use std::collections::HashMap;
2320
2321 fn make_fixture_with_input(id: &str, input: serde_json::Value) -> Fixture {
2322 Fixture {
2323 id: id.to_string(),
2324 category: None,
2325 description: "test fixture".to_string(),
2326 tags: vec![],
2327 skip: None,
2328 env: None,
2329 call: None,
2330 input,
2331 mock_response: None,
2332 source: String::new(),
2333 http: None,
2334 assertions: vec![],
2335 visitor: None,
2336 }
2337 }
2338
2339 #[test]
2342 fn test_java_select_when_routes_to_batch_scrape() {
2343 let mut calls = HashMap::new();
2344 calls.insert(
2345 "batch_scrape".to_string(),
2346 CallConfig {
2347 function: "batchScrape".to_string(),
2348 module: "com.example.kreuzcrawl".to_string(),
2349 select_when: Some(SelectWhen::InputHas("batch_urls".to_string())),
2350 ..CallConfig::default()
2351 },
2352 );
2353
2354 let e2e_config = E2eConfig {
2355 call: CallConfig {
2356 function: "scrape".to_string(),
2357 module: "com.example.kreuzcrawl".to_string(),
2358 ..CallConfig::default()
2359 },
2360 calls,
2361 ..E2eConfig::default()
2362 };
2363
2364 let fixture = make_fixture_with_input("batch_empty_urls", serde_json::json!({ "batch_urls": [] }));
2366
2367 let resolved_call = e2e_config.resolve_call_for_fixture(fixture.call.as_deref(), &fixture.input);
2368 assert_eq!(resolved_call.function, "batchScrape");
2369
2370 let fixture_no_batch =
2372 make_fixture_with_input("simple_scrape", serde_json::json!({ "url": "https://example.com" }));
2373 let resolved_default =
2374 e2e_config.resolve_call_for_fixture(fixture_no_batch.call.as_deref(), &fixture_no_batch.input);
2375 assert_eq!(resolved_default.function, "scrape");
2376 }
2377}