From: harrison.bachrach@...
Date: 2019-12-29T18:15:24+00:00
Subject: [ruby-core:96577] [Ruby master Feature#16460] External names for keyword parameters in method definitions

Issue #16460 has been updated by harrisonb (Harrison Bachrach).


zverok (Victor Shepelev) wrote:
> You propose to introduce the "same" construct, which means something
> completely different, but due to our intutions, **in Ruby** `external_name
> internal_name:` reads as `external_name(internal_name:)`, which is misleading.

I think at the core, I am suggesting introducing something _new_. While there
is certainly merit to discussion of what is the most "Ruby-like" way to
implement that, I don't think it should hamstring the entire conversation
surrounding the proposal. The new `case...in` feature addition in 2.7 is a
fairly radical departure from the previous semantics of Ruby:

* It performs a name binding outside of all the usual places (assignment,
  parameter list, block parameter list, `rescue` block, etc.)
* That binding is set up by a pattern (which often looks very much like an
  array/hash/etc. literal) which would previously never preform a binding

While the pattern matching feature requires one to read up about it, so do other
later added features that provide utility and expressiveness to Ruby developers.

> 1. So, it should be read this way: `(within range):`, where both words are
>    related to colon, with one being "title" and the second "explanation"?
>    This, again unlike anything else in Ruby. It slightly reminds me of some
>    documentation system links (yard, probably? with "Look also
>    [FormalClassName that class]", or something like that... and I honestly
>    never can remember which is which -- which is "link" and which is "title").

The main constant is that the identifier with the colon (`range` in the example
above) remains the one that can be referenced in the body of the method.

I am open to the arrow syntax if we conclude that it is still unclear and/or
sufficiently alien.

> 2. It works well for _some examples you constructed_, but there always can be
>    counter-examples, like: I want to use `when:` argument, renaming it to
>    `time` (becase `when` is a keyword), so... is it `time when:` or `when
>    time:`?..

I would point to the same fact above about the identifier with the colon
remaining the only valid one in the method body. Additionally, this is change
for the sake of improving readability. One can probably come up with confusing
examples, but the proposed syntax, like much of Ruby syntax, is a sharp knife
that one can use to make their code more or less readable.

> 3. It is incredibly confusing for any parser (especially "rename as it is
>    keyword" use-case) and syntax highlighter:`def foo(in time:` would be the
>    only case where standalone `in<space><something>` should be
>    parsed/hihglighted differently.

I think this could be a valid criticism of the arrow-less syntax. However,
parsers for at least two other languages accomplish this without issue, so I
don't think it's an insurmountable issue.

> Some additional point: currently keyword args definitions reads this way:
> `method(arg1: ..., arg2: ...)` -- is how it would be called (e.g. in call-site
> we'll see the same structure `method(arg1: ..., arg2: ...)` -- and everything
> after the `:` is somewhat "how it is implemented" (for example, defaults
> calculation), which `=> put_to_this_variable` follows.

I think this is a very good point and perhaps the most compelling one for the
`external_name: default_value => internal_name` syntax.

> And one more consideration: imagine Ruby introduced one of those syntaxes in
> 2.8. And some rubyist who missed the announcement, comes to a "new" codebase,
> and how they would understand it?
>
> ```ruby
> def foo(in: => time)
> ```
> "...Ugh, what is it?.. Putting something into `time`? Ah, `in` is a keyword, they want to rename it. Got it. Hate it, but got it."
> ```ruby
> def foo(in time: )
> ```
> "...Ugh what?.. What?.. Is it a new keyword?.. Some kind of type hinting?.. No idea..."
> 
> (Pure speculations, obv.)

I am going to assume good faith here in some of the phrasing. With any syntax
introduction, there is going to have to be some learning. The `case...in` syntax
also requires a similar (if not greater) amount of learning, but that was also
deemed acceptable.


----------------------------------------
Feature #16460: External names for keyword parameters in method definitions
https://bugs.ruby-lang.org/issues/16460#change-83542

* Author: harrisonb (Harrison Bachrach)
* Status: Open
* Priority: Normal
* Assignee: 
* Target version: 
----------------------------------------
Hello! This is my first time filing an issue and I was unable to find anything similar. I apologize if one already exists.

In other languages (such as JavaScript, Swift, and Crystal), it is possible to have two names for a given keyword argument: one that is used in method invocation, and one used in the method definition. Here is an example from Crystal (which has syntax very similar to Ruby):

```crystal
def increment(value, by amount)
  value + amount
end

increment(value: 5, by: 10)
```

This helps create more readable method invocations and definitions. It would be especially helpful in Ruby as the language lacks a destructuring syntax for hashes/keyword args. This is unlike JavaScript, where you can do something like:

```javascript
const { nameOfOneProperty: newNameForTheProperty, nameOfAnotherProperty: newNameForTheOtherProperty } = foo;
```

where `foo` is a JavaScript Object that has the properties `nameOfOneProperty` or `nameOfAnotherProperty` (If it did not have either of them, the corresponding identifiers (newNameForTheProperty and newNameForTheOtherProperty would be initialized to undefined).

I'm thinking that such a change would pair nicely with the new 3.0 keyword argument changes. A suggested syntax might be

```ruby
def name(external_name internal_name:)
  # ...
end




-- 
https://bugs.ruby-lang.org/

Unsubscribe: <mailto:ruby-core-request@ruby-lang.org?subject=unsubscribe>
<http://lists.ruby-lang.org/cgi-bin/mailman/options/ruby-core>