use Vtc.Ecto.Postgres.Utils
defpgmodule Vtc.Ecto.Postgres.PgFramestamp.Range.Migrations do
@moduledoc """
Migrations for adding framestamp range types, functions and constraints to a
Postgres database.
"""
alias Ecto.Migration
alias Vtc.Ecto.Postgres
alias Vtc.Ecto.Postgres.PgFramestamp
require Ecto.Migration
@doc section: :migrations_full
@doc """
Adds raw SQL queries to a migration for creating the database types, associated
functions, casts, operators, and operator families.
Safe to run multiple times when new functionality is added in updates to this library.
Existing values will be skipped.
## Types Created
Calling this macro creates the following type definitions:
```sql
CREATE TYPE framestamp_range AS RANGE (
subtype = framestamp,
subtype_diff = framestamp_range_private.subtype_diff
canonical = framestamp_range_private.canonicalization
);
```
```sql
CREATE TYPE framestamp_fastrange AS RANGE (
subtype = double precision,
subtype_diff = float8mi
);
```
## Schemas Created
Up to two schemas are created as detailed by the
[Configuring Database Objects](Vtc.Ecto.Postgres.PgFramestamp.Migrations.html#create_all/0-configuring-database-objects)
section below.
## Configuring Database Objects
To change where supporting functions are created, add the following to your
Repo confiugration:
```elixir
config :vtc, Vtc.Test.Support.Repo,
adapter: Ecto.Adapters.Postgres,
...
vtc: [
framestamp_range: [
functions_schema: :framestamp_range,
functions_prefix: "framestamp_range"
]
]
```
Option definitions are as follows:
- `functions_schema`: The postgres schema to store framestamp_range-related custom
functions.
- `functions_prefix`: A prefix to add before all functions. Defaults to
"framestamp_range" for any function created in the `:public` schema, and ""
otherwise.
## Examples
```elixir
defmodule MyMigration do
use Ecto.Migration
alias Vtc.Ecto.Postgres.PgFramestamp
require PgFramestamp.Migrations
def change do
PgFramestamp.Range.Migrations.create_all()
end
end
```
"""
@spec create_all() :: :ok
def create_all do
create_function_schemas()
create_func_subtype_diff()
create_type_framestamp_range()
create_func_canonicalization()
create_type_framestamp_fastrange()
create_func_framestamp_fastrange_from_stamps()
create_func_framestamp_fastrange_from_range()
# There is a limitation with PL/pgSQL where shell-types cannot be used as either
# arguments OR return types.
#
# However, in the user-facing API flow, the canonical function must be created
# before the range type with a shell type, then passed to the range type upon
# construction. Further, ALTER TYPE does not work on range functions out-of-the
# gate, so we cannot add it later... through the public API.
#
# Instead we are going to edit the pg_catalog directly and supply the function
# after-the-fact ourselves. Since this will all happen in a single transaction
# it should be functionally equivalent to creating it on the type as part of the
# initial call.
canonicalization = private_function(:canonicalization, Migration.repo())
Migration.execute("""
UPDATE pg_catalog.pg_range
SET
rngcanonical = '#{canonicalization}'::regproc
WHERE
pg_catalog.pg_range.rngcanonical = '-'::regproc
AND EXISTS (
SELECT * FROM pg_catalog.pg_type
WHERE pg_catalog.pg_type.oid = pg_catalog.pg_range.rngsubtype
AND pg_catalog.pg_type.typname = 'framestamp'
)
AND EXISTS (
SELECT * FROM pg_catalog.pg_type
WHERE pg_catalog.pg_type.oid = pg_catalog.pg_range.rngtypid
AND pg_catalog.pg_type.typname = 'framestamp_range'
);
""")
:ok
end
@doc section: :migrations_types
@doc """
Adds `framestamp_range` RANGE type.
"""
@spec create_type_framestamp_range() :: :ok
def create_type_framestamp_range do
subtype_diff = private_function(:subtype_diff, Migration.repo())
Migration.execute("""
DO $$ BEGIN
CREATE TYPE framestamp_range AS RANGE (
subtype = framestamp,
subtype_diff = #{subtype_diff}
);
EXCEPTION WHEN duplicate_object
THEN null;
END $$;
""")
end
@doc section: :migrations_types
@doc """
Adds `framestamp_fastrange` RANGE type that uses double-precision floats under the
hood.
"""
@spec create_type_framestamp_fastrange() :: :ok
def create_type_framestamp_fastrange do
Migration.execute("""
DO $$ BEGIN
CREATE TYPE framestamp_fastrange AS RANGE (
subtype = double precision,
subtype_diff = float8mi
);
EXCEPTION WHEN duplicate_object
THEN null;
END $$;
""")
end
@doc section: :migrations_functions
@doc """
Creates `framestamp_fastrange(:framestamp, framestamp)` to construct fast ranges out
of framestamps.
"""
@spec create_func_framestamp_fastrange_from_stamps() :: :ok
def create_func_framestamp_fastrange_from_stamps do
create_func =
Postgres.Utils.create_plpgsql_function(
"framestamp_fastrange",
args: [lower: :framestamp, upper: :framestamp],
returns: :framestamp_fastrange,
body: """
RETURN framestamp_fastrange(
CAST((lower).seconds as double precision),
CAST((upper).seconds as double precision)
);
"""
)
Migration.execute(create_func)
end
@doc section: :migrations_functions
@doc """
Creates `framestamp_fastrange(:framestamp_range)` to construct fast ranges out
of the slower `framestamp_range` type.
"""
@spec create_func_framestamp_fastrange_from_range() :: :ok
def create_func_framestamp_fastrange_from_range do
create_func =
Postgres.Utils.create_plpgsql_function(
"framestamp_fastrange",
args: [input: :framestamp_range],
returns: :framestamp_fastrange,
body: """
RETURN framestamp_fastrange(
CAST((LOWER(input)).seconds as double precision),
CAST((UPPER(input)).seconds as double precision)
);
"""
)
Migration.execute(create_func)
end
@doc section: :migrations_private_functions
@doc """
Creates `framestamp.__private__subtype_diff(a, b)` used by the range type for more
efficient GiST indexes.
"""
@spec create_func_subtype_diff() :: :ok
def create_func_subtype_diff do
create_func =
Postgres.Utils.create_plpgsql_function(
private_function(:subtype_diff, Migration.repo()),
args: [a: :framestamp, b: :framestamp],
declares: [diff: {:framestamp, "a - b"}],
returns: :"double precision",
body: """
RETURN CAST((diff).seconds as double precision);
"""
)
Migration.execute(create_func)
end
@doc section: :migrations_private_functions
@doc """
Creates `framestamp.__private__canonicalization(a, b, type)` used by the range
constructor to normalize ranges.
Output ranges have an inclusive lower bound and an exclusive upper bound.
"""
@spec create_func_canonicalization() :: :ok
def create_func_canonicalization do
Migration.execute("""
DO $$ BEGIN
CREATE TYPE framestamp_range;
EXCEPTION WHEN duplicate_object
THEN null;
END $$;
""")
framestamp_with_frames = PgFramestamp.Migrations.function(:with_frames, Migration.repo())
framestamp_with_seconds = PgFramestamp.Migrations.function(:with_seconds, Migration.repo())
create_func =
Postgres.Utils.create_plpgsql_function(
private_function(:canonicalization, Migration.repo()),
args: [input: :framestamp_range],
declares: [
single_frame: {:framestamp, "#{framestamp_with_frames}(1, (LOWER(input)).rate)"},
upper_stamp: {:framestamp, "UPPER(input)"},
lower_stamp: {:framestamp, "LOWER(input)"},
rates_match: {:boolean, "(lower_stamp).rate === (upper_stamp).rate"},
new_rate: :framerate
],
returns: :framestamp_range,
body: """
CASE
WHEN LOWER_INC(input) AND NOT UPPER_INC(input) AND rates_match THEN
RETURN input;
WHEN LOWER_INC(input) AND NOT UPPER_INC(input) THEN
new_rate := GREATEST((lower_stamp).rate, (upper_stamp).rate);
lower_stamp := #{framestamp_with_seconds}((lower_stamp).seconds, new_rate);
upper_stamp := #{framestamp_with_seconds}((upper_stamp).seconds, new_rate);
RETURN framestamp_range(lower_stamp, upper_stamp, '[)');
WHEN LOWER_INC(input) AND UPPER_INC(input) THEN
RETURN framestamp_range(lower_stamp, upper_stamp + single_frame, '[)');
WHEN NOT LOWER_INC(input) AND UPPER_INC(input) THEN
RETURN framestamp_range(lower_stamp + single_frame, upper_stamp + single_frame, '[)');
WHEN NOT LOWER_INC(input) AND NOT UPPER_INC(input) THEN
RETURN framestamp_range(lower_stamp + single_frame, upper_stamp, '[)');
END CASE;
"""
)
Migration.execute(create_func)
end
@doc section: :migrations_types
@doc """
Creates function schema as described by the
[Configuring Database Objects](Vtc.Ecto.Postgres.PgFramestamp.Range.Migrations.html#create_all/0-configuring-database-objects)
section above.
"""
@spec create_function_schemas() :: :ok
def create_function_schemas, do: Postgres.Utils.create_type_schemas(:framestamp_range)
@spec private_function(atom(), Ecto.Repo.t()) :: String.t()
defp private_function(name, repo) do
function_prefix = Postgres.Utils.type_private_function_prefix(repo, :framestamp_range)
"#{function_prefix}#{name}"
end
end