-
Notifications
You must be signed in to change notification settings - Fork 3.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Prevent crash on zero arity def in protocol #14191
Conversation
lib/elixir/lib/protocol.ex
Outdated
@@ -666,7 +666,7 @@ defmodule Protocol do | |||
end | |||
|
|||
new_signatures = | |||
for {{fun, arity}, :def, _, _} <- definitions do | |||
for {{fun, arity}, :def, _, _} when arity > 0 <- definitions do |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Unfortunately this is not enough to fix it. If people are defining custom functions, then we need to do this modification only for protocol functions. So we need to store somewhere, either in a module attribute or perhaps in the function metadata, the protocol functions and traverse only them here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe we can replace this line:
elixir/lib/elixir/lib/protocol.ex
Line 292 in 7a1a3ab
Kernel.def(unquote(name)(unquote_splicing(args))) |
By:
Kernel.def(unquote({name, [protocol: true], args}))
And see if [protocol: true]
appears in the metadata here.
Another option, which perhaps I like most, is to traverse the implementation of the __protocol__
and find the return for the :functions
clause:
elixir/lib/elixir/lib/protocol.ex
Line 974 in 7a1a3ab
Kernel.def(__protocol__(:functions), do: unquote(:lists.sort(@__functions__))) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
And then we need to add a test here that the signature for the user generated function does not have a :strong
triplet:
test "defines signatures without fallback to any" do |
It should most likely be an :infer
tuple (you can match on {:infer, _, _}
).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Another option, which perhaps I like most, is to traverse the implementation of the protocol and find the return for the :functions clause:
I like it, seems straightforward. This is what you mean right? 609f7cd
And then we need to add a test here that the signature for the user generated function does not have a :strong triplet:
Will have a look!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
lib/elixir/lib/protocol.ex
Outdated
@@ -665,10 +665,13 @@ defmodule Protocol do | |||
end | |||
end | |||
|
|||
fun_arities = :sets.from_list(protocol.__protocol__(:functions), version: 2) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The set might be overkill since most protocols should define very few functions, so perhaps just a list is fine?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Unfortunately we cannot load the code, that would add a bunch of unnecessary work, given we would have to purge it immediately after consolidation anyway. What I proposed is to find the {:__protocol__, 1}
entry inside definitions
, then find the :functions
clauses in there, and get its return type, which should be a keyword list. :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Got it! 6ff5011
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks great, we just need updated tests and we can ship it!
Co-authored-by: José Valim <[email protected]>
defprotocol Protocol.ConsolidationTest.ExtraDef do | ||
def protocol_fun(term) | ||
|
||
Kernel.def(regular_fun(term), do: term + 1) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Or I can add it to NoImpl
if you prefer to avoid fixture proliferation?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, let's add it to to an existing one with a comment on top.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done 9adad46
aaa0100
to
9adad46
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ship it!!!!
Co-authored-by: José Valim <[email protected]>
We can feel the excitement in the commits 😄 1763865.
I tried
main
on a project and got a compiler crash due to this dependency which defines regular functions inside a protocol, basically:The lib should probably be fixed and we should probably not allow this in the first place (same spirit as #14158), but right now this crashes the compiler on consolidation: (
fun: :pack_nan, arity: 0
)I tried to add a test but consolidation tests are a bit high ceremony and I wasn't sure it was worth defining a fixture etc. especially if this is going to be forbidden.