Maxximiliann

Maxximiliann

(FunctionClauseError) no function clause matching in anonymous fn/1

defmodule Sandbox do
  def currencies do
    [
      %{
        "USD" => %{
          asset: "USD",
          exchange_currencies: ["JPY"],
          priced_in: "KHR"
        }
      },
      %{
        "CVE" => %{
          asset: "CVE",
          exchange_currencies: ["JPY"],
          priced_in: "KHR"
        }
      },
      %{
        "CVE" => %{
          asset: "CVE",
          exchange_currencies: ["KHR", "JPY"],
          priced_in: "USD"
        }
      },
      %{
        "AUD" => %{
          asset: "AUD",
          exchange_currencies: ["JPY"],
          priced_in: "KHR"
        }
      },
      %{
        "AUD" => %{
          asset: "AUD",
          exchange_currencies: ["KHR", "USD", "JPY", "ZAR"],
          priced_in: "KRW"
        }
      },
      %{
        "AUD" => %{
          asset: "AUD",
          exchange_currencies: ["KHR", "JPY"],
          priced_in: "USD"
        }
      },
      %{
        "AUD" => %{
          asset: "AUD",
          exchange_currencies: ["KHR", "KRW", "LAK", "MGA", "XOF", "USD", "JPY", "AUD", "EUR"],
          priced_in: "ZAR"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["JPY"],
          priced_in: "KHR"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["KHR", "USD", "JPY", "ZAR"],
          priced_in: "KRW"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["JPY", "ZAR"],
          priced_in: "LAK"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["KHR", "USD", "ZAR"],
          priced_in: "MGA"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["KHR", "USD", "JPY", "ZAR"],
          priced_in: "XOF"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["KHR", "JPY"],
          priced_in: "USD"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["KHR", "KRW", "USD", "JPY", "ZAR"],
          priced_in: "AUD"
        }
      },
      %{
        "ZAR" => %{
          asset: "ZAR",
          exchange_currencies: ["LAK", "ZAR"],
          priced_in: "EUR"
        }
      },
      %{
        "MNT" => %{
          asset: "MNT",
          exchange_currencies: ["JPY"],
          priced_in: "KHR"
        }
      }
    ]
  end

  def foo do
    currencies = currencies()

    Enum.map(currencies, fn {_asset, record} -> record end)
  end

  def bar do
    data = %{"USD" => %{asset: "USD", exchange_currencies: ["JPY"], priced_in: "KHR"}}

    Enum.map(data, fn {_asset, record} -> record end)
  end
end
iex(1)> Sandbox.bar
[%{asset: "USD", exchange_currencies: ["JPY"], priced_in: "KHR"}]
iex(2)> Sandbox.foo
** (FunctionClauseError) no function clause matching in anonymous fn/1 in Sandbox.foo/0    
    
    The following arguments were given to anonymous fn/1 in Sandbox.foo/0:
    
        # 1
        %{"USD" => %{asset: "USD", exchange_currencies: ["JPY"], priced_in: "KHR"}}
    
    (arbit 0.1.0) lib/Sandbox.ex:122: anonymous fn/1 in Sandbox.foo/0
    (elixir 1.10.3) lib/enum.ex:1396: Enum."-map/2-lists^map/1-0-"/2

Why exactly does Sandbox.foo fail but Sandbox.bar succeed? How shouldSandbox.foo be corrected for it to succeed as well?

Marked As Solved

lpil

lpil

Creator of Gleam

foo’s calls Enum.map list of maps, but the anonymous function pattern matches on the argument as if it were a tuple. The tuple pattern doesn’t match the map, so it crashes.

You’ll need to remove the tuple pattern for this not to crash.

  def foo do
    currencies = currencies()

    Enum.map(currencies, fn map -> do_something_with(map) end)
  end

Also Liked

al2o3cr

al2o3cr

currencies returns an unusual shape - a list of maps, each with exactly one key. I’d typically expect either a map of currency name → some value, or a list of tuples ready for List.keyfind if the order is important.

BUT

I also see that the values in currencies don’t have unique keys ("AUD" appears more than once, for instance). What’s the intent of these separate records? That might suggest a better data structure.

lpil

lpil

Creator of Gleam

It’s fairly idiomatic and relatively efficient, but you might be surprised by the behaviour.

Maps are not ordered so the one you get from List.first will be semi-random depending on how many keys are in the map. If this is ok (i.e. if the map only has one key) then that is not a problem.

al2o3cr

al2o3cr

Nothing special here. You could use atoms for currency names, but that makes things more complicated when interacting with other systems.

all_currencies = ["AUD", "CVE", "USD"]

I’ve bolded the key words that suggest the data structure, a list of maps:

currencies = [
    %{name: "USD", exchange_rate: 1.23, value: 0.87},
    %{name: "CVE", exchange_rate: 1.75, value: 0.75},
    %{name: "AUD", exchange_rate: 1.01, value: 0.99}
  ]

