265 lines
7.3 KiB
Elixir
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
|