Hello,
I am looking to more closely understand the advantages of the Repo.transact
pattern and the quoted disadvantages of Ecto.Multi
.
Repo.transact
is quickly described in this blog post.
(Link to the original article by Sasa Juric is included in the post above.)
Here are four (4) points from the article, and my comments. I am quoting @sasajuric and @tomkonidas.
This function commits the transaction if the lambda returns
{:ok, result}
, rolling it back if the lambda returns{:error, reason}
. In both cases, the function returns the result of the lambda.
Does Ecto.Multi
ultimately not do that as well? Sure it’s not a direct response; in case of errors you get a tuple containing various state, including “changes so far”. I’d think that the rich choices offered by Ecto.Multi
are welcome – you get to decide for yourself how do you want to react to problems. How is the Repo.transact
feature the clear winner here?
One thing I can see is if we unify our error-handling code by utilizing shared helpers + including the Repo.transact
in an even higher-level wrappers (business code). Is that the selling point – less boilerplate?
We chose this approach over Ecto.Multi, because we’ve experimentally established that multi adds a lot of noise with no real benefits for our needs.
I am very curious about this empirical evidence; I think having it spelled out somewhere would be hugely valuable both for new learners and more long-term users like myself. I can only imagine it’s the proposition that the else
clauses of the with
statement are hard to maintain? Or having to know the changeset functions (which I don’t view to be as negative thing as the blog post author seems to imply)?
As we can see, it is not the worst, but once we see the
Repo.transact/2
way, it will be clear which is better.
It’s clear only insofar as the code is (1) shorter and definitely easier to read, and (2) has no else
clause(s). Is that all? I am all for shorter and more readable code, I am almost religious about it too, but not all teams are open to code modifications on that basis alone. I am looking to understand if there’s more to this beyond code readability.
Perhaps the error-handling utilities we can combine with Repo.transact
are the true value proposition here?
Another big benefit is that we do not need to go down to the changeset level for inserting, we could use our functions that perform
Repo.insert
s in them (Accounts.new_user_changeset/1
vsAccounts.create_user/1
). This lets us compose many functions together from outside the context modules without having the need to expose your changeset functions.
Hmmm. I’ve been in 3 big-ish Elixir codebases (we’re talking 1000 - 3000 files) and I have never stumbled upon a problem that we the team would describe as “modules outside the Phoenix contexts have access to changeset functions and that is a problem”. I mean they are public; being able to call them is always on the table, this is not Java / C# / Rust et. al. where you can make functions (package|namespace)-private so you can limit who can actually call them.
A notable mention here is the boundaries
library, made by Sasa Juric as well: Boundary - enforcing boundaries in Elixir projects. I’ve used it with success and I do like it but sadly I never managed to convince too many people of its value.
I am not sure I see the clear win here. Help me understand.
Maybe the actual root problem is that I worked as a contractor for several years and had to be very flexible – meaning that things that many of us as programmers would immediately agree on became actual contention points when working with different teams and CTOs. Or maybe I am just a bad developer advocate (would not surprise me ).
My summary of the advantages of Repo.transact
would be (1) less boilerplate and (2) better separation of concerns. I respect and even worship both but I’ve met plenty of people who don’t so I am curious how would I sell them such a coding pattern better and be more convincing.
In any case: I am very curious as to why is the Repo.transact
pattern deemed valuable (or not) by others. What other factors did I miss? Could you share your thoughts, please? Thank you.