use std::borrow::Cow;
use syntax::ast::{Expression, FormatStringPart, Literal, UnaryOperator};
use syntax::program::DotAccessKind;
macro_rules! write_line {
($dst:expr, $($arg:tt)*) => {
{ use std::fmt::Write as _; writeln!($dst, $($arg)*).unwrap() }
};
}
pub(crate) use write_line;
pub(crate) fn receiver_name(type_name: &str) -> String {
type_name
.trim_start_matches('*')
.split('[')
.next()
.unwrap_or(type_name)
.chars()
.next()
.unwrap_or('x')
.to_lowercase()
.to_string()
}
pub(crate) fn output_references_var(output: &str, var: &str) -> bool {
let masked = mask_go_string_literals(output);
let bytes = masked.as_bytes();
masked
.match_indices(var)
.any(|(abs, _)| is_at_token_boundary(bytes, abs, var.len()))
}
fn is_at_token_boundary(bytes: &[u8], pos: usize, token_len: usize) -> bool {
let before_ok = pos == 0 || {
let c = bytes[pos - 1];
!c.is_ascii_alphanumeric() && c != b'_'
};
let after = pos + token_len;
let after_ok = after >= bytes.len() || {
let c = bytes[after];
!c.is_ascii_alphanumeric() && c != b'_'
};
before_ok && after_ok
}
pub(crate) fn mask_go_string_literals(go_text: &str) -> Cow<'_, str> {
if !go_text.bytes().any(|b| matches!(b, b'"' | b'\'' | b'`')) {
return Cow::Borrowed(go_text);
}
let bytes = go_text.as_bytes();
let mut out = String::with_capacity(bytes.len());
let mut i = 0;
while i < bytes.len() {
match bytes[i] {
quote @ (b'"' | b'\'') => i = mask_literal(bytes, i, quote, true, &mut out),
b'`' => i = mask_literal(bytes, i, b'`', false, &mut out),
_ => {
let start = i;
while i < bytes.len() && !matches!(bytes[i], b'"' | b'\'' | b'`') {
i += 1;
}
out.push_str(&go_text[start..i]);
}
}
}
Cow::Owned(out)
}
fn mask_literal(bytes: &[u8], start: usize, quote: u8, escapes: bool, out: &mut String) -> usize {
out.push(quote as char);
let mut i = start + 1;
while i < bytes.len() {
let b = bytes[i];
if escapes && b == b'\\' && i + 1 < bytes.len() {
out.push_str(" ");
i += 2;
} else if b == quote {
out.push(quote as char);
return i + 1;
} else {
out.push(' ');
i += 1;
}
}
i
}
pub(crate) fn group_params(params: &[(String, String)]) -> String {
if params.is_empty() {
return String::new();
}
if params.len() == 1 {
return format!("{} {}", params[0].0, params[0].1);
}
let mut parts: Vec<String> = Vec::new();
let mut names: Vec<&str> = vec![¶ms[0].0];
let mut current_ty = ¶ms[0].1;
for param in ¶ms[1..] {
if param.1 == *current_ty {
names.push(¶m.0);
} else {
parts.push(format!("{} {}", names.join(", "), current_ty));
names.clear();
names.push(¶m.0);
current_ty = ¶m.1;
}
}
parts.push(format!("{} {}", names.join(", "), current_ty));
parts.join(", ")
}
pub(crate) fn contains_call(expression: &Expression) -> bool {
match expression.unwrap_parens() {
Expression::Call { .. } => true,
Expression::Binary { left, right, .. } => contains_call(left) || contains_call(right),
Expression::Unary { expression, .. }
| Expression::DotAccess { expression, .. }
| Expression::Cast { expression, .. }
| Expression::Reference { expression, .. } => contains_call(expression),
Expression::IndexedAccess {
expression, index, ..
} => contains_call(expression) || contains_call(index),
Expression::Tuple { elements, .. } => elements.iter().any(contains_call),
Expression::StructCall {
field_assignments,
spread,
..
} => {
field_assignments.iter().any(|f| contains_call(&f.value))
|| spread.as_expression().is_some_and(contains_call)
}
Expression::Literal {
literal: Literal::Slice(elements),
..
} => elements.iter().any(contains_call),
Expression::Literal {
literal: Literal::FormatString(parts),
..
} => parts.iter().any(|part| match part {
FormatStringPart::Expression(e) => contains_call(e),
FormatStringPart::Text(_) => false,
}),
Expression::Range { start, end, .. } => {
start.as_deref().is_some_and(contains_call) || end.as_deref().is_some_and(contains_call)
}
Expression::If { .. }
| Expression::IfLet { .. }
| Expression::Match { .. }
| Expression::Block { .. }
| Expression::Loop { .. }
| Expression::Propagate { .. }
| Expression::TryBlock { .. }
| Expression::Select { .. } => false,
_ => false,
}
}
pub(crate) fn is_order_sensitive(expression: &Expression) -> bool {
!matches!(
expression.unwrap_parens(),
Expression::Literal { .. } | Expression::Identifier { .. }
)
}
pub(crate) fn observable_after_mutation(expression: &Expression) -> bool {
!matches!(expression.unwrap_parens(), Expression::Literal { .. })
}
pub(crate) fn reads_mutable_operand(expression: &Expression) -> bool {
match expression.unwrap_parens() {
Expression::IndexedAccess { .. } | Expression::Call { .. } => true,
Expression::Unary {
operator: UnaryOperator::Deref,
..
} => true,
Expression::DotAccess {
expression,
dot_access_kind,
..
} => match dot_access_kind {
Some(
DotAccessKind::StructField { .. }
| DotAccessKind::TupleStructField { .. }
| DotAccessKind::TupleElement,
) => true,
_ => reads_mutable_operand(expression),
},
_ => false,
}
}
pub(crate) fn output_ends_with_diverge(output: &str) -> bool {
output
.trim_end()
.lines()
.next_back()
.is_some_and(is_diverge_line)
}
fn is_diverge_line(line: &str) -> bool {
let trimmed = line.trim();
trimmed == "break"
|| trimmed.starts_with("break ")
|| trimmed == "continue"
|| trimmed.starts_with("continue ")
|| trimmed == "return"
|| trimmed.starts_with("return ")
|| trimmed.starts_with("panic(")
}