2017-05-06 15:09:05 +00:00
|
|
|
module Test.Main
|
|
|
|
( main
|
|
|
|
) where
|
|
|
|
|
2017-12-04 21:43:36 +00:00
|
|
|
import Prelude
|
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
import Control.Monad.Aff (Aff, launchAff)
|
|
|
|
import Control.Monad.Aff.AVar (AVAR)
|
|
|
|
import Control.Monad.Aff.Console (CONSOLE)
|
2017-05-06 15:09:05 +00:00
|
|
|
import Control.Monad.Eff (Eff)
|
|
|
|
import Control.Monad.Eff.Class (liftEff)
|
2017-06-03 11:48:00 +00:00
|
|
|
import Control.Monad.Eff.Exception (EXCEPTION, error)
|
2018-04-22 14:47:07 +00:00
|
|
|
import Control.Monad.Eff.Now (NOW)
|
|
|
|
import Control.Monad.Error.Class (catchError, throwError, try)
|
|
|
|
import Data.DateTime.Instant (Instant, unInstant)
|
2018-04-21 11:12:20 +00:00
|
|
|
import Data.Decimal as D
|
2018-04-22 14:47:07 +00:00
|
|
|
import Data.Foldable (all)
|
|
|
|
import Data.JSDate (toInstant)
|
|
|
|
import Data.JSDate as JSDate
|
|
|
|
import Data.Maybe (Maybe(..), fromJust)
|
|
|
|
import Data.Newtype (unwrap)
|
|
|
|
import Database.PostgreSQL (Connection, POSTGRESQL, PoolConfiguration, Query(Query), Row0(Row0), Row1(Row1), Row2(Row2), Row3(Row3), Row9(Row9), execute, newPool, query, scalar, withConnection, withTransaction)
|
|
|
|
import Math ((%))
|
|
|
|
import Partial.Unsafe (unsafePartial)
|
2017-05-06 15:09:05 +00:00
|
|
|
import Test.Assert (ASSERT, assert)
|
2018-04-22 14:47:07 +00:00
|
|
|
import Test.Unit (suite)
|
|
|
|
import Test.Unit as Test.Unit
|
|
|
|
import Test.Unit.Console (TESTOUTPUT)
|
|
|
|
import Test.Unit.Main (runTest)
|
2017-05-06 15:09:05 +00:00
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
withRollback
|
|
|
|
:: ∀ eff
|
|
|
|
. Connection
|
|
|
|
-> Aff (postgreSQL :: POSTGRESQL | eff) Unit
|
|
|
|
-> Aff (postgreSQL :: POSTGRESQL | eff) Unit
|
|
|
|
withRollback conn action = do
|
|
|
|
execute conn (Query "BEGIN TRANSACTION") Row0
|
|
|
|
catchError (action >>= const rollback) (\e -> rollback >>= const (throwError e))
|
|
|
|
where
|
|
|
|
rollback = execute conn (Query "ROLLBACK") Row0
|
|
|
|
|
|
|
|
test
|
|
|
|
:: ∀ eff
|
|
|
|
. Connection
|
|
|
|
-> String
|
|
|
|
-> Aff ( postgreSQL :: POSTGRESQL | eff) Unit
|
|
|
|
-> Test.Unit.TestSuite (postgreSQL :: POSTGRESQL | eff)
|
|
|
|
test conn t a = Test.Unit.test t (withRollback conn a)
|
|
|
|
|
|
|
|
now :: ∀ eff. Eff (now :: NOW | eff) Instant
|
|
|
|
now = unsafePartial $ (fromJust <<< toInstant) <$> JSDate.now
|
|
|
|
|
|
|
|
main :: ∀ eff. Eff (assert :: ASSERT, avar :: AVAR, console :: CONSOLE, exception :: EXCEPTION, now :: NOW, postgreSQL :: POSTGRESQL, testOutput :: TESTOUTPUT | eff) Unit
|
2017-05-06 15:09:05 +00:00
|
|
|
main = void $ launchAff do
|
|
|
|
pool <- newPool config
|
2017-12-05 21:12:01 +00:00
|
|
|
withConnection pool \conn -> do
|
2017-05-06 15:09:05 +00:00
|
|
|
execute conn (Query """
|
2017-12-04 21:43:36 +00:00
|
|
|
CREATE TEMPORARY TABLE foods (
|
|
|
|
name text NOT NULL,
|
|
|
|
delicious boolean NOT NULL,
|
2018-04-21 11:12:20 +00:00
|
|
|
price NUMERIC(4,2) NOT NULL,
|
2018-04-22 14:47:07 +00:00
|
|
|
added TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
2017-12-04 21:43:36 +00:00
|
|
|
PRIMARY KEY (name)
|
|
|
|
)
|
2017-06-03 11:10:15 +00:00
|
|
|
""") Row0
|
2017-05-06 15:09:05 +00:00
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
liftEff $ runTest $ do
|
|
|
|
suite "Postgresql client" $ do
|
|
|
|
let
|
|
|
|
testCount n = do
|
|
|
|
count <- scalar conn (Query """
|
|
|
|
SELECT count(*) = $1
|
|
|
|
FROM foods
|
|
|
|
""") (Row1 n)
|
|
|
|
liftEff <<< assert $ count == Just true
|
2017-06-03 11:48:00 +00:00
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
Test.Unit.test "transaction commit" $ do
|
|
|
|
withTransaction conn do
|
|
|
|
execute conn (Query """
|
|
|
|
INSERT INTO foods (name, delicious, price)
|
|
|
|
VALUES ($1, $2, $3)
|
|
|
|
""") (Row3 "pork" true (D.fromString "8.30"))
|
|
|
|
testCount 1
|
|
|
|
testCount 1
|
|
|
|
execute conn (Query """
|
|
|
|
DELETE FROM foods
|
|
|
|
""") Row0
|
2018-04-21 11:12:20 +00:00
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
Test.Unit.test "transaction rollback" $ do
|
|
|
|
_ <- try $ withTransaction conn do
|
|
|
|
execute conn (Query """
|
|
|
|
INSERT INTO foods (name, delicious, price)
|
|
|
|
VALUES ($1, $2, $3)
|
|
|
|
""") (Row3 "pork" true (D.fromString "8.30"))
|
|
|
|
testCount 1
|
|
|
|
throwError $ error "fail"
|
|
|
|
testCount 0
|
2017-05-06 15:09:05 +00:00
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
let
|
|
|
|
insertFood =
|
|
|
|
execute conn (Query """
|
|
|
|
INSERT INTO foods (name, delicious, price)
|
|
|
|
VALUES ($1, $2, $3), ($4, $5, $6), ($7, $8, $9)
|
|
|
|
""") (Row9
|
|
|
|
"pork" true (D.fromString "8.30")
|
|
|
|
"sauerkraut" false (D.fromString "3.30")
|
|
|
|
"rookworst" true (D.fromString "5.60"))
|
|
|
|
test conn "select column subset" $ do
|
|
|
|
insertFood
|
|
|
|
names <- query conn (Query """
|
|
|
|
SELECT name, delicious
|
|
|
|
FROM foods
|
|
|
|
WHERE delicious
|
|
|
|
ORDER BY name ASC
|
|
|
|
""") Row0
|
|
|
|
liftEff <<< assert $ names == [Row2 "pork" true, Row2 "rookworst" true]
|
2017-06-03 11:48:00 +00:00
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
test conn "select default instant value" $ do
|
|
|
|
before <- liftEff $ (unwrap <<< unInstant) <$> now
|
|
|
|
insertFood
|
|
|
|
added <- query conn (Query """
|
|
|
|
SELECT added
|
|
|
|
FROM foods
|
|
|
|
""") Row0
|
|
|
|
after <- liftEff $ (unwrap <<< unInstant) <$> now
|
|
|
|
-- | timestamps are fetched without milliseconds so we have to
|
|
|
|
-- | round before value down
|
|
|
|
liftEff <<< assert $ all
|
|
|
|
(\(Row1 t) ->
|
|
|
|
( unwrap $ unInstant t) >= (before - before % 1000.0)
|
|
|
|
&& after >= (unwrap $ unInstant t))
|
|
|
|
added
|
2017-06-03 11:48:00 +00:00
|
|
|
|
2018-04-22 14:47:07 +00:00
|
|
|
test conn "select decimal" $ do
|
|
|
|
insertFood
|
|
|
|
sauerkrautPrice <- query conn (Query """
|
|
|
|
SELECT price
|
|
|
|
FROM foods
|
|
|
|
WHERE NOT delicious
|
|
|
|
""") Row0
|
|
|
|
liftEff <<< assert $ sauerkrautPrice == [Row1 (D.fromString "3.30")]
|
2017-06-03 11:48:00 +00:00
|
|
|
|
2017-05-06 15:09:05 +00:00
|
|
|
|
|
|
|
config :: PoolConfiguration
|
|
|
|
config =
|
|
|
|
{ user: "postgres"
|
|
|
|
, password: "lol123"
|
|
|
|
, host: "127.0.0.1"
|
|
|
|
, port: 5432
|
|
|
|
, database: "purspg"
|
|
|
|
, max: 10
|
|
|
|
, idleTimeoutMillis: 1000
|
|
|
|
}
|