{-# OPTIONS_GHC -fno-warn-unused-binds -fno-warn-name-shadowing#-}
{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE UndecidableInstances #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE CPP #-}
module Database.Beam.Sqlite.Syntax
(
SqliteSyntax(..)
, SqliteCommandSyntax(..)
, SqliteSelectSyntax(..), SqliteInsertSyntax(..)
, SqliteUpdateSyntax(..), SqliteDeleteSyntax(..)
, SqliteOnConflictSyntax(..)
, SqliteInsertValuesSyntax(..)
, SqliteColumnSchemaSyntax(..)
, SqliteExpressionSyntax(..), SqliteValueSyntax(..)
, SqliteTableNameSyntax(..)
, SqliteFieldNameSyntax(..)
, SqliteAggregationSetQuantifierSyntax(..)
, fromSqliteExpression
, SqliteDataTypeSyntax(..)
, sqliteTextType, sqliteBlobType
, sqliteBigIntType, sqliteSerialType
, fromSqliteCommand, formatSqliteInsert, formatSqliteInsertOnConflict
, emit, emitValue, parens, commas, quotedIdentifier
, sqliteEscape, withPlaceholders
, sqliteRenderSyntaxScript
) where
import Database.Beam.Backend.Internal.Compat
import Database.Beam.Backend.SQL
import Database.Beam.Backend.SQL.AST (ExtractField(..))
import Database.Beam.Haskell.Syntax
import Database.Beam.Migrate.Checks (HasDataTypeCreatedCheck(..))
import Database.Beam.Migrate.SQL.Builder hiding (fromSqlConstraintAttributes)
import Database.Beam.Migrate.SQL.SQL92
import Database.Beam.Migrate.Serialization
import Database.Beam.Query hiding (ExtractField(..))
import Data.ByteString (ByteString)
import qualified Data.ByteString as B
import Data.ByteString.Builder
import qualified Data.ByteString.Lazy.Char8 as BL
import Data.Coerce
import qualified Data.DList as DL
import Data.Hashable
import Data.Int
import Data.Maybe
import Data.Scientific
import Data.String
import qualified Data.Text as T
import qualified Data.Text.Encoding as TE
import qualified Data.Text.Lazy as TL
import Data.Time
import Data.Word
#if !MIN_VERSION_base(4, 11, 0)
import Data.Semigroup
#endif
import GHC.TypeLits
import Database.SQLite.Simple (SQLData(..))
import GHC.Float
import GHC.Generics
data SqliteSyntax = SqliteSyntax ((SQLData -> Builder) -> Builder) (DL.DList SQLData)
newtype SqliteData = SqliteData SQLData
instance Show SqliteSyntax where
show (SqliteSyntax s d) =
"SqliteSyntax (" <> show (toLazyByteString (withPlaceholders s)) <> ") " <> show d
instance Sql92DisplaySyntax SqliteSyntax where
displaySyntax = BL.unpack . sqliteRenderSyntaxScript
instance Semigroup SqliteSyntax where
(<>) = mappend
instance Monoid SqliteSyntax where
mempty = SqliteSyntax (\_ -> mempty) mempty
mappend (SqliteSyntax ab av) (SqliteSyntax bb bv) =
SqliteSyntax (\v -> ab v <> bb v) (av <> bv)
instance Eq SqliteSyntax where
SqliteSyntax ab av == SqliteSyntax bb bv =
toLazyByteString (withPlaceholders ab) ==
toLazyByteString (withPlaceholders bb) &&
av == bv
instance Hashable SqliteSyntax where
hashWithSalt salt (SqliteSyntax s d) =
hashWithSalt salt ( toLazyByteString (withPlaceholders s)
, map SqliteData (DL.toList d) )
instance Hashable SqliteData where
hashWithSalt salt (SqliteData (SQLInteger i)) = hashWithSalt salt (0 :: Int, i)
hashWithSalt salt (SqliteData (SQLFloat d)) = hashWithSalt salt (1 :: Int, d)
hashWithSalt salt (SqliteData (SQLText t)) = hashWithSalt salt (2 :: Int, t)
hashWithSalt salt (SqliteData (SQLBlob b)) = hashWithSalt salt (3 :: Int, b)
hashWithSalt salt (SqliteData SQLNull) = hashWithSalt salt (4 :: Int)
withPlaceholders :: ((SQLData -> Builder) -> Builder) -> Builder
withPlaceholders build = build (\_ -> "?")
emit :: ByteString -> SqliteSyntax
emit b = SqliteSyntax (\_ -> byteString b) mempty
emit' :: Show a => a -> SqliteSyntax
emit' x = SqliteSyntax (\_ -> byteString (fromString (show x))) mempty
quotedIdentifier :: T.Text -> SqliteSyntax
quotedIdentifier txt = emit "\"" <> SqliteSyntax (\_ -> stringUtf8 (T.unpack (sqliteEscape txt))) mempty <> emit "\""
sqliteEscape :: T.Text -> T.Text
sqliteEscape = T.concatMap (\c -> if c == '"' then "\"\"" else T.singleton c)
emitValue :: SQLData -> SqliteSyntax
emitValue v = SqliteSyntax ($ v) (DL.singleton v)
sqliteRenderSyntaxScript :: SqliteSyntax -> BL.ByteString
sqliteRenderSyntaxScript (SqliteSyntax s _) =
toLazyByteString . s $ \case
SQLInteger i -> int64Dec i
SQLFloat d -> doubleDec d
SQLText t -> TE.encodeUtf8Builder (sqliteEscape t)
SQLBlob b -> char8 'X' <> char8 '\'' <>
foldMap word8Hex (B.unpack b) <>
char8 '\''
SQLNull -> "NULL"
data SqliteCommandSyntax
= SqliteCommandSyntax SqliteSyntax
| SqliteCommandInsert SqliteInsertSyntax
fromSqliteCommand :: SqliteCommandSyntax -> SqliteSyntax
fromSqliteCommand (SqliteCommandSyntax s) = s
fromSqliteCommand (SqliteCommandInsert (SqliteInsertSyntax tbl fields values onConflict)) =
formatSqliteInsertOnConflict tbl fields values onConflict
newtype SqliteSelectSyntax = SqliteSelectSyntax { fromSqliteSelect :: SqliteSyntax }
newtype SqliteOnConflictSyntax = SqliteOnConflictSyntax { fromSqliteOnConflict :: SqliteSyntax }
data SqliteInsertSyntax
= SqliteInsertSyntax
{ sqliteInsertTable :: !SqliteTableNameSyntax
, sqliteInsertFields :: [ T.Text ]
, sqliteInsertValues :: !SqliteInsertValuesSyntax
, sqliteInsertOnConflict :: !(Maybe SqliteOnConflictSyntax)
}
newtype SqliteUpdateSyntax = SqliteUpdateSyntax { fromSqliteUpdate :: SqliteSyntax }
newtype SqliteDeleteSyntax = SqliteDeleteSyntax { fromSqliteDelete :: SqliteSyntax }
newtype SqliteSelectTableSyntax = SqliteSelectTableSyntax { fromSqliteSelectTable :: SqliteSyntax }
data SqliteExpressionSyntax
= SqliteExpressionSyntax SqliteSyntax
| SqliteExpressionDefault
deriving (Show, Eq, Generic)
instance Hashable SqliteExpressionSyntax
newtype SqliteFromSyntax = SqliteFromSyntax { fromSqliteFromSyntax :: SqliteSyntax }
newtype SqliteComparisonQuantifierSyntax = SqliteComparisonQuantifierSyntax { fromSqliteComparisonQuantifier :: SqliteSyntax }
newtype SqliteAggregationSetQuantifierSyntax = SqliteAggregationSetQuantifierSyntax { fromSqliteAggregationSetQuantifier :: SqliteSyntax }
newtype SqliteProjectionSyntax = SqliteProjectionSyntax { fromSqliteProjection :: SqliteSyntax }
newtype SqliteGroupingSyntax = SqliteGroupingSyntax { fromSqliteGrouping :: SqliteSyntax }
newtype SqliteOrderingSyntax = SqliteOrderingSyntax { fromSqliteOrdering :: SqliteSyntax }
newtype SqliteValueSyntax = SqliteValueSyntax { fromSqliteValue :: SqliteSyntax }
newtype SqliteTableSourceSyntax = SqliteTableSourceSyntax { fromSqliteTableSource :: SqliteSyntax }
newtype SqliteFieldNameSyntax = SqliteFieldNameSyntax { fromSqliteFieldNameSyntax :: SqliteSyntax }
data SqliteInsertValuesSyntax
= SqliteInsertExpressions [ [ SqliteExpressionSyntax ] ]
| SqliteInsertFromSql SqliteSelectSyntax
newtype SqliteCreateTableSyntax = SqliteCreateTableSyntax { fromSqliteCreateTable :: SqliteSyntax }
data SqliteTableOptionsSyntax = SqliteTableOptionsSyntax SqliteSyntax SqliteSyntax
data SqliteColumnSchemaSyntax
= SqliteColumnSchemaSyntax
{ fromSqliteColumnSchema :: SqliteSyntax
, sqliteIsSerialColumn :: Bool }
deriving (Show, Eq, Generic)
instance Hashable SqliteColumnSchemaSyntax
instance Sql92DisplaySyntax SqliteColumnSchemaSyntax where
displaySyntax = displaySyntax . fromSqliteColumnSchema
data SqliteDataTypeSyntax
= SqliteDataTypeSyntax
{ fromSqliteDataType :: SqliteSyntax
, sqliteDataTypeToHs :: HsDataType
, sqliteDataTypeSerialized :: BeamSerializedDataType
, sqliteDataTypeSerial :: Bool
} deriving (Show, Eq, Generic)
instance Hashable SqliteDataTypeSyntax where
hashWithSalt salt (SqliteDataTypeSyntax s _ _ _) = hashWithSalt salt s
instance Sql92DisplaySyntax SqliteDataTypeSyntax where
displaySyntax = displaySyntax . fromSqliteDataType
data SqliteColumnConstraintDefinitionSyntax
= SqliteColumnConstraintDefinitionSyntax
{ fromSqliteColumnConstraintDefinition :: SqliteSyntax
, sqliteColumnConstraintDefinitionSerialized :: BeamSerializedConstraintDefinition
} deriving (Show, Eq)
instance Hashable SqliteColumnConstraintDefinitionSyntax where
hashWithSalt salt (SqliteColumnConstraintDefinitionSyntax s _) = hashWithSalt salt s
instance Sql92DisplaySyntax SqliteColumnConstraintDefinitionSyntax where
displaySyntax = displaySyntax . fromSqliteColumnConstraintDefinition
data SqliteColumnConstraintSyntax
= SqliteColumnConstraintSyntax
{ fromSqliteColumnConstraint :: SqlConstraintAttributesBuilder -> SqliteSyntax
, sqliteColumnConstraintSerialized :: BeamSerializedConstraint }
data SqliteTableConstraintSyntax
= SqliteTableConstraintSyntax
{ fromSqliteTableConstraint :: SqliteSyntax
, sqliteTableConstraintPrimaryKey :: Maybe [ T.Text ] }
data SqliteMatchTypeSyntax
= SqliteMatchTypeSyntax
{ fromSqliteMatchType :: SqliteSyntax
, sqliteMatchTypeSerialized :: BeamSerializedMatchType }
data SqliteReferentialActionSyntax
= SqliteReferentialActionSyntax
{ fromSqliteReferentialAction :: SqliteSyntax
, sqliteReferentialActionSerialized :: BeamSerializedReferentialAction }
newtype SqliteAlterTableSyntax = SqliteAlterTableSyntax { fromSqliteAlterTable :: SqliteSyntax }
newtype SqliteAlterTableActionSyntax = SqliteAlterTableActionSyntax { fromSqliteAlterTableAction :: Maybe SqliteSyntax }
newtype SqliteAlterColumnActionSyntax = SqliteAlterColumnActionSyntax { fromSqliteAlterColumnAction :: Maybe SqliteSyntax }
newtype SqliteDropTableSyntax = SqliteDropTableSyntax { fromSqliteDropTable :: SqliteSyntax }
newtype SqliteTableNameSyntax = SqliteTableNameSyntax { fromSqliteTableName :: SqliteSyntax }
fromSqliteExpression :: SqliteExpressionSyntax -> SqliteSyntax
fromSqliteExpression (SqliteExpressionSyntax s) = s
fromSqliteExpression SqliteExpressionDefault = emit "NULL /* DEFAULT */"
sqliteExpressionSerialized :: SqliteExpressionSyntax -> BeamSerializedExpression
sqliteExpressionSerialized = BeamSerializedExpression . TE.decodeUtf8 . BL.toStrict .
sqliteRenderSyntaxScript . fromSqliteExpression
formatSqliteInsert :: SqliteTableNameSyntax -> [ T.Text ] -> SqliteInsertValuesSyntax -> SqliteSyntax
formatSqliteInsert tblNm fields values =
formatSqliteInsertOnConflict tblNm fields values Nothing
formatSqliteInsertOnConflict :: SqliteTableNameSyntax -> [ T.Text ] -> SqliteInsertValuesSyntax -> Maybe SqliteOnConflictSyntax -> SqliteSyntax
formatSqliteInsertOnConflict tblNm fields values onConflict = mconcat
[ emit "INSERT INTO "
, fromSqliteTableName tblNm
, parens (commas (map quotedIdentifier fields))
, emit " "
, case values of
SqliteInsertFromSql (SqliteSelectSyntax select) -> select
SqliteInsertExpressions es ->
emit "VALUES " <> commas (map (\row -> parens (commas (map fromSqliteExpression row)) ) es)
, maybe mempty ((emit " " <>) . fromSqliteOnConflict) onConflict
]
instance IsSql92Syntax SqliteCommandSyntax where
type Sql92SelectSyntax SqliteCommandSyntax = SqliteSelectSyntax
type Sql92InsertSyntax SqliteCommandSyntax = SqliteInsertSyntax
type Sql92UpdateSyntax SqliteCommandSyntax = SqliteUpdateSyntax
type Sql92DeleteSyntax SqliteCommandSyntax = SqliteDeleteSyntax
selectCmd = SqliteCommandSyntax . fromSqliteSelect
insertCmd = SqliteCommandInsert
updateCmd = SqliteCommandSyntax . fromSqliteUpdate
deleteCmd = SqliteCommandSyntax . fromSqliteDelete
instance IsSql92DdlCommandSyntax SqliteCommandSyntax where
type Sql92DdlCommandCreateTableSyntax SqliteCommandSyntax = SqliteCreateTableSyntax
type Sql92DdlCommandAlterTableSyntax SqliteCommandSyntax = SqliteAlterTableSyntax
type Sql92DdlCommandDropTableSyntax SqliteCommandSyntax = SqliteDropTableSyntax
createTableCmd = SqliteCommandSyntax . fromSqliteCreateTable
alterTableCmd = SqliteCommandSyntax . fromSqliteAlterTable
dropTableCmd = SqliteCommandSyntax . fromSqliteDropTable
instance IsSql92TableNameSyntax SqliteTableNameSyntax where
tableName Nothing tbl = SqliteTableNameSyntax (quotedIdentifier tbl)
tableName (Just sch) tbl = SqliteTableNameSyntax (quotedIdentifier sch <> emit "." <> quotedIdentifier tbl)
instance IsSql92DropTableSyntax SqliteDropTableSyntax where
type Sql92DropTableTableNameSyntax SqliteDropTableSyntax = SqliteTableNameSyntax
dropTableSyntax nm = SqliteDropTableSyntax (emit "DROP TABLE " <> fromSqliteTableName nm)
instance IsSql92AlterTableSyntax SqliteAlterTableSyntax where
type Sql92AlterTableAlterTableActionSyntax SqliteAlterTableSyntax = SqliteAlterTableActionSyntax
type Sql92AlterTableTableNameSyntax SqliteAlterTableSyntax = SqliteTableNameSyntax
alterTableSyntax nm action =
SqliteAlterTableSyntax $
case fromSqliteAlterTableAction action of
Just alterTable ->
emit "ALTER TABLE " <> fromSqliteTableName nm <> emit " " <> alterTable
Nothing ->
emit "SELECT 1"
instance IsSql92AlterTableActionSyntax SqliteAlterTableActionSyntax where
type Sql92AlterTableAlterColumnActionSyntax SqliteAlterTableActionSyntax = SqliteAlterColumnActionSyntax
type Sql92AlterTableColumnSchemaSyntax SqliteAlterTableActionSyntax = SqliteColumnSchemaSyntax
alterColumnSyntax columnNm columnAction =
SqliteAlterTableActionSyntax $
case fromSqliteAlterColumnAction columnAction of
Nothing -> Nothing
Just columnAction ->
Just (emit "ALTER COLUMN " <> quotedIdentifier columnNm <> columnAction)
addColumnSyntax columnNm schema =
SqliteAlterTableActionSyntax . Just $
emit "ADD COLUMN " <> quotedIdentifier columnNm <> emit " " <> fromSqliteColumnSchema schema
dropColumnSyntax _ = SqliteAlterTableActionSyntax Nothing
renameTableToSyntax newNm =
SqliteAlterTableActionSyntax . Just $
emit "RENAME TO " <> quotedIdentifier newNm
renameColumnToSyntax oldNm newNm =
SqliteAlterTableActionSyntax . Just $
emit "RENAME COLUMN " <> quotedIdentifier oldNm <>
emit " TO " <> quotedIdentifier newNm
instance IsSql92AlterColumnActionSyntax SqliteAlterColumnActionSyntax where
setNotNullSyntax = SqliteAlterColumnActionSyntax Nothing
setNullSyntax = SqliteAlterColumnActionSyntax Nothing
instance IsSql92ColumnSchemaSyntax SqliteColumnSchemaSyntax where
type Sql92ColumnSchemaColumnTypeSyntax SqliteColumnSchemaSyntax = SqliteDataTypeSyntax
type Sql92ColumnSchemaExpressionSyntax SqliteColumnSchemaSyntax = SqliteExpressionSyntax
type Sql92ColumnSchemaColumnConstraintDefinitionSyntax SqliteColumnSchemaSyntax = SqliteColumnConstraintDefinitionSyntax
columnSchemaSyntax ty defVal constraints collation =
SqliteColumnSchemaSyntax
(fromSqliteDataType ty <>
maybe mempty (\defVal -> emit " DEFAULT " <> parens (fromSqliteExpression defVal)) defVal <>
foldMap (\constraint -> emit " " <> fromSqliteColumnConstraintDefinition constraint <> emit " ") constraints <>
maybe mempty (\c -> emit " COLLATE " <> quotedIdentifier c) collation)
(if sqliteDataTypeSerial ty then True else False)
instance IsSql92ColumnConstraintDefinitionSyntax SqliteColumnConstraintDefinitionSyntax where
type Sql92ColumnConstraintDefinitionConstraintSyntax SqliteColumnConstraintDefinitionSyntax = SqliteColumnConstraintSyntax
type Sql92ColumnConstraintDefinitionAttributesSyntax SqliteColumnConstraintDefinitionSyntax = SqlConstraintAttributesBuilder
constraintDefinitionSyntax nm def attrs =
SqliteColumnConstraintDefinitionSyntax
(maybe mempty (\nm' -> emit "CONSTRAINT " <> quotedIdentifier nm') nm <>
fromSqliteColumnConstraint def (fromMaybe mempty attrs))
(constraintDefinitionSyntax nm (sqliteColumnConstraintSerialized def)
(fmap sqlConstraintAttributesSerialized attrs))
instance Sql92SerializableConstraintDefinitionSyntax SqliteColumnConstraintDefinitionSyntax where
serializeConstraint = fromBeamSerializedConstraintDefinition . sqliteColumnConstraintDefinitionSerialized
instance IsSql92ColumnConstraintSyntax SqliteColumnConstraintSyntax where
type Sql92ColumnConstraintMatchTypeSyntax SqliteColumnConstraintSyntax = SqliteMatchTypeSyntax
type Sql92ColumnConstraintReferentialActionSyntax SqliteColumnConstraintSyntax = SqliteReferentialActionSyntax
type Sql92ColumnConstraintExpressionSyntax SqliteColumnConstraintSyntax = SqliteExpressionSyntax
notNullConstraintSyntax = SqliteColumnConstraintSyntax (\_ -> emit "NOT NULL") notNullConstraintSyntax
uniqueColumnConstraintSyntax = SqliteColumnConstraintSyntax (\_ -> emit "UNIQUE") uniqueColumnConstraintSyntax
primaryKeyColumnConstraintSyntax = SqliteColumnConstraintSyntax (\_ -> emit "PRIMARY KEY") primaryKeyColumnConstraintSyntax
checkColumnConstraintSyntax expr =
SqliteColumnConstraintSyntax (\_ -> emit "CHECK " <> parens (fromSqliteExpression expr))
(checkColumnConstraintSyntax (sqliteExpressionSerialized expr))
referencesConstraintSyntax tbl fields matchType onUpdate onDelete =
SqliteColumnConstraintSyntax sqliteConstraint
(referencesConstraintSyntax tbl fields (fmap sqliteMatchTypeSerialized matchType)
(fmap sqliteReferentialActionSerialized onUpdate)
(fmap sqliteReferentialActionSerialized onDelete))
where
sqliteConstraint (SqlConstraintAttributesBuilder atTime deferrable) =
emit "REFERENCES " <> quotedIdentifier tbl <> parens (commas (map quotedIdentifier fields)) <>
maybe mempty (\matchType' -> emit " MATCH " <> fromSqliteMatchType matchType') matchType <>
maybe mempty (\onUpdate' -> emit " ON UPDATE " <> fromSqliteReferentialAction onUpdate') onUpdate <>
maybe mempty (\onDelete' -> emit " ON DELETE " <> fromSqliteReferentialAction onDelete') onDelete <>
case (deferrable, atTime) of
(_, Just atTime) ->
let deferrable' = fromMaybe False deferrable
in (if deferrable' then emit " DEFERRABLE " else emit " NOT DEFERRABLE ") <>
case atTime of
InitiallyDeferred -> emit "INITIALLY DEFERRED"
InitiallyImmediate -> emit "INITIALLY IMMEDIATE"
(Just deferrable', _) ->
if deferrable' then emit " DEFERRABLE" else emit " NOT DEFERRABLE"
_ -> mempty
instance IsSql92MatchTypeSyntax SqliteMatchTypeSyntax where
fullMatchSyntax = SqliteMatchTypeSyntax (emit "FULL") fullMatchSyntax
partialMatchSyntax = SqliteMatchTypeSyntax (emit "PARTIAL") partialMatchSyntax
instance IsSql92ReferentialActionSyntax SqliteReferentialActionSyntax where
referentialActionCascadeSyntax = SqliteReferentialActionSyntax (emit "CASCADE") referentialActionCascadeSyntax
referentialActionSetNullSyntax = SqliteReferentialActionSyntax (emit "SET NULL") referentialActionSetNullSyntax
referentialActionSetDefaultSyntax = SqliteReferentialActionSyntax (emit "SET DEFAULT") referentialActionSetDefaultSyntax
referentialActionNoActionSyntax = SqliteReferentialActionSyntax (emit "NO ACTION") referentialActionNoActionSyntax
instance IsSql92TableConstraintSyntax SqliteTableConstraintSyntax where
primaryKeyConstraintSyntax fields =
SqliteTableConstraintSyntax
(emit "PRIMARY KEY" <> parens (commas (map quotedIdentifier fields)))
(Just fields)
instance IsSql92CreateTableSyntax SqliteCreateTableSyntax where
type Sql92CreateTableColumnSchemaSyntax SqliteCreateTableSyntax = SqliteColumnSchemaSyntax
type Sql92CreateTableTableConstraintSyntax SqliteCreateTableSyntax = SqliteTableConstraintSyntax
type Sql92CreateTableOptionsSyntax SqliteCreateTableSyntax = SqliteTableOptionsSyntax
type Sql92CreateTableTableNameSyntax SqliteCreateTableSyntax = SqliteTableNameSyntax
createTableSyntax _ nm fields constraints =
let fieldDefs = map mkFieldDef fields
constraintDefs = map fromSqliteTableConstraint constraints
noPkConstraintDefs = map fromSqliteTableConstraint (filter (isNothing . sqliteTableConstraintPrimaryKey) constraints)
constraintPks = mapMaybe sqliteTableConstraintPrimaryKey constraints
fieldPrimaryKey = map fst (filter (sqliteIsSerialColumn . snd) fields)
mkFieldDef (fieldNm, fieldTy) =
quotedIdentifier fieldNm <> emit " " <>
fromSqliteColumnSchema fieldTy
createWithConstraints constraintDefs' =
SqliteCreateTableSyntax $
emit "CREATE TABLE " <> fromSqliteTableName nm <> parens (commas (fieldDefs <> constraintDefs'))
normalCreateTable = createWithConstraints constraintDefs
createTableNoPkConstraint = createWithConstraints noPkConstraintDefs
in case fieldPrimaryKey of
[] -> normalCreateTable
[field] ->
case constraintPks of
[] -> error "A column claims to have a primary key, but there is no key on this table"
[[fieldPk]]
| field /= fieldPk -> error "Two columns claim to be a primary key on this table"
| otherwise -> createTableNoPkConstraint
_ -> error "There are multiple primary key constraints on this table"
_ -> error "More than one column claims to be a primary key on this table"
instance IsSql92DataTypeSyntax SqliteDataTypeSyntax where
domainType nm = SqliteDataTypeSyntax (quotedIdentifier nm) (domainType nm) (domainType nm) False
charType prec charSet = SqliteDataTypeSyntax (emit "CHAR" <> sqliteOptPrec prec <> sqliteOptCharSet charSet)
(charType prec charSet)
(charType prec charSet)
False
varCharType prec charSet = SqliteDataTypeSyntax (emit "VARCHAR" <> sqliteOptPrec prec <> sqliteOptCharSet charSet)
(varCharType prec charSet)
(varCharType prec charSet)
False
nationalCharType prec = SqliteDataTypeSyntax (emit "NATIONAL CHAR" <> sqliteOptPrec prec)
(nationalCharType prec)
(nationalCharType prec)
False
nationalVarCharType prec = SqliteDataTypeSyntax (emit "NATIONAL CHARACTER VARYING" <> sqliteOptPrec prec)
(nationalVarCharType prec)
(nationalVarCharType prec)
False
bitType prec = SqliteDataTypeSyntax (emit "BIT" <> sqliteOptPrec prec) (bitType prec) (bitType prec) False
varBitType prec = SqliteDataTypeSyntax (emit "BIT VARYING" <> sqliteOptPrec prec) (varBitType prec) (varBitType prec) False
numericType prec = SqliteDataTypeSyntax (emit "NUMERIC" <> sqliteOptNumericPrec prec) (numericType prec) (numericType prec) False
decimalType prec = SqliteDataTypeSyntax (emit "DECIMAL" <> sqliteOptNumericPrec prec) (decimalType prec) (decimalType prec) False
intType = SqliteDataTypeSyntax (emit "INTEGER") intType intType False
smallIntType = SqliteDataTypeSyntax (emit "SMALLINT") smallIntType smallIntType False
floatType prec = SqliteDataTypeSyntax (emit "FLOAT" <> sqliteOptPrec prec) (floatType prec) (floatType prec) False
doubleType = SqliteDataTypeSyntax (emit "DOUBLE PRECISION") doubleType doubleType False
realType = SqliteDataTypeSyntax (emit "REAL") realType realType False
dateType = SqliteDataTypeSyntax (emit "DATE") dateType dateType False
timeType prec withTz = SqliteDataTypeSyntax (emit "TIME" <> sqliteOptPrec prec <> if withTz then emit " WITH TIME ZONE" else mempty)
(timeType prec withTz) (timeType prec withTz) False
timestampType prec withTz = SqliteDataTypeSyntax (emit "TIMESTAMP" <> sqliteOptPrec prec <> if withTz then emit " WITH TIME ZONE" else mempty)
(timestampType prec withTz) (timestampType prec withTz) False
instance IsSql99DataTypeSyntax SqliteDataTypeSyntax where
characterLargeObjectType = sqliteTextType
binaryLargeObjectType = sqliteBlobType
booleanType = SqliteDataTypeSyntax (emit "BOOLEAN") booleanType booleanType False
arrayType _ _ = error "SQLite does not support arrayType"
rowType _ = error "SQLite does not support rowType"
instance IsSql2008BigIntDataTypeSyntax SqliteDataTypeSyntax where
bigIntType = sqliteBigIntType
sqliteTextType, sqliteBlobType, sqliteBigIntType :: SqliteDataTypeSyntax
sqliteTextType = SqliteDataTypeSyntax (emit "TEXT")
(HsDataType (hsVarFrom "sqliteText" "Database.Beam.Sqlite")
(HsType (tyConNamed "Text")
(importSome "Data.Text" [importTyNamed "Text"]))
characterLargeObjectType)
characterLargeObjectType
False
sqliteBlobType = SqliteDataTypeSyntax (emit "BLOB")
(HsDataType (hsVarFrom "sqliteBlob" "Database.Beam.Sqlite")
(HsType (tyConNamed "ByteString")
(importSome "Data.ByteString" [importTyNamed "ByteString"]))
binaryLargeObjectType)
binaryLargeObjectType
False
sqliteBigIntType = SqliteDataTypeSyntax (emit "BIGINT")
(HsDataType (hsVarFrom "sqliteBigInt" "Database.Beam.Sqlite")
(HsType (tyConNamed "Int64")
(importSome "Data.Int" [importTyNamed "Int64"]))
bigIntType)
bigIntType
False
instance Sql92SerializableDataTypeSyntax SqliteDataTypeSyntax where
serializeDataType = fromBeamSerializedDataType . sqliteDataTypeSerialized
sqliteOptPrec :: Maybe Word -> SqliteSyntax
sqliteOptPrec Nothing = mempty
sqliteOptPrec (Just x) = parens (emit (fromString (show x)))
sqliteOptNumericPrec :: Maybe (Word, Maybe Word) -> SqliteSyntax
sqliteOptNumericPrec Nothing = mempty
sqliteOptNumericPrec (Just (prec, Nothing)) = sqliteOptPrec (Just prec)
sqliteOptNumericPrec (Just (prec, Just dec)) = parens $ emit (fromString (show prec)) <> emit ", " <> emit (fromString (show dec))
sqliteOptCharSet :: Maybe T.Text -> SqliteSyntax
sqliteOptCharSet Nothing = mempty
sqliteOptCharSet (Just cs) = emit " CHARACTER SET " <> emit (TE.encodeUtf8 cs)
instance IsSql92SelectSyntax SqliteSelectSyntax where
type Sql92SelectSelectTableSyntax SqliteSelectSyntax = SqliteSelectTableSyntax
type Sql92SelectOrderingSyntax SqliteSelectSyntax = SqliteOrderingSyntax
selectStmt tbl ordering limit offset =
SqliteSelectSyntax $
fromSqliteSelectTable tbl <>
(case ordering of
[] -> mempty
_ -> emit " ORDER BY " <> commas (coerce ordering)) <>
case (limit, offset) of
(Nothing, Nothing) -> mempty
(Just limit, Nothing) -> emit " LIMIT " <> emit' limit
(Nothing, Just offset) -> emit " LIMIT -1 OFFSET " <> emit' offset
(Just limit, Just offset) -> emit " LIMIT " <> emit' limit <>
emit " OFFSET " <> emit' offset
instance IsSql92SelectTableSyntax SqliteSelectTableSyntax where
type Sql92SelectTableSelectSyntax SqliteSelectTableSyntax = SqliteSelectSyntax
type Sql92SelectTableExpressionSyntax SqliteSelectTableSyntax = SqliteExpressionSyntax
type Sql92SelectTableProjectionSyntax SqliteSelectTableSyntax = SqliteProjectionSyntax
type Sql92SelectTableFromSyntax SqliteSelectTableSyntax = SqliteFromSyntax
type Sql92SelectTableGroupingSyntax SqliteSelectTableSyntax = SqliteGroupingSyntax
type Sql92SelectTableSetQuantifierSyntax SqliteSelectTableSyntax = SqliteAggregationSetQuantifierSyntax
selectTableStmt setQuantifier proj from where_ grouping having =
SqliteSelectTableSyntax $
emit "SELECT " <>
maybe mempty (<> emit " ") (fromSqliteAggregationSetQuantifier <$> setQuantifier) <>
fromSqliteProjection proj <>
maybe mempty (emit " FROM " <>) (fromSqliteFromSyntax <$> from) <>
maybe mempty (emit " WHERE " <>) (fromSqliteExpression <$> where_) <>
maybe mempty (emit " GROUP BY " <>) (fromSqliteGrouping <$> grouping) <>
maybe mempty (emit " HAVING " <>) (fromSqliteExpression <$> having)
unionTables all = tableOp (if all then "UNION ALL" else "UNION")
intersectTables all = tableOp (if all then "INTERSECT ALL" else "INTERSECT")
exceptTable all = tableOp (if all then "EXCEPT ALL" else "EXCEPT")
tableOp :: ByteString -> SqliteSelectTableSyntax -> SqliteSelectTableSyntax -> SqliteSelectTableSyntax
tableOp op a b =
SqliteSelectTableSyntax $
fromSqliteSelectTable a <> spaces (emit op) <> fromSqliteSelectTable b
instance IsSql92FromSyntax SqliteFromSyntax where
type Sql92FromExpressionSyntax SqliteFromSyntax = SqliteExpressionSyntax
type Sql92FromTableSourceSyntax SqliteFromSyntax = SqliteTableSourceSyntax
fromTable tableSrc Nothing = SqliteFromSyntax (fromSqliteTableSource tableSrc)
fromTable tableSrc (Just (nm, colNms)) =
SqliteFromSyntax (fromSqliteTableSource tableSrc <> emit " AS " <> quotedIdentifier nm <>
maybe mempty (\colNms' -> parens (commas (map quotedIdentifier colNms'))) colNms)
innerJoin = _join "INNER JOIN"
leftJoin = _join "LEFT JOIN"
rightJoin = _join "RIGHT JOIN"
_join :: ByteString -> SqliteFromSyntax -> SqliteFromSyntax -> Maybe SqliteExpressionSyntax -> SqliteFromSyntax
_join joinType a b Nothing =
SqliteFromSyntax (fromSqliteFromSyntax a <> spaces (emit joinType) <> fromSqliteFromSyntax b)
_join joinType a b (Just on) =
SqliteFromSyntax (fromSqliteFromSyntax a <> spaces (emit joinType) <> fromSqliteFromSyntax b <> emit " ON " <> fromSqliteExpression on)
instance IsSql92ProjectionSyntax SqliteProjectionSyntax where
type Sql92ProjectionExpressionSyntax SqliteProjectionSyntax = SqliteExpressionSyntax
projExprs exprs =
SqliteProjectionSyntax $
commas (map (\(expr, nm) -> fromSqliteExpression expr <>
maybe mempty (\nm -> emit " AS " <> quotedIdentifier nm) nm) exprs)
instance IsSql92FieldNameSyntax SqliteFieldNameSyntax where
qualifiedField a b =
SqliteFieldNameSyntax $
quotedIdentifier a <> emit "." <> quotedIdentifier b
unqualifiedField a =
SqliteFieldNameSyntax $
quotedIdentifier a
instance IsSql92TableSourceSyntax SqliteTableSourceSyntax where
type Sql92TableSourceTableNameSyntax SqliteTableSourceSyntax = SqliteTableNameSyntax
type Sql92TableSourceSelectSyntax SqliteTableSourceSyntax = SqliteSelectSyntax
type Sql92TableSourceExpressionSyntax SqliteTableSourceSyntax = SqliteExpressionSyntax
tableNamed = SqliteTableSourceSyntax . fromSqliteTableName
tableFromSubSelect s =
SqliteTableSourceSyntax (parens (fromSqliteSelect s))
tableFromValues vss = SqliteTableSourceSyntax . parens $
emit "VALUES " <>
commas (map (\vs -> parens (commas (map fromSqliteExpression vs))) vss)
instance IsSql92GroupingSyntax SqliteGroupingSyntax where
type Sql92GroupingExpressionSyntax SqliteGroupingSyntax = SqliteExpressionSyntax
groupByExpressions es =
SqliteGroupingSyntax $
commas (map fromSqliteExpression es)
instance IsSql92OrderingSyntax SqliteOrderingSyntax where
type Sql92OrderingExpressionSyntax SqliteOrderingSyntax = SqliteExpressionSyntax
ascOrdering e = SqliteOrderingSyntax (fromSqliteExpression e <> emit " ASC")
descOrdering e = SqliteOrderingSyntax (fromSqliteExpression e <> emit " DESC")
instance HasSqlValueSyntax SqliteValueSyntax Int8 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Int16 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Int32 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Int64 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Word8 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Word16 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Word32 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Word64 where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance HasSqlValueSyntax SqliteValueSyntax Scientific where
sqlValueSyntax s = SqliteValueSyntax (emitValue (SQLText (fromString (show s))))
instance HasSqlValueSyntax SqliteValueSyntax Float where
sqlValueSyntax f = SqliteValueSyntax (emitValue (SQLFloat (float2Double f)))
instance HasSqlValueSyntax SqliteValueSyntax Double where
sqlValueSyntax f = SqliteValueSyntax (emitValue (SQLFloat f))
instance HasSqlValueSyntax SqliteValueSyntax Bool where
sqlValueSyntax = sqlValueSyntax . (\b -> if b then 1 else 0 :: Int32)
instance HasSqlValueSyntax SqliteValueSyntax SqlNull where
sqlValueSyntax _ = SqliteValueSyntax (emit "NULL")
instance HasSqlValueSyntax SqliteValueSyntax String where
sqlValueSyntax s = SqliteValueSyntax (emitValue (SQLText (fromString s)))
instance HasSqlValueSyntax SqliteValueSyntax T.Text where
sqlValueSyntax s = SqliteValueSyntax (emitValue (SQLText s))
instance HasSqlValueSyntax SqliteValueSyntax TL.Text where
sqlValueSyntax s = SqliteValueSyntax (emitValue (SQLText (TL.toStrict s)))
instance HasSqlValueSyntax SqliteValueSyntax x =>
HasSqlValueSyntax SqliteValueSyntax (Maybe x) where
sqlValueSyntax (Just x) = sqlValueSyntax x
sqlValueSyntax Nothing = sqlValueSyntax SqlNull
instance TypeError (PreferExplicitSize Int Int32) => HasSqlValueSyntax SqliteValueSyntax Int where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance TypeError (PreferExplicitSize Word Word32) => HasSqlValueSyntax SqliteValueSyntax Word where
sqlValueSyntax i = SqliteValueSyntax (emitValue (SQLInteger (fromIntegral i)))
instance IsCustomSqlSyntax SqliteExpressionSyntax where
newtype CustomSqlSyntax SqliteExpressionSyntax =
SqliteCustomExpressionSyntax { fromSqliteCustomExpression :: SqliteSyntax }
deriving (Monoid, Semigroup)
customExprSyntax = SqliteExpressionSyntax . fromSqliteCustomExpression
renderSyntax = SqliteCustomExpressionSyntax . fromSqliteExpression
instance IsString (CustomSqlSyntax SqliteExpressionSyntax) where
fromString = SqliteCustomExpressionSyntax . emit . fromString
instance IsSql92QuantifierSyntax SqliteComparisonQuantifierSyntax where
quantifyOverAll = SqliteComparisonQuantifierSyntax (emit "ALL")
quantifyOverAny = SqliteComparisonQuantifierSyntax (emit "ANY")
instance IsSql92ExpressionSyntax SqliteExpressionSyntax where
type Sql92ExpressionValueSyntax SqliteExpressionSyntax = SqliteValueSyntax
type Sql92ExpressionSelectSyntax SqliteExpressionSyntax = SqliteSelectSyntax
type Sql92ExpressionFieldNameSyntax SqliteExpressionSyntax = SqliteFieldNameSyntax
type Sql92ExpressionQuantifierSyntax SqliteExpressionSyntax = SqliteComparisonQuantifierSyntax
type Sql92ExpressionCastTargetSyntax SqliteExpressionSyntax = SqliteDataTypeSyntax
type Sql92ExpressionExtractFieldSyntax SqliteExpressionSyntax = ExtractField
addE = binOp "+"; subE = binOp "-"; mulE = binOp "*"; divE = binOp "/"
modE = binOp "%"; orE = binOp "OR"; andE = binOp "AND"; likeE = binOp "LIKE"
overlapsE = binOp "OVERLAPS"
eqE = compOp "="; neqE = compOp "<>"; ltE = compOp "<"; gtE = compOp ">"
leE = compOp "<="; geE = compOp ">="
negateE = unOp "-"; notE = unOp "NOT"
isNotNullE = postFix "IS NOT NULL"; isNullE = postFix "IS NULL"
isTrueE = postFix "IS 1"; isNotTrueE = postFix "IS NOT 1"
isFalseE = postFix "IS 0"; isNotFalseE = postFix "IS NOT 0"
isUnknownE = postFix "IS NULL"; isNotUnknownE = postFix "IS NOT NULL"
existsE select = SqliteExpressionSyntax (emit "EXISTS " <> parens (fromSqliteSelect select))
uniqueE select = SqliteExpressionSyntax (emit "UNIQUE " <> parens (fromSqliteSelect select))
betweenE a b c = SqliteExpressionSyntax (parens (fromSqliteExpression a) <>
emit " BETWEEN " <>
parens (fromSqliteExpression b) <>
emit " AND " <>
parens (fromSqliteExpression c))
valueE = SqliteExpressionSyntax . fromSqliteValue
rowE vs = SqliteExpressionSyntax (parens (commas (map fromSqliteExpression vs)))
fieldE = SqliteExpressionSyntax . fromSqliteFieldNameSyntax
subqueryE = SqliteExpressionSyntax . parens . fromSqliteSelect
positionE needle haystack =
SqliteExpressionSyntax $
emit "POSITION" <> parens (parens (fromSqliteExpression needle) <> emit " IN " <> parens (fromSqliteExpression haystack))
nullIfE a b =
SqliteExpressionSyntax $
emit "NULLIF" <> parens (fromSqliteExpression a <> emit ", " <> fromSqliteExpression b)
absE x = SqliteExpressionSyntax (emit "ABS" <> parens (fromSqliteExpression x))
bitLengthE x = SqliteExpressionSyntax (emit "8 * LENGTH" <> parens (emit "CAST" <> parens (parens (fromSqliteExpression x) <> emit " AS BLOB")))
charLengthE x = SqliteExpressionSyntax (emit "LENGTH" <> parens (fromSqliteExpression x))
octetLengthE x = SqliteExpressionSyntax (emit "LENGTH" <> parens (emit "CAST" <> parens (parens (fromSqliteExpression x) <> emit " AS BLOB")))
lowerE x = SqliteExpressionSyntax (emit "LOWER" <> parens (fromSqliteExpression x))
upperE x = SqliteExpressionSyntax (emit "UPPER" <> parens (fromSqliteExpression x))
trimE x = SqliteExpressionSyntax (emit "TRIM" <> parens (fromSqliteExpression x))
coalesceE es = SqliteExpressionSyntax (emit "COALESCE" <> parens (commas (map fromSqliteExpression es)))
extractE = sqliteExtract
castE e t = SqliteExpressionSyntax (emit "CAST" <> parens (parens (fromSqliteExpression e) <> emit " AS " <> fromSqliteDataType t))
caseE cases else_ =
SqliteExpressionSyntax $
emit "CASE " <>
foldMap (\(cond, res) -> emit "WHEN " <> fromSqliteExpression cond <> emit " THEN " <> fromSqliteExpression res <> emit " ") cases <>
emit "ELSE " <> fromSqliteExpression else_ <> emit " END"
currentTimestampE = SqliteExpressionSyntax (emit "CURRENT_TIMESTAMP")
defaultE = SqliteExpressionDefault
inE e es = SqliteExpressionSyntax (parens (fromSqliteExpression e) <> emit " IN " <> parens (commas (map fromSqliteExpression es)))
instance IsSql99ConcatExpressionSyntax SqliteExpressionSyntax where
concatE [] = valueE (sqlValueSyntax ("" :: T.Text))
concatE (x:xs) =
SqliteExpressionSyntax $ parens $
foldl (\a b -> a <> emit " || " <> parens (fromSqliteExpression b)) (fromSqliteExpression x) xs
instance IsSql99FunctionExpressionSyntax SqliteExpressionSyntax where
functionCallE fn args =
SqliteExpressionSyntax $
fromSqliteExpression fn <> parens (commas (fmap fromSqliteExpression args))
functionNameE nm = SqliteExpressionSyntax (emit (TE.encodeUtf8 nm))
binOp :: ByteString -> SqliteExpressionSyntax -> SqliteExpressionSyntax -> SqliteExpressionSyntax
binOp op a b =
SqliteExpressionSyntax $
parens (fromSqliteExpression a) <> emit " " <> emit op <> emit " " <> parens (fromSqliteExpression b)
compOp :: ByteString -> Maybe SqliteComparisonQuantifierSyntax
-> SqliteExpressionSyntax -> SqliteExpressionSyntax
-> SqliteExpressionSyntax
compOp op quantifier a b =
SqliteExpressionSyntax $
parens (fromSqliteExpression a) <>
emit op <>
maybe mempty (\q -> emit " " <> fromSqliteComparisonQuantifier q <> emit " ") quantifier <>
parens (fromSqliteExpression b)
unOp, postFix :: ByteString -> SqliteExpressionSyntax -> SqliteExpressionSyntax
unOp op a =
SqliteExpressionSyntax (emit op <> parens (fromSqliteExpression a))
postFix op a =
SqliteExpressionSyntax (parens (fromSqliteExpression a) <> emit " " <> emit op)
instance IsSql92AggregationExpressionSyntax SqliteExpressionSyntax where
type Sql92AggregationSetQuantifierSyntax SqliteExpressionSyntax = SqliteAggregationSetQuantifierSyntax
countAllE = SqliteExpressionSyntax (emit "COUNT(*)")
countE = unAgg "COUNT"
sumE = unAgg "SUM"
avgE = unAgg "AVG"
minE = unAgg "MIN"
maxE = unAgg "MAX"
unAgg :: ByteString -> Maybe SqliteAggregationSetQuantifierSyntax -> SqliteExpressionSyntax
-> SqliteExpressionSyntax
unAgg fn q e =
SqliteExpressionSyntax $
emit fn <> parens (maybe mempty (\q -> fromSqliteAggregationSetQuantifier q <> emit " ") q <>
fromSqliteExpression e)
instance IsSql92AggregationSetQuantifierSyntax SqliteAggregationSetQuantifierSyntax where
setQuantifierDistinct = SqliteAggregationSetQuantifierSyntax (emit "DISTINCT")
setQuantifierAll = SqliteAggregationSetQuantifierSyntax (emit "ALL")
instance IsSql92InsertSyntax SqliteInsertSyntax where
type Sql92InsertTableNameSyntax SqliteInsertSyntax = SqliteTableNameSyntax
type Sql92InsertValuesSyntax SqliteInsertSyntax = SqliteInsertValuesSyntax
insertStmt table fields values = SqliteInsertSyntax table fields values Nothing
instance IsSql92InsertValuesSyntax SqliteInsertValuesSyntax where
type Sql92InsertValuesExpressionSyntax SqliteInsertValuesSyntax = SqliteExpressionSyntax
type Sql92InsertValuesSelectSyntax SqliteInsertValuesSyntax = SqliteSelectSyntax
insertSqlExpressions = SqliteInsertExpressions
insertFromSql = SqliteInsertFromSql
instance IsSql92UpdateSyntax SqliteUpdateSyntax where
type Sql92UpdateTableNameSyntax SqliteUpdateSyntax = SqliteTableNameSyntax
type Sql92UpdateFieldNameSyntax SqliteUpdateSyntax = SqliteFieldNameSyntax
type Sql92UpdateExpressionSyntax SqliteUpdateSyntax = SqliteExpressionSyntax
updateStmt tbl fields where_ =
SqliteUpdateSyntax $
emit "UPDATE " <> fromSqliteTableName tbl <>
(case fields of
[] -> mempty
_ -> emit " SET " <>
commas (map (\(field, val) -> fromSqliteFieldNameSyntax field <> emit "=" <> fromSqliteExpression val) fields)) <>
maybe mempty (\where_ -> emit " WHERE " <> fromSqliteExpression where_) where_
instance IsSql92DeleteSyntax SqliteDeleteSyntax where
type Sql92DeleteTableNameSyntax SqliteDeleteSyntax = SqliteTableNameSyntax
type Sql92DeleteExpressionSyntax SqliteDeleteSyntax = SqliteExpressionSyntax
deleteStmt tbl Nothing where_ =
SqliteDeleteSyntax $
emit "DELETE FROM " <> fromSqliteTableName tbl <>
maybe mempty (\where_ -> emit " WHERE " <> fromSqliteExpression where_) where_
deleteStmt _ (Just _) _ =
error "beam-sqlite: invariant failed: DELETE must not have a table alias"
spaces, parens :: SqliteSyntax -> SqliteSyntax
spaces a = emit " " <> a <> emit " "
parens a = emit "(" <> a <> emit ")"
commas :: [SqliteSyntax] -> SqliteSyntax
commas [] = mempty
commas [x] = x
commas (x:xs) = x <> foldMap (emit ", " <>) xs
strftimeSyntax :: SqliteExpressionSyntax -> SqliteExpressionSyntax -> [ SqliteExpressionSyntax ] -> SqliteExpressionSyntax
strftimeSyntax fmt ts mods =
functionCallE (SqliteExpressionSyntax (emit "strftime"))
(fmt:ts:mods)
sqliteExtract :: ExtractField -> SqliteExpressionSyntax -> SqliteExpressionSyntax
sqliteExtract field from =
case field of
ExtractFieldTimeZoneHour -> error "sqliteExtract: TODO ExtractFieldTimeZoneHour"
ExtractFieldTimeZoneMinute -> error "sqliteExtract: TODO ExtractFieldTimeZoneMinute"
ExtractFieldDateTimeYear -> extractStrftime "%Y"
ExtractFieldDateTimeMonth -> extractStrftime "%m"
ExtractFieldDateTimeDay -> extractStrftime "%d"
ExtractFieldDateTimeHour -> extractStrftime "%H"
ExtractFieldDateTimeMinute -> extractStrftime "%M"
ExtractFieldDateTimeSecond -> extractStrftime "%S"
where
extractStrftime :: String -> SqliteExpressionSyntax
extractStrftime fmt = strftimeSyntax (valueE (sqlValueSyntax fmt)) from []
sqliteSerialType :: SqliteDataTypeSyntax
sqliteSerialType = SqliteDataTypeSyntax (emit "INTEGER PRIMARY KEY AUTOINCREMENT")
intType
(BeamSerializedDataType (beamSerializeJSON "sqlite" "serial"))
True
instance HasSqlValueSyntax SqliteValueSyntax ByteString where
sqlValueSyntax bs = SqliteValueSyntax (emitValue (SQLBlob bs))
instance HasSqlValueSyntax SqliteValueSyntax UTCTime where
sqlValueSyntax tm = SqliteValueSyntax (emitValue (SQLText (fromString tmStr)))
where tmStr = formatTime defaultTimeLocale (iso8601DateFormat (Just "%H:%M:%S%Q")) tm
instance HasSqlValueSyntax SqliteValueSyntax LocalTime where
sqlValueSyntax tm = SqliteValueSyntax (emitValue (SQLText (fromString tmStr)))
where tmStr = formatTime defaultTimeLocale (iso8601DateFormat (Just "%H:%M:%S%Q")) tm
instance HasSqlValueSyntax SqliteValueSyntax Day where
sqlValueSyntax tm = SqliteValueSyntax (emitValue (SQLText (fromString tmStr)))
where tmStr = formatTime defaultTimeLocale (iso8601DateFormat Nothing) tm
instance HasDataTypeCreatedCheck SqliteDataTypeSyntax where
dataTypeHasBeenCreated _ _ = True