Ecto v2.0.0-beta.2 Ecto.Multi
Ecto.Multi is a data structure that allows grouping multiple Repo operations together.
Ecto.Multi makes it possible to pack operations that should be performed together (in a single database transaction) and gives a way to introspect the queued operations without actually performing them. Each operation is given a name that is unique and will identify it’s result or will help to identify place of failure in case it occurs.
All operations will be executed in the order they were added.
The Ecto.Multi
structure should be considered opaque. You can use
%Ecto.Multi{}
to pattern match the type, but accessing fields or
directly modifying them is not advised.
Ecto.Multi.to_list/1
returns a canonical representation of the structure
that can be used for introspection.
Changesets
If Multi contains operations that accept changesets (like insert/4
,
update/4
or delete/4
) they will be checked before starting transaction
if any of them isn’t invalid. In case any has errors, the transaction won’t
even be started and the error will be immediately returned.
Run
Multi allows you to run arbitrary functions as part of your transaction via
the run/3
and run/5
. Those functions will receive changes so far as the
first argument and have to return {:ok, value}
or {:error, value}
as
their result. Returning an error will abort any further operations and
make the whole multi fail.
Example
Let’s look at an example definition and usage. The use case we’ll be looking into is resetting password. We need to update the account with proper information, log the request and remove all current sessions. We define a function creating the Multi structure probably in some sort of service layer:
defmodule Service do
alias Ecto.Multi
import Ecto
def password_reset(account, params) do
Multi.new
|> Multi.update(:account, Account.password_reset_changeset(account, params))
|> Multi.insert(:log, Log.password_reset_changeset(account, params))
|> Multi.delete_all(:sessions, assoc(account, :sessions))
end
end
We can later execute it in the integration layer using Repo:
Repo.transaction(Service.password_reset(account, params))
By pattern matching on the result we can differentiate different conditions:
case result do
{:ok, %{account: account, log: log, sessions: sessions}} ->
# Operation was successful, we can access results (exactly the same
# we would get from running corresponding Repo functions)
# under keys we used for naming the operations.
{:error, failed_operation, failed_value, changes_so_far} ->
# One of the operations failed. We can access the operation's failure
# value (like changeset for operations on changesets) to prepare
# proper response. We also get access to results of any operations
# that succeeded before the indicated operation failed.
end
We can also easily unit test our transaction without actually running it. Since changesets can use in-memory-data, we can use an account that is constructed in memory as well (without persisting it to the database):
test "dry run password_reset" do
account = %Account{password: "letmein"}
multi = Service.password_reset(account, params)
assert [
{:account, {:update, account_changeset, []}},
{:log, {:insert, log_changeset, []}},
{:sessions, {:delete_all, query, []}}
] = Ecto.Multi.to_list(multi)
# We can introspect changesets and query to see if everything
# is as expected, for example:
assert account_changeset.valid?
assert log_changeset.valid?
assert inspect(query) == "#Ecto.Query<from a in Session>"
end
Summary
Functions
Appends the second multi to the first one
Adds a delete operation to the multi
Adds an delete_all operation to the multi
Adds an insert operation to the multi
Adds an insert_all operation to the multi
Returns an empty Ecto.Multi
struct
Prepends the second multi to the first one
Adds a function to run as part of the multi
Adds a function to run as part of the multi
Adds an update operation to the multi
Adds an update_all operation to the multi
Types
Functions
Appends the second multi to the first one.
All names must be unique between both structures.
Example
iex> lhs = Ecto.Multi.new |> Ecto.Multi.run(:left, &{:ok, &1})
iex> rhs = Ecto.Multi.new |> Ecto.Multi.run(:right, &{:error, &1})
iex> Ecto.Multi.append(lhs, rhs) |> Ecto.Multi.to_list |> Keyword.keys
[:left, :right]
Specs
delete(t, name, Ecto.Changeset.t | Ecto.Schema.t, Keyword.t) :: t
Adds a delete operation to the multi.
Accepts the same arguments and options as Ecto.Repo.delete/3
does.
Specs
delete_all(t, name, Ecto.Queryable.t, Keyword.t) :: t
Adds an delete_all operation to the multi.
Accepts the same arguments and options as Ecto.Repo.delete_all/4
does.
Specs
insert(t, name, Ecto.Changeset.t | Ecto.Schema.t, Keyword.t) :: t
Adds an insert operation to the multi.
Accepts the same arguments and options as Ecto.Repo.insert/3
does.
Specs
Adds an insert_all operation to the multi.
Accepts the same arguments and options as Ecto.Repo.insert_all/4
does.
Specs
new :: t
Returns an empty Ecto.Multi
struct.
Example
iex> Ecto.Multi.new |> Ecto.Multi.to_list
[]
Prepends the second multi to the first one.
All names must be unique between both structures.
Example
iex> lhs = Ecto.Multi.new |> Ecto.Multi.run(:left, &{:ok, &1})
iex> rhs = Ecto.Multi.new |> Ecto.Multi.run(:right, &{:error, &1})
iex> Ecto.Multi.prepend(lhs, rhs) |> Ecto.Multi.to_list |> Keyword.keys
[:right, :left]
Adds a function to run as part of the multi
The function should return either {:ok, value}
or {:error, value}
, and
receives changes so far as an argument.
Adds a function to run as part of the multi
Similar to run/3
, but allows to pass module name, function and arguments.
The function should return either {:ok, value}
or {:error, value}
, and
will receive changes so far as the first argument (prepened to those passed in
the call to the function).
Specs
update(t, name, Ecto.Changeset.t, Keyword.t) :: t
Adds an update operation to the multi.
Accepts the same arguments and options as Ecto.Repo.update/3
does.