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

Ecto schema default value is not included in changeset changes

考虑以下架构:

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 的变更集结构包含应用于 schema/struct 的变更。默认值未显示为更改的原因是,当您执行 %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 也会尝试插入它。