feat: query builder, strict build

This commit is contained in:
bingus 2023-12-17 18:38:30 -06:00
parent 6a7350e665
commit 825eaa0079
Signed by: orion
GPG Key ID: 6D4165AE4C928719
5 changed files with 99 additions and 35 deletions

View File

@ -1,19 +1,25 @@
package: package:
build:
strict: true
pedantic_packages: true
dependencies: dependencies:
- aff - aff
- console - arrays
- bifunctors
- control
- effect - effect
- either - either
- foldable-traversable - exceptions
- foreign - foreign
- fork - fork
- maybe - maybe
- mmorph - mmorph
- newtype - newtype
- parallel
- postgresql-client - postgresql-client
- prelude - prelude
- strings - record
- stringutils - tailrec
- transformers - transformers
- tuples - tuples
- typelevel-prelude - typelevel-prelude
@ -195,4 +201,4 @@ workspace:
ref: "v0.2.0" ref: "v0.2.0"
package_set: package_set:
url: https://raw.githubusercontent.com/purescript/package-sets/psc-0.15.10-20230930/packages.json url: https://raw.githubusercontent.com/purescript/package-sets/psc-0.15.10-20230930/packages.json
hash: sha256-nTsd44o7/hrTdk0c6dh0wyBqhFFDJJIeKdQU6L1zv/A= hash: sha256-hp58GPoH+qX3eUsk2ecoHBZpQ50rFeZCCMTdKkYTr/Y=

View File

