2025-04-16 10:03:13 -03:00

265 lines
7.3 KiB
Elixir

Code.require_file "../support/file_helpers.exs", __DIR__
defmodule Ecto.Integration.MigratorTest do
use Ecto.Integration.Case
import Support.FileHelpers
import ExUnit.CaptureLog
import Ecto.Migrator
alias Ecto.Integration.{TestRepo, PoolRepo}
alias Ecto.Migration.SchemaMigration
setup config do
Process.register(self(), config.test)
PoolRepo.delete_all(SchemaMigration)
:ok
end
defmodule AnotherSchemaMigration do
use Ecto.Migration
def change do
execute TestRepo.create_prefix("bad_schema_migrations"),
TestRepo.drop_prefix("bad_schema_migrations")
create table(:schema_migrations, prefix: "bad_schema_migrations") do
add :version, :string
add :inserted_at, :integer
end
end
end
defmodule BrokenLinkMigration do
use Ecto.Migration
def change do
Task.start_link(fn -> raise "oops" end)
Process.sleep(:infinity)
end
end
defmodule GoodMigration do
use Ecto.Migration
def up do
create table(:good_migration)
end
def down do
drop table(:good_migration)
end
end
defmodule BadMigration do
use Ecto.Migration
def change do
execute "CREATE WHAT"
end
end
test "migrations up and down" do
assert migrated_versions(PoolRepo) == []
assert up(PoolRepo, 31, GoodMigration, log: false) == :ok
[migration] = PoolRepo.all(SchemaMigration)
assert migration.version == 31
assert migration.inserted_at
assert migrated_versions(PoolRepo) == [31]
assert up(PoolRepo, 31, GoodMigration, log: false) == :already_up
assert migrated_versions(PoolRepo) == [31]
assert down(PoolRepo, 32, GoodMigration, log: false) == :already_down
assert migrated_versions(PoolRepo) == [31]
assert down(PoolRepo, 31, GoodMigration, log: false) == :ok
assert migrated_versions(PoolRepo) == []
end
@tag :prefix
test "does not commit migration if insert into schema migration fails" do
# First we create a new schema migration table in another prefix
assert up(PoolRepo, 33, AnotherSchemaMigration, log: false) == :ok
assert migrated_versions(PoolRepo) == [33]
catch_error(up(PoolRepo, 34, GoodMigration, log: false, prefix: "bad_schema_migrations"))
catch_error(PoolRepo.all("good_migration"))
catch_error(PoolRepo.all("good_migration", prefix: "bad_schema_migrations"))
assert down(PoolRepo, 33, AnotherSchemaMigration, log: false) == :ok
end
test "ecto-generated migration queries pass schema_migration in telemetry options" do
handler = fn _event_name, _measurements, metadata ->
send(self(), metadata)
end
# migration table creation
Process.put(:telemetry, handler)
migrated_versions(PoolRepo, log: false)
assert_received %{options: [schema_migration: true]}
# transaction begin statement
Process.put(:telemetry, handler)
migrated_versions(PoolRepo, skip_table_creation: true, log: false)
assert_received %{options: [schema_migration: true]}
# retrieving the migration versions
Process.put(:telemetry, handler)
migrated_versions(PoolRepo, migration_lock: false, skip_table_creation: true, log: false)
assert_received %{options: [schema_migration: true]}
end
test "bad execute migration" do
assert catch_error(up(PoolRepo, 31, BadMigration, log: false))
assert DynamicSupervisor.which_children(Ecto.MigratorSupervisor) == []
end
test "broken link migration" do
Process.flag(:trap_exit, true)
assert capture_log(fn ->
{:ok, pid} = Task.start_link(fn -> up(PoolRepo, 31, BrokenLinkMigration, log: false) end)
assert_receive {:EXIT, ^pid, _}
end) =~ "oops"
assert capture_log(fn ->
catch_exit(up(PoolRepo, 31, BrokenLinkMigration, log: false))
end) =~ "oops"
end
test "run up to/step migration", config do
in_tmp fn path ->
create_migration(47, config)
create_migration(48, config)
assert [47] = run(PoolRepo, path, :up, step: 1, log: false)
assert count_entries() == 1
assert [48] = run(PoolRepo, path, :up, to: 48, log: false)
end
end
test "run down to/step migration", config do
in_tmp fn path ->
migrations = [
create_migration(49, config),
create_migration(50, config),
]
assert [49, 50] = run(PoolRepo, path, :up, all: true, log: false)
purge migrations
assert [50] = run(PoolRepo, path, :down, step: 1, log: false)
purge migrations
assert count_entries() == 1
assert [50] = run(PoolRepo, path, :up, to: 50, log: false)
end
end
test "runs all migrations", config do
in_tmp fn path ->
migrations = [
create_migration(53, config),
create_migration(54, config),
]
assert [53, 54] = run(PoolRepo, path, :up, all: true, log: false)
assert [] = run(PoolRepo, path, :up, all: true, log: false)
purge migrations
assert [54, 53] = run(PoolRepo, path, :down, all: true, log: false)
purge migrations
assert count_entries() == 0
assert [53, 54] = run(PoolRepo, path, :up, all: true, log: false)
end
end
test "does not commit half transactions on bad syntax", config do
in_tmp fn path ->
migrations = [
create_migration(64, config),
create_migration("65_+", config)
]
assert_raise SyntaxError, fn ->
run(PoolRepo, path, :up, all: true, log: false)
end
refute_received {:up, _}
assert count_entries() == 0
purge migrations
end
end
@tag :lock_for_migrations
test "raises when connection pool is too small" do
config = Application.fetch_env!(:ecto_sql, PoolRepo)
config = Keyword.merge(config, pool_size: 1)
Application.put_env(:ecto_sql, __MODULE__.SingleConnectionRepo, config)
defmodule SingleConnectionRepo do
use Ecto.Repo, otp_app: :ecto_sql, adapter: PoolRepo.__adapter__()
end
{:ok, _pid} = SingleConnectionRepo.start_link()
in_tmp fn path ->
exception_message = ~r/Migrations failed to run because the connection pool size is less than 2/
assert_raise Ecto.MigrationError, exception_message, fn ->
run(SingleConnectionRepo, path, :up, all: true, log: false)
end
end
end
test "does not raise when connection pool is too small but there is no lock" do
config = Application.fetch_env!(:ecto_sql, PoolRepo)
config = Keyword.merge(config, pool_size: 1, migration_lock: nil)
Application.put_env(:ecto_sql, __MODULE__.SingleConnectionNoLockRepo, config)
defmodule SingleConnectionNoLockRepo do
use Ecto.Repo, otp_app: :ecto_sql, adapter: PoolRepo.__adapter__()
end
{:ok, _pid} = SingleConnectionNoLockRepo.start_link()
in_tmp fn path ->
run(SingleConnectionNoLockRepo, path, :up, all: true, log: false)
end
end
defp count_entries() do
PoolRepo.aggregate(SchemaMigration, :count, :version)
end
defp create_migration(num, config) do
module = Module.concat(__MODULE__, "Migration#{num}")
File.write! "#{num}_migration_#{num}.exs", """
defmodule #{module} do
use Ecto.Migration
def up do
send #{inspect config.test}, {:up, #{inspect num}}
end
def down do
send #{inspect config.test}, {:down, #{inspect num}}
end
end
"""
module
end
defp purge(modules) do
Enum.each(List.wrap(modules), fn m ->
:code.delete m
:code.purge m
end)
end
end