# `PhoenixKit.Migration`
[🔗](https://github.com/BeamLabEU/phoenix_kit/blob/v1.7.164/lib/phoenix_kit/migration.ex#L8)

Migrations create and modify the database tables PhoenixKit needs to function.

## Usage

To use migrations in your application you'll need to generate an `Ecto.Migration` that wraps
calls to `PhoenixKit.Migration`:

```bash
mix ecto.gen.migration add_phoenix_kit
```

Open the generated migration in your editor and call the `up` and `down` functions on
`PhoenixKit.Migration`:

```elixir
defmodule MyApp.Repo.Migrations.AddPhoenixKit do
  use Ecto.Migration

  def up, do: PhoenixKit.Migrations.up()

  def down, do: PhoenixKit.Migrations.down()
end
```

This will run all of PhoenixKit's versioned migrations for your database.

Now, run the migration to create the table:

```bash
mix ecto.migrate
```

Migrations between versions are idempotent. As new versions are released, you may need to run
additional migrations. To do this, generate a new migration:

```bash
mix ecto.gen.migration upgrade_phoenix_kit_to_v2
```

Open the generated migration in your editor and call the `up` and `down` functions on
`PhoenixKit.Migration`, passing a version number:

```elixir
defmodule MyApp.Repo.Migrations.UpgradePhoenixKitToV2 do
  use Ecto.Migration

  def up, do: PhoenixKit.Migrations.up(version: 2)

  def down, do: PhoenixKit.Migrations.down(version: 2)
end
```

## Isolation with Prefixes

PhoenixKit supports namespacing through PostgreSQL schemas, also called "prefixes" in Ecto. With
prefixes your auth tables can reside outside of your primary schema (usually public) and you can
have multiple separate auth systems.

To use a prefix you first have to specify it within your migration:

```elixir
defmodule MyApp.Repo.Migrations.AddPrefixedPhoenixKitTables do
  use Ecto.Migration

  def up, do: PhoenixKit.Migrations.up(prefix: "auth")

  def down, do: PhoenixKit.Migrations.down(prefix: "auth")
end
```

The migration will create the "auth" schema and all tables within
that schema. With the database migrated you'll then specify the prefix in your configuration:

```elixir
config :phoenix_kit,
  prefix: "auth",
  ...
```

In some cases, for example if your "auth" schema already exists and your database user in
production doesn't have permissions to create a new schema, trying to create the schema from the
migration will result in an error. In such situations, it may be useful to inhibit the creation
of the "auth" schema:

```elixir
defmodule MyApp.Repo.Migrations.AddPrefixedPhoenixKitTables do
  use Ecto.Migration

  def up, do: PhoenixKit.Migrations.up(prefix: "auth", create_schema: false)

  def down, do: PhoenixKit.Migrations.down(prefix: "auth")
end
```

## Migrating Without Ecto

If your application uses something other than Ecto for migrations, be it an external system or
another ORM, it may be helpful to create plain SQL migrations for PhoenixKit database schema changes.

The simplest mechanism for obtaining the SQL changes is to create the migration locally and run
`mix ecto.migrate --log-migrations-sql`. That will log all of the generated SQL, which you can
then paste into your migration system of choice.

# `down`

```elixir
@callback down(Keyword.t()) :: :ok
```

Migrates storage down to the previous version.

# `migrated_version`

```elixir
@callback migrated_version(Keyword.t()) :: non_neg_integer()
```

Identifies the last migrated version.

# `up`

```elixir
@callback up(Keyword.t()) :: :ok
```

Migrates storage up to the latest version.

# `down`

Run the `down` changes for all migrations between the current version and the initial version.

## Example

Run all migrations from current version down to the first:

    PhoenixKit.Migration.down()

Run migrations down to and including a specified version:

    PhoenixKit.Migration.down(version: 1)

Run migrations in an alternate prefix:

    PhoenixKit.Migration.down(prefix: "auth")

# `ensure_current`

```elixir
@spec ensure_current(
  Ecto.Repo.t(),
  keyword()
) :: :ok
```

Idempotently brings `repo` up to the latest PhoenixKit migration version.

Designed for test helpers and re-runnable boot paths where the
database is long-lived but the calling process restarts on every
invocation.

## Why this exists

The natural-looking pattern

    Ecto.Migrator.run(repo, [{0, PhoenixKit.Migration}], :up, all: true)

is broken for re-runnable contexts: `Ecto.Migrator` records "version
0 applied" in `schema_migrations` after the first call and filters
`{0, PhoenixKit.Migration}` out of pending on every subsequent call.
`PhoenixKit.Migration.up/1` is never re-invoked, so newly-shipped
Vxxx migrations don't get applied even though PhoenixKit's own marker
(the comment on the `phoenix_kit` table) is itself idempotent.

`ensure_current/2` works around that by passing a fresh wall-clock
version (`:os.system_time(:microsecond)`) to `Ecto.Migrator.up/4` on
every call. Ecto sees a "new" migration each time and invokes the
inner runner; PhoenixKit's marker then short-circuits if there's
nothing new to apply. The `schema_migrations` table accumulates one
row per call — cosmetic noise acceptable for the test-DB use case.
Microsecond precision keeps the collision and clock-skew windows
small enough that an NTP correction would have to rewind the clock
by µs at exactly the wrong moment to hide a newly-shipped migration.

For one-shot production migrations, prefer the documented
`mix ecto.migrate` path with a hand-rolled migration that calls
`PhoenixKit.Migration.up/1` directly.

## Options

Forwarded verbatim to `Ecto.Migrator.up/4` and through to
`PhoenixKit.Migration.up/1`. Common values:

  * `:log` — Ecto-level migration log (`:info` default; `false` to
    silence)
  * `:prefix` — runs PhoenixKit's tables under a non-default schema

## Return contract

Returns `:ok` on success. Failures (advisory-lock contention,
migration crashes, connection errors) propagate as raises from
`Ecto.Migrator.up/4`; `ensure_current/2` does not wrap them in
`{:error, _}`.

## Example

    # In test/test_helper.exs
    PhoenixKit.Migration.ensure_current(MyApp.Test.Repo, log: false)

# `migrated_version`

Check the latest version the database is migrated to.

## Example

    PhoenixKit.Migration.migrated_version()

# `up`

Run the `up` changes for all migrations between the initial version and the current version.

## Example

Run all migrations up to the current version:

    PhoenixKit.Migration.up()

Run migrations up to a specified version:

    PhoenixKit.Migration.up(version: 2)

Run migrations in an alternate prefix:

    PhoenixKit.Migration.up(prefix: "auth")

Run migrations in an alternate prefix but don't try to create the schema:

    PhoenixKit.Migration.up(prefix: "auth", create_schema: false)

---

*Consult [api-reference.md](api-reference.md) for complete listing*
