EDIT: re-read the question and it seems it’s more about the Ash way of doing it. But the docs above is how to do it with raw Ecto, just need to figure out how to call it from Ash. Maybe: Manual Actions — ash v2.15.19
Yeah I am also not super sure about how to do this in Ash, but as a minor database note TRUNCATE is the ideal way to delete all rows in a table. delete_all has to essentially write an update for every row, marking it as deleted, where as TRUNCATE basically just deletes the file associated with the table data.
I would agree that TRUNCATE would be the first choice if all conditions allow… and in many basic scenarios you’ll be fine; but one needs to use TRUNCATE with some care in more complex scenarios. (Assuming PostgreSQL)
There are special considerations if the table is used in other table foreign key references.
If you have ON DELETE triggers, they will not fire.
TRUNCATE is not fully MVCC safe in relation to certain concurrent transactions.
Because of the special nature of TRUNCATE and the efficiency it tries to bring to mass delete operations, there can be similar issues in other vendor’s databases as well. For example, in PostgreSQL, while fully not MVCC safe, you can roll back a TRUNCATE operation, but in Oracle you cannot.
So I would advise anyone needing to delete all the data in a table to TRUNCATE if they can, but it’s one of those commands that you really need to understand and use with care.
So just to close the case, I ended up using following generic action:
actions do
defaults [:create, :read, :update, :destroy]
action :destroy_all, :term do
run(fn _, _ ->
case __MODULE__
|> MyApp.Repo.to_ecto()
|> MyApp.Repo.delete_all() do
{n, result} when is_integer(n) -> {:ok, result}
error -> {:error, error}
end
end)
end
end
And that gives me MyApp.Api.Resource.destroy_all()