validation - 使用 Ecto,验证具有 2 个不同相关模型的变更集是否具有相同的父模型

标签 validation elixir ecto changeset

在我的应用程序中,我有一个方法可以创建一个新的responseresponseplayermatch 都有 belongs_to 关系。

此外,playermatch 都与 teambelongs_to 关系。

看起来像这样:

schema

插入新的response 时,我想验证 playermatch 是否具有 player_idmatch_id 变更集中的外键属于同一个 team

目前我正在按如下方式实现这一目标。首先,定义一个自定义验证来检查属于外键的记录:

def validate_match_player(changeset) do
  player_team =
    Player
    |> Repo.get(get_field(changeset, :player_id))
    |> Map.get(:team_id)

  match_team =
    Match
    |> Repo.get(get_field(changeset, :match_id))
    |> Map.get(:team_id)

  cond do
    match_team == player_team -> changeset
    true -> changeset |> add_error(:player, "does not belong to the same team as the match")
  end
end

并将验证用作变更集的一部分:

def changeset(model, params \\ %{}) do
  model
  |> cast(params, [:player_id, :match_id, :message])
  |> validate_required([:player_id, :match_id, :message])
  |> foreign_key_constraint(:match_id)
  |> foreign_key_constraint(:player_id)
  |> validate_match_player()
  |> unique_constraint(
    :player,
    name: :responses_player_id_match_id_unique,
    message: "already has an response for this match"
  )
end

这工作正常,但涉及几个额外的 SQL 查询来查找相关记录,以便获取他们的 team_id 外键来比较它们。

有没有更好的方法来避免额外的查询,也许是使用约束?

最佳答案

我有两个可能的改进:

  • 应用程序级解决方案:您只需查询一次,而不是两次查询。
  • 数据库级解决方案:您为数据库中的检查创建一个触发器。

应用级解决方案

现在您有两个查询来检查球员和比赛是否属于同一个球队。这意味着两次往返数据库。如果你只使用一个查询,你可以减少一半,例如给定以下查询:

    SELECT COUNT(*)
      FROM players AS p
INNER JOIN matches AS m
        ON p.team_id = m.team_id
     WHERE p.id = NEW.player_id AND m.id = NEW.match_id

你会改变你的功能如下:

def validate_match_player(changeset) do
  player_id = get_field(changeset, :player_id)
  match_id = get_field(changeset, :match_id)

  [result] =
    Player
    |> join(:inner, [p], m in Match, on: p.team_id == m.team_id)
    |> where([p, m], p.id == ^player_id and m.id == ^match_id)
    |> select([p, m], %{count: count(p.id)})
    |> Repo.all()

  case result do
    %{count: 0} ->
      add_error(changeset, :player, "does not belong to the same team as the match")
    _ ->
      changeset
  end
end

数据库级解决方案

我假设您使用的是 PostgreSQL,因此我的回答将与您在 PostgreSQL 手册中找到的内容相对应。

没有(干净的)方法来在执行此操作的表中定义约束。约束只能访问定义它们的表。某些约束只能从它们定义的内容访问列,仅此而已 (CHECK CONSTRAINT)。

最好的方法是编写一个触发器来验证这两个字段,例如:

CREATE OR REPLACE FUNCTION trigger_validate_match_player()
  RETURNS TRIGGER AS $$
    IF (
         SELECT COUNT(*)
         FROM players AS p
         INNER JOIN matches AS m
         ON p.team_id = m.team_id
         WHERE p.id = NEW.player_id AND m.id = NEW.match_id
       ) = 0
    THEN
      RAISE 'does not belong to the same team as the match'
        USING ERRCODE 'invalid_match_player';
    END IF;

    RETURN NEW;
  $$ LANGUAGE plpgsql;

CREATE TRIGGER responses_validate_match_player
  BEFORE INSERT OR UPDATE ON responses
  FOR EACH ROW
  EXECUTE PROCEDURE trigger_validate_match_player();

前一个触发器在失败时会引发异常。这也意味着 Ecto 将引发异常。你可以看到如何处理这个异常here .

最后,除非您使用类似 sqitch 的东西,否则维护触发器并不容易。用于数据库迁移。

PS: If you're curious, the very dirty way of doing this in a CHECK constraint is by defining a PostgreSQL function that basically bypasses the limitation. I wouldn't recommend it.

希望对您有所帮助:)

关于validation - 使用 Ecto,验证具有 2 个不同相关模型的变更集是否具有相同的父模型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58889077/

相关文章:

elixir - Map.get(分配, :key) returns nil instead of expected value in the view

mysql - OS X、Elixir、Ecto、Crypto、MySQL

javascript - 使用 JavaScript 验证 URL?

json - 将Play框架与Scala结合使用,如何使Json验证消息人性化?

javascript - 逐行验证

elixir - 如何验证 ecto 中指定关联的长度?

elixir - 使用 Ecto 运行自定义 sql 查询

php - 文件类型 PHP 验证

elixir - 在 Elixir 中使用多个键从 map 中选择多个项目

reactjs - 图片未加载但路径正确