唯一性的 Phoenix 控制器测试不起作用

Phoenix Controller Testing for Uniqueness Not Working

我的用户 table 中的用户名和电子邮件都有标准的唯一性约束。现在,我正在测试我的 UserController 在用户创建期间的重复,似乎我的模型上的 unique_constraint 在单元期间不起作用测试但在手动测试中很好。我在这里做错了什么?

迁移:

defmodule MyApi.Repo.Migrations.CreateUser do
  use Ecto.Migration

  def change do
    create table(:users) do
      add :username, :string, null: false
      add :email, :string, null: false
      add :password, :string, null: false
      add :first_name, :string, null: false
      add :last_name, :string, null: false

      timestamps
    end

    create unique_index(:users, [:username])
    create unique_index(:users, [:email])
  end
end

控制器:

defmodule MyApi.V1.UserController do
  use MyApi.Web, :controller

  alias MyApi.User

  def create(conn, %{"data" => data}) do
    user_attrs = JaSerializer.Params.to_attributes(data)
    user_changeset = User.changeset(%User{}, user_attrs)

    case Repo.insert(user_changeset) do
      {:ok, user} ->
        conn
        |> put_status(201)
        |> render(:show, data: user)
      {:error, user_changeset} ->
        conn
        |> put_status(500)
        |> render(:errors, data: user_changeset)
    end
  end
end

测试:

...

test "User creation with non-unique username", %{conn: conn} do
    user_changeset = User.changeset(%User{}, @valid_attrs)
    Repo.insert(user_changeset)

    conn = post(conn, v1_user_path(conn, :create), user_params(@valid_attrs))
    response = json_response(conn, 500)

    errors = response["errors"]
    assert errors
    assert Enum.count(errors) == 1

    {:ok, error} = Enum.fetch(errors, 0)
    assert error["title"] == "has already been taken"

    refute Repo.get_by(User, username: "name1")
end
...

混合测试

1) test User creation with non-unique username (MyApi.V1.UserControllerTest)
     test/controllers/v1/user_controller_test.exs:57
     ** (RuntimeError) expected response with status 500, got: 201, with body:
     {"jsonapi":{"version":"1.0"},"data":{"type":"user","id":"269","attributes":{"username":"name1","last-name":"lastname1","first-name":"first","email":"first@email.com"}}}
     stacktrace:
       (phoenix) lib/phoenix/test/conn_test.ex:362: Phoenix.ConnTest.response/2
       (phoenix) lib/phoenix/test/conn_test.ex:408: Phoenix.ConnTest.json_response/2
       test/controllers/v1/user_controller_test.exs:61: (test)

如果您 运行 mix test 在创建迁移后编辑相同的迁移以添加 unique_index 调用,您的数据库将不会自动迁移到新版本同样的迁移。

有2种解决方案:

  1. 创建一个单独的迁移,将 unique_index 添加到 table。

    您的测试数据库将在您下次 运行 mix test 时自动迁移。

  2. 手动 drop/create 或 rollback/migrate(如果这是您的最新迁移)测试数据库:

    或者:

    MIX_ENV=test mix do ecto.drop, ecto.create, ecto.migrate
    

    或(如果这是您的最新迁移):

    MIX_ENV=test mix do ecto.rollback, ecto.migrate