Skip to content
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

Add Access.values/0 #14350

Merged
merged 5 commits into from
Mar 24, 2025
Merged

Conversation

xxdavid
Copy link
Contributor

@xxdavid xxdavid commented Mar 20, 2025

Hi, I am adding Access.values/0 as discussed here, and also Access.keys/0 for parity with keys. If you think Access.keys/0 is not useful and should not be added, I can remove it from the PR.

I am not adding any explicit unit tests as I find the coverage by doctests enough, but I can write some if you wish so.

Thank you for reviewing the changes!

iex> get_in([1, 2, 3], [Access.values()])
** (RuntimeError) Access.values/0 expected a map, got: [1, 2, 3]
"""
@spec values() :: Access.access_fun(data :: map(), current_value :: list())
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
@spec values() :: Access.access_fun(data :: map(), current_value :: list())
@doc since: "1.19.0"
@spec values() :: Access.access_fun(data :: map(), current_value :: list())

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added in f7705f4. Thanks for noticing!

iex> get_in([1, 2, 3], [Access.keys()])
** (RuntimeError) Access.keys/0 expected a map, got: [1, 2, 3]
"""
@spec keys() :: Access.access_fun(data :: map(), current_value :: list())
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
@spec keys() :: Access.access_fun(data :: map(), current_value :: list())
@doc since: "1.19.0"
@spec keys() :: Access.access_fun(data :: map(), current_value :: list())

Copy link
Contributor

@sabiwara sabiwara left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM, thanks @xxdavid ! 💜

@josevalim
Copy link
Member

Let's not add keys for now but I'd suggest to make it so values works both with keyword lists and maps. WDYT?

@xxdavid
Copy link
Contributor Author

xxdavid commented Mar 20, 2025

Okay, I'll drop keys/0. Supporting also keyword lists sounds reasonable. I'll add it to the PR.

@xxdavid
Copy link
Contributor Author

xxdavid commented Mar 20, 2025

I added the support for keyword lists in the last commit. I was thinking about what to allow as a keyword list and I've come to a pragmatic decision to check whether the first element of a list is a 2-tuple and the first element of that tuple is an atom. If not, we'll raise with Access.values/0 expected a map or a keyword list, got: …. It may fail with FunctionClauseError if given for example [{:a, 1}, {:b, 2, :haha}] but I think that's acceptable as for example Keyword.filter/1 would fail as well.

I've also added unit tests as it got a bit more complex. And I've also found that some doctests for values/0 were relying on a particular or of keys in a map, which I fixed also in the last commit.

Comment on lines 1074 to 1079
Enum.map(data, fn {_key, value} -> next.(value) end)
end

defp values_keyword(:get_and_update, data, next) do
{reverse_gets, reverse_updated_data} =
Enum.reduce(data, {[], []}, fn {key, value}, {gets, data_acc} ->
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perhaps we should still have guards to be sure it's a keyword?

Suggested change
Enum.map(data, fn {_key, value} -> next.(value) end)
end
defp values_keyword(:get_and_update, data, next) do
{reverse_gets, reverse_updated_data} =
Enum.reduce(data, {[], []}, fn {key, value}, {gets, data_acc} ->
Enum.map(data, fn {key, value} when is_atom(key) -> next.(value) end)
end
defp values_keyword(:get_and_update, data, next) do
{reverse_gets, reverse_updated_data} =
Enum.reduce(data, {[], []}, fn {key, value}, {gets, data_acc} when is_atom(key) ->

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I looked at Keyword.filter/2 and Keyword.values/1 and they don't seem to be that strict. But why not. 🙂 Added in 8b4aa0f.

@xxdavid xxdavid changed the title Add Access.values/0 and Access.keys/0 Add Access.values/0 Mar 21, 2025
@josevalim josevalim merged commit e0c016c into elixir-lang:main Mar 24, 2025
10 checks passed
@josevalim
Copy link
Member

💚 💙 💜 💛 ❤️

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

3 participants