[前][次][番号順一覧][スレッド一覧]

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/

[前][次][番号順一覧][スレッド一覧]