Ecto:如何使用put_assoc在一个更新中更新和插入关联

时间:2016-02-19 21:08:54

标签: elixir ecto

我试图编写一个更新模型并插入关联的变更集。我找不到有关如何使用put_assoc/4

的示例
  order = order
    |> Proj.Order.changeset(%{state: "error", error_count: order.error_count + 1})
    |> Ecto.Changeset.put_assoc(
      :order_errors,
      [Proj.OrderError.changeset(%Proj.OrderError{}, %{reason: "not_found"})])
    |> Proj.Repo.update!

这会打印出以下错误:

** (Ecto.InvalidChangesetError) could not perform update because changeset is invalid.

* Changeset changes

%{order_errors: [%Ecto.Changeset{action: :insert, changes: %{id: nil, inserted_at: nil, order_id: nil, reason: "not_found", updated_at: nil}, constraints: [], errors: [order_id: "can't be blank"], filters: %{}, model: %Proj.OrderError{__meta__: #Ecto.Schema.Metadata<:built>, id: nil, inserted_at: nil, order: #Ecto.Association.NotLoaded<association :order is not loaded>, order_id: nil, reason: nil, updated_at: nil}, optional: [], opts: [], params: %{"reason" => "not_found"}, prepare: [], repo: nil, required: [:order_id, :reason], types: %{id: :id, inserted_at: Ecto.DateTime, order_id: :id, reason: :string, updated_at: Ecto.DateTime}, valid?: false, validations: []}], state: "error"}

* Changeset params

%{"error_count" => 1, "state" => "error"}

* Changeset errors

[]

我可以查看put_assoc/4的所有示例?如何找到变更集无效的原因?

以这种方式做事的目标是希望新的order预先加载order_errors

2 个答案:

答案 0 :(得分:5)

我找到了如何让它在Ecto tests中运行。简短的故事,只需删除changeset创建并直接使用新模型。

  order = order
    |> Proj.Order.changeset(%{state: "error", error_count: order.error_count + 1})
    |> Ecto.Changeset.put_assoc(
      :order_errors,
      [%Proj.OrderError{reason: "not_found"}])
    |> Proj.Repo.update!

我仍然想知道如何理解原始帖子中的错误消息

答案 1 :(得分:0)

由于put_assoc/4描述了有效值:

  

给定值可以是关联结构,给定关联的变更集,也可以是要应用于当前关联的地图或关键字变更列表。在所有情况下,预计关键是原子。如果给出了地图或关键字列表且没有关联,则会创建一个。

如果您使用changeset,则应使用现有的%Proj.OrderError{}put_assoc/4会自动为此update执行changeset

但对于其他情况,尤其是记录需要insert,请使用struct  ,关键字列表或地图是更好的主意。