gpt4 book ai didi

elixir - has_many,通过Ecto中的关联

转载 作者:行者123 更新时间:2023-12-04 07:34:54 26 4
gpt4 key购买 nike

我仍然在尝试如何处理在Ecto中创建/更新has_many, through:关联。我已经重新阅读了José's关于关联以及the docs的帖子,但我仍在努力。

我所拥有的是:

网站/模型/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

网站/模型/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

网站/模型/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帖子删除了 "dish"的其他参数。)

如何在 Controller 中处理此问题?具体来说,我想要这种行为:
  • 对于POST请求(创建操作),请在dish_dietary_prefs中创建必要的记录,以将此新Dish与给定的DietaryPref关联。逗号分隔的字符串是id记录的DietaryPref
  • 对于PUT/PATCH请求(更新),请在dish_dietary_prefs中创建/销毁必要的记录以更新关联(用户可以将菜肴重新分配给不同的饮食偏好)。
  • 对于DELETE请求,销毁dish_dietary_prefs。我认为这种情况已经通过模型中的on_delete配置进行了处理。

  • 我已经在 Controller 中有了为给定供应商创建/更新菜肴的逻辑(这只是一个简单的 has_many/belongs_to关系),但是我仍然不知道如何为给定菜肴创建/更新/销毁这些关联。

    任何帮助将不胜感激。

    如果我将 "need to receive the IDs and manually build the intermediate association for each" DietaryPref关联到 Dish,是否可以在我的 Controller 中获得有关如何对上述规范执行此操作的示例?

    更新:仅看到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

    关于elixir - has_many,通过Ecto中的关联,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35756250/

    26 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com