Hi everyone, it seems I can’t get out of this forum this week (you guys are awesome btw).
So, I got surprised by this behavior while using Ecto and I’m still banging my head against the wall with some conflicting information.
A coworker opened this issue: Updating a record in a Repo transaction failure · Issue #3944 · elixir-ecto/ecto · GitHub yesterday where he describes something unexpected we found with how transactions work.
The context of the whole thing is in the issue, but to summarize… The docs about transaction/2
say:
If an unhandled error occurs the transaction will be rolled back and the error will bubble up from the transaction function. If no error occurred the transaction will be committed when the function returns. A transaction can be explicitly rolled back by calling
rollback/1
, this will immediately leave the function and return the value given torollback
as{:error, value}
This is what we are trying to do… We always have a “tag” persisted after trying to save a “post”. If we are unable to persist the “tag”, we also don’t care about saving the “post”:
Repo.transaction(fn repo ->
case repo.insert(changeset) do
{:ok, post} ->
# Great! Please also save the tag confirming it
# If we can't just explode and rollback...
tag
|> Ecto.Changeset.change(%{name: "success"})
|> repo.update!()
{:error, changeset} ->
# Oh noes! Please save some information about it
# If we can't save just explode and rollback...
tag
|> Ecto.Changeset.change(%{name: "failure"})
|> repo.update!()
end
end)
What surprised us that is that this operation fails, even though we are handling the errors in the first insert. This is the exception raised when the code gets to the repo.update!
(this is from the adapter, so it also happens with the safe alternative repo.update
):
** (Postgrex.Error) ERROR 25P02 (in_failed_sql_transaction) current transaction is aborted, commands ignored until end of transaction block
Ok, apparently there’s something preventing the transaction from succeeding. After conducting some tests it seems that repo.insert
is putting the transaction in a bad state and repo.update!
can’t succeed because of it. We had that confirmed by jose and as it seems, this error is happening because an operation already failed inside the transaction. He also seems to agree that the docs are talking about exceptions and not just “errors”. So, what exactly are we missing here?
If the docs are right about the transaction rolling back on exceptions, why can’t this code succeed? If not, and we agree that there’s something to be improved in the docs, what should the correct behavior be?
I’m certain that there’s some important information I’m missing, but I’m feeling a bit misguided by the docs and we want to improve it for posterity.