The numbers here are written as floats, but that may not be what you want. Consider the Decimal library for doing financial arithmetic.

“Record” could also be satisfied with a defstruct (or an Erlang record if you’re feeling exotic).

Looking up a record is spelled Enum.find(currencies, & &1.name == "USD"). If the code does this a lot, consider making a map out of currencies:

currencies_map = Map.new(currencies, fn v -> {v.name, v} end)
  exchange_rates = %{
    "AUD" => %{"USD" => 1.14, "CVN" => 1.2},
    # OR, if an exchange rate is more complicated than just a number
    "USD" => %{"CVN" => %{rate: 0.67, quote_id: 1234}, "AUD" => %{rate: 1.06, quote_id: 4567}}
  }

The benefit of using nested maps here is that looking up an exchange rate is spelled exchange_rates[source_currency][destination_currency].

If there is more than one exchange rate for a given currency pair, this approach will not work. Consider using lists for that case.

exchanged_values =
  exchange_rates
  |> Map.new(fn {src_currency, dest_currencies} ->
    {
      src_currency,
      Map.new(dest_currencies, fn {dest_currency, exchange_rate} ->
        # do something with src_currency, dst_currency, and exchange_rate
        # probably look things up in currencies
        result = %{
          name: dst_currency, # or "#{src_currency} -> #{dst_currency}"
          exchange_rate: ... # calculate this
          value: ... # and this
        }

        {dest_currency, result}
      end
    }
  end

This results in a new map of maps; exchanged_values[source_currency][dest_currency] is a map just like the ones in currencies.

That last part about adding initial and final values to the same list seems tricky; how do consumers of R know how to interpret the {name: ..., exchange_rate:..., value:...} results?

One option would be to have R contain tuples:

exchanged_values
|> Enum.flat_map(fn {src_currency, dest_currencies} ->
  dest_currencies
  |> Enum.map(fn {dest_currency, final_value} ->
    # get initial_value for dest_currency from currencies
    {initial_value, final_value}
  end)
  |> Enum.filter(fn {initial_value, final_value} ->
    # decide if record should be included or not
  end)
end)

This uses Enum.flat_map for two reasons:

  • we’re building a list, but a single key of the input may return many values
  • some keys in the input may return NO values

Where Next?

Popular in Questions Top

sen
Hi All, I set a environment variables in dev.exs , like below code. when i start server, how can i set the ${enable} value? thanks. d...
New
qwerescape
Is there a way to get the call stack or stack trace at any point in the code? Not from exceptions, but an expression that returns how the...
New
chrisalley
ExUnit now has describe blocks which is a welcome addition coming from RSpec. In the docs, it states that nested hierarchies of describe ...
New
greenz1
I have a phoenix application from which a user can download multiple(5-6) files of size 1MB. I couldn’t find anything related to sending ...
New
electic
Hi, I am new to Elixir. I am trying to use the DateTime component to insert a date into MySQL however the there seems to be no way to fo...
New
jononomo
I am trying to figure out how Mix knows whether the environment is test, dev, or prod – where is this set? Thanks.
New
vac
Hi, I’m quite new in Elixir and I’m trying to format a string to a PEM format. I have the certificate value like MIIDBTCCAe2...... and I...
New
nobody
How to bind a phoenix app to a specific ip address? could not find anything about that, nowhere, unfortunately, but for me this is quite...
New
jay1
Why is it that the mnesia database isn’t the most preferred database for use in Elixir/Phoenix?
New
marick
I had some trouble figuring out how to make many-to-many associations work. Once I got it working, I wrote a blog post. Because I’m a nov...
New

Other popular topics Top

sen
Hi All, I set a environment variables in dev.exs , like below code. when i start server, how can i set the ${enable} value? thanks. d...
New
danschultzer
None of the current solutions worked well for me, so I went ahead and built a user management system from scratch. This project took far...
548 29377 241
New
vertexbuffer
Hello, can anybody help here..? I have a list of players and I what to delete an element, but every for loop the list is reverting to ori...
New
AstonJ
Posting this to see if we can make things easier for people to get into Neovim. If you use Neovim and have a favourite distro please let ...
New
shahryarjb
Hello, I have map which I want to convert it to string like this: the map: %{last_name: "tavakkoli", name: "shahryar"} the string I ne...
New
minhajuddin
I have seen a lot of code which picks the first element from a list using Enum.at(0) instead of List.first. Is there a reason why people ...
New
msaraiva
Surface is an experimental library built on top of Phoenix LiveView and its new LiveComponent API that aims to provide a more declarative...
564 43622 214
New
JorisKok
I have a server on AWS, and was running a load test using artillery. When looking at the Phoenix dashboard I see the Ports going to 100% ...
New
freewebwithme
Using vs code and installed ElixirLS: support and debugger. And I got an error popped up on start up says Failed to run ‘elixir’ comma...
New
axelson
This post is a wiki (feel free to hit the edit button near the bottom right of this post to add your own changes!) This post collects co...
239 47930 226
New

We're in Beta

About us Mission Statement