Ecto变更集中的转换错误

ank*_*981 3 elixir ecto phoenix-framework

我有以下模型(用于聊天室),我在使用更改集时遇到问题:

defmodule Elemental.TxChat.Room do
  use Elemental.TxChat.Web, :model

  schema "rooms" do
    field :name, :string

    # Foreign key indicating which user created this room
    # One user can create any number of rooms 
    belongs_to :created_by, Elemental.TxChat.User
    field :created_from_app, :integer

    many_to_many :members, Elemental.TxChat.User, join_through: "rooms_users"

    timestamps()
  end

  def changeset(struct, params \\ %{}) do
    struct
    |> cast(params, [:name, :created_by, :created_from_app])
    |> validate_required([:name, :created_by, :created_from_app])
  end
end
Run Code Online (Sandbox Code Playgroud)

然后我想,我会尝试将一个空结构传递给变更集并查看它产生的错误.所以我做了(在别名之后):

iex(4)> c = Room.changeset(%Room{}, %{})
** (RuntimeError) casting assocs with cast/3 is not supported, use cast_assoc/3 instead
       (ecto) lib/ecto/changeset.ex:440: Ecto.Changeset.type!/2
       (ecto) lib/ecto/changeset.ex:415: Ecto.Changeset.process_param/8
     (elixir) lib/enum.ex:1151: Enum."-map_reduce/3-lists^mapfoldl/2-0-"/3
     (elixir) lib/enum.ex:1151: Enum."-map_reduce/3-lists^mapfoldl/2-0-"/3
       (ecto) lib/ecto/changeset.ex:391: Ecto.Changeset.do_cast/7
    (tx_chat) web/models/room.ex:19: Elemental.TxChat.Room.changeset/2
Run Code Online (Sandbox Code Playgroud)

然后我认为有belongs_to等等的模型可以使用cast_assoc,所以我更改了函数名称.现在:

iex(4)> c = Room.changeset(%Room{}, %{})
** (FunctionClauseError) no function clause matching in Ecto.Changeset.cast_assoc/3
       (ecto) lib/ecto/changeset.ex:518: Ecto.Changeset.cast_assoc(%Elemental.TxChat.Room{__meta__: #Ecto.Schema.Metadata<:built, "rooms">, created_by: #Ecto.Association.NotLoaded<association :created_by is not loaded>, created_by_id: nil, created_from_app: nil, id: nil, inserted_at: nil, members: #Ecto.Association.NotLoaded<association :members is not loaded>, name: nil, updated_at: nil}, %{}, [:name, :created_by, :created_from_app])
    (tx_chat) web/models/room.ex:19: Elemental.TxChat.Room.changeset/2
Run Code Online (Sandbox Code Playgroud)

我究竟做错了什么?

Dog*_*ert 5

如评论中所述,您希望将现有用户与新房间相关联.您可以通过添加做到这一点created_by_id传递给字段列表cast中,assoc_constraint(:created_by)然后在关键发送用户的ID created_by_idRoom.changeset/2.这应该工作:

def changeset(struct, params \\ %{}) do
  struct
  |> cast(params, [:name, :created_by_id, :created_from_app])
  |> validate_required([:name, :created_by_id, :created_from_app])
  |> assoc_constraint(:created_by)
end
Run Code Online (Sandbox Code Playgroud)

然后创建一个房间:

Room.changeset(%Room{}, %{name: "foo", created_by_id: 1, created_from_app: 1})
|> Repo.insert
Run Code Online (Sandbox Code Playgroud)