ash/test/filter/filter_interaction_test.exs

220 lines
4.9 KiB
Elixir
Raw Normal View History

2020-06-30 09:20:29 +12:00
defmodule Ash.Test.Filter.FilterInteractionTest do
use ExUnit.Case, async: false
import ExUnit.CaptureLog
2020-06-30 09:42:01 +12:00
alias Ash.DataLayer.Mnesia
2020-06-30 09:20:29 +12:00
defmodule Profile do
@moduledoc false
use Ash.Resource, data_layer: Ash.DataLayer.Ets
ets do
private?(true)
end
actions do
read :default
create :default
update :default
end
attributes do
attribute :id, :uuid, primary_key?: true, default: &Ecto.UUID.generate/0
attribute :bio, :string
end
relationships do
belongs_to :user, Ash.Test.Filter.FilterInteractionTest.User
end
end
defmodule User do
@moduledoc false
use Ash.Resource, data_layer: Ash.DataLayer.Ets
ets do
private?(true)
end
actions do
read :default
create :default
update :default
end
attributes do
attribute :id, :uuid, primary_key?: true, default: &Ecto.UUID.generate/0
attribute :name, :string
attribute :allow_second_author, :boolean
end
relationships do
has_many :posts, Ash.Test.Filter.FilterInteractionTest.Post, destination_field: :author_id
has_many :second_posts, Ash.Test.Filter.FilterInteractionTest.Post,
destination_field: :author_id
has_one :profile, Profile, destination_field: :user_id
end
end
defmodule PostLink do
@moduledoc false
use Ash.Resource, data_layer: Ash.DataLayer.Mnesia
actions do
read :default
create :default
update :default
end
relationships do
belongs_to :source_post, Ash.Test.Filter.FilterInteractionTest.Post, primary_key?: true
belongs_to :destination_post, Ash.Test.Filter.FilterInteractionTest.Post, primary_key?: true
end
end
defmodule Post do
@moduledoc false
use Ash.Resource, data_layer: Ash.DataLayer.Mnesia
actions do
read :default
create :default
update :default
destroy :default
end
attributes do
attribute :id, :uuid, primary_key?: true, default: &Ecto.UUID.generate/0
attribute :title, :string
attribute :contents, :string
attribute :points, :integer
end
relationships do
belongs_to :author, User,
destination_field: :id,
source_field: :author_id
many_to_many :related_posts, __MODULE__,
through: PostLink,
source_field_on_join_table: :source_post_id,
destination_field_on_join_table: :destination_post_id
end
end
defmodule Api do
@moduledoc false
use Ash.Api
resources do
resource(Post)
resource(User)
resource(Profile)
resource(PostLink)
end
end
setup do
capture_log(fn ->
Mnesia.start(Api)
end)
2020-06-30 09:20:29 +12:00
on_exit(fn ->
capture_log(fn ->
:mnesia.stop()
:mnesia.delete_schema([node()])
end)
2020-06-30 09:20:29 +12:00
end)
end
test "mnesia data layer sanity test" do
post = Api.create!(Post, attributes: %{title: "best"})
assert [^post] = Api.read!(Post)
Api.update!(post, attributes: %{title: "worst"})
new_post = %{post | title: "worst"}
assert [^new_post] = Api.read!(Post)
Api.destroy!(post)
assert [] = Api.read!(Post)
end
describe "cross data layer filtering" do
test "it properly filters with a simple filter" do
author = Api.create!(User, attributes: %{name: "best author"})
post1 =
Api.create!(Post,
attributes: %{title: "best"},
relationships: %{author: author}
)
Api.create!(Post, attributes: %{title: "worst"})
post1 = Api.reload!(post1)
query =
Post
|> Ash.Query.filter(author: [name: "best author"])
assert [^post1] = Api.read!(query)
end
test "parallelizable filtering of related resources with a data layer that cannot join" do
post2 = Api.create!(Post, attributes: %{title: "two"})
Api.create!(Post, attributes: %{title: "three"})
2020-06-30 09:20:29 +12:00
post1 =
Api.create!(Post,
attributes: %{title: "one"},
2020-06-30 09:42:01 +12:00
relationships: %{related_posts: [post2]}
2020-06-30 09:20:29 +12:00
)
query =
Post
|> Ash.Query.filter(related_posts: [title: "two"])
2020-06-30 09:42:01 +12:00
post1 = Api.reload!(post1)
2020-06-30 09:20:29 +12:00
assert [^post1] = Api.read!(query)
end
test "parallelizable filter with filtered side loads" do
post2 = Api.create!(Post, attributes: %{title: "two"})
post3 = Api.create!(Post, attributes: %{title: "three"})
post1 =
Api.create!(Post,
attributes: %{title: "one"},
relationships: %{related_posts: [post2, post3]}
)
posts_query =
Post
|> Ash.Query.filter(title: "three")
query =
Post
|> Ash.Query.filter(related_posts: [title: "two"])
|> Ash.Query.side_load(related_posts: posts_query)
post1_id = post1.id
post3_id = post3.id
assert [%{id: ^post1_id, related_posts: [%{id: ^post3_id}]}] = Api.read!(query)
end
2020-06-30 09:20:29 +12:00
end
end