use crate::config::E2eConfig;
use crate::escape::{go_string_literal, sanitize_filename};
use crate::field_access::FieldResolver;
use crate::fixture::{Assertion, CallbackAction, Fixture, FixtureGroup, HttpFixture};
use alef_codegen::naming::{go_param_name, to_go_name};
use alef_core::backend::GeneratedFile;
use alef_core::config::AlefConfig;
use alef_core::hash::{self, CommentStyle};
use anyhow::Result;
use heck::ToUpperCamelCase;
use std::fmt::Write as FmtWrite;
use std::path::PathBuf;
use super::E2eCodegen;
pub struct GoCodegen;
impl E2eCodegen for GoCodegen {
fn generate(
&self,
groups: &[FixtureGroup],
e2e_config: &E2eConfig,
alef_config: &AlefConfig,
) -> Result<Vec<GeneratedFile>> {
let lang = self.language_name();
let output_base = PathBuf::from(e2e_config.effective_output()).join(lang);
let mut files = Vec::new();
let call = &e2e_config.call;
let overrides = call.overrides.get(lang);
let module_path = overrides
.and_then(|o| o.module.as_ref())
.cloned()
.unwrap_or_else(|| call.module.clone());
let import_alias = overrides
.and_then(|o| o.alias.as_ref())
.cloned()
.unwrap_or_else(|| "pkg".to_string());
let go_pkg = e2e_config.resolve_package("go");
let go_module_path = go_pkg
.as_ref()
.and_then(|p| p.module.as_ref())
.cloned()
.unwrap_or_else(|| module_path.clone());
let replace_path = go_pkg.as_ref().and_then(|p| p.path.as_ref()).cloned();
let go_version = go_pkg
.as_ref()
.and_then(|p| p.version.as_ref())
.cloned()
.unwrap_or_else(|| {
alef_config
.resolved_version()
.map(|v| format!("v{v}"))
.unwrap_or_else(|| "v0.0.0".to_string())
});
let field_resolver = FieldResolver::new(
&e2e_config.fields,
&e2e_config.fields_optional,
&e2e_config.result_fields,
&e2e_config.fields_array,
);
let effective_replace = match e2e_config.dep_mode {
crate::config::DependencyMode::Registry => None,
crate::config::DependencyMode::Local => replace_path.as_deref().map(String::from),
};
files.push(GeneratedFile {
path: output_base.join("go.mod"),
content: render_go_mod(&go_module_path, effective_replace.as_deref(), &go_version),
generated_header: false,
});
for group in groups {
let active: Vec<&Fixture> = group
.fixtures
.iter()
.filter(|f| f.skip.as_ref().is_none_or(|s| !s.should_skip(lang)))
.collect();
if active.is_empty() {
continue;
}
let filename = format!("{}_test.go", sanitize_filename(&group.category));
let content = render_test_file(
&group.category,
&active,
&module_path,
&import_alias,
&field_resolver,
e2e_config,
);
files.push(GeneratedFile {
path: output_base.join(filename),
content,
generated_header: true,
});
}
Ok(files)
}
fn language_name(&self) -> &'static str {
"go"
}
}
fn render_go_mod(go_module_path: &str, replace_path: Option<&str>, version: &str) -> String {
let mut out = String::new();
let _ = writeln!(out, "module e2e_go");
let _ = writeln!(out);
let _ = writeln!(out, "go 1.26");
let _ = writeln!(out);
let _ = writeln!(out, "require (");
let _ = writeln!(out, "\t{go_module_path} {version}");
let _ = writeln!(out, "\tgithub.com/stretchr/testify v1.11.1");
let _ = writeln!(out, ")");
if let Some(path) = replace_path {
let _ = writeln!(out);
let _ = writeln!(out, "replace {go_module_path} => {path}");
}
out
}
fn render_test_file(
category: &str,
fixtures: &[&Fixture],
go_module_path: &str,
import_alias: &str,
field_resolver: &FieldResolver,
e2e_config: &crate::config::E2eConfig,
) -> String {
let mut out = String::new();
out.push_str(&hash::header(CommentStyle::DoubleSlash));
let _ = writeln!(out);
let needs_pkg = fixtures.iter().any(|f| f.mock_response.is_some());
let needs_os = fixtures.iter().any(|f| {
if f.is_http_test() {
return true;
}
let call_args = &e2e_config.resolve_call(f.call.as_deref()).args;
call_args.iter().any(|a| a.arg_type == "mock_url")
});
let needs_json = fixtures.iter().any(|f| {
let call = e2e_config.resolve_call(f.call.as_deref());
let call_args = &call.args;
let has_handle = call_args.iter().any(|a| a.arg_type == "handle") && {
call_args.iter().filter(|a| a.arg_type == "handle").any(|a| {
let field = a.field.strip_prefix("input.").unwrap_or(&a.field);
let v = f.input.get(field).unwrap_or(&serde_json::Value::Null);
!(v.is_null() || v.is_object() && v.as_object().is_some_and(|o| o.is_empty()))
})
};
let go_override = call.overrides.get("go");
let opts_type = go_override.and_then(|o| o.options_type.as_deref()).or_else(|| {
e2e_config
.call
.overrides
.get("go")
.and_then(|o| o.options_type.as_deref())
});
let has_json_obj = call_args.iter().any(|a| {
if a.arg_type != "json_object" {
return false;
}
let field = a.field.strip_prefix("input.").unwrap_or(&a.field);
let v = f.input.get(field).unwrap_or(&serde_json::Value::Null);
if v.is_array() {
return true;
} opts_type.is_some() && v.is_object() && !v.as_object().is_some_and(|o| o.is_empty())
});
has_handle || has_json_obj
});
let needs_base64 = fixtures.iter().any(|f| {
let call_args = &e2e_config.resolve_call(f.call.as_deref()).args;
call_args.iter().any(|a| {
if a.arg_type != "bytes" {
return false;
}
let field = a.field.strip_prefix("input.").unwrap_or(&a.field);
matches!(f.input.get(field), Some(serde_json::Value::String(_)))
})
});
let needs_fmt = fixtures.iter().any(|f| {
f.visitor.as_ref().is_some_and(|v| {
v.callbacks.values().any(|action| {
if let CallbackAction::CustomTemplate { template } = action {
template.contains('{')
} else {
false
}
})
})
});
let needs_strings = fixtures.iter().any(|f| {
f.assertions.iter().any(|a| {
let type_needs_strings = if a.assertion_type == "equals" {
a.value.as_ref().is_some_and(|v| v.is_string())
} else {
matches!(
a.assertion_type.as_str(),
"contains" | "contains_all" | "contains_any" | "not_contains" | "starts_with" | "ends_with"
)
};
let field_valid = a
.field
.as_ref()
.map(|f| f.is_empty() || field_resolver.is_valid_for_result(f))
.unwrap_or(true);
type_needs_strings && field_valid
})
});
let needs_assert = fixtures.iter().any(|f| {
f.assertions.iter().any(|a| {
let field_valid = a
.field
.as_ref()
.map(|f| f.is_empty() || field_resolver.is_valid_for_result(f))
.unwrap_or(true);
let type_needs_assert = matches!(
a.assertion_type.as_str(),
"count_min"
| "count_max"
| "is_true"
| "is_false"
| "method_result"
| "min_length"
| "max_length"
| "matches_regex"
);
type_needs_assert && field_valid
})
});
let has_http_fixtures = fixtures.iter().any(|f| f.is_http_test());
let needs_http = has_http_fixtures;
let needs_io = fixtures
.iter()
.any(|f| f.http.as_ref().is_some_and(|h| h.expected_response.body.is_some()));
let needs_reflect = fixtures.iter().any(|f| {
if let Some(http) = &f.http {
if let Some(body) = &http.expected_response.body {
matches!(body, serde_json::Value::Object(_) | serde_json::Value::Array(_))
} else {
false
}
} else {
false
}
});
let _ = writeln!(out, "// E2e tests for category: {category}");
let _ = writeln!(out, "package e2e_test");
let _ = writeln!(out);
let _ = writeln!(out, "import (");
if needs_base64 {
let _ = writeln!(out, "\t\"encoding/base64\"");
}
if needs_json || needs_reflect {
let _ = writeln!(out, "\t\"encoding/json\"");
}
if needs_fmt {
let _ = writeln!(out, "\t\"fmt\"");
}
if needs_io {
let _ = writeln!(out, "\t\"io\"");
}
if needs_http {
let _ = writeln!(out, "\t\"net/http\"");
}
if needs_os {
let _ = writeln!(out, "\t\"os\"");
}
if needs_reflect {
let _ = writeln!(out, "\t\"reflect\"");
}
if needs_strings || needs_http {
let _ = writeln!(out, "\t\"strings\"");
}
let _ = writeln!(out, "\t\"testing\"");
if needs_assert {
let _ = writeln!(out);
let _ = writeln!(out, "\t\"github.com/stretchr/testify/assert\"");
}
if needs_pkg {
let _ = writeln!(out);
let _ = writeln!(out, "\t{import_alias} \"{go_module_path}\"");
}
let _ = writeln!(out, ")");
let _ = writeln!(out);
for fixture in fixtures.iter() {
if let Some(visitor_spec) = &fixture.visitor {
let struct_name = visitor_struct_name(&fixture.id);
emit_go_visitor_struct(&mut out, &struct_name, visitor_spec, import_alias);
let _ = writeln!(out);
}
}
for (i, fixture) in fixtures.iter().enumerate() {
render_test_function(&mut out, fixture, import_alias, field_resolver, e2e_config);
if i + 1 < fixtures.len() {
let _ = writeln!(out);
}
}
while out.ends_with("\n\n") {
out.pop();
}
if !out.ends_with('\n') {
out.push('\n');
}
out
}
fn render_test_function(
out: &mut String,
fixture: &Fixture,
import_alias: &str,
field_resolver: &FieldResolver,
e2e_config: &crate::config::E2eConfig,
) {
let fn_name = fixture.id.to_upper_camel_case();
let description = &fixture.description;
if let Some(http) = &fixture.http {
render_http_test_function(out, fixture, http);
return;
}
if fixture.mock_response.is_none() {
let _ = writeln!(out, "func Test_{fn_name}(t *testing.T) {{");
let _ = writeln!(out, "\t// {description}");
let _ = writeln!(
out,
"\tt.Skip(\"non-HTTP fixture: Go binding does not expose a callable for the configured `[e2e.call]` function\")"
);
let _ = writeln!(out, "}}");
return;
}
let call_config = e2e_config.resolve_call(fixture.call.as_deref());
let lang = "go";
let overrides = call_config.overrides.get(lang);
let function_name = to_go_name(
overrides
.and_then(|o| o.function.as_ref())
.map(String::as_str)
.unwrap_or(&call_config.function),
);
let result_var = &call_config.result_var;
let args = &call_config.args;
let returns_result = overrides
.and_then(|o| o.returns_result)
.unwrap_or(call_config.returns_result);
let returns_void = call_config.returns_void;
let result_is_simple = overrides.map(|o| o.result_is_simple).unwrap_or_else(|| {
call_config
.overrides
.get("rust")
.map(|o| o.result_is_simple)
.unwrap_or(false)
});
let result_is_array = overrides.map(|o| o.result_is_array).unwrap_or(false);
let call_options_type = overrides.and_then(|o| o.options_type.as_deref()).or_else(|| {
e2e_config
.call
.overrides
.get("go")
.and_then(|o| o.options_type.as_deref())
});
let expects_error = fixture.assertions.iter().any(|a| a.assertion_type == "error");
let (mut setup_lines, args_str) =
build_args_and_setup(&fixture.input, args, import_alias, call_options_type, &fixture.id);
let mut visitor_arg = String::new();
if fixture.visitor.is_some() {
let struct_name = visitor_struct_name(&fixture.id);
setup_lines.push(format!("visitor := &{struct_name}{{}}"));
visitor_arg = "visitor".to_string();
}
let final_args = if visitor_arg.is_empty() {
args_str
} else {
format!("{args_str}, {visitor_arg}")
};
let _ = writeln!(out, "func Test_{fn_name}(t *testing.T) {{");
let _ = writeln!(out, "\t// {description}");
for line in &setup_lines {
let _ = writeln!(out, "\t{line}");
}
if expects_error {
if returns_result && !returns_void {
let _ = writeln!(out, "\t_, err := {import_alias}.{function_name}({final_args})");
} else {
let _ = writeln!(out, "\terr := {import_alias}.{function_name}({final_args})");
}
let _ = writeln!(out, "\tif err == nil {{");
let _ = writeln!(out, "\t\tt.Errorf(\"expected an error, but call succeeded\")");
let _ = writeln!(out, "\t}}");
let _ = writeln!(out, "}}");
return;
}
let has_usable_assertion = fixture.assertions.iter().any(|a| {
if a.assertion_type == "not_error" || a.assertion_type == "error" {
return false;
}
if a.assertion_type == "method_result" {
return true;
}
match &a.field {
Some(f) if !f.is_empty() => field_resolver.is_valid_for_result(f),
_ => true,
}
});
if !returns_result && result_is_simple {
let result_binding = if has_usable_assertion {
result_var.to_string()
} else {
"_".to_string()
};
let assign_op = if result_binding == "_" { "=" } else { ":=" };
let _ = writeln!(
out,
"\t{result_binding} {assign_op} {import_alias}.{function_name}({final_args})"
);
if has_usable_assertion && result_binding != "_" {
let _ = writeln!(out, "\tif {result_var} == nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"expected non-nil result\")");
let _ = writeln!(out, "\t}}");
let _ = writeln!(out, "\tvalue := *{result_var}");
}
} else if !returns_result || returns_void {
let _ = writeln!(out, "\terr := {import_alias}.{function_name}({final_args})");
let _ = writeln!(out, "\tif err != nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"call failed: %v\", err)");
let _ = writeln!(out, "\t}}");
let _ = writeln!(out, "}}");
return;
} else {
let result_binding = if has_usable_assertion {
result_var.to_string()
} else {
"_".to_string()
};
let _ = writeln!(
out,
"\t{result_binding}, err := {import_alias}.{function_name}({final_args})"
);
let _ = writeln!(out, "\tif err != nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"call failed: %v\", err)");
let _ = writeln!(out, "\t}}");
if result_is_simple && has_usable_assertion && result_binding != "_" {
let _ = writeln!(out, "\tif {result_var} == nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"expected non-nil result\")");
let _ = writeln!(out, "\t}}");
let _ = writeln!(out, "\tvalue := *{result_var}");
}
}
let effective_result_var = if result_is_simple && has_usable_assertion {
"value".to_string()
} else {
result_var.to_string()
};
let mut optional_locals: std::collections::HashMap<String, String> = std::collections::HashMap::new();
for assertion in &fixture.assertions {
if let Some(f) = &assertion.field {
if !f.is_empty() {
let resolved = field_resolver.resolve(f);
if field_resolver.is_optional(resolved) && !optional_locals.contains_key(f.as_str()) {
let is_string_field = assertion.value.as_ref().is_some_and(|v| v.is_string());
let is_array_field = field_resolver.is_array(resolved);
if !is_string_field || is_array_field {
continue;
}
let field_expr = field_resolver.accessor(f, "go", &effective_result_var);
let local_var = go_param_name(&resolved.replace(['.', '[', ']'], "_"));
if field_resolver.has_map_access(f) {
let _ = writeln!(out, "\t{local_var} := {field_expr}");
} else {
let _ = writeln!(out, "\tvar {local_var} string");
let _ = writeln!(out, "\tif {field_expr} != nil {{");
let _ = writeln!(out, "\t\t{local_var} = *{field_expr}");
let _ = writeln!(out, "\t}}");
}
optional_locals.insert(f.clone(), local_var);
}
}
}
}
for assertion in &fixture.assertions {
if let Some(f) = &assertion.field {
if !f.is_empty() && !optional_locals.contains_key(f.as_str()) {
let parts: Vec<&str> = f.split('.').collect();
let mut guard_expr: Option<String> = None;
for i in 1..parts.len() {
let prefix = parts[..i].join(".");
let resolved_prefix = field_resolver.resolve(&prefix);
if field_resolver.is_optional(resolved_prefix) {
let accessor = field_resolver.accessor(&prefix, "go", &effective_result_var);
guard_expr = Some(accessor);
break;
}
}
if let Some(guard) = guard_expr {
if field_resolver.is_valid_for_result(f) {
let _ = writeln!(out, "\tif {guard} != nil {{");
let mut nil_buf = String::new();
render_assertion(
&mut nil_buf,
assertion,
&effective_result_var,
import_alias,
field_resolver,
&optional_locals,
result_is_simple,
result_is_array,
);
for line in nil_buf.lines() {
let _ = writeln!(out, "\t{line}");
}
let _ = writeln!(out, "\t}}");
} else {
render_assertion(
out,
assertion,
&effective_result_var,
import_alias,
field_resolver,
&optional_locals,
result_is_simple,
result_is_array,
);
}
continue;
}
}
}
render_assertion(
out,
assertion,
&effective_result_var,
import_alias,
field_resolver,
&optional_locals,
result_is_simple,
result_is_array,
);
}
let _ = writeln!(out, "}}");
}
fn render_http_test_function(out: &mut String, fixture: &Fixture, http: &HttpFixture) {
let fn_name = fixture.id.to_upper_camel_case();
let description = &fixture.description;
let request = &http.request;
let expected = &http.expected_response;
let method = request.method.to_uppercase();
let fixture_id = &fixture.id;
let expected_status = expected.status_code;
let _ = writeln!(out, "func Test_{fn_name}(t *testing.T) {{");
let _ = writeln!(out, "\t// {description}");
let _ = writeln!(out, "\tbaseURL := os.Getenv(\"MOCK_SERVER_URL\")");
let _ = writeln!(out, "\tif baseURL == \"\" {{");
let _ = writeln!(out, "\t\tbaseURL = \"http://localhost:8080\"");
let _ = writeln!(out, "\t}}");
let body_expr = if let Some(body) = &request.body {
let json = serde_json::to_string(body).unwrap_or_default();
let escaped = go_string_literal(&json);
format!("strings.NewReader({})", escaped)
} else {
"strings.NewReader(\"\")".to_string()
};
let _ = writeln!(out, "\tbody := {body_expr}");
let _ = writeln!(
out,
"\treq, err := http.NewRequest(\"{method}\", baseURL+\"/fixtures/{fixture_id}\", body)"
);
let _ = writeln!(out, "\tif err != nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"new request failed: %v\", err)");
let _ = writeln!(out, "\t}}");
let content_type = request.content_type.as_deref().unwrap_or("application/json");
if request.body.is_some() {
let _ = writeln!(out, "\treq.Header.Set(\"Content-Type\", \"{content_type}\")");
}
for (name, value) in &request.headers {
let escaped_name = go_string_literal(name);
let escaped_value = go_string_literal(value);
let _ = writeln!(out, "\treq.Header.Set({escaped_name}, {escaped_value})");
}
if !request.cookies.is_empty() {
for (name, value) in &request.cookies {
let escaped_name = go_string_literal(name);
let escaped_value = go_string_literal(value);
let _ = writeln!(
out,
"\treq.AddCookie(&http.Cookie{{Name: {escaped_name}, Value: {escaped_value}}})"
);
}
}
let _ = writeln!(out, "\tnoRedirectClient := &http.Client{{");
let _ = writeln!(
out,
"\t\tCheckRedirect: func(req *http.Request, via []*http.Request) error {{"
);
let _ = writeln!(out, "\t\t\treturn http.ErrUseLastResponse");
let _ = writeln!(out, "\t\t}},");
let _ = writeln!(out, "\t}}");
let _ = writeln!(out, "\tresp, err := noRedirectClient.Do(req)");
let _ = writeln!(out, "\tif err != nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"request failed: %v\", err)");
let _ = writeln!(out, "\t}}");
let _ = writeln!(out, "\tdefer resp.Body.Close()");
let body_used = expected.body.is_some();
if body_used {
let _ = writeln!(out, "\tbodyBytes, err := io.ReadAll(resp.Body)");
let _ = writeln!(out, "\tif err != nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"read body failed: %v\", err)");
let _ = writeln!(out, "\t}}");
}
let _ = writeln!(out, "\tif resp.StatusCode != {expected_status} {{");
let _ = writeln!(
out,
"\t\tt.Fatalf(\"status: got %d want {expected_status}\", resp.StatusCode)"
);
let _ = writeln!(out, "\t}}");
if let Some(expected_body) = &expected.body {
match expected_body {
serde_json::Value::Object(_) | serde_json::Value::Array(_) => {
let json_str = serde_json::to_string(expected_body).unwrap_or_default();
let escaped = go_string_literal(&json_str);
let _ = writeln!(out, "\tvar got any");
let _ = writeln!(out, "\tvar want any");
let _ = writeln!(out, "\tif err := json.Unmarshal(bodyBytes, &got); err != nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"json unmarshal got: %v\", err)");
let _ = writeln!(out, "\t}}");
let _ = writeln!(
out,
"\tif err := json.Unmarshal([]byte({escaped}), &want); err != nil {{"
);
let _ = writeln!(out, "\t\tt.Fatalf(\"json unmarshal want: %v\", err)");
let _ = writeln!(out, "\t}}");
let _ = writeln!(out, "\tif !reflect.DeepEqual(got, want) {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"body mismatch: got %v want %v\", got, want)");
let _ = writeln!(out, "\t}}");
}
serde_json::Value::String(s) => {
let escaped = go_string_literal(s);
let _ = writeln!(out, "\twant := {escaped}");
let _ = writeln!(out, "\tif strings.TrimSpace(string(bodyBytes)) != want {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"body: got %q want %q\", string(bodyBytes), want)");
let _ = writeln!(out, "\t}}");
}
other => {
let escaped = go_string_literal(&other.to_string());
let _ = writeln!(out, "\twant := {escaped}");
let _ = writeln!(out, "\tif strings.TrimSpace(string(bodyBytes)) != want {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"body: got %q want %q\", string(bodyBytes), want)");
let _ = writeln!(out, "\t}}");
}
}
}
for (name, value) in &expected.headers {
if value == "<<absent>>" || value == "<<present>>" || value == "<<uuid>>" {
continue;
}
let lower = name.to_ascii_lowercase();
if lower == "content-encoding" || lower == "connection" {
continue;
}
let escaped_name = go_string_literal(name);
let escaped_value = go_string_literal(value);
let _ = writeln!(
out,
"\tif !strings.Contains(resp.Header.Get({escaped_name}), {escaped_value}) {{"
);
let _ = writeln!(
out,
"\t\tt.Fatalf(\"header %s mismatch: got %q want to contain %q\", {escaped_name}, resp.Header.Get({escaped_name}), {escaped_value})"
);
let _ = writeln!(out, "\t}}");
}
let _ = writeln!(out, "}}");
}
fn build_args_and_setup(
input: &serde_json::Value,
args: &[crate::config::ArgMapping],
import_alias: &str,
options_type: Option<&str>,
fixture_id: &str,
) -> (Vec<String>, String) {
use heck::ToUpperCamelCase;
if args.is_empty() {
return (Vec::new(), String::new());
}
let mut setup_lines: Vec<String> = Vec::new();
let mut parts: Vec<String> = Vec::new();
for arg in args {
if arg.arg_type == "mock_url" {
setup_lines.push(format!(
"{} := os.Getenv(\"MOCK_SERVER_URL\") + \"/fixtures/{fixture_id}\"",
arg.name,
));
parts.push(arg.name.clone());
continue;
}
if arg.arg_type == "handle" {
let constructor_name = format!("Create{}", arg.name.to_upper_camel_case());
let field = arg.field.strip_prefix("input.").unwrap_or(&arg.field);
let config_value = input.get(field).unwrap_or(&serde_json::Value::Null);
if config_value.is_null()
|| config_value.is_object() && config_value.as_object().is_some_and(|o| o.is_empty())
{
setup_lines.push(format!(
"{name}, createErr := {import_alias}.{constructor_name}(nil)\n\tif createErr != nil {{\n\t\tt.Fatalf(\"create handle failed: %v\", createErr)\n\t}}",
name = arg.name,
));
} else {
let json_str = serde_json::to_string(config_value).unwrap_or_default();
let go_literal = go_string_literal(&json_str);
let name = &arg.name;
setup_lines.push(format!(
"var {name}Config {import_alias}.CrawlConfig\n\tif err := json.Unmarshal([]byte({go_literal}), &{name}Config); err != nil {{\n\t\tt.Fatalf(\"config parse failed: %v\", err)\n\t}}"
));
setup_lines.push(format!(
"{name}, createErr := {import_alias}.{constructor_name}(&{name}Config)\n\tif createErr != nil {{\n\t\tt.Fatalf(\"create handle failed: %v\", createErr)\n\t}}"
));
}
parts.push(arg.name.clone());
continue;
}
let field = arg.field.strip_prefix("input.").unwrap_or(&arg.field);
let val = input.get(field);
if arg.arg_type == "bytes" {
let var_name = format!("{}Bytes", arg.name);
match val {
None | Some(serde_json::Value::Null) => {
if arg.optional {
parts.push("nil".to_string());
} else {
parts.push("[]byte{}".to_string());
}
}
Some(serde_json::Value::String(s)) => {
let go_b64 = go_string_literal(s);
setup_lines.push(format!("{var_name}, _ := base64.StdEncoding.DecodeString({go_b64})"));
parts.push(var_name);
}
Some(other) => {
parts.push(format!("[]byte({})", json_to_go(other)));
}
}
continue;
}
match val {
None | Some(serde_json::Value::Null) if arg.optional => {
match arg.arg_type.as_str() {
"string" => {
parts.push("nil".to_string());
}
"json_object" => {
if let Some(opts_type) = options_type {
parts.push(format!("{import_alias}.{opts_type}{{}}"));
} else {
parts.push("nil".to_string());
}
}
_ => {
parts.push("nil".to_string());
}
}
}
None | Some(serde_json::Value::Null) => {
let default_val = match arg.arg_type.as_str() {
"string" => "\"\"".to_string(),
"int" | "integer" | "i64" => "0".to_string(),
"float" | "number" => "0.0".to_string(),
"bool" | "boolean" => "false".to_string(),
"json_object" => {
if let Some(opts_type) = options_type {
format!("{import_alias}.{opts_type}{{}}")
} else {
"nil".to_string()
}
}
_ => "nil".to_string(),
};
parts.push(default_val);
}
Some(v) => {
match arg.arg_type.as_str() {
"json_object" => {
let is_array = v.is_array();
let is_empty_obj = !is_array && v.is_object() && v.as_object().is_some_and(|o| o.is_empty());
if is_empty_obj {
if let Some(opts_type) = options_type {
parts.push(format!("{import_alias}.{opts_type}{{}}"));
} else {
parts.push("nil".to_string());
}
} else if is_array {
let json_str = serde_json::to_string(v).unwrap_or_default();
let go_literal = go_string_literal(&json_str);
let var_name = &arg.name;
setup_lines.push(format!(
"var {var_name} []string\n\tif err := json.Unmarshal([]byte({go_literal}), &{var_name}); err != nil {{\n\t\tt.Fatalf(\"config parse failed: %v\", err)\n\t}}"
));
parts.push(var_name.to_string());
} else if let Some(opts_type) = options_type {
let json_str = serde_json::to_string(v).unwrap_or_default();
let go_literal = go_string_literal(&json_str);
let var_name = &arg.name;
setup_lines.push(format!(
"var {var_name} {import_alias}.{opts_type}\n\tif err := json.Unmarshal([]byte({go_literal}), &{var_name}); err != nil {{\n\t\tt.Fatalf(\"config parse failed: %v\", err)\n\t}}"
));
parts.push(var_name.to_string());
} else {
parts.push(json_to_go(v));
}
}
"string" if arg.optional => {
let var_name = format!("{}Val", arg.name);
let go_val = json_to_go(v);
setup_lines.push(format!("{var_name} := {go_val}"));
parts.push(format!("&{var_name}"));
}
_ => {
parts.push(json_to_go(v));
}
}
}
}
}
(setup_lines, parts.join(", "))
}
#[allow(clippy::too_many_arguments)]
fn render_assertion(
out: &mut String,
assertion: &Assertion,
result_var: &str,
import_alias: &str,
field_resolver: &FieldResolver,
optional_locals: &std::collections::HashMap<String, String>,
result_is_simple: bool,
result_is_array: bool,
) {
if !result_is_simple {
if let Some(f) = &assertion.field {
let embed_deref = format!("(*{result_var})");
match f.as_str() {
"chunks_have_content" => {
let pred = format!(
"func() bool {{ chunks := {result_var}.Chunks; if chunks == nil {{ return false }}; for _, c := range *chunks {{ if c.Content == \"\" {{ return false }} }}; return true }}()"
);
match assertion.assertion_type.as_str() {
"is_true" => {
let _ = writeln!(out, "\tassert.True(t, {pred}, \"expected true\")");
}
"is_false" => {
let _ = writeln!(out, "\tassert.False(t, {pred}, \"expected false\")");
}
_ => {
let _ = writeln!(out, "\t// skipped: unsupported assertion type on synthetic field '{f}'");
}
}
return;
}
"chunks_have_embeddings" => {
let pred = format!(
"func() bool {{ chunks := {result_var}.Chunks; if chunks == nil {{ return false }}; for _, c := range *chunks {{ if c.Embedding == nil || len(*c.Embedding) == 0 {{ return false }} }}; return true }}()"
);
match assertion.assertion_type.as_str() {
"is_true" => {
let _ = writeln!(out, "\tassert.True(t, {pred}, \"expected true\")");
}
"is_false" => {
let _ = writeln!(out, "\tassert.False(t, {pred}, \"expected false\")");
}
_ => {
let _ = writeln!(out, "\t// skipped: unsupported assertion type on synthetic field '{f}'");
}
}
return;
}
"embeddings" => {
match assertion.assertion_type.as_str() {
"count_equals" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
let _ = writeln!(
out,
"\tassert.Equal(t, {n}, len({embed_deref}), \"expected exactly {n} elements\")"
);
}
}
}
"count_min" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
let _ = writeln!(
out,
"\tassert.GreaterOrEqual(t, len({embed_deref}), {n}, \"expected at least {n} elements\")"
);
}
}
}
"not_empty" => {
let _ = writeln!(
out,
"\tassert.NotEmpty(t, {embed_deref}, \"expected non-empty embeddings\")"
);
}
"is_empty" => {
let _ = writeln!(out, "\tassert.Empty(t, {embed_deref}, \"expected empty embeddings\")");
}
_ => {
let _ = writeln!(
out,
"\t// skipped: unsupported assertion type on synthetic field 'embeddings'"
);
}
}
return;
}
"embedding_dimensions" => {
let expr = format!(
"func() int {{ if len({embed_deref}) == 0 {{ return 0 }}; return len({embed_deref}[0]) }}()"
);
match assertion.assertion_type.as_str() {
"equals" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
let _ = writeln!(
out,
"\tif {expr} != {n} {{\n\t\tt.Errorf(\"equals mismatch: got %v\", {expr})\n\t}}"
);
}
}
}
"greater_than" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
let _ = writeln!(out, "\tassert.Greater(t, {expr}, {n}, \"expected > {n}\")");
}
}
}
_ => {
let _ = writeln!(
out,
"\t// skipped: unsupported assertion type on synthetic field 'embedding_dimensions'"
);
}
}
return;
}
"embeddings_valid" | "embeddings_finite" | "embeddings_non_zero" | "embeddings_normalized" => {
let pred = match f.as_str() {
"embeddings_valid" => {
format!(
"func() bool {{ for _, e := range {embed_deref} {{ if len(e) == 0 {{ return false }} }}; return true }}()"
)
}
"embeddings_finite" => {
format!(
"func() bool {{ for _, e := range {embed_deref} {{ for _, v := range e {{ if v != v || v == float32(1.0/0.0) || v == float32(-1.0/0.0) {{ return false }} }} }}; return true }}()"
)
}
"embeddings_non_zero" => {
format!(
"func() bool {{ for _, e := range {embed_deref} {{ hasNonZero := false; for _, v := range e {{ if v != 0 {{ hasNonZero = true; break }} }}; if !hasNonZero {{ return false }} }}; return true }}()"
)
}
"embeddings_normalized" => {
format!(
"func() bool {{ for _, e := range {embed_deref} {{ var n float64; for _, v := range e {{ n += float64(v) * float64(v) }}; if n < 0.999 || n > 1.001 {{ return false }} }}; return true }}()"
)
}
_ => unreachable!(),
};
match assertion.assertion_type.as_str() {
"is_true" => {
let _ = writeln!(out, "\tassert.True(t, {pred}, \"expected true\")");
}
"is_false" => {
let _ = writeln!(out, "\tassert.False(t, {pred}, \"expected false\")");
}
_ => {
let _ = writeln!(out, "\t// skipped: unsupported assertion type on synthetic field '{f}'");
}
}
return;
}
"keywords" | "keywords_count" => {
let _ = writeln!(out, "\t// skipped: field '{f}' not available on Go ExtractionResult");
return;
}
_ => {}
}
}
}
if !result_is_simple {
if let Some(f) = &assertion.field {
if !f.is_empty() && !field_resolver.is_valid_for_result(f) {
let _ = writeln!(out, "\t// skipped: field '{f}' not available on result type");
return;
}
}
}
let field_expr = if result_is_simple {
result_var.to_string()
} else {
match &assertion.field {
Some(f) if !f.is_empty() => {
if let Some(local_var) = optional_locals.get(f.as_str()) {
local_var.clone()
} else {
field_resolver.accessor(f, "go", result_var)
}
}
_ => result_var.to_string(),
}
};
let is_optional = assertion
.field
.as_ref()
.map(|f| {
let resolved = field_resolver.resolve(f);
let check_path = resolved
.strip_suffix(".length")
.or_else(|| resolved.strip_suffix(".count"))
.or_else(|| resolved.strip_suffix(".size"))
.unwrap_or(resolved);
field_resolver.is_optional(check_path) && !optional_locals.contains_key(f.as_str())
})
.unwrap_or(false);
let field_expr = if is_optional && field_expr.starts_with("len(") && field_expr.ends_with(')') {
let inner = &field_expr[4..field_expr.len() - 1];
format!("len(*{inner})")
} else {
field_expr
};
let nil_guard_expr = if is_optional && field_expr.starts_with("len(*") {
Some(field_expr[5..field_expr.len() - 1].to_string())
} else {
None
};
let deref_field_expr = if is_optional && !field_expr.starts_with("len(") {
format!("*{field_expr}")
} else {
field_expr.clone()
};
let array_guard: Option<String> = if let Some(idx) = field_expr.find("[0]") {
let array_expr = &field_expr[..idx];
Some(array_expr.to_string())
} else {
None
};
let mut assertion_buf = String::new();
let out_ref = &mut assertion_buf;
match assertion.assertion_type.as_str() {
"equals" => {
if let Some(expected) = &assertion.value {
let go_val = json_to_go(expected);
if expected.is_string() {
let trimmed_field = if is_optional && !field_expr.starts_with("len(") {
format!("strings.TrimSpace(*{field_expr})")
} else {
format!("strings.TrimSpace({field_expr})")
};
if is_optional && !field_expr.starts_with("len(") {
let _ = writeln!(out_ref, "\tif {field_expr} != nil && {trimmed_field} != {go_val} {{");
} else {
let _ = writeln!(out_ref, "\tif {trimmed_field} != {go_val} {{");
}
} else if is_optional && !field_expr.starts_with("len(") {
let _ = writeln!(out_ref, "\tif {field_expr} != nil && {deref_field_expr} != {go_val} {{");
} else {
let _ = writeln!(out_ref, "\tif {field_expr} != {go_val} {{");
}
let _ = writeln!(out_ref, "\t\tt.Errorf(\"equals mismatch: got %v\", {field_expr})");
let _ = writeln!(out_ref, "\t}}");
}
}
"contains" => {
if let Some(expected) = &assertion.value {
let go_val = json_to_go(expected);
let resolved_field = assertion.field.as_deref().unwrap_or("");
let resolved_name = field_resolver.resolve(resolved_field);
let field_is_array = result_is_array || field_resolver.is_array(resolved_name);
let is_opt =
is_optional && !optional_locals.contains_key(assertion.field.as_ref().unwrap_or(&String::new()));
let field_for_contains = if is_opt && field_is_array {
format!("strings.Join(*{field_expr}, \" \")")
} else if is_opt {
format!("string(*{field_expr})")
} else if field_is_array {
format!("strings.Join({field_expr}, \" \")")
} else {
format!("string({field_expr})")
};
if is_opt {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(out_ref, "\tif !strings.Contains({field_for_contains}, {go_val}) {{");
let _ = writeln!(
out_ref,
"\t\tt.Errorf(\"expected to contain %s, got %v\", {go_val}, {field_expr})"
);
let _ = writeln!(out_ref, "\t}}");
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(out_ref, "\tif !strings.Contains({field_for_contains}, {go_val}) {{");
let _ = writeln!(
out_ref,
"\t\tt.Errorf(\"expected to contain %s, got %v\", {go_val}, {field_expr})"
);
let _ = writeln!(out_ref, "\t}}");
}
}
}
"contains_all" => {
if let Some(values) = &assertion.values {
let resolved_field = assertion.field.as_deref().unwrap_or("");
let resolved_name = field_resolver.resolve(resolved_field);
let field_is_array = result_is_array || field_resolver.is_array(resolved_name);
let is_opt =
is_optional && !optional_locals.contains_key(assertion.field.as_ref().unwrap_or(&String::new()));
for val in values {
let go_val = json_to_go(val);
let field_for_contains = if is_opt && field_is_array {
format!("strings.Join(*{field_expr}, \" \")")
} else if is_opt {
format!("string(*{field_expr})")
} else if field_is_array {
format!("strings.Join({field_expr}, \" \")")
} else {
format!("string({field_expr})")
};
if is_opt {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(out_ref, "\tif !strings.Contains({field_for_contains}, {go_val}) {{");
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected to contain %s\", {go_val})");
let _ = writeln!(out_ref, "\t}}");
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(out_ref, "\tif !strings.Contains({field_for_contains}, {go_val}) {{");
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected to contain %s\", {go_val})");
let _ = writeln!(out_ref, "\t}}");
}
}
}
}
"not_contains" => {
if let Some(expected) = &assertion.value {
let go_val = json_to_go(expected);
let resolved_field = assertion.field.as_deref().unwrap_or("");
let resolved_name = field_resolver.resolve(resolved_field);
let field_is_array = result_is_array || field_resolver.is_array(resolved_name);
let is_opt =
is_optional && !optional_locals.contains_key(assertion.field.as_ref().unwrap_or(&String::new()));
let field_for_contains = if is_opt && field_is_array {
format!("strings.Join(*{field_expr}, \" \")")
} else if is_opt {
format!("string(*{field_expr})")
} else if field_is_array {
format!("strings.Join({field_expr}, \" \")")
} else {
format!("string({field_expr})")
};
let _ = writeln!(out_ref, "\tif strings.Contains({field_for_contains}, {go_val}) {{");
let _ = writeln!(
out_ref,
"\t\tt.Errorf(\"expected NOT to contain %s, got %v\", {go_val}, {field_expr})"
);
let _ = writeln!(out_ref, "\t}}");
}
}
"not_empty" => {
let field_is_array = {
let rf = assertion.field.as_deref().unwrap_or("");
let rn = field_resolver.resolve(rf);
field_resolver.is_array(rn)
};
if is_optional && !field_is_array {
let _ = writeln!(out_ref, "\tif {field_expr} == nil {{");
} else if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} == nil || len(*{field_expr}) == 0 {{");
} else {
let _ = writeln!(out_ref, "\tif len({field_expr}) == 0 {{");
}
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected non-empty value\")");
let _ = writeln!(out_ref, "\t}}");
}
"is_empty" => {
let field_is_array = {
let rf = assertion.field.as_deref().unwrap_or("");
let rn = field_resolver.resolve(rf);
field_resolver.is_array(rn)
};
if is_optional && !field_is_array {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
} else if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil && len(*{field_expr}) != 0 {{");
} else {
let _ = writeln!(out_ref, "\tif len({field_expr}) != 0 {{");
}
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected empty value, got %v\", {field_expr})");
let _ = writeln!(out_ref, "\t}}");
}
"contains_any" => {
if let Some(values) = &assertion.values {
let resolved_field = assertion.field.as_deref().unwrap_or("");
let resolved_name = field_resolver.resolve(resolved_field);
let field_is_array = field_resolver.is_array(resolved_name);
let is_opt =
is_optional && !optional_locals.contains_key(assertion.field.as_ref().unwrap_or(&String::new()));
let field_for_contains = if is_opt && field_is_array {
format!("strings.Join(*{field_expr}, \" \")")
} else if is_opt {
format!("*{field_expr}")
} else if field_is_array {
format!("strings.Join({field_expr}, \" \")")
} else {
field_expr.clone()
};
let _ = writeln!(out_ref, "\t{{");
let _ = writeln!(out_ref, "\t\tfound := false");
for val in values {
let go_val = json_to_go(val);
let _ = writeln!(
out_ref,
"\t\tif strings.Contains({field_for_contains}, {go_val}) {{ found = true }}"
);
}
let _ = writeln!(out_ref, "\t\tif !found {{");
let _ = writeln!(
out_ref,
"\t\t\tt.Errorf(\"expected to contain at least one of the specified values\")"
);
let _ = writeln!(out_ref, "\t\t}}");
let _ = writeln!(out_ref, "\t}}");
}
}
"greater_than" => {
if let Some(val) = &assertion.value {
let go_val = json_to_go(val);
if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
if let Some(n) = val.as_u64() {
let next = n + 1;
let _ = writeln!(out_ref, "\t\tif {deref_field_expr} < {next} {{");
} else {
let _ = writeln!(out_ref, "\t\tif {deref_field_expr} <= {go_val} {{");
}
let _ = writeln!(
out_ref,
"\t\t\tt.Errorf(\"expected > {go_val}, got %v\", {deref_field_expr})"
);
let _ = writeln!(out_ref, "\t\t}}");
let _ = writeln!(out_ref, "\t}}");
} else if let Some(n) = val.as_u64() {
let next = n + 1;
let _ = writeln!(out_ref, "\tif {field_expr} < {next} {{");
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected > {go_val}, got %v\", {field_expr})");
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(out_ref, "\tif {field_expr} <= {go_val} {{");
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected > {go_val}, got %v\", {field_expr})");
let _ = writeln!(out_ref, "\t}}");
}
}
}
"less_than" => {
if let Some(val) = &assertion.value {
let go_val = json_to_go(val);
let _ = writeln!(out_ref, "\tif {field_expr} >= {go_val} {{");
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected < {go_val}, got %v\", {field_expr})");
let _ = writeln!(out_ref, "\t}}");
}
}
"greater_than_or_equal" => {
if let Some(val) = &assertion.value {
let go_val = json_to_go(val);
if let Some(ref guard) = nil_guard_expr {
let _ = writeln!(out_ref, "\tif {guard} != nil {{");
let _ = writeln!(out_ref, "\t\tif {field_expr} < {go_val} {{");
let _ = writeln!(
out_ref,
"\t\t\tt.Errorf(\"expected >= {go_val}, got %v\", {field_expr})"
);
let _ = writeln!(out_ref, "\t\t}}");
let _ = writeln!(out_ref, "\t}}");
} else if is_optional && !field_expr.starts_with("len(") {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(out_ref, "\t\tif {deref_field_expr} < {go_val} {{");
let _ = writeln!(
out_ref,
"\t\t\tt.Errorf(\"expected >= {go_val}, got %v\", {deref_field_expr})"
);
let _ = writeln!(out_ref, "\t\t}}");
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(out_ref, "\tif {field_expr} < {go_val} {{");
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected >= {go_val}, got %v\", {field_expr})");
let _ = writeln!(out_ref, "\t}}");
}
}
}
"less_than_or_equal" => {
if let Some(val) = &assertion.value {
let go_val = json_to_go(val);
let _ = writeln!(out_ref, "\tif {field_expr} > {go_val} {{");
let _ = writeln!(out_ref, "\t\tt.Errorf(\"expected <= {go_val}, got %v\", {field_expr})");
let _ = writeln!(out_ref, "\t}}");
}
}
"starts_with" => {
if let Some(expected) = &assertion.value {
let go_val = json_to_go(expected);
let field_for_prefix = if is_optional
&& !optional_locals.contains_key(assertion.field.as_ref().unwrap_or(&String::new()))
{
format!("string(*{field_expr})")
} else {
format!("string({field_expr})")
};
let _ = writeln!(out_ref, "\tif !strings.HasPrefix({field_for_prefix}, {go_val}) {{");
let _ = writeln!(
out_ref,
"\t\tt.Errorf(\"expected to start with %s, got %v\", {go_val}, {field_expr})"
);
let _ = writeln!(out_ref, "\t}}");
}
}
"count_min" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(
out_ref,
"\t\tassert.GreaterOrEqual(t, len(*{field_expr}), {n}, \"expected at least {n} elements\")"
);
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(
out_ref,
"\tassert.GreaterOrEqual(t, len({field_expr}), {n}, \"expected at least {n} elements\")"
);
}
}
}
}
"count_equals" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(
out_ref,
"\t\tassert.Equal(t, len(*{field_expr}), {n}, \"expected exactly {n} elements\")"
);
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(
out_ref,
"\tassert.Equal(t, len({field_expr}), {n}, \"expected exactly {n} elements\")"
);
}
}
}
}
"is_true" => {
if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(out_ref, "\t\tassert.True(t, *{field_expr}, \"expected true\")");
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(out_ref, "\tassert.True(t, {field_expr}, \"expected true\")");
}
}
"is_false" => {
if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(out_ref, "\t\tassert.False(t, *{field_expr}, \"expected false\")");
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(out_ref, "\tassert.False(t, {field_expr}, \"expected false\")");
}
}
"method_result" => {
if let Some(method_name) = &assertion.method {
let info = build_go_method_call(result_var, method_name, assertion.args.as_ref(), import_alias);
let check = assertion.check.as_deref().unwrap_or("is_true");
let deref_expr = if info.is_pointer {
format!("*{}", info.call_expr)
} else {
info.call_expr.clone()
};
match check {
"equals" => {
if let Some(val) = &assertion.value {
if val.is_boolean() {
if val.as_bool() == Some(true) {
let _ = writeln!(out_ref, "\tassert.True(t, {deref_expr}, \"expected true\")");
} else {
let _ = writeln!(out_ref, "\tassert.False(t, {deref_expr}, \"expected false\")");
}
} else {
let go_val = if let Some(cast) = info.value_cast {
if val.is_number() {
format!("{cast}({})", json_to_go(val))
} else {
json_to_go(val)
}
} else {
json_to_go(val)
};
let _ = writeln!(
out_ref,
"\tassert.Equal(t, {go_val}, {deref_expr}, \"method_result equals assertion failed\")"
);
}
}
}
"is_true" => {
let _ = writeln!(out_ref, "\tassert.True(t, {deref_expr}, \"expected true\")");
}
"is_false" => {
let _ = writeln!(out_ref, "\tassert.False(t, {deref_expr}, \"expected false\")");
}
"greater_than_or_equal" => {
if let Some(val) = &assertion.value {
let n = val.as_u64().unwrap_or(0);
let cast = info.value_cast.unwrap_or("uint");
let _ = writeln!(
out_ref,
"\tassert.GreaterOrEqual(t, {deref_expr}, {cast}({n}), \"expected >= {n}\")"
);
}
}
"count_min" => {
if let Some(val) = &assertion.value {
let n = val.as_u64().unwrap_or(0);
let _ = writeln!(
out_ref,
"\tassert.GreaterOrEqual(t, len({deref_expr}), {n}, \"expected at least {n} elements\")"
);
}
}
"contains" => {
if let Some(val) = &assertion.value {
let go_val = json_to_go(val);
let _ = writeln!(
out_ref,
"\tassert.Contains(t, {deref_expr}, {go_val}, \"expected result to contain value\")"
);
}
}
"is_error" => {
let _ = writeln!(out_ref, "\t{{");
let _ = writeln!(out_ref, "\t\t_, methodErr := {}", info.call_expr);
let _ = writeln!(out_ref, "\t\tassert.Error(t, methodErr)");
let _ = writeln!(out_ref, "\t}}");
}
other_check => {
panic!("Go e2e generator: unsupported method_result check type: {other_check}");
}
}
} else {
panic!("Go e2e generator: method_result assertion missing 'method' field");
}
}
"min_length" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(
out_ref,
"\t\tassert.GreaterOrEqual(t, len(*{field_expr}), {n}, \"expected length >= {n}\")"
);
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(
out_ref,
"\tassert.GreaterOrEqual(t, len({field_expr}), {n}, \"expected length >= {n}\")"
);
}
}
}
}
"max_length" => {
if let Some(val) = &assertion.value {
if let Some(n) = val.as_u64() {
if is_optional {
let _ = writeln!(out_ref, "\tif {field_expr} != nil {{");
let _ = writeln!(
out_ref,
"\t\tassert.LessOrEqual(t, len(*{field_expr}), {n}, \"expected length <= {n}\")"
);
let _ = writeln!(out_ref, "\t}}");
} else {
let _ = writeln!(
out_ref,
"\tassert.LessOrEqual(t, len({field_expr}), {n}, \"expected length <= {n}\")"
);
}
}
}
}
"ends_with" => {
if let Some(expected) = &assertion.value {
let go_val = json_to_go(expected);
let field_for_suffix = if is_optional
&& !optional_locals.contains_key(assertion.field.as_ref().unwrap_or(&String::new()))
{
format!("string(*{field_expr})")
} else {
format!("string({field_expr})")
};
let _ = writeln!(out_ref, "\tif !strings.HasSuffix({field_for_suffix}, {go_val}) {{");
let _ = writeln!(
out_ref,
"\t\tt.Errorf(\"expected to end with %s, got %v\", {go_val}, {field_expr})"
);
let _ = writeln!(out_ref, "\t}}");
}
}
"matches_regex" => {
if let Some(expected) = &assertion.value {
let go_val = json_to_go(expected);
let field_for_regex = if is_optional
&& !optional_locals.contains_key(assertion.field.as_ref().unwrap_or(&String::new()))
{
format!("*{field_expr}")
} else {
field_expr.clone()
};
let _ = writeln!(
out_ref,
"\tassert.Regexp(t, {go_val}, {field_for_regex}, \"expected value to match regex\")"
);
}
}
"not_error" => {
}
"error" => {
}
other => {
panic!("Go e2e generator: unsupported assertion type: {other}");
}
}
if let Some(ref arr) = array_guard {
if !assertion_buf.is_empty() {
let _ = writeln!(out, "\tif len({arr}) > 0 {{");
for line in assertion_buf.lines() {
let _ = writeln!(out, "\t{line}");
}
let _ = writeln!(out, "\t}}");
}
} else {
out.push_str(&assertion_buf);
}
}
struct GoMethodCallInfo {
call_expr: String,
is_pointer: bool,
value_cast: Option<&'static str>,
}
fn build_go_method_call(
result_var: &str,
method_name: &str,
args: Option<&serde_json::Value>,
import_alias: &str,
) -> GoMethodCallInfo {
match method_name {
"root_node_type" => GoMethodCallInfo {
call_expr: format!("{import_alias}.RootNodeInfo({result_var}).Kind"),
is_pointer: false,
value_cast: None,
},
"named_children_count" => GoMethodCallInfo {
call_expr: format!("{import_alias}.RootNodeInfo({result_var}).NamedChildCount"),
is_pointer: false,
value_cast: Some("uint"),
},
"has_error_nodes" => GoMethodCallInfo {
call_expr: format!("{import_alias}.TreeHasErrorNodes({result_var})"),
is_pointer: true,
value_cast: None,
},
"error_count" | "tree_error_count" => GoMethodCallInfo {
call_expr: format!("{import_alias}.TreeErrorCount({result_var})"),
is_pointer: true,
value_cast: Some("uint"),
},
"tree_to_sexp" => GoMethodCallInfo {
call_expr: format!("{import_alias}.TreeToSexp({result_var})"),
is_pointer: true,
value_cast: None,
},
"contains_node_type" => {
let node_type = args
.and_then(|a| a.get("node_type"))
.and_then(|v| v.as_str())
.unwrap_or("");
GoMethodCallInfo {
call_expr: format!("{import_alias}.TreeContainsNodeType({result_var}, \"{node_type}\")"),
is_pointer: true,
value_cast: None,
}
}
"find_nodes_by_type" => {
let node_type = args
.and_then(|a| a.get("node_type"))
.and_then(|v| v.as_str())
.unwrap_or("");
GoMethodCallInfo {
call_expr: format!("{import_alias}.FindNodesByType({result_var}, \"{node_type}\")"),
is_pointer: true,
value_cast: None,
}
}
"run_query" => {
let query_source = args
.and_then(|a| a.get("query_source"))
.and_then(|v| v.as_str())
.unwrap_or("");
let language = args
.and_then(|a| a.get("language"))
.and_then(|v| v.as_str())
.unwrap_or("");
let query_lit = go_string_literal(query_source);
let lang_lit = go_string_literal(language);
GoMethodCallInfo {
call_expr: format!("{import_alias}.RunQuery({result_var}, {lang_lit}, {query_lit}, []byte(source))"),
is_pointer: false,
value_cast: None,
}
}
other => {
let method_pascal = other.to_upper_camel_case();
GoMethodCallInfo {
call_expr: format!("{result_var}.{method_pascal}()"),
is_pointer: false,
value_cast: None,
}
}
}
}
fn json_to_go(value: &serde_json::Value) -> String {
match value {
serde_json::Value::String(s) => go_string_literal(s),
serde_json::Value::Bool(b) => b.to_string(),
serde_json::Value::Number(n) => n.to_string(),
serde_json::Value::Null => "nil".to_string(),
other => go_string_literal(&other.to_string()),
}
}
fn visitor_struct_name(fixture_id: &str) -> String {
use heck::ToUpperCamelCase;
format!("testVisitor{}", fixture_id.to_upper_camel_case())
}
fn emit_go_visitor_struct(
out: &mut String,
struct_name: &str,
visitor_spec: &crate::fixture::VisitorSpec,
import_alias: &str,
) {
let _ = writeln!(out, "type {struct_name} struct{{}}");
for (method_name, action) in &visitor_spec.callbacks {
emit_go_visitor_method(out, struct_name, method_name, action, import_alias);
}
}
fn emit_go_visitor_method(
out: &mut String,
struct_name: &str,
method_name: &str,
action: &CallbackAction,
import_alias: &str,
) {
let camel_method = method_to_camel(method_name);
let params = match method_name {
"visit_link" => format!("_ {import_alias}.NodeContext, href, text, title string"),
"visit_image" => format!("_ {import_alias}.NodeContext, src, alt, title string"),
"visit_heading" => format!("_ {import_alias}.NodeContext, level int, text, id string"),
"visit_code_block" => format!("_ {import_alias}.NodeContext, lang, code string"),
"visit_code_inline"
| "visit_strong"
| "visit_emphasis"
| "visit_strikethrough"
| "visit_underline"
| "visit_subscript"
| "visit_superscript"
| "visit_mark"
| "visit_button"
| "visit_summary"
| "visit_figcaption"
| "visit_definition_term"
| "visit_definition_description" => format!("_ {import_alias}.NodeContext, text string"),
"visit_text" => format!("_ {import_alias}.NodeContext, text string"),
"visit_list_item" => {
format!("_ {import_alias}.NodeContext, ordered bool, marker, text string")
}
"visit_blockquote" => format!("_ {import_alias}.NodeContext, content string, depth int"),
"visit_table_row" => format!("_ {import_alias}.NodeContext, cells []string, isHeader bool"),
"visit_custom_element" => format!("_ {import_alias}.NodeContext, tagName, html string"),
"visit_form" => format!("_ {import_alias}.NodeContext, actionUrl, method string"),
"visit_input" => format!("_ {import_alias}.NodeContext, inputType, name, value string"),
"visit_audio" | "visit_video" | "visit_iframe" => {
format!("_ {import_alias}.NodeContext, src string")
}
"visit_details" => format!("_ {import_alias}.NodeContext, isOpen bool"),
"visit_element_end" | "visit_table_end" | "visit_definition_list_end" | "visit_figure_end" => {
format!("_ {import_alias}.NodeContext, output string")
}
"visit_list_start" => format!("_ {import_alias}.NodeContext, ordered bool"),
"visit_list_end" => format!("_ {import_alias}.NodeContext, ordered bool, output string"),
_ => format!("_ {import_alias}.NodeContext"),
};
let _ = writeln!(
out,
"func (v *{struct_name}) {camel_method}({params}) {import_alias}.VisitResult {{"
);
match action {
CallbackAction::Skip => {
let _ = writeln!(out, "\treturn {import_alias}.VisitResultSkip");
}
CallbackAction::Continue => {
let _ = writeln!(out, "\treturn {import_alias}.VisitResultContinue");
}
CallbackAction::PreserveHtml => {
let _ = writeln!(out, "\treturn {import_alias}.VisitResultPreserveHtml");
}
CallbackAction::Custom { output } => {
let escaped = go_string_literal(output);
let _ = writeln!(out, "\treturn {import_alias}.VisitResultCustom({escaped})");
}
CallbackAction::CustomTemplate { template } => {
let (fmt_str, fmt_args) = template_to_sprintf(template);
let escaped_fmt = go_string_literal(&fmt_str);
if fmt_args.is_empty() {
let _ = writeln!(out, "\treturn {import_alias}.VisitResultCustom({escaped_fmt})");
} else {
let args_str = fmt_args.join(", ");
let _ = writeln!(
out,
"\treturn {import_alias}.VisitResultCustom(fmt.Sprintf({escaped_fmt}, {args_str}))"
);
}
}
}
let _ = writeln!(out, "}}");
}
fn template_to_sprintf(template: &str) -> (String, Vec<String>) {
let mut fmt_str = String::new();
let mut args: Vec<String> = Vec::new();
let mut chars = template.chars().peekable();
while let Some(c) = chars.next() {
if c == '{' {
let mut name = String::new();
for inner in chars.by_ref() {
if inner == '}' {
break;
}
name.push(inner);
}
fmt_str.push_str("%s");
args.push(name);
} else {
fmt_str.push(c);
}
}
(fmt_str, args)
}
fn method_to_camel(snake: &str) -> String {
use heck::ToUpperCamelCase;
snake.to_upper_camel_case()
}
#[cfg(test)]
mod tests {
use super::*;
use crate::config::{CallConfig, E2eConfig};
use crate::field_access::FieldResolver;
use crate::fixture::{Assertion, Fixture};
fn make_fixture(id: &str) -> Fixture {
Fixture {
id: id.to_string(),
category: None,
description: "test fixture".to_string(),
tags: vec![],
skip: None,
call: None,
input: serde_json::Value::Null,
mock_response: Some(crate::fixture::MockResponse {
status: 200,
body: Some(serde_json::Value::Null),
stream_chunks: None,
headers: std::collections::HashMap::new(),
}),
source: String::new(),
http: None,
assertions: vec![Assertion {
assertion_type: "not_error".to_string(),
field: None,
value: None,
values: None,
method: None,
args: None,
check: None,
}],
visitor: None,
}
}
#[test]
fn test_go_method_name_uses_go_casing() {
let e2e_config = E2eConfig {
call: CallConfig {
function: "clean_extracted_text".to_string(),
module: "github.com/example/mylib".to_string(),
result_var: "result".to_string(),
r#async: false,
path: None,
method: None,
args: vec![],
overrides: std::collections::HashMap::new(),
returns_result: true,
returns_void: false,
skip_languages: vec![],
},
..E2eConfig::default()
};
let fixture = make_fixture("basic_text");
let resolver = FieldResolver::new(
&std::collections::HashMap::new(),
&std::collections::HashSet::new(),
&std::collections::HashSet::new(),
&std::collections::HashSet::new(),
);
let mut out = String::new();
render_test_function(&mut out, &fixture, "kreuzberg", &resolver, &e2e_config);
assert!(
out.contains("kreuzberg.CleanExtractedText("),
"expected Go-cased method name 'CleanExtractedText', got:\n{out}"
);
assert!(
!out.contains("kreuzberg.clean_extracted_text("),
"must not emit raw snake_case method name, got:\n{out}"
);
}
}