OBS: Esse post é uma continuação das associações. Se você não fez o setup do projeto, você pode voltar para o post anterior para fazê-lo.
Além das associações, o Ecto também suporta embeds (incorporações)
em alguns bancos de dados. Com embeds
, o filho é incorporado no pai, em vez de ser armazenado em outra tabela.
Bancos de dados como o PostgreSQL usa uma combinação de colunas JSONB (embeds_one/3) e ARRAY para fornecer esta funcionalidade (tanto JSONB como ARRAY são suportados por padrão e são cidadãos de primeira classe no Ecto).
Trabalhar com embeds
é na maioria das vezes o mesmo que trabalhar com outro campo em um schema
, exceto quando se trata de manipulá-los. Vamos ver um exemplo:
Crie um arquivo com o conteúdo:
defmodule Blog.Permalink do
use Ecto.Schema
embedded_schema do
field :url
timestamps
end
end
Altere o schema Blog.Post
:
defmodule Blog.Post do
use Ecto.Schema
schema "posts" do
field :title
field :body
has_many :comments, Blog.Comment
# adicione essa linha
embeds_many :permalinks, Blog.Permalink
timestamps()
end
end
Gere uma migração pra criar o campo permalinks
:
$ mix ecto.gen.migration add_permalinks_to_post
Altere o arquivo de migração:
defmodule Blog.Repo.Migrations.AddPermalinksToPost do
use Ecto.Migration
def change do
# adicionando o campo `permalinks` na tabela `posts`
alter table(:posts) do
add :permalinks, :jsonb
end
end
end
O :jsonb
já aceita uma lista, então não precisa colocar {:array, :jsonb}
.
$ mix ecto.migrate
É possível inserir diretamente um post com múltiplos permalinks:
iex> Blog.Repo.insert!(%Blog.Post{
title: "Hello",
permalinks: [
%Blog.Permalink{url: "example.com/thebest"},
%Blog.Permalink{url: "another.com/mostaccessed"}
]
})
Vamos ver como ficou no banco de dados:
Semelhante às associações, você também pode gerenciar essas entradas usando changesets (conjuntos de alterações)
:
Preste atenção no Ecto.Changeset.put_embed
.
# Generate a changeset for the post
changeset = Ecto.Changeset.change(post)
# Let's track the new permalinks
changeset = Ecto.Changeset.put_embed(changeset, :permalinks,
[%Permalink{url: "example.com/thebest"},
%Permalink{url: "another.com/mostaccessed"}]
)
# Now insert the post with permalinks at once
post = Repo.insert!(changeset)
Se você quiser substituir ou remover um permalink em particular, você pode trabalhar com permalinks como coleção e depois simplesmente colocá-lo como uma mudança novamente:
# Remove all permalinks from example.com
permalinks = Enum.reject post.permalinks, fn permalink ->
permalink.url =~ "example.com"
end
# Let's create a new changeset
changeset =
post
|> Ecto.Changeset.change
|> Ecto.Changeset.put_embed(:permalinks, permalinks)
# And update the entry
post = Repo.update!(changeset)
Se você não conhece Enum.reject
, veja aqui como funciona de forma visual.
A beleza de trabalhar com os changesets é que eles acompanham todas as mudanças que serão enviadas para o banco de dados e nós podemos introspectá-los a qualquer momento. Por exemplo, se chamássemos antes Repo.update!/3
:
IO.inspect(changeset.changes.permalinks)
Nós veríamos algo parecido:
[
%Ecto.Changeset{
action: :delete,
changes: %{},
model: %Permalink{url: "example.com/thebest"}
},
%Ecto.Changeset{
action: :update,
changes: %{},
model: %Permalink{url: "another.com/mostaccessed"}
}
]
Se, por acaso, também estivéssemos inserindo um permalink nesta operação, veríamos ali outro changeset com a ação :inserir
.
Os changesets contêm uma visão completa do que está mudando, como eles estão mudando e você pode manipulá-los diretamente.
Top comments (2)
Show Maiqui Tomé!
Valeeu Romenig :)