From 6ca53d7b3963d4f6aede60963aaa794e480ebae9 Mon Sep 17 00:00:00 2001 From: Larpon Date: Wed, 27 May 2020 11:05:50 +0200 Subject: [PATCH] scanner: improve @FN substitution for anon functions --- vlib/v/scanner/scanner.v | 58 +++++++++++++++++++++++------------ vlib/v/scanner/scanner_test.v | 33 +++++++++++++++++++- 2 files changed, 70 insertions(+), 21 deletions(-) diff --git a/vlib/v/scanner/scanner.v b/vlib/v/scanner/scanner.v index 6150ad26f5..65c765b4e5 100644 --- a/vlib/v/scanner/scanner.v +++ b/vlib/v/scanner/scanner.v @@ -105,39 +105,57 @@ fn (mut s Scanner) ident_fn_name() string { start := s.pos mut pos := s.pos pos++ - // Search for function scope start - for pos < s.text.len && s.text[pos] != `{` { + + if s.current_column() - 2 != 0 { + return s.fn_name + } + + has_struct_name := s.struct_name != '' + + if has_struct_name { + for pos < s.text.len && s.text[pos] != `(` { + pos++ + } + if pos >= s.text.len { + return '' + } + pos++ + } + + for pos < s.text.len && s.text[pos] != `(` { pos++ } if pos >= s.text.len { - return "" + return '' } - // Search backwards for "first" occurrence of function open paranthesis - for pos > start && s.text[pos] != `(` { + pos-- + + // Eat whitespaces + for pos > start && s.text[pos].is_space() { pos-- } if pos < start { - return "" - } - // Search backwards for end position of function name - for pos > start && !util.is_func_char(s.text[pos]) { - pos-- + return '' } + end_pos := pos + 1 - if pos < start { - return "" - } + + pos-- // Search for the start position for pos > start && util.is_func_char(s.text[pos]) { pos-- } - start_pos := pos + 1 - if pos < start || pos >= s.text.len { - return "" + pos++ + + start_pos := pos + + if pos <= start || pos >= s.text.len { + return '' } - if s.text[start_pos].is_digit() || end_pos > s.text.len || end_pos <= start_pos || end_pos <= start || start_pos <= start { - return "" + if s.text[start_pos].is_digit() || end_pos > s.text.len || end_pos <= start_pos || end_pos <= start || start_pos < start { + return '' } + fn_name := s.text[start_pos..end_pos] return fn_name } @@ -202,7 +220,7 @@ fn (mut s Scanner) ident_struct_name() string { return '' } - // Search for closing paranthesis + // Search for closing parenthesis for pos < s.text.len && s.text[pos] != `)` { pos++ } @@ -554,8 +572,8 @@ pub fn (mut s Scanner) scan() token.Token { if token.is_key(name) { kind := token.key_to_token(name) if kind == .key_fn { - s.fn_name = s.ident_fn_name() s.struct_name = s.ident_struct_name() + s.fn_name = s.ident_fn_name() } else if kind == .key_module { s.mod_name = s.ident_mod_name() } diff --git a/vlib/v/scanner/scanner_test.v b/vlib/v/scanner/scanner_test.v index a5d5324278..6afe2400c3 100644 --- a/vlib/v/scanner/scanner_test.v +++ b/vlib/v/scanner/scanner_test.v @@ -24,6 +24,25 @@ fn (mut t TestStruct) test_struct_w_high_order(cb fn(int)string) string { return 'test'+cb(2) } +struct TestFn { } + +fn (mut t TestFn) tst_1() { + assert @FN == 'tst_1' +} + +fn (mut t TestFn) tst_2(cb fn(int)) { + assert @FN == 'tst_2' + cb(1) +} + +fn fn_name_mod_level() { + assert @FN == 'fn_name_mod_level' +} + +fn fn_name_mod_level_high_order(cb fn(int)) { + assert @FN == 'fn_name_mod_level_high_order' + cb(1) +} fn test_scan() { text := 'println(2 + 3)' @@ -64,6 +83,19 @@ fn test_scan() { // Test @FN assert @FN == 'test_scan' + fn_name_mod_level() + fn_name_mod_level_high_order(fn(i int){ + t := i + 1 + assert t == 2 + }) + + tfn := TestFn{} + tfn.tst_1() + tfn.tst_2(fn(i int){ + t := i + 1 + assert t == 2 + }) + // Test @MOD assert @MOD == 'scanner' @@ -80,7 +112,6 @@ fn test_scan() { assert r1 == 'test' assert r2 == 'test2' - } fn test_vmod_file() {