hasql/library/Hasql.hs

136 lines
4.6 KiB
Haskell
Raw Normal View History

2015-11-14 13:58:30 +03:00
module Hasql
2015-11-08 21:09:42 +03:00
(
-- * Connection
2015-11-08 21:09:42 +03:00
Connection,
Settings.Settings(..),
acquire,
release,
-- * Query
Query(..),
query,
2015-11-08 21:09:42 +03:00
-- * Errors
AcquisitionError(..),
ResultsError(..),
ResultError(..),
RowError(..),
)
where
import Hasql.Prelude
import qualified Database.PostgreSQL.LibPQ as LibPQ
import qualified Hasql.PreparedStatementRegistry as PreparedStatementRegistry
import qualified Hasql.Deserialization.Results as ResultsDeserialization
import qualified Hasql.Deserialization as Deserialization
import qualified Hasql.Serialization.Params as ParamsSerialization
import qualified Hasql.Serialization as Serialization
import qualified Hasql.Settings as Settings
import qualified Hasql.IO as IO
-- |
-- A single connection to the database.
data Connection =
Connection !LibPQ.Connection !Bool !PreparedStatementRegistry.PreparedStatementRegistry
data ResultsError =
-- |
-- An error on the client-side,
-- with a message generated by the \"libpq\" library.
-- Usually indicates problems with connection.
2015-11-08 21:18:59 +03:00
ClientError !(Maybe ByteString) |
2015-11-08 21:09:42 +03:00
ResultError !ResultError
2015-11-15 12:10:28 +03:00
deriving (Show, Eq)
2015-11-08 21:09:42 +03:00
data ResultError =
-- |
-- An error reported by the DB. Code, message, details, hint.
--
-- * The SQLSTATE code for the error. The SQLSTATE code identifies the type of error that has occurred;
-- it can be used by front-end applications to perform specific operations (such as error handling)
-- in response to a particular database error.
-- For a list of the possible SQLSTATE codes, see Appendix A.
-- This field is not localizable, and is always present.
--
-- * The primary human-readable error message (typically one line). Always present.
--
-- * Detail: an optional secondary error message carrying more detail about the problem.
-- Might run to multiple lines.
--
-- * Hint: an optional suggestion what to do about the problem.
-- This is intended to differ from detail in that it offers advice (potentially inappropriate)
-- rather than hard facts. Might run to multiple lines.
2015-11-08 21:18:59 +03:00
ServerError !ByteString !ByteString !(Maybe ByteString) !(Maybe ByteString) |
2015-11-08 21:09:42 +03:00
-- |
-- The database returned an unexpected result.
-- Indicates an improper statement or a schema mismatch.
UnexpectedResult !Text |
-- |
-- An error of the row reader, preceded by the index of the row.
RowError !Int !RowError |
-- |
-- An unexpected amount of rows.
UnexpectedAmountOfRows !Int
2015-11-15 12:10:28 +03:00
deriving (Show, Eq)
2015-11-08 21:09:42 +03:00
data RowError =
EndOfInput |
UnexpectedNull |
ValueError !Text
2015-11-15 12:10:28 +03:00
deriving (Show, Eq)
2015-11-08 21:09:42 +03:00
-- |
-- A connection acquistion error.
2015-11-15 12:10:28 +03:00
type AcquisitionError =
Maybe ByteString
2015-11-08 21:09:42 +03:00
-- |
-- Acquire a connection using the provided settings.
2015-11-08 21:18:59 +03:00
acquire :: Settings.Settings -> IO (Either AcquisitionError Connection)
2015-11-08 21:09:42 +03:00
acquire settings =
runEitherT $ do
pqConnection <- lift (IO.acquireConnection settings)
2015-11-15 12:10:28 +03:00
lift (IO.checkConnectionStatus pqConnection) >>= traverse left
2015-11-08 21:09:42 +03:00
lift (IO.initConnection pqConnection)
integerDatetimes <- lift (IO.getIntegerDatetimes pqConnection)
registry <- lift (IO.acquirePreparedStatementRegistry)
pure (Connection pqConnection integerDatetimes registry)
-- |
-- Release the connection.
release :: Connection -> IO ()
release (Connection pqConnection _ _) =
LibPQ.finish pqConnection
2015-11-14 13:58:30 +03:00
-- |
-- A strictly single-statement query, which can be parameterized and prepared.
--
-- SQL template, params serializer, results deserializer and a flag, determining whether it should be prepared.
--
type Query a b =
2015-11-14 13:58:30 +03:00
(ByteString, Serialization.Params a, Deserialization.Results b, Bool)
2015-11-08 21:09:42 +03:00
-- |
-- Execute a parametric query, producing either a deserialization failure or a successful result.
query :: Connection -> Query a b -> a -> IO (Either ResultsError b)
query (Connection pqConnection integerDatetimes registry) (template, serializer, deserializer, preparable) params =
2015-11-08 21:09:42 +03:00
fmap (mapLeft coerceResultsError) $ runEitherT $ do
EitherT $ IO.sendParametricQuery pqConnection integerDatetimes registry template (coerceSerializer serializer) preparable params
EitherT $ IO.getResults pqConnection integerDatetimes (coerceDeserializer deserializer)
-- |
-- WARNING: We need to take special care that the structure of
-- the "ResultsDeserialization.Error" type in the public API is an exact copy of
-- "Error", since we're using coercion.
coerceResultsError :: ResultsDeserialization.Error -> ResultsError
coerceResultsError =
unsafeCoerce
coerceDeserializer :: Deserialization.Results a -> ResultsDeserialization.Results a
coerceDeserializer =
unsafeCoerce
coerceSerializer :: Serialization.Params a -> ParamsSerialization.Params a
coerceSerializer =
unsafeCoerce