我Xyz
有country_id
,federal_state_id
或city_id
。但只有其中一个(不是全部三个而不是两个)。我该如何进行验证呢?另外,我如何才为那个领域做assoc_constraint/1
?
defmodule Example.Location.Xyz do
use Ecto.Schema
import Ecto.Changeset
alias Example.Location.Xyz
schema "xyzs" do
field :name, :string
belongs_to :country, Example.Location.Country
belongs_to :federal_state, Example.Location.FederalState
belongs_to :city, Example.Location.City
timestamps()
end
@doc false
def changeset(%Xyz{} = xyz, attrs) do
school
|> cast(attrs, [:name, :country_id, :federal_state_id, :city_id])
|> validate_required([:name, :country_id, :federal_state_id, :city_id])
|> assoc_constraint(:country)
|> assoc_constraint(:federal_state)
|> assoc_constraint(:city)
end
end
答案 0 :(得分:3)
我创建了一个函数来检查3个字段中是否有1个存在并且还添加了正确的assoc_constraint
:
@doc false
def changeset(%Xyz{} = xyz, attrs) do
school
|> cast(attrs, [:name, :country_id, :federal_state_id, :city_id])
|> validate_required([:name])
|> validate_one_of_present([:country_id, :federal_state_id, :city_id])
end
def validate_one_of_present(changeset, fields) do
fields
|> Enum.filter(fn field ->
# Checks if a field is "present".
# The logic is copied from `validate_required` in Ecto.
case get_field(changeset, field) do
nil -> false
binary when is_binary(binary) -> String.trim_leading(binary) == ""
_ -> true
end
end)
|> case do
# Exactly one field was present.
[field] ->
without_id = field |> Atom.to_string |> String.replace_suffix("_id", "") |> String.to_existing_atom
assoc_constraint(changeset, without_id)
# Zero or more than one fields were present.
_ ->
add_error(changeset, hd(fields), "expected exactly one of #{inspect(fields)} to be present")
end
end
代码未经测试,如果您发现任何错误,请告诉我们!