ruby-changes:70667
From: Jeremy <ko1@a...>
Date: Fri, 31 Dec 2021 07:37:55 +0900 (JST)
Subject: [ruby-changes:70667] f53dfab95c (master): Add support for anonymous rest and keyword rest argument forwarding
https://git.ruby-lang.org/ruby.git/commit/?id=f53dfab95c From f53dfab95c30e222f67e610234f63d3e9189234d Mon Sep 17 00:00:00 2001 From: Jeremy Evans <code@j...> Date: Fri, 19 Nov 2021 09:38:22 -0800 Subject: Add support for anonymous rest and keyword rest argument forwarding This allows for the following syntax: ```ruby def foo(*) bar(*) end def baz(**) quux(**) end ``` This is a natural addition after the introduction of anonymous block forwarding. Anonymous rest and keyword rest arguments were already supported in method parameters, this just allows them to be used as arguments to other methods. The same advantages of anonymous block forwarding apply to rest and keyword rest argument forwarding. This has some minor changes to #parameters output. Now, instead of `[:rest], [:keyrest]`, you get `[:rest, :*], [:keyrest, :**]`. These were already used for `...` forwarding, so I think it makes it more consistent to include them in other cases. If we want to use `[:rest], [:keyrest]` in both cases, that is also possible. I don't think the previous behavior of `[:rest], [:keyrest]` in the non-... case and `[:rest, :*], [:keyrest, :**]` in the ... case makes sense, but if we did want that behavior, we'll have to make more substantial changes, such as using a different ID in the ... forwarding case. Implements [Feature #18351] --- NEWS.md | 15 ++++++++++++ doc/syntax/methods.rdoc | 14 ++++++++++++ parse.y | 39 ++++++++++++++++++++++++++++---- proc.c | 16 +++++++++++-- spec/ruby/core/method/parameters_spec.rb | 15 +++++++++--- spec/ruby/core/proc/parameters_spec.rb | 12 ++++++++-- test/ruby/test_iseq.rb | 14 +++++++++--- test/ruby/test_method.rb | 12 +++++----- test/ruby/test_proc.rb | 2 +- test/ruby/test_syntax.rb | 28 +++++++++++++++++++++++ 10 files changed, 146 insertions(+), 21 deletions(-) diff --git a/NEWS.md b/NEWS.md index 69f57d5997f..06696129a58 100644 --- a/NEWS.md +++ b/NEWS.md @@ -7,6 +7,19 @@ Note that each entry is kept to a minimum, see links for details. https://github.com/ruby/ruby/blob/trunk/NEWS.md#L7 ## Language changes +* Anonymous rest and keyword rest arguments can now be passed as + arguments, instead of just used in method parameters. + [[Feature #18351]] + + ```ruby + def foo(*) + bar(*) + end + def baz(**) + quux(**) + end + ``` + ## Command line options ## Core classes updates @@ -52,3 +65,5 @@ Note: Excluding feature bug fixes. https://github.com/ruby/ruby/blob/trunk/NEWS.md#L65 ## IRB Autocomplete and Document Display ## Miscellaneous changes + +[Feature #18351]: https://bugs.ruby-lang.org/issues/18351 diff --git a/doc/syntax/methods.rdoc b/doc/syntax/methods.rdoc index 2bb350def18..8dafa6bb0cf 100644 --- a/doc/syntax/methods.rdoc +++ b/doc/syntax/methods.rdoc @@ -441,6 +441,13 @@ Also, note that a bare <code>*</code> can be used to ignore arguments: https://github.com/ruby/ruby/blob/trunk/doc/syntax/methods.rdoc#L441 def ignore_arguments(*) end +You can also use a bare <code>*</code> when calling a method to pass the +arguments directly to another method: + + def delegate_arguments(*) + other_method(*) + end + === Keyword Arguments Keyword arguments are similar to positional arguments with default values: @@ -481,6 +488,13 @@ Also, note that <code>**</code> can be used to ignore keyword arguments: https://github.com/ruby/ruby/blob/trunk/doc/syntax/methods.rdoc#L488 def ignore_keywords(**) end +You can also use <code>**</code> when calling a method to delegate +keyword arguments to another method: + + def delegate_keywords(**) + other_method(**) + end + To mark a method as accepting keywords, but not actually accepting keywords, you can use the <code>**nil</code>: diff --git a/parse.y b/parse.y index 7555d0db166..794c818e349 100644 --- a/parse.y +++ b/parse.y @@ -427,6 +427,8 @@ static void token_info_drop(struct parser_params *p, const char *token, rb_code_ https://github.com/ruby/ruby/blob/trunk/parse.y#L427 #define lambda_beginning_p() (p->lex.lpar_beg == p->lex.paren_nest) #define ANON_BLOCK_ID '&' +#define ANON_REST_ID '*' +#define ANON_KEYWORD_REST_ID idPow static enum yytokentype yylex(YYSTYPE*, YYLTYPE*, struct parser_params*); @@ -2890,6 +2892,16 @@ args : arg_value https://github.com/ruby/ruby/blob/trunk/parse.y#L2892 /*% %*/ /*% ripper: args_add_star!(args_new!, $2) %*/ } + | tSTAR + { + /*%%%*/ + if (!local_id(p, ANON_REST_ID)) { + compile_error(p, "no anonymous rest parameter"); + } + $$ = NEW_SPLAT(NEW_LVAR(ANON_REST_ID, &@1), &@$); + /*% %*/ + /*% ripper: args_add_star!(args_new!, Qnil) %*/ + } | args ',' arg_value { /*%%%*/ @@ -2904,6 +2916,16 @@ args : arg_value https://github.com/ruby/ruby/blob/trunk/parse.y#L2916 /*% %*/ /*% ripper: args_add_star!($1, $4) %*/ } + | args ',' tSTAR + { + /*%%%*/ + if (!local_id(p, ANON_REST_ID)) { + compile_error(p, "no anonymous rest parameter"); + } + $$ = rest_arg_append(p, $1, NEW_LVAR(ANON_REST_ID, &@3), &@$); + /*% %*/ + /*% ripper: args_add_star!($1, Qnil) %*/ + } ; /* value */ @@ -5479,8 +5501,7 @@ f_kwrest : kwrest_mark tIDENTIFIER https://github.com/ruby/ruby/blob/trunk/parse.y#L5501 | kwrest_mark { /*%%%*/ - $$ = internal_id(p); - arg_var(p, $$); + arg_var(p, shadowing_lvar(p, get_id(ANON_KEYWORD_REST_ID))); /*% %*/ /*% ripper: kwrest_param!(Qnil) %*/ } @@ -5555,8 +5576,7 @@ f_rest_arg : restarg_mark tIDENTIFIER https://github.com/ruby/ruby/blob/trunk/parse.y#L5576 | restarg_mark { /*%%%*/ - $$ = internal_id(p); - arg_var(p, $$); + arg_var(p, shadowing_lvar(p, get_id(ANON_REST_ID))); /*% %*/ /*% ripper: rest_param!(Qnil) %*/ } @@ -5710,6 +5730,17 @@ assoc : arg_value tASSOC arg_value https://github.com/ruby/ruby/blob/trunk/parse.y#L5730 /*% %*/ /*% ripper: assoc_splat!($2) %*/ } + | tDSTAR + { + /*%%%*/ + if (!local_id(p, ANON_KEYWORD_REST_ID)) { + compile_error(p, "no anonymous keyword rest parameter"); + } + $$ = list_append(p, NEW_LIST(0, &@$), + NEW_LVAR(ANON_KEYWORD_REST_ID, &@$)); + /*% %*/ + /*% ripper: assoc_splat!(Qnil) %*/ + } ; operation : tIDENTIFIER diff --git a/proc.c b/proc.c index d075b7382e6..6cdd7bd836b 100644 --- a/proc.c +++ b/proc.c @@ -3124,6 +3124,16 @@ method_inspect(VALUE method) https://github.com/ruby/ruby/blob/trunk/proc.c#L3124 rb_str_buf_cat2(str, "("); + if (RARRAY_LEN(params) == 3 && + RARRAY_AREF(RARRAY_AREF(params, 0), 0) == rest && + RARRAY_AREF(RARRAY_AREF(params, 0), 1) == ID2SYM('*') && + RARRAY_AREF(RARRAY_AREF(params, 1), 0) == keyrest && + RARRAY_AREF(RARRAY_AREF(params, 1), 1) == ID2SYM(idPow) && + RARRAY_AREF(RARRAY_AREF(params, 2), 0) == block && + RARRAY_AREF(RARRAY_AREF(params, 2), 1) == ID2SYM('&')) { + forwarding = 1; + } + for (int i = 0; i < RARRAY_LEN(params); i++) { pair = RARRAY_AREF(params, i); kind = RARRAY_AREF(pair, 0); @@ -3159,8 +3169,7 @@ method_inspect(VALUE method) https://github.com/ruby/ruby/blob/trunk/proc.c#L3169 } else if (kind == rest) { if (name == ID2SYM('*')) { - forwarding = 1; - rb_str_cat_cstr(str, "..."); + rb_str_cat_cstr(str, forwarding ? "..." : "*"); } else { rb_str_catf(str, "*%"PRIsVALUE, name); @@ -3173,6 +3182,9 @@ method_inspect(VALUE method) https://github.com/ruby/ruby/blob/trunk/proc.c#L3182 else if (i > 0) { rb_str_set_len(str, RSTRING_LEN(str) - 2); } + else { + rb_str_cat_cstr(str, "**"); + } } else if (kind == block) { if (name == ID2SYM('&')) { diff --git a/spec/ruby/core/method/parameters_spec.rb b/spec/ruby/core/method/parameters_spec.rb index 3fdaf9ce6f5..67b42afafa6 100644 --- a/spec/ruby/core/method/parameters_spec.rb +++ b/spec/ruby/core/method/parameters_spec.rb @@ -222,9 +222,18 @@ describe "Method#parameters" do https://github.com/ruby/ruby/blob/trunk/spec/ruby/core/method/parameters_spec.rb#L222 m.method(:handled_via_method_missing).parameters.should == [[:rest]] end - it "adds nameless rest arg for \"star\" argument" do - m = MethodSpecs::Methods.new - m.method(:one_unnamed_splat).parameters.should == [[:rest]] + ruby_version_is '3.1' do + it "adds * rest arg for \"star\" argument" do + m = MethodSpecs::Methods.new + m.method(:one_unnamed_splat).parameters.should == [[:rest, :*]] + end + end + + ruby_version_is ''...'3.1' do + it "adds nameless rest arg for \"star\" argument" do + m = MethodSpecs::Methods.new + m.method(:one_unnamed_splat).parameters.should == [[:rest]] + end end it "returns the args and block for a splat and block argument" do diff --git a/spec/ruby/core/proc/parameters_spec.rb b/spec/ruby/core/proc/parameters_spec.rb index 5fb5cf418dc..60620722f81 100644 --- a/spec/ruby/core/proc/parameters_spec.rb +++ b/spec/ruby/core/proc/parameters_spec.rb @@ -80,8 +80,16 @@ describe "Proc#parameters" do https://github.com/ruby/ruby/blob/trunk/spec/ruby/core/proc/parameters_spec.rb#L80 -> x {}.parameters.should == [[:re (... truncated) -- ML: ruby-changes@q... Info: http://www.atdot.net/~ko1/quickml/