use crate::config::E2eConfig;
use crate::escape::{go_string_literal, sanitize_filename};
use crate::field_access::FieldResolver;
use crate::fixture::{Assertion, Fixture, FixtureGroup};
use alef_core::backend::GeneratedFile;
use alef_core::config::AlefConfig;
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.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 function_name = overrides
.and_then(|o| o.function.as_ref())
.cloned()
.unwrap_or_else(|| call.function.clone());
let import_alias = overrides
.and_then(|o| o.alias.as_ref())
.cloned()
.unwrap_or_else(|| "pkg".to_string());
let result_var = &call.result_var;
let go_pkg = e2e_config.packages.get("go");
let go_module_path = go_pkg
.and_then(|p| p.module.as_ref())
.cloned()
.unwrap_or_else(|| module_path.clone());
let replace_path = go_pkg.and_then(|p| p.path.as_ref()).cloned();
let go_version = go_pkg
.and_then(|p| p.version.as_ref())
.cloned()
.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,
);
files.push(GeneratedFile {
path: output_base.join("go.mod"),
content: render_go_mod(&go_module_path, replace_path.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,
&function_name,
result_var,
&e2e_config.call.args,
&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.23");
let _ = writeln!(out);
let _ = writeln!(out, "require {go_module_path} {version}");
if let Some(path) = replace_path {
let _ = writeln!(out);
let _ = writeln!(out, "replace {go_module_path} => {path}");
}
out
}
#[allow(clippy::too_many_arguments)]
fn render_test_file(
category: &str,
fixtures: &[&Fixture],
go_module_path: &str,
import_alias: &str,
function_name: &str,
result_var: &str,
args: &[crate::config::ArgMapping],
field_resolver: &FieldResolver,
e2e_config: &crate::config::E2eConfig,
) -> String {
let mut out = String::new();
let _ = writeln!(out, "// Code generated by alef. DO NOT EDIT.");
let _ = writeln!(out);
let needs_os = args.iter().any(|a| a.arg_type == "mock_url");
let needs_json = args.iter().any(|a| a.arg_type == "handle")
&& fixtures.iter().any(|f| {
args.iter().filter(|a| a.arg_type == "handle").any(|a| {
let v = f.input.get(&a.field).unwrap_or(&serde_json::Value::Null);
!(v.is_null() || v.is_object() && v.as_object().is_some_and(|o| o.is_empty()))
})
});
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" | "not_contains" | "starts_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);
matches!(a.assertion_type.as_str(), "count_min" | "count_max") && field_valid
})
});
let _ = writeln!(out, "// E2e tests for category: {category}");
let _ = writeln!(out, "package e2e_test");
let _ = writeln!(out);
let _ = writeln!(out, "import (");
if needs_json {
let _ = writeln!(out, "\t\"encoding/json\"");
}
if needs_os {
let _ = writeln!(out, "\t\"os\"");
}
if needs_strings {
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\"");
}
let _ = writeln!(out);
let _ = writeln!(out, "\t{import_alias} \"{go_module_path}\"");
let _ = writeln!(out, ")");
let _ = writeln!(out);
for (i, fixture) in fixtures.iter().enumerate() {
render_test_function(
&mut out,
fixture,
import_alias,
function_name,
result_var,
args,
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
}
#[allow(clippy::too_many_arguments)]
fn render_test_function(
out: &mut String,
fixture: &Fixture,
import_alias: &str,
function_name: &str,
result_var: &str,
args: &[crate::config::ArgMapping],
field_resolver: &FieldResolver,
e2e_config: &crate::config::E2eConfig,
) {
let fn_name = fixture.id.to_upper_camel_case();
let description = &fixture.description;
let expects_error = fixture.assertions.iter().any(|a| a.assertion_type == "error");
let (setup_lines, args_str) = build_args_and_setup(&fixture.input, args, import_alias, e2e_config, &fixture.id);
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 {
let _ = writeln!(out, "\t_, err := {import_alias}.{function_name}({args_str})");
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;
}
match &a.field {
Some(f) if !f.is_empty() => field_resolver.is_valid_for_result(f),
_ => true,
}
});
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}({args_str})"
);
let _ = writeln!(out, "\tif err != nil {{");
let _ = writeln!(out, "\t\tt.Fatalf(\"call failed: %v\", err)");
let _ = writeln!(out, "\t}}");
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());
if !is_string_field {
continue;
}
let field_expr = field_resolver.accessor(f, "go", result_var);
let local_var = go_local_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", 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 {{");
render_assertion(out, assertion, result_var, field_resolver, &optional_locals);
let _ = writeln!(out, "\t}}");
} else {
render_assertion(out, assertion, result_var, field_resolver, &optional_locals);
}
continue;
}
}
}
render_assertion(out, assertion, result_var, field_resolver, &optional_locals);
}
let _ = writeln!(out, "}}");
}
fn build_args_and_setup(
input: &serde_json::Value,
args: &[crate::config::ArgMapping],
import_alias: &str,
e2e_config: &crate::config::E2eConfig,
fixture_id: &str,
) -> (Vec<String>, String) {
use heck::ToUpperCamelCase;
if args.is_empty() {
return (Vec::new(), json_to_go(input));
}
let overrides = e2e_config.call.overrides.get("go");
let options_type = overrides.and_then(|o| o.options_type.as_deref());
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 config_value = input.get(&arg.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}()\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 val = input.get(&arg.field);
match val {
None | Some(serde_json::Value::Null) if arg.optional => {
continue;
}
None | Some(serde_json::Value::Null) => {
let default_val = match arg.arg_type.as_str() {
"string" => "\"\"".to_string(),
"int" | "integer" => "0".to_string(),
"float" | "number" => "0.0".to_string(),
"bool" | "boolean" => "false".to_string(),
_ => "nil".to_string(),
};
parts.push(default_val);
}
Some(v) => {
if let (Some(opts_type), "json_object") = (options_type, arg.arg_type.as_str()) {
if let Some(obj) = v.as_object() {
let with_calls: Vec<String> = obj
.iter()
.map(|(k, vv)| {
let func_name = format!("With{}{}", opts_type, k.to_upper_camel_case());
let go_val = json_to_go(vv);
format!("htmd.{func_name}({go_val})")
})
.collect();
let new_fn = format!("New{opts_type}");
parts.push(format!("htmd.{new_fn}({})", with_calls.join(", ")));
continue;
}
}
parts.push(json_to_go(v));
}
}
}
(setup_lines, parts.join(", "))
}
fn render_assertion(
out: &mut String,
assertion: &Assertion,
result_var: &str,
field_resolver: &FieldResolver,
optional_locals: &std::collections::HashMap<String, String>,
) {
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 = 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 field_for_contains = 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.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 {
for val in values {
let go_val = json_to_go(val);
let field_for_contains = 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.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 field_for_contains = 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.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" => {
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" => {
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 field_for_contains = 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, "\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 let Some(n) = val.as_u64() {
let next = n + 1;
let _ = writeln!(out_ref, "\tif {field_expr} < {next} {{");
} 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 {
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\")"
);
}
}
}
}
"not_error" => {
}
"error" => {
}
other => {
let _ = writeln!(out_ref, "\t// TODO: 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);
}
}
const GO_INITIALISMS: &[&str] = &[
"ACL", "API", "ASCII", "CPU", "CSS", "DNS", "EOF", "GUID", "HTML", "HTTP", "HTTPS", "ID", "IDS", "IP", "JSON",
"LHS", "QPS", "RAM", "RHS", "RPC", "SLA", "SMTP", "SQL", "SSH", "TCP", "TLS", "TTL", "UDP", "UI", "UID", "UUID",
"URI", "URL", "UTF8", "VM", "XML", "XMPP", "XSRF", "XSS",
];
fn go_local_name(snake: &str) -> String {
let words: Vec<&str> = snake.split('_').filter(|w| !w.is_empty()).collect();
if words.is_empty() {
return String::new();
}
let mut result = String::new();
for (i, word) in words.iter().enumerate() {
let upper = word.to_uppercase();
if GO_INITIALISMS.contains(&upper.as_str()) {
if i == 0 {
result.push_str(&upper.to_lowercase());
} else {
result.push_str(&upper);
}
} else if i == 0 {
result.push_str(&word.to_lowercase());
} else {
let mut chars = word.chars();
if let Some(c) = chars.next() {
result.extend(c.to_uppercase());
result.push_str(&chars.as_str().to_lowercase());
}
}
}
result
}
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()),
}
}