module Hasql.Statement ( Statement(..), refineResult, -- * Recipies -- ** Insert many -- $insertMany -- ** IN and NOT IN -- $inAndNotIn ) where import Hasql.Private.Prelude import qualified Hasql.Decoders as Decoders import qualified Hasql.Encoders as Encoders import qualified Hasql.Private.Decoders as Decoders {-| Specification of a strictly single-statement query, which can be parameterized and prepared. Consists of the following: * SQL template, * params encoder, * result decoder, * a flag, determining whether it should be prepared. The SQL template must be formatted according to Postgres' standard, with any non-ASCII characters of the template encoded using UTF-8. According to the format, parameters must be referred to using a positional notation, as in the following: @$1@, @$2@, @$3@ and etc. Those references must be used in accordance with the order in which the value encoders are specified in 'Encoders.Params'. Following is an example of a declaration of a prepared statement with its associated codecs. @ selectSum :: 'Statement' (Int64, Int64) Int64 selectSum = 'Statement' sql encoder decoder True where sql = "select ($1 + $2)" encoder = ('fst' '>$<' Encoders.'Hasql.Encoders.param' (Encoders.'Hasql.Encoders.nonNullable' Encoders.'Hasql.Encoders.int8')) '<>' ('snd' '>$<' Encoders.'Hasql.Encoders.param' (Encoders.'Hasql.Encoders.nonNullable' Encoders.'Hasql.Encoders.int8')) decoder = Decoders.'Hasql.Decoders.singleRow' (Decoders.'Hasql.Decoders.column' (Decoders.'Hasql.Decoders.nonNullable' Decoders.'Hasql.Decoders.int8')) @ The statement above accepts a product of two parameters of type 'Int64' and produces a single result of type 'Int64'. -} data Statement a b = Statement ByteString (Encoders.Params a) (Decoders.Result b) Bool instance Functor (Statement a) where {-# INLINE fmap #-} fmap :: (a -> b) -> Statement a a -> Statement a b fmap = (a -> b) -> Statement a a -> Statement a b forall (p :: * -> * -> *) b c a. Profunctor p => (b -> c) -> p a b -> p a c rmap instance Profunctor Statement where {-# INLINE dimap #-} dimap :: (a -> b) -> (c -> d) -> Statement b c -> Statement a d dimap a -> b f1 c -> d f2 (Statement ByteString template Params b encoder Result c decoder Bool preparable) = ByteString -> Params a -> Result d -> Bool -> Statement a d forall a b. ByteString -> Params a -> Result b -> Bool -> Statement a b Statement ByteString template ((a -> b) -> Params b -> Params a forall (f :: * -> *) a b. Contravariant f => (a -> b) -> f b -> f a contramap a -> b f1 Params b encoder) ((c -> d) -> Result c -> Result d forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b fmap c -> d f2 Result c decoder) Bool preparable {-| Refine a result of a statement, causing the running session to fail with the `UnexpectedResult` error in case of refinement failure. This function is especially useful for refining the results of statements produced with <http://hackage.haskell.org/package/hasql-th the \"hasql-th\" library>. -} refineResult :: (a -> Either Text b) -> Statement params a -> Statement params b refineResult :: (a -> Either Text b) -> Statement params a -> Statement params b refineResult a -> Either Text b refiner (Statement ByteString template Params params encoder Result a decoder Bool preparable) = ByteString -> Params params -> Result b -> Bool -> Statement params b forall a b. ByteString -> Params a -> Result b -> Bool -> Statement a b Statement ByteString template Params params encoder ((a -> Either Text b) -> Result a -> Result b forall a b. (a -> Either Text b) -> Result a -> Result b Decoders.refineResult a -> Either Text b refiner Result a decoder) Bool preparable {- $insertMany It is not currently possible to pass in an array of encodable values to use in an insert many statement. Instead, PostgreSQL's (9.4 or later) @unnest@ function can be used in an analogous way to haskell's `zip` function by passing in multiple arrays of values to be zipped into the rows we want to insert: @ insertMultipleLocations :: 'Statement' (Vector (UUID, Double, Double)) () insertMultipleLocations = 'Statement' sql encoder decoder True where sql = "insert into location (id, x, y) select * from unnest ($1, $2, $3)" encoder = contramap Vector.'Data.Vector.unzip3' $ contrazip3 (vector Encoders.'Encoders.uuid') (vector Encoders.'Encoders.float8') (vector Encoders.'Encoders.float8') where vector = Encoders.'Encoders.param' . Encoders.'Encoders.nonNullable' . Encoders.'Encoders.array' . Encoders.'Encoders.dimension' 'foldl'' . Encoders.'Encoders.element' . Encoders.'Encoders.nonNullable' decoder = Decoders.'Decoders.noResult' @ This approach is much more efficient than executing a single-row Insert statement multiple times. -} {- $inAndNotIn There is a common misconception that Postgresql supports array as a parameter for the @IN@ operator. However Postgres only supports a syntactical list of values with it, i.e., you have to specify each option as an individual parameter (@something IN ($1, $2, $3)@). Clearly it would be much more convenient to provide an array as a single parameter, but the @IN@ operator does not support that. Fortunately, Postgres does provide such functionality with other operators: * Use @something = ANY($1)@ instead of @something IN ($1)@ * Use @something <> ALL($1)@ instead of @something NOT IN ($1)@ For details see <https://www.postgresql.org/docs/9.6/static/functions-comparisons.html#AEN20944 the Postgresql docs>. -}