@ -5,26 +5,21 @@ import Prelude
import Control.Alternative (class Alt, class Alternative, class Plus) import Control.Alternative (class Alt, class Alternative, class Plus)
import Control.Monad.Cont (class MonadCont) import Control.Monad.Cont (class MonadCont)
import Control.Monad.Error.Class (class MonadError, class MonadThrow, liftEither, throwError) import Control.Monad.Error.Class (class MonadError, class MonadThrow, liftEither, throwError)
import Control.Monad.Fork.Class import Control.Monad.Fork.Class (class MonadBracket, class MonadFork, class MonadKill, bracket, kill, never, uninterruptible)
( class MonadBracket
, class MonadFork
, class MonadKill
, bracket
, kill
, never
, uninterruptible
)
import Control.Monad.Morph (class MFunctor, class MMonad, embed, hoist) import Control.Monad.Morph (class MFunctor, class MMonad, embed, hoist)
import Control.Monad.Reader (class MonadAsk, class MonadReader, ReaderT, ask) import Control.Monad.Reader (class MonadAsk, class MonadReader, ReaderT, ask)
import Control.Monad.Rec.Class (class MonadRec) import Control.Monad.Rec.Class (class MonadRec)
import Control.Monad.State (StateT(..))
import Control.Monad.Trans.Class (class MonadTrans, lift) import Control.Monad.Trans.Class (class MonadTrans, lift)
import Control.Monad.Writer (class MonadTell, class MonadWriter) import Control.Monad.Writer (class MonadTell, class MonadWriter, WriterT(..))
import Control.MonadPlus (class MonadPlus) import Control.MonadPlus (class MonadPlus)
import Control.Parallel (class Parallel, parallel, sequential) import Control.Parallel (class Parallel, parallel, sequential)
import Data.Bifunctor (lmap) import Data.Bifunctor (lmap)
import Data.Either (Either(..)) import Data.Either (Either(..))
import Data.Maybe (maybe) import Data.Maybe (maybe)
import Data.Newtype (class Newtype, unwrap, wrap) import Data.Newtype (class Newtype, unwrap, wrap)
import Data.Postgres.Query.Builder as Query
import Data.Tuple.Nested ((/\))
import Database.PostgreSQL as Pg import Database.PostgreSQL as Pg
import Database.PostgreSQL.Aff as Pg.Aff import Database.PostgreSQL.Aff as Pg.Aff
import Effect.Aff.Class (class MonadAff, liftAff) import Effect.Aff.Class (class MonadAff, liftAff)
@ -152,28 +147,20 @@ else instance
-- | A monad which can perform logs -- | A monad which can perform logs
class (Monad m, MonadThrow Error m) <= MonadPostgres m where class (Monad m, MonadThrow Error m) <= MonadPostgres m where
query :: forall @a. Pg.FromSQLRow a => String -> m (Array a) query' :: forall @a n. Monad n => (n ~> m) -> Pg.FromSQLRow a => Query.BuilderT n String -> m (Array a)
queryP :: forall @a. Pg.FromSQLRow a => Array Foreign -> String -> m (Array a) exec' :: forall @n. Monad n => (n ~> m) -> Query.BuilderT n String -> m Unit
exec :: String -> m Unit
execP :: Array Foreign -> String -> m Unit
transaction :: forall a. m a -> m a transaction :: forall a. m a -> m a
instance (MonadUnliftAff m, MonadThrow Error m) => MonadPostgres (PostgresT m) where instance (MonadUnliftAff m, MonadThrow Error m) => MonadPostgres (PostgresT m) where
query q = do query' nm q = do
conn <- ask conn <- ask
res <- liftAff $ Pg.Aff.query conn (Pg.Query q) Pg.Row0 qs /\ ps <- Query.runBuilder $ hoist nm q
res <- liftAff $ Pg.Aff.query conn (Pg.Query qs) ps
liftEither $ lmap (error <<< show) $ res liftEither $ lmap (error <<< show) $ res
queryP ps q = do exec' nm q = do
conn <- ask conn <- ask
res <- liftAff $ Pg.Aff.query conn (Pg.Query q) ps qs /\ ps <- Query.runBuilder $ hoist nm q
liftEither $ lmap (error <<< show) $ res res <- liftAff $ Pg.Aff.execute conn (Pg.Query qs) ps
exec q = do
conn <- ask
res <- liftAff $ Pg.Aff.execute conn (Pg.Query q) Pg.Row0
liftEither $ lmap (error <<< show) $ maybe (Right unit) Left $ res
execP ps q = do
conn <- ask
res <- liftAff $ Pg.Aff.execute conn (Pg.Query q) ps
liftEither $ lmap (error <<< show) $ maybe (Right unit) Left $ res liftEither $ lmap (error <<< show) $ maybe (Right unit) Left $ res
transaction pg = do transaction pg = do
conn :: Pg.Connection <- ask conn :: Pg.Connection <- ask
@ -184,10 +171,14 @@ instance (MonadUnliftAff m, MonadThrow Error m) => MonadPostgres (PostgresT m) w
res <- liftAff $ Pg.Aff.withTransaction pool $ mToAff <<< pgToM res <- liftAff $ Pg.Aff.withTransaction pool $ mToAff <<< pgToM
liftEither $ lmap (error <<< show) $ res liftEither $ lmap (error <<< show) $ res
else instance (MonadUnliftAff m, Monad (t m), MonadThrow Error (t m), MMonad t, MonadPostgres m) => MonadPostgres (t m) where else instance (MonadUnliftAff m, Monad (t m), MonadThrow Error (t m), MMonad t, MonadPostgres m) => MonadPostgres (t m) where
query q = lift $ query q query' nt q = do
queryP ps q = lift $ queryP ps q ran <- nt $ Query.runBuilder q
exec q = lift $ exec q let builderInM = StateT $ const $ pure ran
execP ps q = lift $ execP ps q lift $ query' identity builderInM
exec' nt q = do
ran <- nt $ Query.runBuilder q
let builderInM = StateT $ const $ pure ran
lift $ exec' identity builderInM
transaction = hoist transaction transaction = hoist transaction
runPostgres :: forall m a. Pg.Connection -> PostgresT m a -> m a runPostgres :: forall m a. Pg.Connection -> PostgresT m a -> m a
@ -195,3 +186,15 @@ runPostgres conn t = (unwrap $ unwrap t) conn
runHasPostgres :: forall m a. Pg.Connection -> HasPostgresT m a -> m a runHasPostgres :: forall m a. Pg.Connection -> HasPostgresT m a -> m a
runHasPostgres conn t = (unwrap $ unwrap t) conn runHasPostgres conn t = (unwrap $ unwrap t) conn
query :: forall m @a. MonadPostgres m => Pg.FromSQLRow a => Query.BuilderT m String -> m (Array a)
query = query' identity
exec :: forall m @a. MonadPostgres m => Pg.FromSQLRow a => Query.BuilderT m String -> m Unit
exec = exec' identity
query0 :: forall m @a. MonadPostgres m => Pg.FromSQLRow a => String -> m (Array a)
query0 q = query' identity (pure q)
exec0 :: forall m @a. MonadPostgres m => Pg.FromSQLRow a => String -> m Unit
exec0 q = exec' identity (pure q)

