{-# LANGUAGE OverloadedLists #-}
{-# LANGUAGE Strict #-}
module Database.PostgreSQL.Entity
(
Entity (..)
, Field
, selectById
, selectOneByField
, selectManyByField
, selectWhereNotNull
, selectWhereNull
, selectOneWhereIn
, joinSelectById
, joinSelectOneByField
, selectOrderBy
, insert
, insertMany
, upsert
, update
, updateFieldsBy
, delete
, deleteByField
, _select
, _selectWithFields
, _where
, _selectWhere
, _selectWhereNotNull
, _selectWhereNull
, _selectWhereIn
, _joinSelect
, _innerJoin
, _joinSelectWithFields
, _joinSelectOneByField
, _insert
, _onConflictDoUpdate
, _update
, _updateBy
, _updateFields
, _updateFieldsBy
, _delete
, _deleteWhere
, _orderBy
, _orderByMany
)
where
import Control.Monad (void)
import Control.Monad.IO.Class (MonadIO)
import Data.Foldable (fold)
import Data.Int (Int64)
import Data.Text.Display (display)
import Data.Vector (Vector)
import qualified Data.Vector as V
import Database.PostgreSQL.Simple (Only (..))
import Database.PostgreSQL.Simple.FromRow (FromRow)
import Database.PostgreSQL.Simple.ToField (ToField)
import Database.PostgreSQL.Simple.ToRow (ToRow (..))
import Database.PostgreSQL.Simple.Types (Query (..))
import Database.PostgreSQL.Transact (DBT)
import Data.Text (Text)
import Database.PostgreSQL.Entity.DBT (QueryNature (..), execute, executeMany, query, queryOne, queryOne_, query_)
import Database.PostgreSQL.Entity.Internal
import Database.PostgreSQL.Entity.Types
selectById
:: forall e value m
. (Entity e, FromRow e, MonadIO m, ToRow value)
=> value
-> DBT m (Maybe e)
selectById :: forall e value (m :: * -> *).
(Entity e, FromRow e, MonadIO m, ToRow value) =>
value -> DBT m (Maybe e)
selectById value
value = Field -> value -> DBT m (Maybe e)
forall e value (m :: * -> *).
(Entity e, FromRow e, MonadIO m, ToRow value) =>
Field -> value -> DBT m (Maybe e)
selectOneByField (forall e. Entity e => Field
primaryKey @e) value
value
selectOneByField
:: forall e value m
. (Entity e, FromRow e, MonadIO m, ToRow value)
=> Field
-> value
-> DBT m (Maybe e)
selectOneByField :: forall e value (m :: * -> *).
(Entity e, FromRow e, MonadIO m, ToRow value) =>
Field -> value -> DBT m (Maybe e)
selectOneByField Field
f value
value = QueryNature -> Query -> value -> DBT m (Maybe e)
forall params result (m :: * -> *).
(ToRow params, FromRow result, MonadIO m) =>
QueryNature -> Query -> params -> DBT m (Maybe result)
queryOne QueryNature
Select (forall e. Entity e => Vector Field -> Query
_selectWhere @e [Item (Vector Field)
Field
f]) value
value
selectManyByField
:: forall e value m
. (Entity e, FromRow e, MonadIO m, ToRow value)
=> Field
-> value
-> DBT m (Vector e)
selectManyByField :: forall e value (m :: * -> *).
(Entity e, FromRow e, MonadIO m, ToRow value) =>
Field -> value -> DBT m (Vector e)
selectManyByField Field
f value
value = QueryNature -> Query -> value -> DBT m (Vector e)
forall params result (m :: * -> *).
(ToRow params, FromRow result, MonadIO m) =>
QueryNature -> Query -> params -> DBT m (Vector result)
query QueryNature
Select (forall e. Entity e => Vector Field -> Query
_selectWhere @e [Item (Vector Field)
Field
f]) value
value
selectWhereNotNull
:: forall e m
. (Entity e, FromRow e, MonadIO m)
=> Vector Field
-> DBT m (Vector e)
selectWhereNotNull :: forall e (m :: * -> *).
(Entity e, FromRow e, MonadIO m) =>
Vector Field -> DBT m (Vector e)
selectWhereNotNull Vector Field
fs = QueryNature -> Query -> DBT m (Vector e)
forall result (m :: * -> *).
(FromRow result, MonadIO m) =>
QueryNature -> Query -> DBT m (Vector result)
query_ QueryNature
Select (forall e. Entity e => Vector Field -> Query
_selectWhereNotNull @e Vector Field
fs)
selectWhereNull
:: forall e m
. (Entity e, FromRow e, MonadIO m)
=> Vector Field
-> DBT m (Vector e)
selectWhereNull :: forall e (m :: * -> *).
(Entity e, FromRow e, MonadIO m) =>
Vector Field -> DBT m (Vector e)
selectWhereNull Vector Field
fs = QueryNature -> Query -> DBT m (Vector e)
forall result (m :: * -> *).
(FromRow result, MonadIO m) =>
QueryNature -> Query -> DBT m (Vector result)
query_ QueryNature
Select (forall e. Entity e => Vector Field -> Query
_selectWhereNull @e Vector Field
fs)
selectOneWhereIn
:: forall e m
. (Entity e, FromRow e, MonadIO m)
=> Field
-> Vector Text
-> DBT m (Maybe e)
selectOneWhereIn :: forall e (m :: * -> *).
(Entity e, FromRow e, MonadIO m) =>
Field -> Vector Text -> DBT m (Maybe e)
selectOneWhereIn Field
f Vector Text
values = QueryNature -> Query -> DBT m (Maybe e)
forall result (m :: * -> *).
(FromRow result, MonadIO m) =>
QueryNature -> Query -> DBT m (Maybe result)
queryOne_ QueryNature
Select (forall e. Entity e => Field -> Vector Text -> Query
_selectWhereIn @e Field
f Vector Text
values)
joinSelectById
:: forall e1 e2 m
. (Entity e1, Entity e2, FromRow e1, MonadIO m)
=> DBT m (Vector e1)
joinSelectById :: forall e1 e2 (m :: * -> *).
(Entity e1, Entity e2, FromRow e1, MonadIO m) =>
DBT m (Vector e1)
joinSelectById = QueryNature -> Query -> DBT m (Vector e1)
forall result (m :: * -> *).
(FromRow result, MonadIO m) =>
QueryNature -> Query -> DBT m (Vector result)
query_ QueryNature
Select (forall e1 e2. (Entity e1, Entity e2) => Query
_joinSelect @e1 @e2)
joinSelectOneByField
:: forall e1 e2 value m
. (Entity e1, Entity e2, FromRow e1, MonadIO m, ToField value)
=> Field
-> Field
-> value
-> DBT m (Vector e1)
joinSelectOneByField :: forall e1 e2 value (m :: * -> *).
(Entity e1, Entity e2, FromRow e1, MonadIO m, ToField value) =>
Field -> Field -> value -> DBT m (Vector e1)
joinSelectOneByField Field
pivot Field
whereClause value
value =
QueryNature -> Query -> Only value -> DBT m (Vector e1)
forall params result (m :: * -> *).
(ToRow params, FromRow result, MonadIO m) =>
QueryNature -> Query -> params -> DBT m (Vector result)
query QueryNature
Select (forall e1 e2. (Entity e1, Entity e2) => Field -> Field -> Query
_joinSelectOneByField @e1 @e2 Field
pivot Field
whereClause) (value -> Only value
forall a. a -> Only a
Only value
value)
selectOrderBy
:: forall e m
. (Entity e, FromRow e, MonadIO m)
=> Vector (Field, SortKeyword)
-> DBT m (Vector e)
selectOrderBy :: forall e (m :: * -> *).
(Entity e, FromRow e, MonadIO m) =>
Vector (Field, SortKeyword) -> DBT m (Vector e)
selectOrderBy Vector (Field, SortKeyword)
sortSpec = QueryNature -> Query -> DBT m (Vector e)
forall result (m :: * -> *).
(FromRow result, MonadIO m) =>
QueryNature -> Query -> DBT m (Vector result)
query_ QueryNature
Select (forall e. Entity e => Query
_select @e Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Vector (Field, SortKeyword) -> Query
_orderByMany Vector (Field, SortKeyword)
sortSpec)
insert
:: forall e values m
. (Entity e, ToRow values, MonadIO m)
=> values
-> DBT m ()
insert :: forall e values (m :: * -> *).
(Entity e, ToRow values, MonadIO m) =>
values -> DBT m ()
insert values
fs = DBT m Int64 -> DBT m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (DBT m Int64 -> DBT m ()) -> DBT m Int64 -> DBT m ()
forall a b. (a -> b) -> a -> b
$ QueryNature -> Query -> values -> DBT m Int64
forall params (m :: * -> *).
(ToRow params, MonadIO m) =>
QueryNature -> Query -> params -> DBT m Int64
execute QueryNature
Insert (forall e. Entity e => Query
_insert @e) values
fs
upsert
:: forall e values m
. (Entity e, ToRow values, MonadIO m)
=> values
-> Vector Field
-> DBT m ()
upsert :: forall e values (m :: * -> *).
(Entity e, ToRow values, MonadIO m) =>
values -> Vector Field -> DBT m ()
upsert values
entity Vector Field
fieldsToReplace = DBT m Int64 -> DBT m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (DBT m Int64 -> DBT m ()) -> DBT m Int64 -> DBT m ()
forall a b. (a -> b) -> a -> b
$ QueryNature -> Query -> values -> DBT m Int64
forall params (m :: * -> *).
(ToRow params, MonadIO m) =>
QueryNature -> Query -> params -> DBT m Int64
execute QueryNature
Insert (forall e. Entity e => Query
_insert @e Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Vector Field -> Query
_onConflictDoUpdate Vector Field
conflictTarget Vector Field
fieldsToReplace) values
entity
where
conflictTarget :: Vector Field
conflictTarget = Field -> Vector Field
forall a. a -> Vector a
V.singleton (Field -> Vector Field) -> Field -> Vector Field
forall a b. (a -> b) -> a -> b
$ forall e. Entity e => Field
primaryKey @e
insertMany
:: forall e values m
. (Entity e, ToRow values, MonadIO m)
=> [values]
-> DBT m ()
insertMany :: forall e values (m :: * -> *).
(Entity e, ToRow values, MonadIO m) =>
[values] -> DBT m ()
insertMany [values]
values = DBT m Int64 -> DBT m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (DBT m Int64 -> DBT m ()) -> DBT m Int64 -> DBT m ()
forall a b. (a -> b) -> a -> b
$ QueryNature -> Query -> [values] -> DBT m Int64
forall params (m :: * -> *).
(ToRow params, MonadIO m) =>
QueryNature -> Query -> [params] -> DBT m Int64
executeMany QueryNature
Insert (forall e. Entity e => Query
_insert @e) [values]
values
update
:: forall e newValue m
. (Entity e, ToRow newValue, MonadIO m)
=> newValue
-> DBT m ()
update :: forall e values (m :: * -> *).
(Entity e, ToRow values, MonadIO m) =>
values -> DBT m ()
update newValue
fs = DBT m Int64 -> DBT m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (DBT m Int64 -> DBT m ()) -> DBT m Int64 -> DBT m ()
forall a b. (a -> b) -> a -> b
$ QueryNature -> Query -> UpdateRow newValue -> DBT m Int64
forall params (m :: * -> *).
(ToRow params, MonadIO m) =>
QueryNature -> Query -> params -> DBT m Int64
execute QueryNature
Update (forall e. Entity e => Query
_update @e) (newValue -> UpdateRow newValue
forall a. a -> UpdateRow a
UpdateRow newValue
fs)
updateFieldsBy
:: forall e v1 v2 m
. (Entity e, MonadIO m, ToRow v2, ToField v1)
=> Vector Field
-> (Field, v1)
-> v2
-> DBT m Int64
updateFieldsBy :: forall e v1 v2 (m :: * -> *).
(Entity e, MonadIO m, ToRow v2, ToField v1) =>
Vector Field -> (Field, v1) -> v2 -> DBT m Int64
updateFieldsBy Vector Field
fs (Field
f, v1
oldValue) v2
newValue = QueryNature -> Query -> [Action] -> DBT m Int64
forall params (m :: * -> *).
(ToRow params, MonadIO m) =>
QueryNature -> Query -> params -> DBT m Int64
execute QueryNature
Update (forall e. Entity e => Vector Field -> Field -> Query
_updateFieldsBy @e Vector Field
fs Field
f) (v2 -> [Action]
forall a. ToRow a => a -> [Action]
toRow v2
newValue [Action] -> [Action] -> [Action]
forall a. [a] -> [a] -> [a]
++ Only v1 -> [Action]
forall a. ToRow a => a -> [Action]
toRow (v1 -> Only v1
forall a. a -> Only a
Only v1
oldValue))
delete
:: forall e value m
. (Entity e, ToRow value, MonadIO m)
=> value
-> DBT m ()
delete :: forall e values (m :: * -> *).
(Entity e, ToRow values, MonadIO m) =>
values -> DBT m ()
delete value
value = forall e values (m :: * -> *).
(Entity e, ToRow values, MonadIO m) =>
Vector Field -> values -> DBT m ()
deleteByField @e [forall e. Entity e => Field
primaryKey @e] value
value
deleteByField
:: forall e values m
. (Entity e, ToRow values, MonadIO m)
=> Vector Field
-> values
-> DBT m ()
deleteByField :: forall e values (m :: * -> *).
(Entity e, ToRow values, MonadIO m) =>
Vector Field -> values -> DBT m ()
deleteByField Vector Field
fs values
values = DBT m Int64 -> DBT m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (DBT m Int64 -> DBT m ()) -> DBT m Int64 -> DBT m ()
forall a b. (a -> b) -> a -> b
$ QueryNature -> Query -> values -> DBT m Int64
forall params (m :: * -> *).
(ToRow params, MonadIO m) =>
QueryNature -> Query -> params -> DBT m Int64
execute QueryNature
Delete (forall e. Entity e => Vector Field -> Query
_deleteWhere @e Vector Field
fs) values
values
_select :: forall e. Entity e => Query
_select :: forall e. Entity e => Query
_select = Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$ Text
"SELECT " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
expandQualifiedFields @e Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" FROM " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e
_selectWithFields :: forall e. Entity e => Vector Field -> Query
_selectWithFields :: forall e. Entity e => Vector Field -> Query
_selectWithFields Vector Field
fs = Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$ Text
"SELECT " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Text -> Text
expandQualifiedFields' Vector Field
fs Text
tn Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" FROM " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text -> Text
quoteName Text
tn
where
tn :: Text
tn = forall e. Entity e => Text
getTableName @e
_where :: Vector Field -> Query
_where :: Vector Field -> Query
_where Vector Field
fs' = Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$ Text
" WHERE " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
clauseFields
where
clauseFields :: Text
clauseFields = Vector Text -> Text
forall m. Monoid m => Vector m -> m
forall (t :: * -> *) m. (Foldable t, Monoid m) => t m -> m
fold (Vector Text -> Text) -> Vector Text -> Text
forall a b. (a -> b) -> a -> b
$ Text -> Vector Text -> Vector Text
intercalateVector Text
" AND " ((Field -> Text) -> Vector Field -> Vector Text
forall a b. (a -> b) -> Vector a -> Vector b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Field -> Text
placeholder Vector Field
fs')
_selectWhere :: forall e. Entity e => Vector Field -> Query
_selectWhere :: forall e. Entity e => Vector Field -> Query
_selectWhere Vector Field
fs = forall e. Entity e => Query
_select @e Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Query
_where Vector Field
fs
_selectWhereNotNull :: forall e. Entity e => Vector Field -> Query
_selectWhereNotNull :: forall e. Entity e => Vector Field -> Query
_selectWhereNotNull Vector Field
fs = forall e. Entity e => Query
_select @e Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Text -> Query
textToQuery (Text
" WHERE " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Text
isNotNull Vector Field
fs)
_selectWhereNull :: forall e. Entity e => Vector Field -> Query
_selectWhereNull :: forall e. Entity e => Vector Field -> Query
_selectWhereNull Vector Field
fs = forall e. Entity e => Query
_select @e Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Text -> Query
textToQuery (Text
" WHERE " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Text
isNull Vector Field
fs)
_selectWhereIn :: forall e. Entity e => Field -> Vector Text -> Query
_selectWhereIn :: forall e. Entity e => Field -> Vector Text -> Query
_selectWhereIn Field
f Vector Text
values = forall e. Entity e => Query
_select @e Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Text -> Query
textToQuery (Text
" WHERE " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Field -> Vector Text -> Text
isIn Field
f Vector Text
values)
_joinSelect :: forall e1 e2. (Entity e1, Entity e2) => Query
_joinSelect :: forall e1 e2. (Entity e1, Entity e2) => Query
_joinSelect =
Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$
Text
"SELECT "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
expandQualifiedFields @e1
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
", "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
expandQualifiedFields @e2
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" FROM "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e1
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Query -> Text
queryToText (forall e. Entity e => Field -> Query
_innerJoin @e2 (forall e. Entity e => Field
primaryKey @e2))
_innerJoin :: forall e. Entity e => Field -> Query
_innerJoin :: forall e. Entity e => Field -> Query
_innerJoin Field
f =
Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$
Text
" INNER JOIN "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" USING("
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Field -> Text
fieldName Field
f
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
")"
_joinSelectWithFields
:: forall e1 e2
. (Entity e1, Entity e2)
=> Vector Field
-> Vector Field
-> Query
_joinSelectWithFields :: forall e1 e2.
(Entity e1, Entity e2) =>
Vector Field -> Vector Field -> Query
_joinSelectWithFields Vector Field
fs1 Vector Field
fs2 =
Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$
Text
"SELECT "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Text -> Text
expandQualifiedFields' Vector Field
fs1 Text
tn1
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
", "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Text -> Text
expandQualifiedFields' Vector Field
fs2 Text
tn2
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" FROM "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e1
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Query -> Text
queryToText (forall e. Entity e => Field -> Query
_innerJoin @e2 (forall e. Entity e => Field
primaryKey @e2))
where
tn1 :: Text
tn1 = forall e. Entity e => Text
getTableName @e1
tn2 :: Text
tn2 = forall e. Entity e => Text
getTableName @e2
_joinSelectOneByField
:: forall e1 e2
. (Entity e1, Entity e2)
=> Field
-> Field
-> Query
_joinSelectOneByField :: forall e1 e2. (Entity e1, Entity e2) => Field -> Field -> Query
_joinSelectOneByField Field
pivotField Field
whereField =
Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$
Text
"SELECT "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
expandQualifiedFields @e1
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" FROM "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> (forall e. Entity e => Text
getTableName @e1)
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" INNER JOIN "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e2
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" ON "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> (forall e. Entity e => Text
getTableName @e1)
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
"."
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Field -> Text
getFieldName Field
pivotField
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" = "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> (forall e. Entity e => Text
getTableName @e2)
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
"."
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Field -> Text
getFieldName Field
pivotField
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" WHERE "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Field -> Text
placeholder' @e2 Field
whereField
_insert :: forall e. Entity e => Query
_insert :: forall e. Entity e => Query
_insert = Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$ Text
"INSERT INTO " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
fs Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" VALUES " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
ps
where
fs :: Text
fs = Text -> Text
inParens (forall e. Entity e => Text
expandFields @e)
ps :: Text
ps = Text -> Text
inParens (Vector Field -> Text
generatePlaceholders (Vector Field -> Text) -> Vector Field -> Text
forall a b. (a -> b) -> a -> b
$ forall e. Entity e => Vector Field
fields @e)
_onConflictDoUpdate :: Vector Field -> Vector Field -> Query
_onConflictDoUpdate :: Vector Field -> Vector Field -> Query
_onConflictDoUpdate Vector Field
conflictTarget Vector Field
fieldsToReplace =
Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$ Text
" ON CONFLICT (" Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
targetNames Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
") DO UPDATE SET " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
replacedFields
where
targetNames :: Text
targetNames = Vector Text -> Text
forall m. Monoid m => Vector m -> m
forall (t :: * -> *) m. (Foldable t, Monoid m) => t m -> m
fold (Vector Text -> Text) -> Vector Text -> Text
forall a b. (a -> b) -> a -> b
$ Text -> Vector Text -> Vector Text
intercalateVector Text
", " ((Field -> Text) -> Vector Field -> Vector Text
forall a b. (a -> b) -> Vector a -> Vector b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Field -> Text
fieldName Vector Field
conflictTarget)
replacedFields :: Text
replacedFields = Vector Text -> Text
forall m. Monoid m => Vector m -> m
forall (t :: * -> *) m. (Foldable t, Monoid m) => t m -> m
fold (Vector Text -> Text) -> Vector Text -> Text
forall a b. (a -> b) -> a -> b
$ Text -> Vector Text -> Vector Text
intercalateVector Text
", " ((Field -> Text) -> Vector Field -> Vector Text
forall a b. (a -> b) -> Vector a -> Vector b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Text -> Text
replaceField (Text -> Text) -> (Field -> Text) -> Field -> Text
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Field -> Text
fieldName) Vector Field
fieldsToReplace)
replaceField :: Text -> Text
replaceField :: Text -> Text
replaceField Text
f = Text
f Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" = EXCLUDED." Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
f
_update :: forall e. Entity e => Query
_update :: forall e. Entity e => Query
_update = forall e. Entity e => Field -> Query
_updateBy @e (forall e. Entity e => Field
primaryKey @e)
_updateBy :: forall e. Entity e => Field -> Query
_updateBy :: forall e. Entity e => Field -> Query
_updateBy Field
f = forall e. Entity e => Vector Field -> Field -> Query
_updateFieldsBy @e (forall e. Entity e => Vector Field
fields @e) Field
f
_updateFields :: forall e. Entity e => Vector Field -> Query
_updateFields :: forall e. Entity e => Vector Field -> Query
_updateFields Vector Field
fs = forall e. Entity e => Vector Field -> Field -> Query
_updateFieldsBy @e Vector Field
fs (forall e. Entity e => Field
primaryKey @e)
_updateFieldsBy
:: forall e
. Entity e
=> Vector Field
-> Field
-> Query
_updateFieldsBy :: forall e. Entity e => Vector Field -> Field -> Query
_updateFieldsBy Vector Field
fs' Field
f =
Text -> Query
textToQuery
( Text
"UPDATE "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" SET "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
updatedFields
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" = "
Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
newValues
)
Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Query
_where [Item (Vector Field)
Field
f]
where
fs :: Vector Field
fs = (Field -> Bool) -> Vector Field -> Vector Field
forall a. (a -> Bool) -> Vector a -> Vector a
V.filter (Field -> Field -> Bool
forall a. Eq a => a -> a -> Bool
/= (forall e. Entity e => Field
primaryKey @e)) Vector Field
fs'
newValues :: Text
newValues = Text
"ROW" Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text -> Text
inParens (Vector Field -> Text
generatePlaceholders Vector Field
fs)
updatedFields :: Text
updatedFields =
Text -> Text
inParens (Text -> Text) -> Text -> Text
forall a b. (a -> b) -> a -> b
$
(Text -> Text -> Text) -> Vector Text -> Text
forall a. (a -> a -> a) -> Vector a -> a
V.foldl1' (\Text
element Text
acc -> Text
element Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
", " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
acc) (Text -> Text
quoteName (Text -> Text) -> (Field -> Text) -> Field -> Text
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Field -> Text
fieldName (Field -> Text) -> Vector Field -> Vector Text
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Vector Field
fs)
_delete :: forall e. Entity e => Query
_delete :: forall e. Entity e => Query
_delete = Text -> Query
textToQuery (Text
"DELETE FROM " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> forall e. Entity e => Text
getTableName @e) Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Query
_where [forall e. Entity e => Field
primaryKey @e]
_deleteWhere :: forall e. Entity e => Vector Field -> Query
_deleteWhere :: forall e. Entity e => Vector Field -> Query
_deleteWhere Vector Field
fs = Text -> Query
textToQuery (Text
"DELETE FROM " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> (forall e. Entity e => Text
getTableName @e)) Query -> Query -> Query
forall a. Semigroup a => a -> a -> a
<> Vector Field -> Query
_where Vector Field
fs
_orderBy :: (Field, SortKeyword) -> Query
_orderBy :: (Field, SortKeyword) -> Query
_orderBy (Field
f, SortKeyword
sort) = Text -> Query
textToQuery (Text
" ORDER BY " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text -> Text
quoteName (Field -> Text
fieldName Field
f) Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
" " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> SortKeyword -> Text
forall a. Display a => a -> Text
display SortKeyword
sort)
_orderByMany :: Vector (Field, SortKeyword) -> Query
_orderByMany :: Vector (Field, SortKeyword) -> Query
_orderByMany Vector (Field, SortKeyword)
sortExpressions = Text -> Query
textToQuery (Text -> Query) -> Text -> Query
forall a b. (a -> b) -> a -> b
$ Text
" ORDER BY " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Vector Text -> Text
forall m. Monoid m => Vector m -> m
forall (t :: * -> *) m. (Foldable t, Monoid m) => t m -> m
fold (Text -> Vector Text -> Vector Text
intercalateVector Text
", " (Vector Text -> Vector Text) -> Vector Text -> Vector Text
forall a b. (a -> b) -> a -> b
$ ((Field, SortKeyword) -> Text)
-> Vector (Field, SortKeyword) -> Vector Text
forall a b. (a -> b) -> Vector a -> Vector b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Field, SortKeyword) -> Text
renderSortExpression Vector (Field, SortKeyword)
sortExpressions)