defmodule Ash.Test.Actions.UpdateTest do @moduledoc false use ExUnit.Case, async: true import Ash.Changeset import Ash.Test.Helpers require Ash.Query defmodule Authorized do @moduledoc false use Ash.Resource, data_layer: Ash.DataLayer.Ets, authorizers: [Ash.Test.Authorizer] ets do private?(true) end attributes do uuid_primary_key :id attribute :name, :string end actions do read :read create :create update :update end end defmodule Profile do @moduledoc false use Ash.Resource, data_layer: Ash.DataLayer.Ets ets do private?(true) end actions do read :read create :create update :update, primary?: true update :set_private_attribute_to_nil do change set_attribute(:non_nil_private, nil) end update :set_private_attribute_from_arg do argument :private, :string change set_attribute(:private, {:arg, :private}) end end attributes do uuid_primary_key :id attribute :bio, :string, allow_nil?: false attribute :non_nil_private, :string, allow_nil?: false, default: "non_nil" attribute :private, :string, default: "non_nil" end relationships do belongs_to :author, Ash.Test.Actions.UpdateTest.Author end end defmodule DuplicateName do @moduledoc false use Ash.Resource.Change def change(changeset, _, _) do case Ash.Changeset.fetch_change(changeset, :name) do :error -> changeset {:ok, name} -> Ash.Changeset.change_attribute(changeset, :name, name <> name) end end end defmodule ManualUpdateAuthor do @moduledoc false use Ash.Resource.Change def change(changeset, _, _) do Ash.Changeset.after_action(changeset, fn _changeset, data -> {:ok, data |> Ash.Changeset.new() |> Ash.Changeset.change_attribute(:name, "manual") |> Ash.Test.Actions.UpdateTest.Api.update!()} end) end end defmodule Author do @moduledoc false use Ash.Resource, data_layer: Ash.DataLayer.Ets ets do private?(true) end actions do read :read create :create update :update, primary?: true update :only_allow_name do accept([:name]) end update :duplicate_name do change {DuplicateName, []} end update :manual_update do accept [] manual? true change ManualUpdateAuthor end end attributes do uuid_primary_key :id attribute :name, :string attribute :bio, :string end relationships do has_one :profile, Profile, destination_field: :author_id has_many :posts, Ash.Test.Actions.UpdateTest.Post, destination_field: :author_id end end defmodule PostLink do @moduledoc false use Ash.Resource, data_layer: Ash.DataLayer.Ets ets do private?(true) end attributes do attribute :type, :string end actions do read :read create :create update :update end relationships do belongs_to :source_post, Ash.Test.Actions.UpdateTest.Post, primary_key?: true, required?: true belongs_to :destination_post, Ash.Test.Actions.UpdateTest.Post, primary_key?: true, required?: true end end defmodule Post do @moduledoc false use Ash.Resource, data_layer: Ash.DataLayer.Ets ets do private?(true) end actions do read :read create :create update :update end attributes do uuid_primary_key :id attribute :title, :string attribute :contents, :string end relationships do belongs_to :author, Author many_to_many :related_posts, __MODULE__, through: PostLink, source_field_on_join_table: :source_post_id, destination_field_on_join_table: :destination_post_id, join_attributes: [:type] end end defmodule Api do @moduledoc false use Ash.Api resources do resource(Author) resource(Post) resource(Profile) resource(PostLink) resource(Authorized) end end describe "simple updates" do test "allows updating a record with valid attributes" do post = Post |> new(%{title: "foo", contents: "bar"}) |> Api.create!() assert %Post{title: "bar", contents: "foo"} = post |> new(%{title: "bar", contents: "foo"}) |> Api.update!() end end describe "manual updates" do test "the update occurs properly" do author = Author |> new(%{name: "auto"}) |> Api.create!() assert %Author{name: "manual"} = author |> new() |> Api.update!(action: :manual_update) end end describe "allow_nil?" do test "it does not allow updating a value to `nil` when `allow_nil?: false`" do profile = Profile |> new(%{bio: "foobar"}) |> Api.create!() assert_raise Ash.Error.Invalid, ~r/attribute bio is required/, fn -> profile |> new(%{bio: ""}) |> Api.update!() end end test "it does not allow updating a private attribute's value to `nil` when `allow_nil?: false`" do profile = Profile |> new(%{bio: "foobar"}) |> Api.create!() assert_raise Ash.Error.Invalid, ~r/attribute non_nil_private is required/, fn -> profile |> new(%{bio: "foobar"}) |> Api.update!(action: :set_private_attribute_to_nil) end end test "it passes through an argument's value" do profile = Profile |> new(%{bio: "foobar"}) |> Api.create!() profile = profile |> new(%{bio: "foobar", private: "blah"}) |> Api.update!(action: :set_private_attribute_from_arg) assert profile.private == "blah" end end describe "select" do test "allows selecting fields on the changeset" do post = Post |> new(%{title: "foo", contents: "bar"}) |> Api.create!() assert %Post{title: "bar", contents: nil} = post |> new(%{title: "bar", contents: "foo"}) |> Ash.Changeset.select(:title) |> Api.update!() end end describe "allow" do test "allows attributes in the list" do author = Author |> new(%{name: "fred"}) |> Api.create!() author |> new(%{name: "joe"}) |> Api.update!(action: :only_allow_name) end test "does not allow attributes in the list" do author = Author |> new(%{name: "fred"}) |> Api.create!() assert_raise Ash.Error.Invalid, ~r/Invalid value provided for bio: cannot be changed/, fn -> author |> new(%{bio: "bio"}) |> Api.update!(action: :only_allow_name) end end end describe "changeset" do test "changes are run properly" do author = Author |> new(%{name: "fred"}) |> Api.create!() author = author |> new(%{name: "joe"}) |> Api.update!(action: :duplicate_name) assert author.name == "joejoe" end end describe "updating many to many relationships" do test "allows updating with a many_to_many relationship" do post = Post |> new(%{title: "title"}) |> Api.create!() post2 = Post |> new(%{title: "title2"}) |> Api.create!() post3 = Post |> new(%{title: "title3"}) |> Api.create!() post |> new() |> replace_relationship(:related_posts, [post2, post3]) |> Api.update!() end test "allows directly managing a many_to_many relationship" do post = Post |> new(%{title: "title"}) |> manage_relationship(:related_posts, [%{title: "title0"}], type: :direct_control) |> Api.create!() other_post = Post |> Ash.Query.filter(title == "title0") |> Api.read_one!() post |> new() |> manage_relationship( :related_posts, [%{title: "title3", id: other_post.id}, %{title: "title1"}], type: :direct_control ) |> Api.update!() assert ["title", "title1", "title3"] = Post |> Ash.Query.sort(:title) |> Api.read!() |> Enum.map(& &1.title) end test "it updates the join table properly" do post = Post |> new(%{title: "title"}) |> Api.create!() post2 = Post |> new(%{title: "title2"}) |> Api.create!() post3 = Post |> new(%{title: "title3"}) |> Api.create!() post |> new() |> replace_relationship(:related_posts, [post2, post3]) |> Api.update!() assert [_, _] = Api.read!(PostLink) end test "it responds with the relationship filled in" do post = Post |> new(%{title: "title"}) |> Api.create!() post2 = Post |> new(%{title: "title2"}) |> Api.create!() post3 = Post |> new(%{title: "title3"}) |> Api.create!() new_post = post |> new() |> replace_relationship(:related_posts, [post2, post3]) |> Api.update!() assert Enum.sort(new_post.related_posts) == Enum.sort([ Api.get!(Post, post2.id), Api.get!(Post, post3.id) ]) |> clear_meta() end test "it updates any join fields" do post = Post |> new(%{title: "title"}) |> Api.create!() post2 = Post |> new(%{title: "title2"}) |> Api.create!() post3 = Post |> new(%{title: "title3"}) |> Api.create!() new_post = post |> new() |> replace_relationship(:related_posts, [ Ash.Resource.Info.set_metadata(post2, %{join_keys: %{type: "a"}}), Ash.Resource.Info.set_metadata(post3, %{join_keys: %{type: "b"}}) ]) |> Api.update!() |> Api.load!(:related_posts_join_assoc) types = Enum.sort(Enum.map(new_post.related_posts_join_assoc, &Map.get(&1, :type))) assert types == ["a", "b"] new_post = new_post |> new() |> replace_relationship( :related_posts, [ Ash.Resource.Info.set_metadata(post2, %{join_keys: %{type: "c"}}), Ash.Resource.Info.set_metadata(post3, %{join_keys: %{type: "d"}}) ], on_match: :update, on_lookup: :relate ) |> Api.update!() |> Api.load!(:related_posts_join_assoc) types = Enum.sort(Enum.map(new_post.related_posts_join_assoc, &Map.get(&1, :type))) assert types == ["c", "d"] end end describe "updating with has_one relationships" do test "allows updating with has_one relationship" do profile = Profile |> new(%{bio: "best dude"}) |> Api.create!() profile2 = Profile |> new(%{bio: "second best dude"}) |> Api.create!() author = Author |> new(%{name: "fred"}) |> replace_relationship(:profile, profile) |> Api.create!() author |> new() |> replace_relationship(:profile, profile2) |> Api.update!() end test "it sets the relationship on the destination record accordingly" do profile = Profile |> new(%{bio: "best dude"}) |> Api.create!() profile2 = Profile |> new(%{bio: "second best dude"}) |> Api.create!() author = Author |> new(%{name: "fred"}) |> replace_relationship(:profile, profile) |> Api.create!() author |> new() |> replace_relationship(:profile, profile2) |> Api.update!() assert Api.get!(Profile, profile.id).author_id == nil assert Api.get!(Profile, profile2.id).author_id == author.id end test "it responds with the relationship filled in" do profile = Profile |> new(%{bio: "best dude"}) |> Api.create!() profile2 = Profile |> new(%{bio: "second best dude"}) |> Api.create!() author = Author |> new(%{name: "fred"}) |> replace_relationship(:profile, profile) |> Api.create!() updated_author = author |> new() |> replace_relationship(:profile, profile2) |> Api.update!() assert %{updated_author.profile | __metadata__: nil} == %{ profile2 | author_id: author.id, __metadata__: nil } end end describe "updating with a has_many relationship" do test "allows updating with a has_many relationship" do post = Post |> new(%{title: "sup"}) |> Api.create!() post2 = Post |> new(%{title: "sup2"}) |> Api.create!() author = Author |> new(%{name: "foobar"}) |> replace_relationship(:posts, [post]) |> Api.create!() author |> new() |> replace_relationship(:posts, [post, post2]) |> Api.update!() end test "it sets the relationship on the destination records accordingly" do post = Post |> new(%{title: "sup"}) |> Api.create!() post2 = Post |> new(%{title: "sup2"}) |> Api.create!() author = Author |> new(%{name: "foobar"}) |> replace_relationship(:posts, [post]) |> Api.create!() author = author |> new() |> replace_relationship(:posts, [post2.id]) |> Api.update!() assert Api.get!(Post, post.id).author_id == nil assert Api.get!(Post, post2.id).author_id == author.id end test "it responds with the relationship field filled in" do post = Post |> new(%{title: "sup"}) |> Api.create!() post2 = Post |> new(%{title: "sup2"}) |> Api.create!() author = Author |> new(%{name: "foobar"}) |> replace_relationship(:posts, [post]) |> Api.create!() updated_author = author |> new() |> replace_relationship(:posts, [post2]) |> Api.update!() post = Api.get!(Post, post2.id) assert Enum.map(updated_author.posts, &%{&1 | __metadata__: nil}) == [ %{post | __metadata__: nil} ] end end describe "updating with belongs_to relationships" do test "allows updating with belongs_to relationship" do author = Author |> new(%{name: "best dude"}) |> Api.create!() author2 = Author |> new(%{name: "best dude2"}) |> Api.create!() post = Post |> new(%{title: "foobar"}) |> replace_relationship(:author, author) |> Api.create!() post |> new() |> replace_relationship(:author, author2) |> Api.update!() end test "sets the relationship on the destination records accordingly" do author = Author |> new(%{name: "best dude"}) |> Api.create!() author2 = Author |> new(%{name: "best dude2"}) |> Api.create!() post = Post |> new(%{title: "foobar"}) |> replace_relationship(:author, author) |> Api.create!() post |> new() |> replace_relationship(:author, author2) |> Api.update!() author2 = Api.get!(Author, author2.id, load: :posts) assert Enum.map(author2.posts, & &1.id) == [ post.id ] end test "it responds with the relationship field filled in" do author = Author |> new(%{name: "best dude"}) |> Api.create!() author2 = Author |> new(%{name: "best dude2"}) |> Api.create!() post = Post |> new(%{title: "foobar"}) |> replace_relationship(:author, author) |> Api.create!() updated_post = post |> new() |> replace_relationship(:author, author2) |> Api.update!() assert updated_post.author.id == Api.get!(Author, author2.id).id end end describe "unauthorized update" do test "it does not update the record" do record = Authorized |> new(%{name: "bar"}) |> Api.create!() start_supervised({Ash.Test.Authorizer, check: :forbidden, strict_check: :continue}) assert_raise(Ash.Error.Forbidden, fn -> record |> new(%{name: "foo"}) |> Api.update!(authorize?: true) end) assert Api.get!(Authorized, record.id).name == "bar" end end end