View File

@ -0,0 +1,21 @@
module Data.Postgres.Query.Builder where
import Prelude
import Control.Monad.State (StateT, get, put, runStateT)
import Data.Array as Array
import Data.Tuple.Nested (type (/\))
import Database.PostgreSQL (class ToSQLValue, toSQLValue)
import Foreign (Foreign)
type BuilderT m a = StateT (Array Foreign) m a
runBuilder :: forall m a. BuilderT m a -> m (a /\ Array Foreign)
runBuilder = flip runStateT []
param :: forall m a. Monad m => ToSQLValue a => a -> BuilderT m String
param p =
do
ps <- get
put $ ps <> [ toSQLValue p ]
pure $ "$" <> show (Array.length ps + 1)

View File

@ -0,0 +1,32 @@
module Spec.Data.Postgres.Query.Builder where
import Prelude
import Control.Monad.Trans.Class (lift)
import Data.Postgres.Query.Builder (param, runBuilder)
import Data.Tuple.Nested ((/\))
import Foreign (unsafeFromForeign)
import Foreign.Internal.Stringify (unsafeStringify)
import Test.Spec (Spec, describe, it)
import Test.Spec.Assertions (shouldEqual)
spec :: Spec Unit
spec =
describe "Data.Postgres.Query.Builder" do
describe "runBuilder" do
it "empty" do
_ /\ ps <- runBuilder (pure unit)
map unsafeStringify ps `shouldEqual` []
describe "param" do
it "single" do
p /\ ps <- runBuilder $ param 123
p `shouldEqual` "$1"
map unsafeFromForeign ps `shouldEqual` [ 123 ]
it "many" do
_ /\ ps <- runBuilder do
a <- param 123
b <- param "abc"
c <- param [ 123 ]
d <- param true
lift $ [ a, b, c, d ] `shouldEqual` [ "$1", "$2", "$3", "$4" ]
map unsafeStringify ps `shouldEqual` [ "123", "\"abc\"", "[123]", "true" ]

View File

@ -4,6 +4,7 @@ import Prelude
import Effect (Effect) import Effect (Effect)
import Effect.Aff (launchAff_) import Effect.Aff (launchAff_)
import Spec.Data.Postgres.Query.Builder as Spec.Data.Postgres.Query.Builder
import Spec.Data.Postgres.Record as Spec.Data.Postgres.Record import Spec.Data.Postgres.Record as Spec.Data.Postgres.Record
import Test.Spec.Reporter (consoleReporter) import Test.Spec.Reporter (consoleReporter)
import Test.Spec.Runner (runSpec) import Test.Spec.Runner (runSpec)
@ -11,3 +12,4 @@ import Test.Spec.Runner (runSpec)
main :: Effect Unit main :: Effect Unit
main = launchAff_ $ runSpec [ consoleReporter ] do main = launchAff_ $ runSpec [ consoleReporter ] do
Spec.Data.Postgres.Record.spec Spec.Data.Postgres.Record.spec
Spec.Data.Postgres.Query.Builder.spec