【问题标题】:has_many, through associations in Ectohas_many,通过 Ecto 中的关联
【发布时间】:2016-03-02 19:21:32
【问题描述】:

我仍在尝试了解如何处理在 Ecto 中创建/更新 has_many, through: 关联。我已经重新阅读了José'sthe docs 的关联帖子,但我仍在苦苦挣扎。

我拥有的是这样的:

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 帖子中删除了 "dish" 的附加参数。)


如何在我的控制器中处理这个问题?具体来说,我想要这种行为:

  1. 对于POST 请求(创建操作),在dish_dietary_prefs 中创建必要的记录,以将这个新的Dish 与给定的DietaryPrefs 相关联。对于 DietaryPref 记录,逗号分隔的字符串是 ids。
  2. 对于PUT/PATCH 请求(更新),创建/销毁dish_dietary_prefs 中的必要记录以更新关联(用户可以将菜肴重新分配给不同的饮食偏好)。
  3. 对于DELETE 请求,销毁dish_dietary_prefs。我认为这个案例已经通过模型中的on_delete 配置处理了。

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

任何帮助将不胜感激。


如果我将"need to receive the IDs and manually build the intermediate association for each" DietaryPref 关联到Dish,我能否获得一个示例,说明如何在我的控制器中对上述规范执行此操作?


更新:刚刚看到 Ecto 2.0.0-beta.1 已经发布,那就是 supports many_to_many,这看起来可以解决我的问题问题。任何人都有使用它的例子,就像我上面描述的那样?

【问题讨论】:

    标签: elixir phoenix-framework ecto


    【解决方案1】:

    多亏了无与伦比的绝地大师何塞·瓦利姆本人,我才弄清楚了这一点(在 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

    【讨论】:

    • 请注意,您还必须使用many_to_many(而不是has_many X, through: Y)才能使put_assoc/3 正常工作。有关显示此内容的列表,请参阅 Google 群组链接。
    猜你喜欢
    • 2018-05-23
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多