module Database.Mallard.Types where
import           Control.Lens
import           Control.Monad.Catch
import           Crypto.Hash
import           Data.ByteString     (ByteString)
import           Data.FileEmbed
import           Data.Hashable
import           Data.HashMap.Strict (HashMap)
import qualified Data.HashMap.Strict as Map
import           Data.Int
import           Data.Text           (Text)
import           Data.Text.Lens
type MigrationTable = HashMap MigrationId Migration
inflateMigrationIds :: (MonadThrow m) => MigrationTable -> [MigrationId] -> m [Migration]
inflateMigrationIds mTable = mapM inflate
    where
        inflate mid =
            case Map.lookup mid mTable of
                Nothing -> throwM $ MigrationMissingFromTable mid
                Just m  -> return m
data MigrationMissingFromTable
    = MigrationMissingFromTable MigrationId
    deriving (Show)
instance Exception MigrationMissingFromTable
newtype MigrationId = MigrationId { unMigrationId :: Text }
    deriving (Eq, Ord, Hashable)
instance Show MigrationId where
    show (MigrationId txt) = txt ^. unpacked
type MigrationDigest = Digest SHA256
data Migration
    = Migration
        { _migrationName        :: MigrationId
        , _migrationDescription :: Text
        , _migrationRequires    :: [MigrationId]
        , _migrationChecksum    :: MigrationDigest
        , _migrationScript      :: Text
        }
    deriving (Show)
$(makeClassy ''Migration)
type TestTable = HashMap TestId Test
newtype TestId = TestId { unTestId :: Text }
    deriving (Eq, Ord, Hashable)
instance Show TestId where
    show (TestId txt) = txt ^. unpacked
data Test
    = Test
        { _testName        :: TestId
        , _testDescription :: Text
        , _testScript      :: Text
        }
$(makeClassy ''Test)
scriptsMigrationSchema :: [(Int64, ByteString)]
scriptsMigrationSchema =
    [ (0, $(embedFile "sql/mallard/0000-setup.sql"))
    , (1, $(embedFile "sql/mallard/0001-applied-migrations.sql"))
    ]