use crate::Emitter;
use crate::control_flow::fallible::{
Fallible, FallibleEmitter, PARTIAL_BOTH_CTOR, PARTIAL_OK_CTOR,
};
use crate::expressions::context::ExpressionContext;
use crate::is_order_sensitive;
use crate::names::go_name;
use crate::write_line;
use syntax::ast::Expression;
use syntax::types::Type;
use super::GoCallStrategy;
#[derive(Clone, Copy)]
pub(crate) enum WrapperTarget<'a> {
FreshSlot,
Slot(&'a str),
Return,
}
pub(crate) enum WrapperOutcome {
Slot(String),
Returned,
}
impl WrapperOutcome {
pub(crate) fn into_slot(self) -> Option<String> {
match self {
WrapperOutcome::Slot(s) => Some(s),
WrapperOutcome::Returned => None,
}
}
pub(crate) fn expect_slot(self) -> String {
match self {
WrapperOutcome::Slot(s) => s,
WrapperOutcome::Returned => unreachable!("expected slot, got Returned"),
}
}
}
pub(super) enum ResolvedSink {
Slot(String),
Return,
}
pub(super) fn write_leaf(output: &mut String, sink: &ResolvedSink, value: &str) {
match sink {
ResolvedSink::Slot(name) => write_line!(output, "{} = {}", name, value),
ResolvedSink::Return => write_line!(output, "return {}", value),
}
}
impl Emitter<'_> {
pub(super) fn open_wrapper_slot(
&mut self,
output: &mut String,
target: WrapperTarget<'_>,
type_str: &str,
name_hint: &'static str,
) -> (ResolvedSink, WrapperOutcome) {
match target {
WrapperTarget::FreshSlot => {
let var = self.fresh_var(Some(name_hint));
self.declare(&var);
write_line!(output, "var {} {}", var, type_str);
(ResolvedSink::Slot(var.clone()), WrapperOutcome::Slot(var))
}
WrapperTarget::Slot(name) => {
write_line!(output, "var {} {}", name, type_str);
self.declare(name);
let owned = name.to_string();
(
ResolvedSink::Slot(owned.clone()),
WrapperOutcome::Slot(owned),
)
}
WrapperTarget::Return => (ResolvedSink::Return, WrapperOutcome::Returned),
}
}
pub(super) fn emit_simple_wrapper_value(
&mut self,
output: &mut String,
target: WrapperTarget<'_>,
name_hint: &'static str,
value_expr: &str,
) -> WrapperOutcome {
match target {
WrapperTarget::FreshSlot => {
WrapperOutcome::Slot(self.hoist_tmp_value(output, name_hint, value_expr))
}
WrapperTarget::Slot(name) => {
self.declare(name);
write_line!(output, "{} := {}", name, value_expr);
WrapperOutcome::Slot(name.to_string())
}
WrapperTarget::Return => {
write_line!(output, "return {}", value_expr);
WrapperOutcome::Returned
}
}
}
}
impl Emitter<'_> {
pub(super) fn emit_go_tuple_call_wrapped(
&mut self,
output: &mut String,
call_expression: &Expression,
arity: usize,
) -> String {
let Expression::Call { ty, .. } = call_expression else {
unreachable!("emit_go_tuple_call_wrapped called with non-call expression");
};
let call_str = self.emit_call(output, call_expression, None, ExpressionContext::value());
let temp_vars = self.create_temp_vars("ret", arity);
write_line!(output, "{} := {}", temp_vars.join(", "), call_str);
self.emit_tuple_from_vars(output, &temp_vars, ty)
}
pub(super) fn emit_go_partial_call_wrapped(
&mut self,
output: &mut String,
call_expression: &Expression,
partial_ty: &Type,
) -> String {
self.requirements.require_stdlib();
let call_str = self.emit_call(output, call_expression, None, ExpressionContext::value());
self.emit_partial_wrapping(output, &call_str, partial_ty, WrapperTarget::FreshSlot)
.expect_slot()
}
pub(crate) fn emit_partial_wrapping(
&mut self,
output: &mut String,
call_str: &str,
partial_ty: &Type,
target: WrapperTarget<'_>,
) -> WrapperOutcome {
let ok_ty = partial_ty.ok_type();
let err_ty = partial_ty.err_type();
let ok_ty_str = self.go_type_as_string(&ok_ty);
let err_ty_str = self.go_type_as_string(&err_ty);
let pkg = go_name::GO_STDLIB_PKG;
let (err_var, val_var) = self.extract_go_returns(output, call_str, &ok_ty);
let type_params = format!("{}, {}", ok_ty_str, err_ty_str);
let result_ty_str = format!("{pkg}.Partial[{type_params}]");
let (sink, outcome) = self.open_wrapper_slot(output, target, &result_ty_str, "result");
write_line!(output, "if {} != nil {{", err_var);
write_leaf(
output,
&sink,
&format!(
"{PARTIAL_BOTH_CTOR}[{type_params}]({}, {})",
val_var, err_var
),
);
output.push_str("} else {\n");
write_leaf(
output,
&sink,
&format!("{PARTIAL_OK_CTOR}[{type_params}]({})", val_var),
);
output.push_str("}\n");
outcome
}
pub(super) fn emit_go_result_call_wrapped(
&mut self,
output: &mut String,
call_expression: &Expression,
result_ty: &Type,
) -> String {
self.requirements.require_stdlib();
let call_str = self.emit_call(output, call_expression, None, ExpressionContext::value());
self.emit_result_wrapping(output, &call_str, result_ty, WrapperTarget::FreshSlot)
.expect_slot()
}
pub(crate) fn emit_result_wrapping(
&mut self,
output: &mut String,
call_str: &str,
result_ty: &Type,
target: WrapperTarget<'_>,
) -> WrapperOutcome {
let fallible = Fallible::from_type(result_ty).expect("Result type expected");
if fallible.ok_ty().is_unit() {
return self.emit_unit_result_wrapping(output, call_str, &fallible, target);
}
let ok_ty = fallible.ok_ty();
let (err_var, ok_val) = self.extract_go_returns(output, call_str, ok_ty);
let result_ty_str = {
let mut fe = FallibleEmitter::new(self, &fallible);
fe.full_type_string()
};
let interface_id = self.facts.as_interface(ok_ty);
let needs_nil_guard = ok_ty.is_ref()
|| interface_id
.as_deref()
.is_some_and(|id| id != go_name::PRELUDE_ERROR_ID);
let (sink, outcome) = self.open_wrapper_slot(output, target, &result_ty_str, "result");
write_line!(output, "if {} != nil {{", err_var);
let err_wrapper = {
let mut fe = FallibleEmitter::new(self, &fallible);
fe.emit_failure(Some(&err_var))
};
write_leaf(output, &sink, &err_wrapper);
if needs_nil_guard {
self.emit_nil_guard(output, &ok_val, ok_ty, &sink, &fallible);
}
output.push_str("} else {\n");
let ok_wrapper = {
let mut fe = FallibleEmitter::new(self, &fallible);
fe.emit_success(&ok_val)
};
write_leaf(output, &sink, &ok_wrapper);
output.push_str("}\n");
outcome
}
fn emit_unit_result_wrapping(
&mut self,
output: &mut String,
call_str: &str,
fallible: &Fallible,
target: WrapperTarget<'_>,
) -> WrapperOutcome {
let err_var = self.hoist_tmp_value(output, "ret", call_str);
let result_ty_str = {
let mut fe = FallibleEmitter::new(self, fallible);
fe.full_type_string()
};
let (sink, outcome) = self.open_wrapper_slot(output, target, &result_ty_str, "result");
write_line!(output, "if {} != nil {{", err_var);
let err_wrapper = {
let mut fe = FallibleEmitter::new(self, fallible);
fe.emit_failure(Some(&err_var))
};
write_leaf(output, &sink, &err_wrapper);
output.push_str("} else {\n");
let ok_wrapper = {
let mut fe = FallibleEmitter::new(self, fallible);
fe.emit_success("struct{}{}")
};
write_leaf(output, &sink, &ok_wrapper);
output.push_str("}\n");
outcome
}
fn extract_go_returns(
&mut self,
output: &mut String,
call_str: &str,
ok_ty: &Type,
) -> (String, String) {
if let Type::Tuple(elements) = ok_ty {
let tuple_arity = elements.len();
let temp_vars = self.create_temp_vars("ret", tuple_arity + 1);
write_line!(output, "{} := {}", temp_vars.join(", "), call_str);
let tuple_var = self.emit_tuple_from_vars(output, &temp_vars[..tuple_arity], ok_ty);
(temp_vars.last().unwrap().clone(), tuple_var)
} else {
let val_var = self.fresh_var(Some("ret"));
self.declare(&val_var);
let err_var = self.fresh_var(Some("ret"));
self.declare(&err_var);
write_line!(output, "{}, {} := {}", val_var, err_var, call_str);
(err_var, val_var)
}
}
fn emit_nil_guard(
&mut self,
output: &mut String,
ok_val: &str,
ok_ty: &Type,
sink: &ResolvedSink,
fallible: &Fallible,
) {
let nil_check = if ok_ty.is_tuple() {
format!("{}.First", ok_val)
} else {
ok_val.to_string()
};
let is_interface = self.facts.is_interface(ok_ty);
if is_interface {
write_line!(
output,
"}} else if lisette.IsNilInterface({}) {{",
nil_check
);
} else {
write_line!(output, "}} else if {} == nil {{", nil_check);
}
self.requirements.require_errors();
let mut fe = FallibleEmitter::new(self, fallible);
let nil_err = fe.emit_failure(Some("errors.New(\"unexpected nil\")"));
write_leaf(output, sink, &nil_err);
}
pub(crate) fn classify_go_fn_value(&self, expression: &Expression) -> Option<GoCallStrategy> {
let inner = expression.unwrap_parens();
if let Expression::DotAccess {
expression: receiver,
..
} = inner
&& Self::is_go_receiver(receiver)
{
let fn_type = expression.get_type();
let Type::Function { return_type, .. } = fn_type.unwrap_forall() else {
return None;
};
let return_type = return_type.clone();
let go_hints = if let Expression::DotAccess {
expression: receiver_expression,
member,
..
} = inner
{
self.go_qualified_name(receiver_expression, member)
.and_then(|name| self.facts.definition(name.as_str()))
.map(|d| d.go_hints().to_vec())
.unwrap_or_default()
} else {
vec![]
};
return self.facts.classify_go_return_type(&return_type, &go_hints);
}
None
}
pub(crate) fn is_go_array_return_value(&self, expression: &Expression) -> bool {
if let Expression::DotAccess {
expression: receiver,
member,
..
} = expression.unwrap_parens()
&& Self::is_go_receiver(receiver)
{
return self.has_go_array_return(receiver, member);
}
false
}
fn hoist_go_fn_if_needed(&mut self, output: &mut String, expression: &Expression) -> String {
let go_fn_str = self.emit_operand(output, expression, ExpressionContext::value());
let is_go_module_fn = matches!(
expression.unwrap_parens(),
Expression::DotAccess { expression, .. }
if expression.get_type().as_import_namespace()
.is_some_and(|m| m.starts_with(go_name::GO_IMPORT_PREFIX))
);
if is_go_module_fn {
return go_fn_str;
}
if is_order_sensitive(expression) {
self.hoist_tmp_value(output, "fn", &go_fn_str)
} else {
go_fn_str
}
}
pub(crate) fn build_wrapper_params(&mut self, params: &[Type]) -> (Vec<String>, Vec<String>) {
let mut param_strs = Vec::new();
let mut arg_names = Vec::new();
let last_idx = params.len().saturating_sub(1);
for (i, param_ty) in params.iter().enumerate() {
let name = format!("arg{}", i);
let ty_str = self.go_type_as_string(param_ty);
param_strs.push(format!("{} {}", name, ty_str));
if i == last_idx && param_ty.get_name() == Some("VarArgs") {
arg_names.push(format!("{}...", name));
} else {
arg_names.push(name);
}
}
(param_strs, arg_names)
}
pub(crate) fn emit_array_return_wrapper(
&mut self,
output: &mut String,
expression: &Expression,
) -> String {
let fn_type = expression.get_type();
let (params, return_type) = match fn_type.unwrap_forall() {
Type::Function {
params,
return_type,
..
} => (params.clone(), (**return_type).clone()),
_ => return self.emit_operand(output, expression, ExpressionContext::value()),
};
let go_fn_str = self.hoist_go_fn_if_needed(output, expression);
let (param_strs, arg_names) = self.build_wrapper_params(¶ms);
let ret_ty_str = self.go_type_as_string(&return_type);
let call_str = format!("{}({})", go_fn_str, arg_names.join(", "));
let arr_var = self.fresh_var(Some("arr"));
self.declare(&arr_var);
format!(
"func({}) {} {{\n{} := {}\nreturn {}[:]\n}}",
param_strs.join(", "),
ret_ty_str,
arr_var,
call_str,
arr_var,
)
}
pub(crate) fn emit_go_fn_wrapper(
&mut self,
output: &mut String,
expression: &Expression,
strategy: &GoCallStrategy,
) -> String {
self.requirements.require_stdlib();
let fn_type = expression.get_type();
let (params, return_type) = match fn_type.unwrap_forall() {
Type::Function {
params,
return_type,
..
} => (params.clone(), (**return_type).clone()),
_ => unreachable!("expected function type"),
};
let go_fn_str = self.hoist_go_fn_if_needed(output, expression);
let (param_strs, arg_names) = self.build_wrapper_params(¶ms);
let ret_ty_str = self.go_type_as_string(&return_type);
let call_str = format!("{}({})", go_fn_str, arg_names.join(", "));
let mut body = String::new();
let outcome = match strategy {
GoCallStrategy::Result => {
self.emit_result_wrapping(&mut body, &call_str, &return_type, WrapperTarget::Return)
}
GoCallStrategy::CommaOk => self.emit_comma_ok_wrapping(
&mut body,
&call_str,
&return_type,
true,
WrapperTarget::Return,
),
GoCallStrategy::NullableReturn => {
let raw_var = self.hoist_tmp_value(&mut body, "raw", &call_str);
self.emit_nil_check_option_wrap(
&mut body,
&raw_var,
&return_type,
WrapperTarget::Return,
)
}
GoCallStrategy::Tuple { arity } => {
let temp_vars = self.create_temp_vars("ret", *arity);
write_line!(body, "{} := {}", temp_vars.join(", "), call_str);
let tuple_str = self.emit_tuple_from_vars(&mut body, &temp_vars, &return_type);
WrapperOutcome::Slot(tuple_str)
}
GoCallStrategy::Partial => self.emit_partial_wrapping(
&mut body,
&call_str,
&return_type,
WrapperTarget::Return,
),
GoCallStrategy::Sentinel { value } => self.emit_sentinel_wrapping(
&mut body,
&call_str,
&return_type,
*value,
WrapperTarget::Return,
),
};
if let Some(result_var) = outcome.into_slot() {
write_line!(body, "return {}", result_var);
}
format!(
"func({}) {} {{\n{}}}",
param_strs.join(", "),
ret_ty_str,
body
)
}
}