From 39412884ef52af16624a47c496a591db3895398b Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Sat, 22 May 2021 19:28:13 +0800 Subject: [PATCH 1/2] Allow setter methods in more proc pointers --- spec/compiler/formatter/formatter_spec.cr | 15 ++++++++------ spec/compiler/parser/parser_spec.cr | 14 +++++++------ src/compiler/crystal/syntax/parser.cr | 24 ++++++++++++++++++++--- 3 files changed, 38 insertions(+), 15 deletions(-) diff --git a/spec/compiler/formatter/formatter_spec.cr b/spec/compiler/formatter/formatter_spec.cr index 45f91d00ea54..8e657e2e07c5 100644 --- a/spec/compiler/formatter/formatter_spec.cr +++ b/spec/compiler/formatter/formatter_spec.cr @@ -724,13 +724,19 @@ describe Crystal::Formatter do assert_format "lib Bar\n enum Foo\n A\n end\nend" assert_format "lib Bar\n enum Foo\n A = 1\n end\nend" - assert_format "->foo=" - assert_format "foo = 1\n->foo.bar" - assert_format "foo = 1\n->foo.bar=" + %w(foo foo= foo? foo!).each do |method| + assert_format "->#{method}" + assert_format "foo = 1\n->foo.#{method}" + assert_format "->Foo.#{method}" + assert_format "->@foo.#{method}" + assert_format "->@@foo.#{method}" + end + assert_format "foo = 1\n->foo.bar(Int32)" assert_format "foo = 1\n->foo.bar(Int32*)" assert_format "foo = 1\n->foo.bar=(Int32)" assert_format "foo = 1\n->foo.[](Int32)" + assert_format "foo = 1\n->foo.[]=(Int32)" assert_format "->{ x }" assert_format "->{\nx\n}", "->{\n x\n}" assert_format "->do\nx\nend", "->do\n x\nend" @@ -739,9 +745,6 @@ describe Crystal::Formatter do assert_format "->( x , y ) { x }", "->(x, y) { x }" assert_format "->( x : Int32 , y ) { x }", "->(x : Int32, y) { x }" - assert_format "->@foo.foo" - assert_format "->@@foo.foo" - {:+, :-, :*, :/, :^, :>>, :<<, :|, :&, :&+, :&-, :&*, :&**}.each do |sym| assert_format ":#{sym}" end diff --git a/spec/compiler/parser/parser_spec.cr b/spec/compiler/parser/parser_spec.cr index 515195e4e498..2779a43b9e67 100644 --- a/spec/compiler/parser/parser_spec.cr +++ b/spec/compiler/parser/parser_spec.cr @@ -1271,18 +1271,20 @@ module Crystal it_parses "x = 1; ->{ x }", [Assign.new("x".var, 1.int32), ProcLiteral.new(Def.new("->", body: "x".var))] it_parses "f ->{ a do\n end\n }", Call.new(nil, "f", ProcLiteral.new(Def.new("->", body: Call.new(nil, "a", block: Block.new)))) - it_parses "->foo", ProcPointer.new(nil, "foo") - it_parses "->Foo.foo", ProcPointer.new("Foo".path, "foo") + %w(foo foo= foo? foo!).each do |method| + it_parses "->#{method}", ProcPointer.new(nil, method) + it_parses "foo = 1; ->foo.#{method}", [Assign.new("foo".var, 1.int32), ProcPointer.new("foo".var, method)] + it_parses "->Foo.#{method}", ProcPointer.new("Foo".path, method) + it_parses "->@foo.#{method}", ProcPointer.new("@foo".instance_var, method) + it_parses "->@@foo.#{method}", ProcPointer.new("@@foo".class_var, method) + end + it_parses "->Foo::Bar::Baz.foo", ProcPointer.new(["Foo", "Bar", "Baz"].path, "foo") it_parses "->foo(Int32, Float64)", ProcPointer.new(nil, "foo", ["Int32".path, "Float64".path] of ASTNode) it_parses "foo = 1; ->foo.bar(Int32)", [Assign.new("foo".var, 1.int32), ProcPointer.new("foo".var, "bar", ["Int32".path] of ASTNode)] it_parses "->foo(Void*)", ProcPointer.new(nil, "foo", ["Void".path.pointer_of] of ASTNode) it_parses "call ->foo", Call.new(nil, "call", ProcPointer.new(nil, "foo")) it_parses "[] of ->\n", ArrayLiteral.new(of: ProcNotation.new) - it_parses "->foo=", ProcPointer.new(nil, "foo=") - it_parses "foo = 1; ->foo.foo=", [Assign.new("foo".var, 1.int32), ProcPointer.new("foo".var, "foo=")] - it_parses "->@foo.foo", [ProcPointer.new("@foo".instance_var, "foo")] - it_parses "->@@foo.foo", [ProcPointer.new("@@foo".class_var, "foo")] it_parses "foo &->bar", Call.new(nil, "foo", block_arg: ProcPointer.new(nil, "bar")) diff --git a/src/compiler/crystal/syntax/parser.cr b/src/compiler/crystal/syntax/parser.cr index 9c64cc9e6712..cfd1e90819e9 100644 --- a/src/compiler/crystal/syntax/parser.cr +++ b/src/compiler/crystal/syntax/parser.cr @@ -1912,19 +1912,37 @@ module Crystal obj = parse_generic check :"." name = consume_def_or_macro_name - next_token_skip_space + next_token + if @token.type == :"=" + name = "#{name}=" + next_token_skip_space + else + skip_space + end when :INSTANCE_VAR obj = InstanceVar.new(@token.value.to_s) next_token_skip_space check :"." name = consume_def_or_macro_name - next_token_skip_space + next_token + if @token.type == :"=" + name = "#{name}=" + next_token_skip_space + else + skip_space + end when :CLASS_VAR obj = ClassVar.new(@token.value.to_s) next_token_skip_space check :"." name = consume_def_or_macro_name - next_token_skip_space + next_token + if @token.type == :"=" + name = "#{name}=" + next_token_skip_space + else + skip_space + end else unexpected_token end From 37b9c77dcf8c3f99e8b0dd8635e515775037ca7e Mon Sep 17 00:00:00 2001 From: Quinton Miller Date: Sat, 22 May 2021 19:34:36 +0800 Subject: [PATCH 2/2] consume_def_equals_sign_skip_space helper --- src/compiler/crystal/syntax/parser.cr | 86 ++++++++------------------- 1 file changed, 25 insertions(+), 61 deletions(-) diff --git a/src/compiler/crystal/syntax/parser.cr b/src/compiler/crystal/syntax/parser.cr index cfd1e90819e9..84698dffade3 100644 --- a/src/compiler/crystal/syntax/parser.cr +++ b/src/compiler/crystal/syntax/parser.cr @@ -1886,63 +1886,34 @@ module Crystal case @token.type when :IDENT name = @token.value.to_s - next_token - if @token.type == :"=" + if consume_def_equals_sign_skip_space name = "#{name}=" - next_token_skip_space - else - skip_space - if @token.type == :"." - second_name = consume_def_or_macro_name - if name != "self" && !@def_vars.last.includes?(name) - raise "undefined variable '#{name}'", location.line_number, location.column_number - end - obj = Var.new(name) - name = second_name - next_token - if @token.type == :"=" - name = "#{name}=" - next_token_skip_space - else - skip_space - end + elsif @token.type == :"." + if name != "self" && !@def_vars.last.includes?(name) + raise "undefined variable '#{name}'", location.line_number, location.column_number end + obj = Var.new(name) + + name = consume_def_or_macro_name + name = "#{name}=" if consume_def_equals_sign_skip_space end when :CONST obj = parse_generic check :"." name = consume_def_or_macro_name - next_token - if @token.type == :"=" - name = "#{name}=" - next_token_skip_space - else - skip_space - end + name = "#{name}=" if consume_def_equals_sign_skip_space when :INSTANCE_VAR obj = InstanceVar.new(@token.value.to_s) next_token_skip_space check :"." name = consume_def_or_macro_name - next_token - if @token.type == :"=" - name = "#{name}=" - next_token_skip_space - else - skip_space - end + name = "#{name}=" if consume_def_equals_sign_skip_space when :CLASS_VAR obj = ClassVar.new(@token.value.to_s) next_token_skip_space check :"." name = consume_def_or_macro_name - next_token - if @token.type == :"=" - name = "#{name}=" - next_token_skip_space - else - skip_space - end + name = "#{name}=" if consume_def_equals_sign_skip_space else unexpected_token end @@ -2990,12 +2961,7 @@ module Crystal raise "macro can't have a receiver" when :IDENT check_valid_def_name - next_token - if @token.type == :"=" - name += '=' - next_token - end - skip_space + name = "#{name}=" if consume_def_equals_sign_skip_space else check_valid_def_op_name next_token_skip_space @@ -3428,14 +3394,7 @@ module Crystal elsif @token.type == :IDENT check_valid_def_name name = @token.value.to_s - - next_token - if @token.type == :"=" - name = "#{name}=" - next_token_skip_space - else - skip_space - end + name = "#{name}=" if consume_def_equals_sign_skip_space else check_valid_def_op_name name = @token.type.to_s @@ -3462,13 +3421,7 @@ module Crystal name = @token.value.to_s name_location = @token.location - next_token - if @token.type == :"=" - name = "#{name}=" - next_token_skip_space - else - skip_space - end + name = "#{name}=" if consume_def_equals_sign_skip_space else check DefOrMacroCheck2 check_valid_def_op_name @@ -5964,6 +5917,17 @@ module Crystal @token.to_s end + def consume_def_equals_sign_skip_space + next_token + if @token.type == :"=" + next_token_skip_space + true + else + skip_space + false + end + end + def push_def @def_vars.push(Set(String).new) end