Elixir ecto 2 crea una asociación many_to_many
¿Cómo puedo hacer una relación de muchos a muchos con ecto 2? Como una aplicación de ejemplo, quiero crear una publicación que puede estar en varias categorías. Las categorías ya existen. Por ejemplo:
[%Category{id: "1", name: "elixir"}, %Category{id: "2", name: "erlang"}]
Estoy usando Ecto 2 beta 0. El proyecto de ejemplo se llama Ecto2.
Definí dos modelos:
defmodule Ecto2.Post do
use Ecto2.Web, :model
use Ecto.Schema
schema "posts" do
field :title, :string
many_to_many :categories, Ecto2.Category, join_through: "posts_categories", on_replace: :delete
timestamps
end
@required_fields ~w(title)
@optional_fields ~w()
def changeset(model, params \\ :empty) do
model
|> cast(params, @required_fields, @optional_fields)
|> cast_assoc(:categories) # not suitable?
end
end
defmodule Ecto2.Category do
use Ecto2.Web, :model
schema "categories" do
field :name, :string
timestamps
end
@required_fields ~w(name)
@optional_fields ~w()
def changeset(model, params \\ :empty) do
model
|> cast(params, @required_fields, @optional_fields)
end
end
Intenté hacerlo así:
post = Repo.get!(Post, 1) |> Repo.preload(:categories)
changeset = Post.changeset(post, %{"title"=> "bla", "categories"=> [%{id: "1"}]})
Repo.update!(changeset)
Pero cast_assoc en Post.changeset no es adecuado para esta tarea, quiere crear una categoría completamente nueva en lugar de una asociada. ¿Qué debo usar en su lugar? build_assoc? Pero los documentos build_assoc no mencionan que es útil con many_to_many. ¿Como lo uso? ¿Debo poner build_assoc en Post.changeset entonces, o debería usarlo en un controlador Phoenix?