This example overrides Kernel.def/2, so I am aware it is only suitable as an experiment. However here are the results.
Given the functions fetch_user and fetch_cart that both return {:ok, value} or {:error, reason}.
defmodule MyApp do
use OK.Kernel
def checkout(user_id, cart_id) do
user <- fetch_user(user_id) # `<-` will bind user when fetch_user returns {:ok, user}
cart <- fetch_cart(cart_id) # `<-` will shortcut to else clause if returned {:error, reason}
order = checkout(cart, user) # Lines without `<-` behave normally
order.invoice_id
else
:user_not_found ->
IO.puts("No user for user_id: #{user_id}")
nil
:cart_not_found ->
IO.puts("User has no cart")
nil
end
end
By using OK.Kernel errors can be grouped in what I consider to be a very natural way.
I guess it’s natural because it looks like how I would arrange raise and catch in previous languages. However with the magic of Elixir macros I can do it all without using exception raising.
To try it yourself add the release candidate to your mix.exs
{:ok, "~> 1.7.0-rc.1"}
Comments welcome.
My motivation for trying this was to separate unhappy paths due to code errors ( I still raise exceptions for them) and the unhappy path due to bad input.
I suggest to change else to something … else
How about: catch_error?
and also maybe: <- to for example: <=
Maybe also do not override def and defp and add edef and edefp where: e <= extended
Imagine:
defmodule Example do
use OK.Kernel
def sample do
result <- do_something()
string_results = for something <- result, do: to_string(something)
result2 <- do_something_else(string_results)
if result2, do: :success
else
:an_error ->
IO.puts "An error occurred"
nil
end
# retunrs :success or nil
end
Compare it to:
defmodule Example do
use OK.Kernel
edef sample do
result <= do_something()
string_results = for something <- result, do: to_string(something)
result2 <= do_something_else(string_results)
if result2, do: :success
catch_error
:an_error ->
IO.puts "An error occurred"
nil
end
# retunrs :success or nil
end
Personally, I think this leads to hard to reason about code due to the loss of locality between the function body and the else that may or may not be triggered. It’s the same reason goto is “considered harmful” (caveats and exceptions to that aside )
This feels like a place where with would be a good solution. A bit of syntatic sugar around it to tease the behavior of with to automatically succeed on {:ok, anything} and fail on {:error, anything} would be nicer imho, and iirc someone recently did exactly that.
Also, I find that if I have long(er) functions where there are conditions somewhere in the middle that need to branch, such as on failure or success, that this is a signal that perhaps I should break that function there and send processing on to another function which will continue on success or return on failure.
IOW, each function goes as far as it can until need to react to a success/failure and at that point calls another function. So each function ends up being a bit-sized bit of code that does execute completely, allowing each function to be reasoned about simply and clearly.
Exceptions (or, what they really are: early returns) are a tool of “last resort” for me, used when they are really required only. Similarly for case/with/etc. If those start nesting, or a function ends up being a series of them, I take that as a code smell to start chopping that function up into more atomic sets of code to stuff into functions.
@Crowdhailer:
2. oh, I see … How about store method name in atom instead of code block?
defmodule MyModule do
use OK.Kernel
def sample(arg1, arg2) catch_with :fun_name do
# block
end
defp fun_name(error), do: # ...
Is it (or something similar possible?
2. and 3. because new Elixir developer could be confused when see that some operators do multiple works
I added it in example codes. Compare for item <- items ... with item <- get_items() and that part when if is directly above else - as a … not new developer I don’t have a problem with it, but I can see that it could be confused for new users.
4. yes, I saw that - btw. good job