1use crate::error::TypeError;
17
18#[derive(Debug, Clone, Copy)]
20pub struct RuleInfo {
21 pub tag: &'static str,
22 pub explanation: &'static str,
23}
24
25impl TypeError {
26 pub fn rule_tag(&self) -> &'static str {
29 match self {
30 TypeError::TypeMismatch { .. } => "type-mismatch",
31 TypeError::EffectRowMismatch { .. } => "effect-row-mismatch",
32 TypeError::UnknownIdentifier { .. } => "unknown-identifier",
33 TypeError::ArityMismatch { .. } => "arity-mismatch",
34 TypeError::NonExhaustiveMatch { .. } => "non-exhaustive-match",
35 TypeError::UnknownField { .. } => "unknown-field",
36 TypeError::DuplicateField { .. } => "duplicate-field",
37 TypeError::UnknownVariant { .. } => "unknown-variant",
38 TypeError::EffectNotDeclared { .. } => "effect-not-declared",
39 TypeError::InfiniteType { .. } => "infinite-type",
40 TypeError::AmbiguousType { .. } => "ambiguous-type",
41 TypeError::RecursiveTypeWithoutConstructor { .. } => "recursive-type-without-constructor",
42 TypeError::RefinementViolation { .. } => "refinement-violation",
43 TypeError::ExamplesOnEffectfulFn { .. } => "examples-on-effectful-fn",
44 TypeError::ExampleArityMismatch { .. } => "example-arity-mismatch",
45 TypeError::ExampleMismatch { .. } => "example-mismatch",
46 }
47 }
48
49 pub fn rule_explanation(&self) -> &'static str {
53 explanation_for_tag(self.rule_tag())
54 }
55}
56
57fn explanation_for_tag(tag: &str) -> &'static str {
58 match tag {
59 "type-mismatch" => "An expression's inferred type doesn't match what the surrounding context requires \
60(return type, let-binding annotation, function argument, operator operand, etc.). Fix by changing \
61the expression to produce the expected type, or by adjusting the declared/inferred expected type \
62to match.",
63 "effect-row-mismatch" => EFFECT_ROW_MISMATCH,
64 "unknown-identifier" => "A name referenced in scope is not declared. Either the binding is missing, \
65the name is misspelled, or an `import` is missing. Check for typos first; then verify the relevant \
66`let`, parameter, or top-level `fn` is in scope.",
67 "arity-mismatch" => "A call site supplies a different number of arguments than the function or \
68constructor accepts. Either add the missing arguments or remove the extras.",
69 "non-exhaustive-match" => "A `match` expression doesn't cover every case of its scrutinee's type. \
70Add the missing arms listed in the error, or add a `_` wildcard if catching the remainder is intended.",
71 "unknown-field" => "A record field access or literal references a field name that isn't part of \
72the record type. Verify spelling and that the type really has that field — check the type declaration.",
73 "duplicate-field" => "A record literal lists the same field name twice. Each field must appear \
74exactly once. Remove the duplicate or rename one of them.",
75 "unknown-variant" => "A constructor pattern or expression references a variant name that isn't \
76part of the union type. Verify spelling and that the variant exists on this union.",
77 "effect-not-declared" => "A function body invokes an effect (io, fs_read, net, …) that the \
78function's signature doesn't declare. Either add the effect to the function's `[effects]` annotation \
79or remove the call that produces it.",
80 "infinite-type" => "Inference would require a type to contain itself (e.g. `t = List<t>` with no \
81constructor). Add a nominal type wrapper or restructure the data so the recursion is mediated by a \
82named type.",
83 "ambiguous-type" => "Inference couldn't pick a single concrete type for an expression. Add a type \
84annotation to disambiguate.",
85 "recursive-type-without-constructor" => "A type alias references itself with no constructor in \
86between, so no value of the type can ever be built. Make the recursive position carry a constructor \
87(e.g. `Cons<T, List<T>> | Nil`).",
88 "refinement-violation" => "A literal argument provably violates a refinement-type predicate \
89(#209). Adjust the argument to satisfy the predicate, or relax the predicate at the function \
90signature.",
91 "examples-on-effectful-fn" => "A function with an `examples { ... }` block (#369) also \
92declares effects. Signature-level examples are pure-only in v1 — they must be deterministic so the \
93contract is reproducible. Either remove the effects from the signature, or remove the examples block \
94and rely on external tests.",
95 "example-arity-mismatch" => "A case inside an `examples { ... }` block (#369) supplies a \
96different number of arguments than the function declares. Match the call's argument count to the \
97function's parameter count.",
98 "example-mismatch" => "A case inside an `examples { ... }` block (#369) ran successfully \
99but the function body's actual return value differs from the declared `expected` value. Either \
100update the example to match the new behavior, or fix the body to produce the declared value.",
101 _ => "Unknown rule. The rule_tag may have been introduced after this Lex release.",
102 }
103}
104
105pub fn all_rules() -> &'static [RuleInfo] {
109 &[
110 RuleInfo { tag: "type-mismatch", explanation: TYPE_MISMATCH },
111 RuleInfo { tag: "effect-row-mismatch", explanation: EFFECT_ROW_MISMATCH },
112 RuleInfo { tag: "unknown-identifier", explanation: UNKNOWN_IDENT },
113 RuleInfo { tag: "arity-mismatch", explanation: ARITY_MISMATCH },
114 RuleInfo { tag: "non-exhaustive-match", explanation: NON_EXHAUSTIVE },
115 RuleInfo { tag: "unknown-field", explanation: UNKNOWN_FIELD },
116 RuleInfo { tag: "duplicate-field", explanation: DUPLICATE_FIELD },
117 RuleInfo { tag: "unknown-variant", explanation: UNKNOWN_VARIANT },
118 RuleInfo { tag: "effect-not-declared", explanation: EFFECT_NOT_DECLARED },
119 RuleInfo { tag: "infinite-type", explanation: INFINITE_TYPE },
120 RuleInfo { tag: "ambiguous-type", explanation: AMBIGUOUS_TYPE },
121 RuleInfo { tag: "recursive-type-without-constructor", explanation: RECURSIVE_NO_CTOR },
122 RuleInfo { tag: "refinement-violation", explanation: REFINEMENT_VIOLATION },
123 RuleInfo { tag: "examples-on-effectful-fn", explanation: EXAMPLES_ON_EFFECTFUL_FN },
124 RuleInfo { tag: "example-arity-mismatch", explanation: EXAMPLE_ARITY_MISMATCH },
125 RuleInfo { tag: "example-mismatch", explanation: EXAMPLE_MISMATCH },
126 ]
127}
128
129pub fn suggested_transform_for(rule_tag: &str) -> Option<serde_json::Value> {
148 let (kind_hint, summary, details) = match rule_tag {
149 "type-mismatch" => (
150 "ReplaceMatchArm",
151 "Replace the offending match arm (or expression) so its body produces the expected type.",
152 "When a function body's inferred type doesn't match its signature, the easiest \
153typed-transform fix is `ReplaceMatchArm` — rebuild whichever arm produces the wrong type so it \
154returns the expected one. For non-match expressions, the LLM-driven `lex repair --apply` flow \
155can rewrite the body via `ModifyBody`.",
156 ),
157 "unknown-identifier" => (
158 "RenameLocal",
159 "If the name is a typo, rename a similarly-spelled in-scope binding to match.",
160 "An `unknown-identifier` error is most often a typo. Search the function's lexical \
161scope for a binding whose name is a single edit away and apply `RenameLocal` to switch references. \
162If no nearby name exists, the missing binding probably needs a `let` or an `import` — fall back to \
163LLM-driven repair.",
164 ),
165 "non-exhaustive-match" => (
166 "ReplaceMatchArm",
167 "Add the missing match arms (or a `_` wildcard) covering the unhandled variants.",
168 "Use `ReplaceMatchArm` to append arms for the variants listed in the error's \
169`missing` field. If catching the remainder is intended, a single `_` wildcard arm suffices; \
170otherwise add one explicit arm per missing variant so the audit trail records the new semantics.",
171 ),
172 "effect-not-declared" => (
173 "ChangeEffectSig",
174 "Add the inferred effect to the function's `[effects]` declaration.",
175 "The function body invokes an effect that the signature doesn't declare. Either add \
176the effect to the signature via `ChangeEffectSig` (preferred — the effect is genuinely needed) or \
177remove the call that produces it via `ModifyBody` (preferred when the effect was unintentional).",
178 ),
179 "effect-row-mismatch" => (
180 "ModifyBody",
181 "Narrow the closure body to use only the effects in the declared row; do NOT broaden the row.",
182 "Effect rows are invariant: the declared row must match exactly, so the intuitive repair \
183(adding the missing effect to the declared row) is wrong when that row is fixed by a record field or \
184other annotation — e.g. `Skill.handle`, `Tool.execute`. Use `ModifyBody` to remove or replace the \
185calls whose effects fall outside the declared row. Reach for `ChangeEffectSig` only when you own the \
186annotation and the extra effect is genuinely required.",
187 ),
188 "arity-mismatch" => (
189 "ModifyBody",
190 "Match the call site's argument count to the function's declared arity.",
191 "The number of arguments at the call site doesn't match the declared signature. \
192Add the missing arguments or remove the extras. No typed transform directly applies — \
193use the LLM-driven `lex repair --apply` flow with `ModifyBody` to rewrite the call site.",
194 ),
195 "unknown-field" => (
196 "ModifyBody",
197 "Verify the field spelling and the record type's declaration; rewrite the access.",
198 "The field name isn't part of the record type. Either correct the spelling or add \
199the missing field to the type declaration. Use the LLM-driven `lex repair --apply` flow with \
200`ModifyBody` to rewrite the field access once the correct name is known.",
201 ),
202 "ambiguous-type" => (
203 "ModifyBody",
204 "Add a type annotation at the ambiguous expression to disambiguate inference.",
205 "Inference couldn't pick a single concrete type. Add an explicit type annotation on \
206the offending `let` binding, function parameter, or function return type via `ModifyBody`. The \
207LLM-driven `lex repair --apply` flow can synthesize the annotation from the surrounding context.",
208 ),
209 "example-mismatch" => (
210 "ModifyBody",
211 "Reconcile the body with its declared example — fix one or the other.",
212 "An `examples { ... }` case ran but produced a value different from the declared \
213`expected`. Either the example is stale (LLM should rewrite the case via `ReplaceMatchArm` against \
214the `examples` block) or the body regressed (LLM should rewrite the body via `ModifyBody` to make \
215the case pass). The `case_index` field in the error identifies which case to act on.",
216 ),
217 _ => return None,
218 };
219 Some(serde_json::json!({
220 "kind_hint": kind_hint,
221 "rule_tag": rule_tag,
222 "summary": summary,
223 "details": details,
224 }))
225}
226
227const TYPE_MISMATCH: &str = "An expression's inferred type doesn't match what the surrounding context requires \
231(return type, let-binding annotation, function argument, operator operand, etc.). Fix by changing \
232the expression to produce the expected type, or by adjusting the declared/inferred expected type \
233to match.";
234const EFFECT_ROW_MISMATCH: &str = "Two function types failed to unify because their effect rows differ. \
235Effect rows unify by equality, not subtyping: a concrete row must match exactly — a superset or subset \
236is rejected. This most often surfaces on record-field closures (e.g. `Skill.handle`, `Tool.execute`), \
237whose declared effect row is fixed by the record type. Fix by narrowing the function body so it uses \
238only the effects in the declared row — do NOT broaden the declared row to match the body.";
239const UNKNOWN_IDENT: &str = "A name referenced in scope is not declared. Either the binding is missing, \
240the name is misspelled, or an `import` is missing. Check for typos first; then verify the relevant \
241`let`, parameter, or top-level `fn` is in scope.";
242const ARITY_MISMATCH: &str = "A call site supplies a different number of arguments than the function or \
243constructor accepts. Either add the missing arguments or remove the extras.";
244const NON_EXHAUSTIVE: &str = "A `match` expression doesn't cover every case of its scrutinee's type. \
245Add the missing arms listed in the error, or add a `_` wildcard if catching the remainder is intended.";
246const UNKNOWN_FIELD: &str = "A record field access or literal references a field name that isn't part of \
247the record type. Verify spelling and that the type really has that field — check the type declaration.";
248const DUPLICATE_FIELD: &str = "A record literal lists the same field name twice. Each field must appear \
249exactly once. Remove the duplicate or rename one of them.";
250const UNKNOWN_VARIANT: &str = "A constructor pattern or expression references a variant name that isn't \
251part of the union type. Verify spelling and that the variant exists on this union.";
252const EFFECT_NOT_DECLARED: &str = "A function body invokes an effect (io, fs_read, net, …) that the \
253function's signature doesn't declare. Either add the effect to the function's `[effects]` annotation \
254or remove the call that produces it.";
255const INFINITE_TYPE: &str = "Inference would require a type to contain itself (e.g. `t = List<t>` with no \
256constructor). Add a nominal type wrapper or restructure the data so the recursion is mediated by a \
257named type.";
258const AMBIGUOUS_TYPE: &str = "Inference couldn't pick a single concrete type for an expression. Add a type \
259annotation to disambiguate.";
260const RECURSIVE_NO_CTOR: &str = "A type alias references itself with no constructor in \
261between, so no value of the type can ever be built. Make the recursive position carry a constructor \
262(e.g. `Cons<T, List<T>> | Nil`).";
263const REFINEMENT_VIOLATION: &str = "A literal argument provably violates a refinement-type predicate \
264(#209). Adjust the argument to satisfy the predicate, or relax the predicate at the function \
265signature.";
266const EXAMPLES_ON_EFFECTFUL_FN: &str = "A function with an `examples { ... }` block (#369) also \
267declares effects. Signature-level examples are pure-only in v1 — they must be deterministic so the \
268contract is reproducible. Either remove the effects from the signature, or remove the examples block \
269and rely on external tests.";
270const EXAMPLE_ARITY_MISMATCH: &str = "A case inside an `examples { ... }` block (#369) supplies a \
271different number of arguments than the function declares. Match the call's argument count to the \
272function's parameter count.";
273const EXAMPLE_MISMATCH: &str = "A case inside an `examples { ... }` block (#369) ran successfully \
274but the function body's actual return value differs from the declared `expected` value. Either \
275update the example to match the new behavior, or fix the body to produce the declared value.";
276
277#[cfg(test)]
278mod tests {
279 use super::*;
280
281 #[test]
282 fn every_variant_has_a_distinct_tag() {
283 let tags: Vec<&str> = all_rules().iter().map(|r| r.tag).collect();
286 let unique: std::collections::BTreeSet<&str> = tags.iter().copied().collect();
287 assert_eq!(unique.len(), tags.len(), "rule tags must be unique: {tags:?}");
288 }
289
290 #[test]
291 fn every_variant_has_a_nonempty_explanation() {
292 for rule in all_rules() {
293 assert!(!rule.explanation.is_empty(), "rule `{}` lacks an explanation", rule.tag);
294 assert!(
295 rule.explanation.len() > 40,
296 "rule `{}` explanation is too short to be useful for LLM repair",
297 rule.tag
298 );
299 }
300 }
301
302 #[test]
303 fn type_error_methods_match_catalog() {
304 let cases: Vec<TypeError> = vec![
307 TypeError::TypeMismatch {
308 at_node: "n_0".into(),
309 expected: "Int".into(),
310 got: "Str".into(),
311 context: vec![],
312 },
313 TypeError::EffectRowMismatch {
314 at_node: "n_0".into(),
315 expected: "[io]".into(),
316 got: "[net]".into(),
317 context: vec![],
318 },
319 TypeError::UnknownIdentifier { at_node: "n_0".into(), name: "x".into() },
320 TypeError::ArityMismatch { at_node: "n_0".into(), expected: 1, got: 2 },
321 TypeError::NonExhaustiveMatch { at_node: "n_0".into(), missing: vec!["None".into()] },
322 TypeError::UnknownField {
323 at_node: "n_0".into(),
324 record_type: "User".into(),
325 field: "ag".into(),
326 },
327 TypeError::DuplicateField { at_node: "n_0".into(), field: "name".into() },
328 TypeError::UnknownVariant { at_node: "n_0".into(), constructor: "Nada".into() },
329 TypeError::EffectNotDeclared { at_node: "n_0".into(), effect: "io".into() },
330 TypeError::InfiniteType { at_node: "n_0".into() },
331 TypeError::AmbiguousType { at_node: "n_0".into() },
332 TypeError::RecursiveTypeWithoutConstructor {
333 at_node: "n_0".into(),
334 name: "Bad".into(),
335 },
336 TypeError::RefinementViolation {
337 at_node: "n_0".into(),
338 fn_name: "f".into(),
339 param_index: 0,
340 binding: "x".into(),
341 reason: "x > 0".into(),
342 },
343 TypeError::ExamplesOnEffectfulFn {
344 at_node: "n_0".into(),
345 fn_name: "f".into(),
346 },
347 TypeError::ExampleArityMismatch {
348 at_node: "n_0".into(),
349 fn_name: "f".into(),
350 case_index: 0,
351 expected: 2,
352 got: 1,
353 },
354 TypeError::ExampleMismatch {
355 at_node: "n_0".into(),
356 fn_name: "f".into(),
357 case_index: 0,
358 expected: "1".into(),
359 got: "2".into(),
360 },
361 ];
362 let catalog: std::collections::BTreeMap<&str, &str> =
363 all_rules().iter().map(|r| (r.tag, r.explanation)).collect();
364 assert_eq!(cases.len(), catalog.len(), "every variant must be covered");
365 for e in &cases {
366 let tag = e.rule_tag();
367 let expl = catalog.get(tag).unwrap_or_else(|| panic!("tag `{tag}` not in catalog"));
368 assert_eq!(e.rule_explanation(), *expl, "tag/explanation mismatch on {tag}");
369 }
370 }
371
372 #[test]
373 fn suggested_transform_covers_at_least_five_rules() {
374 let mut covered = 0;
377 for rule in all_rules() {
378 if suggested_transform_for(rule.tag).is_some() {
379 covered += 1;
380 }
381 }
382 assert!(
383 covered >= 5,
384 "suggested_transform must cover ≥5 rule_tags; got {covered}"
385 );
386 }
387
388 #[test]
389 fn suggested_transform_shape_is_consistent() {
390 for rule in all_rules() {
393 let Some(s) = suggested_transform_for(rule.tag) else { continue };
394 for field in ["kind_hint", "rule_tag", "summary", "details"] {
395 assert!(
396 s.get(field).and_then(|v| v.as_str()).is_some_and(|v| !v.is_empty()),
397 "rule `{}` suggestion missing/empty `{field}`: {s}",
398 rule.tag
399 );
400 }
401 assert_eq!(
402 s.get("rule_tag").and_then(|v| v.as_str()),
403 Some(rule.tag),
404 "suggestion's rule_tag must echo the input tag"
405 );
406 }
407 }
408
409 #[test]
410 fn unknown_rule_tag_returns_none() {
411 assert!(suggested_transform_for("does-not-exist").is_none());
412 }
413}