💧 Elixir: Trabalhando com Ecto Embeds (Campos Json)

maiquitome

Dev Maiqui 🇧🇷

Posted on March 20, 2022

💧 Elixir: Trabalhando com Ecto Embeds (Campos Json)

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).

A Jornada do Autodidata em Inglês

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
Enter fullscreen mode Exit fullscreen mode

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
Enter fullscreen mode Exit fullscreen mode

Gere uma migração pra criar o campo permalinks:

$ mix ecto.gen.migration add_permalinks_to_post
Enter fullscreen mode Exit fullscreen mode

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
Enter fullscreen mode Exit fullscreen mode

O :jsonb já aceita uma lista, então não precisa colocar {:array, :jsonb}.

$ mix ecto.migrate
Enter fullscreen mode Exit fullscreen mode

É 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"}
  ]
})
Enter fullscreen mode Exit fullscreen mode

Image description

Vamos ver como ficou no banco de dados:

Image description

Image description

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)
Enter fullscreen mode Exit fullscreen mode

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)
Enter fullscreen mode Exit fullscreen mode

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)
Enter fullscreen mode Exit fullscreen mode

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"}
    }
]
Enter fullscreen mode Exit fullscreen mode

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.

💖 💪 🙅 🚩
maiquitome
Dev Maiqui 🇧🇷

Posted on March 20, 2022

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related