From e947d5e8c8a726a32478fbd2407f0953bae9008e Mon Sep 17 00:00:00 2001 From: Delyan Angelov Date: Tue, 14 Apr 2020 16:46:58 +0300 Subject: [PATCH] checker: fully exhaustive matches for sumtypes and enums Also change the vlib/v/checker/tests/inout/match_expr_else.out to reflex the new error details. --- vlib/v/ast/ast.v | 4 +- vlib/v/checker/checker.v | 73 +++++++++++++++++-- .../v/checker/tests/inout/match_expr_else.out | 4 +- 3 files changed, 70 insertions(+), 11 deletions(-) diff --git a/vlib/v/ast/ast.v b/vlib/v/ast/ast.v index 454e9a36a4..0371d76dbc 100644 --- a/vlib/v/ast/ast.v +++ b/vlib/v/ast/ast.v @@ -395,8 +395,8 @@ mut: pub struct MatchBranch { pub: - exprs []Expr - stmts []Stmt + exprs []Expr // left side + stmts []Stmt // right side pos token.Position comment Comment // comment above `xxx {` is_else bool diff --git a/vlib/v/checker/checker.v b/vlib/v/checker/checker.v index b53973fb0f..1905f10555 100644 --- a/vlib/v/checker/checker.v +++ b/vlib/v/checker/checker.v @@ -1332,26 +1332,85 @@ pub fn (c mut Checker) match_expr(node mut ast.MatchExpr) table.Type { if cond_type == 0 { c.error('match 0 cond type', node.pos) } + type_sym := c.table.get_type_symbol(cond_type) + + // all_possible_left_subtypes is a histogram of + // type => how many times it was used in the match + mut all_possible_left_subtypes := map[string]int + // all_possible_left_enum_vals is a histogram of + // enum value name => how many times it was used in the match + mut all_possible_left_enum_vals := map[string]int + match type_sym.info { + table.SumType { + for v in it.variants { + all_possible_left_subtypes[ int(v).str() ] = 0 + } + } + table.Enum { + for v in it.vals { + all_possible_left_enum_vals[v] = 0 + } + } + else {} + } if !node.branches[node.branches.len - 1].is_else { mut used_values_count := 0 - for branch in node.branches { + for bi, branch in node.branches { used_values_count += branch.exprs.len + for bi_ei, bexpr in branch.exprs { + match bexpr { + ast.Type { + tidx := table.type_idx(it.typ) + stidx := tidx.str() + all_possible_left_subtypes[ stidx ] = all_possible_left_subtypes[ stidx ] + 1 + } + ast.EnumVal { + all_possible_left_enum_vals[ it.val ] = all_possible_left_enum_vals[ it.val ] + 1 + } + else{} + } + } } - type_sym := c.table.get_type_symbol(cond_type) - mut err := false + mut err := false + mut err_details := 'match must be exhaustive' + unhandled := []string match type_sym.info { table.SumType { - err = used_values_count < it.variants.len + for k,v in all_possible_left_subtypes { + if v == 0 { + err = true + unhandled << '`' + c.table.type_to_str( table.new_type( k.int() ) ) + '`' + } + if v > 1 { + err = true + multiple_type_name := '`' + c.table.type_to_str( table.new_type( k.int() ) ) + '`' + c.error('a match case for $multiple_type_name is handled more than once', node.pos) + } + } } table.Enum { - err = used_values_count < it.vals.len + for k,v in all_possible_left_enum_vals { + if v == 0 { + err = true + unhandled << '`.$k`' + } + if v > 1 { + err = true + multiple_enum_val := '`.$k`' + c.error('a match case for $multiple_enum_val is handled more than once', node.pos) + } + } } - else { err = false } + else { err = true } } if err { - c.error('match must be exhaustive', node.pos) + if unhandled.len > 0 { + err_details += ' (add match branches for: ' + unhandled.join(', ') + ' or an else{} branch)' + } + c.error(err_details, node.pos) } } + c.expected_type = cond_type mut ret_type := table.void_type for branch in node.branches { diff --git a/vlib/v/checker/tests/inout/match_expr_else.out b/vlib/v/checker/tests/inout/match_expr_else.out index 3ac1bbf9cc..962a3b44ce 100644 --- a/vlib/v/checker/tests/inout/match_expr_else.out +++ b/vlib/v/checker/tests/inout/match_expr_else.out @@ -1,7 +1,7 @@ -vlib/v/checker/tests/inout/match_expr_else.v:5:9: error: match must be exhaustive +vlib/v/checker/tests/inout/match_expr_else.v:5:9: error: match must be exhaustive (add match branches for: `f64` or an else{} branch) 3| fn main() { 4| x := A('test') 5| res := match x { ~~~~~~~~~ 6| int { - 7| 'int' \ No newline at end of file + 7| 'int'