elixir - Ecto 模式默认值不包含在变更集更改中

标签 elixir ecto

考虑以下架构:

defmodule EctoBug.Post do
  use Ecto.Schema
  import Ecto.Changeset

  schema "posts" do
    field :title, :string, default: "test"

    timestamps()
  end

  def changeset(post, attrs) do
    post
    |> cast(attrs, [:title])
    |> validate_required([:title])
  end
end

如果我这样做

changeset = EctoBug.Post.changeset(%EctoBug.Post{}, %{title: "test"})

title 字段在 changes 中不存在:

#Ecto.Changeset<action: nil, changes: %{}, errors: [], data: #EctoBug.Post<>, valid?: true>

我找不到关于此行为的任何信息。

这是一个错误吗?

最佳答案

Ecto 的变更集结构保存应用于模式/结构的变更。默认值未显示为更改的原因是,当您执行 %EctoBug.Post{} 时,它会填充您在架构上设置的默认值。然后,当您转换参数时,由于该字段的原始值和转换值相同,所以它并不是真正的更改,也没有标记为更改。

如果你这样做

changeset = EctoBug.Post.changeset(%EctoBug.Post{}, %{title: "test"})
Ecto.Changeset.get_field(changeset, :title)

您应该看到标题已设置。虽然 Ecto.Changeset.get_change(changeset, :title) 会给你 nil 因为它没有从最初给出的原始结构中改变。

如果您为字段 title 提供的值不同于默认值,您应该会看到它将作为更改进行跟踪。

因为它是这样工作的,现在如果你有一个你试图更新而不是创建的记录,如果你针对该记录转换一些实际上与它已经拥有的参数相同的参数,ecto 可以跳过尝试更新记录,因为它可以看到没有任何改变。在插入的情况下,即使它没有变化,Ecto 也会尝试插入它。

关于elixir - Ecto 模式默认值不包含在变更集更改中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52793233/

相关文章:

elixir - 使用 shell 脚本调用 Elixir 方法

postgresql - 无法为 Elixir 的 Phoenix 创建新的 Postgres 表

elixir - 使用 ecto 2.0 更新多条记录

elixir - 在 Elixir Ecto 中加入属于两个数据库的两个表

cron - 如何在本地运行 Oban 作业?

erlang - Elixir 解析二进制数据?

elixir - 使用 Postgrex 和 Ecto 准备好的语句

postgresql - Elixir/Phoenix 无法连接到云托管的 Postgres 数据库

postgresql - 将 postgres 日期表示转换为 ISO 8601 字符串