has_many,通过 Ecto 中的关联
has_many, through associations in Ecto
我仍在努力理解如何处理 Ecto. I've re-read José's post on associations as well as the docs 中的 creating/updating has_many, through:
关联,但我仍在苦苦挣扎。
我有的是这个:
web/models/dish.ex
defmodule Mp.Dish do
use Mp.Web, :model
schema "dishes" do
# ...
has_many :dish_dietary_prefs, Mp.DishDietaryPref, on_delete: :delete_all,
on_replace: :delete
has_many :dietary_prefs, through: [:dish_dietary_prefs, :dietary_pref]
end
# ...
end
web/models/dietary_pref.ex
defmodule Mp.DietaryPref do
use Mp.Web, :model
schema "dietary_prefs" do
# ...
has_many :dish_dietary_prefs, Mp.DishDietaryPref, on_delete: :delete_all,
on_replace: :delete
has_many :dishes, through: [:dish_dietary_prefs, :dish]
end
# ...
end
web/models/dish_dietary_pref.ex
defmodule Mp.DishDietaryPref do
use Ecto.Schema
schema "dish_dietary_prefs" do
belongs_to :dish, Mp.Dish
belongs_to :dietary_pref, Mp.DietaryPref
end
end
我有一个 JSON 端点接收 Dish
的参数,在其中我有一个名为 dietary_prefs
的键,它作为逗号分隔的字符串传递,因此,对于示例:
[info] POST /api/vendors/4/dishes
[debug] Processing by Mp.Api.DishController.create/2
Parameters: %{"dish" => %{"dietary_prefs" => "2,1"}, "vendor_id" => "4"}
(为此 SO post 删除了 "dish"
的附加参数。)
我如何在我的控制器中处理这个问题?具体来说,我想要这种行为:
- 对于
POST
请求(创建操作),在 dish_dietary_prefs
中创建必要的记录以将这个新的 Dish
与给定的 DietaryPref
相关联。 DietaryPref
条记录的逗号分隔字符串为 id
s。
- 对于
PUT/PATCH
请求(更新),create/destroy dish_dietary_prefs
中的必要记录以更新关联(用户可以将菜肴重新分配给不同的饮食偏好)。
- 对于
DELETE
个请求,销毁 dish_dietary_prefs
。我认为这种情况已经用模型中的 on_delete
配置处理了。
我的控制器中已经有了为给定供应商 create/update 菜肴的逻辑(这只是一个简单的 has_many/belongs_to
关系),但我仍然不知道如何 create/update/销毁给定菜肴的这些关联。
如有任何帮助,我们将不胜感激。
如果我愿意 DietaryPref
我正在关联到 Dish
,我可以举个例子说明如何在我的控制器中执行上述规范吗?
更新:刚看到 Ecto 2.0.0-beta.1 已经出来了,那就是 supports many_to_many
,这看起来像是解决我的问题的方法。任何人都有实际使用它的例子,就像我上面描述的那样?
多亏了无与伦比的绝地大师何塞·瓦利姆本人,我才弄明白了这一点(在 Ecto 2.0.0-beta.1
中):
这是我的最终控制器:
def create(conn, %{"dish" => dish_params }, vendor) do
dietary_prefs = get_dietary_pref_changeset(dish_params["dietary_prefs"])
changeset = vendor
|> build_assoc(:dishes)
|> Repo.preload(:dietary_prefs)
|> Dish.changeset(dish_params)
|> Ecto.Changeset.put_assoc(:dietary_prefs, dietary_prefs)
case Repo.insert(changeset) do
{:ok, dish} ->
conn
|> put_status(:created)
|> render("show.json", dish: dish)
{:error, changeset} ->
conn
|> put_status(:unprocessable_entity)
|> render(ChangesetView, "error.json", changeset: changeset)
end
end
def update(conn, %{"id" => id, "dish" => dish_params}, vendor) do
dish = Repo.get!(vendor_dishes(vendor), id)
dietary_prefs = get_dietary_pref_changeset(dish_params["dietary_prefs"])
changeset = dish
|> Repo.preload(:dietary_prefs)
|> Dish.changeset(dish_params)
|> Ecto.Changeset.put_assoc(:dietary_prefs, dietary_prefs)
case Repo.update(changeset) do
{ :ok, dish } ->
render(conn, "show.json", dish: dish)
{ :error, changeset } ->
conn
|> put_status(:unprocessable_entity)
|> render(ChangesetView, "error.json", changeset: changeset)
end
end
defp vendor_dishes(vendor) do
assoc(vendor, :dishes)
end
defp parse_dietary_pref_ids(ids) do
ids
|> String.split(",")
|> Enum.map(fn(x) -> Integer.parse(x) |> Kernel.elem(0) end)
end
defp get_dietary_prefs_with_ids(ids) do
from(dp in DietaryPref, where: dp.id in ^ids) |> Repo.all
end
defp get_dietary_pref_changeset(param) do
param
|> parse_dietary_pref_ids
|> get_dietary_prefs_with_ids
|> Enum.map(&Ecto.Changeset.change/1)
end
https://groups.google.com/forum/#!topic/elixir-ecto/3cAi6nrsawk
我仍在努力理解如何处理 Ecto. I've re-read José's post on associations as well as the docs 中的 creating/updating has_many, through:
关联,但我仍在苦苦挣扎。
我有的是这个:
web/models/dish.ex
defmodule Mp.Dish do
use Mp.Web, :model
schema "dishes" do
# ...
has_many :dish_dietary_prefs, Mp.DishDietaryPref, on_delete: :delete_all,
on_replace: :delete
has_many :dietary_prefs, through: [:dish_dietary_prefs, :dietary_pref]
end
# ...
end
web/models/dietary_pref.ex
defmodule Mp.DietaryPref do
use Mp.Web, :model
schema "dietary_prefs" do
# ...
has_many :dish_dietary_prefs, Mp.DishDietaryPref, on_delete: :delete_all,
on_replace: :delete
has_many :dishes, through: [:dish_dietary_prefs, :dish]
end
# ...
end
web/models/dish_dietary_pref.ex
defmodule Mp.DishDietaryPref do
use Ecto.Schema
schema "dish_dietary_prefs" do
belongs_to :dish, Mp.Dish
belongs_to :dietary_pref, Mp.DietaryPref
end
end
我有一个 JSON 端点接收 Dish
的参数,在其中我有一个名为 dietary_prefs
的键,它作为逗号分隔的字符串传递,因此,对于示例:
[info] POST /api/vendors/4/dishes
[debug] Processing by Mp.Api.DishController.create/2
Parameters: %{"dish" => %{"dietary_prefs" => "2,1"}, "vendor_id" => "4"}
(为此 SO post 删除了 "dish"
的附加参数。)
我如何在我的控制器中处理这个问题?具体来说,我想要这种行为:
- 对于
POST
请求(创建操作),在dish_dietary_prefs
中创建必要的记录以将这个新的Dish
与给定的DietaryPref
相关联。DietaryPref
条记录的逗号分隔字符串为id
s。 - 对于
PUT/PATCH
请求(更新),create/destroydish_dietary_prefs
中的必要记录以更新关联(用户可以将菜肴重新分配给不同的饮食偏好)。 - 对于
DELETE
个请求,销毁dish_dietary_prefs
。我认为这种情况已经用模型中的on_delete
配置处理了。
我的控制器中已经有了为给定供应商 create/update 菜肴的逻辑(这只是一个简单的 has_many/belongs_to
关系),但我仍然不知道如何 create/update/销毁给定菜肴的这些关联。
如有任何帮助,我们将不胜感激。
如果我愿意 DietaryPref
我正在关联到 Dish
,我可以举个例子说明如何在我的控制器中执行上述规范吗?
更新:刚看到 Ecto 2.0.0-beta.1 已经出来了,那就是 supports many_to_many
,这看起来像是解决我的问题的方法。任何人都有实际使用它的例子,就像我上面描述的那样?
多亏了无与伦比的绝地大师何塞·瓦利姆本人,我才弄明白了这一点(在 Ecto 2.0.0-beta.1
中):
这是我的最终控制器:
def create(conn, %{"dish" => dish_params }, vendor) do
dietary_prefs = get_dietary_pref_changeset(dish_params["dietary_prefs"])
changeset = vendor
|> build_assoc(:dishes)
|> Repo.preload(:dietary_prefs)
|> Dish.changeset(dish_params)
|> Ecto.Changeset.put_assoc(:dietary_prefs, dietary_prefs)
case Repo.insert(changeset) do
{:ok, dish} ->
conn
|> put_status(:created)
|> render("show.json", dish: dish)
{:error, changeset} ->
conn
|> put_status(:unprocessable_entity)
|> render(ChangesetView, "error.json", changeset: changeset)
end
end
def update(conn, %{"id" => id, "dish" => dish_params}, vendor) do
dish = Repo.get!(vendor_dishes(vendor), id)
dietary_prefs = get_dietary_pref_changeset(dish_params["dietary_prefs"])
changeset = dish
|> Repo.preload(:dietary_prefs)
|> Dish.changeset(dish_params)
|> Ecto.Changeset.put_assoc(:dietary_prefs, dietary_prefs)
case Repo.update(changeset) do
{ :ok, dish } ->
render(conn, "show.json", dish: dish)
{ :error, changeset } ->
conn
|> put_status(:unprocessable_entity)
|> render(ChangesetView, "error.json", changeset: changeset)
end
end
defp vendor_dishes(vendor) do
assoc(vendor, :dishes)
end
defp parse_dietary_pref_ids(ids) do
ids
|> String.split(",")
|> Enum.map(fn(x) -> Integer.parse(x) |> Kernel.elem(0) end)
end
defp get_dietary_prefs_with_ids(ids) do
from(dp in DietaryPref, where: dp.id in ^ids) |> Repo.all
end
defp get_dietary_pref_changeset(param) do
param
|> parse_dietary_pref_ids
|> get_dietary_prefs_with_ids
|> Enum.map(&Ecto.Changeset.change/1)
end
https://groups.google.com/forum/#!topic/elixir-ecto/3cAi6nrsawk