src/strata_mysql.erl

-module(strata_mysql).

-behaviour(strata_driver).

-export([up/1, down/1, list/1, add/3, delete/2, last/1, exec/2]).

-spec exec(any(), iodata()) -> ok | {error, any()}.
exec(Conn, Query) ->
    case mysql:query(Conn, Query, []) of
        {error, _Err} = E ->
            E;
        ok ->
            ok
    end.

-spec up(any()) -> ok | {error, any()}.
up(Conn) ->
    case mysql:query(Conn,
                     [<<"CREATE TABLE IF NOT EXISTS __strata_migrations ( ">>,
                      <<"  id SERIAL PRIMARY KEY, ">>,
                      <<"  name TEXT NOT NULL, ">>,
                      <<"  timestamp INTEGER NOT NULL ">>,
                      <<")">>],
                     [])
    of
        {error, _Err} = E ->
            E;
        ok ->
            ok
    end.

-spec down(any()) -> ok | {error, any()}.
down(Conn) ->
    case mysql:query(Conn, <<"DROP TABLE IF EXISTS __strata_migrations">>, []) of
        {error, _Err} = E ->
            E;
        ok ->
            ok
    end.

-spec list(any()) -> {ok, list()} | {error, any()}.
list(Conn) ->
    case mysql:query(Conn,
                     <<"SELECT id, name, timestamp FROM __strata_migrations ORDER BY timestamp ASC">>,
                     [])
    of
        {error, _Err} = E ->
            E;
        {ok, _, Rows} ->
            {ok, format_rows(Rows)}
    end.

-spec format_rows(list()) -> list().
format_rows(Rows) ->
    lists:map(fun format_row/1, Rows).

-spec format_row(list()) -> {integer(), module(), integer()} | {error, invalid_row}.
format_row([Id, Name, Timestamp]) ->
    {Id, binary_to_atom(Name), Timestamp};
format_row(_) ->
    {error, invalid_row}.

-spec add(any(), binary(), integer()) -> ok | {error, any()}.
add(Conn, Name, Timestamp) ->
    case mysql:query(Conn,
                     <<"INSERT INTO __strata_migrations (name, timestamp) VALUES (?, ?)">>,
                     [Name, Timestamp])
    of
        {error, _Err} = E ->
            E;
        ok ->
            ok
    end.

-spec last(any()) -> {error, any()} | {ok, {integer(), module(), integer()}}.
last(Conn) ->
    case mysql:query(Conn,
                     <<"SELECT id, name, timestamp FROM __strata_migrations ORDER BY timestamp DESC LIMIT 1">>,
                     [])
    of
        {error, _Err} = E ->
            E;
        {ok, _, [[Id, Name, Timestamp]]} ->
            {ok, {Id, binary_to_atom(Name), Timestamp}}
    end.

-spec delete(any(), integer()) -> ok | {error, any()}.
delete(Conn, Id) ->
    case mysql:query(Conn, <<"DELETE FROM __strata_migrations WHERE id = ?">>, [Id]) of
        {error, _Err} = E ->
            E;
        ok ->
            ok
    end.