Feature #16182
closedShould `expr in a, b, c` be allowed or not?
Description
In #15865, a new syntax <expr> in <pattern>
was introduced. By using this, we can write:
json = { foo: 1, bar: 2}
if json in { foo:, bar: }
p [foo, bar] #=> [1, 2]
end
However, we cannot write:
p(json in { foo:, bar: }) #=> expected: true, actual: syntax error
This is because <expr> in <pattern>
is an expression but not an argument. For example, foo(json in a, b, c)
is ambiguous: it is considered foo((json in a), b, c)
and foo((json in a, b, c))
.
What should we do?
- Do nothing; we admit that it is a spec
- Revert the feature
- Disallow a pattern like
a, b, c
ora:, b:, c:
in this one-line pattern matching syntax; we ask a user to writejson in [a, b, c]
orjson in {a:, b:, c:}
Updated by shevegen (Robert A. Heiler) over 5 years ago
I can not comment/answer on the issue and questions; I think this is for
matz and the core team to decide either way, whatever the way.
I did, however had, want to add that:
json in {a:, b:, c:}
is quite difficult to read (for me). So even if this may not be an ideal explanation,
but ... I would not be at all opposed to disallowing that, merely syntax-wise
alone. ;-)
(I do not really have a big opinion on the functionality in general but ideally
my personal taste is to prefer simpler syntax, whenever that is possible. We have
in general quite some suggestions that combine a lot of complex syntax together,
which I think is not ideal, in general; also in other proposals.)
Updated by baweaver (Brandon Weaver) over 5 years ago
I wonder if it would make sense to reverse this to be left-to-right (LTR) rather than right-to-left (RTL) to make it easier to parse.
I cannot think of another RTL syntax in Ruby at the moment, including the current for ... in
statement:
for item in collection
A full example might be:
for a, b in { a: 1, b: 2 }
p a, b
end
:a
1
:b
2
=> {:a=>1, :b=>2}
Of course this does not currently work with keyword arguments:
[2] pry(main)> for a: 1, b: 2 in [{ a: 1 }, { b: 2 }]
SyntaxError: unexpected ':', expecting '.' or &. or :: or '['
for a: 1, b: 2 in [{ a: 1 }, { b: 2...
^
[2] pry(main)> for a:, b: in [{ a: 1 }, { b: 2 }]
SyntaxError: unexpected tSYMBEG, expecting do or '{' or '('
for a:, b: in [{ a: 1 }, { b: 2 }]
What if we leveraged some of the current logic for parsing a for ... in
statement to make single-line pattern matching into a LTR syntax? This may be a solution for the parsing difficulties, as well as build on the intuition of Ruby developers expecting LTR syntaxes naturally.
Updated by matz (Yukihiro Matsumoto) over 5 years ago
I vote for 3 in the OP.
Matz.
Updated by ktsj (Kazuki Tsujimoto) about 5 years ago
- Status changed from Open to Closed
Applied in changeset git|6e70fa49b111e2a2839297b057a3df354cae265a.
Disallow omission of parentheses/brackets in single line pattern matching [Feature #16182]
Updated by ktsj (Kazuki Tsujimoto) about 5 years ago
- Status changed from Closed to Open
Updated by ktsj (Kazuki Tsujimoto) about 5 years ago
- Related to Feature #15865: `<expr> in <pattern>` expression added
Updated by ktsj (Kazuki Tsujimoto) over 3 years ago
How about allowing brackets/braces to be omitted in one-line pattern matching?
Now that we use =>
in one-line pattern matching, that syntax can't be made into an argument whether we allow omission or not.
Updated by matz (Yukihiro Matsumoto) over 3 years ago
I agree with allowing to omit parentheses in the pattern.
Matz.
Updated by ktsj (Kazuki Tsujimoto) over 3 years ago
- Status changed from Open to Closed
Applied in changeset git|ecb6d6a4ef058b5598a7633c3921eeab08ce11c6.
Allow omission of parentheses in one line pattern matching [Feature #16182]