module PostgreSQL.Binary.Encoding
(
Encoding,
encodingBytes,
array,
array_foldable,
array_vector,
nullableArray_vector,
hStore_foldable,
hStore_hashMap,
hStore_map,
bool,
int2_int16,
int2_word16,
int4_int32,
int4_word32,
int8_int64,
int8_word64,
float4,
float8,
numeric,
uuid,
inet,
char_utf8,
text_strict,
text_lazy,
bytea_strict,
bytea_lazy,
date,
time_int,
time_float,
timetz_int,
timetz_float,
timestamp_int,
timestamp_float,
timestamptz_int,
timestamptz_float,
interval_int,
interval_float,
json_bytes,
json_ast,
jsonb_bytes,
jsonb_ast,
Array,
encodingArray,
nullArray,
dimensionArray,
)
where
import PostgreSQL.Binary.Prelude hiding (bool, length)
import qualified ByteString.StrictBuilder as C
import qualified Data.Vector as A
import qualified PostgreSQL.Binary.Encoding.Builders as B
import qualified Data.ByteString.Builder as M
import qualified Data.ByteString.Lazy as N
import qualified Data.Text.Lazy as L
import qualified Data.Aeson as R
import qualified Network.IP.Addr as G
type Encoding =
C.Builder
{-# INLINE encodingBytes #-}
encodingBytes :: Encoding -> ByteString
encodingBytes =
C.builderBytes
{-# INLINE array #-}
array :: Word32 -> Array -> Encoding
array oid (Array payload dimensions nulls) =
B.array oid dimensions nulls payload
{-# INLINE array_foldable #-}
array_foldable :: Foldable foldable => Word32 -> (element -> Maybe Encoding) -> foldable element -> Encoding
array_foldable oid elementBuilder =
array oid . dimensionArray foldl' (maybe nullArray encodingArray . elementBuilder)
{-# INLINE array_vector #-}
array_vector :: Word32 -> (element -> Encoding) -> Vector element -> Encoding
array_vector oid elementBuilder vector =
B.array_vector oid elementBuilder vector
{-# INLINE nullableArray_vector #-}
nullableArray_vector :: Word32 -> (element -> Encoding) -> Vector (Maybe element) -> Encoding
nullableArray_vector oid elementBuilder vector =
B.nullableArray_vector oid elementBuilder vector
{-# INLINE hStore_foldable #-}
hStore_foldable :: Foldable foldable => foldable (Text, Maybe Text) -> Encoding
hStore_foldable =
B.hStoreUsingFoldl foldl
{-# INLINE hStore_hashMap #-}
hStore_hashMap :: HashMap Text (Maybe Text) -> Encoding
hStore_hashMap =
B.hStore_hashMap
{-# INLINE hStore_map #-}
hStore_map :: Map Text (Maybe Text) -> Encoding
hStore_map =
B.hStore_map
{-# INLINE bool #-}
bool :: Bool -> Encoding
bool =
B.bool
{-# INLINE int2_int16 #-}
int2_int16 :: Int16 -> Encoding
int2_int16 =
B.int2_int16
{-# INLINE int2_word16 #-}
int2_word16 :: Word16 -> Encoding
int2_word16 =
B.int2_word16
{-# INLINE int4_int32 #-}
int4_int32 :: Int32 -> Encoding
int4_int32 =
B.int4_int32
{-# INLINE int4_word32 #-}
int4_word32 :: Word32 -> Encoding
int4_word32 =
B.int4_word32
{-# INLINE int8_int64 #-}
int8_int64 :: Int64 -> Encoding
int8_int64 =
B.int8_int64
{-# INLINE int8_word64 #-}
int8_word64 :: Word64 -> Encoding
int8_word64 =
B.int8_word64
{-# INLINE float4 #-}
float4 :: Float -> Encoding
float4 =
B.float4
{-# INLINE float8 #-}
float8 :: Double -> Encoding
float8 =
B.float8
{-# INLINE numeric #-}
numeric :: Scientific -> Encoding
numeric =
B.numeric
{-# INLINE uuid #-}
uuid :: UUID -> Encoding
uuid =
B.uuid
{-# INLINE inet #-}
inet :: G.NetAddr G.IP -> Encoding
inet =
B.inet
{-# INLINE char_utf8 #-}
char_utf8 :: Char -> Encoding
char_utf8 =
B.char_utf8
{-# INLINE text_strict #-}
text_strict :: Text -> Encoding
text_strict =
B.text_strict
{-# INLINE text_lazy #-}
text_lazy :: L.Text -> Encoding
text_lazy =
B.text_lazy
{-# INLINE bytea_strict #-}
bytea_strict :: ByteString -> Encoding
bytea_strict =
B.bytea_strict
{-# INLINE bytea_lazy #-}
bytea_lazy :: N.ByteString -> Encoding
bytea_lazy =
B.bytea_lazy
{-# INLINE bytea_lazyBuilder #-}
bytea_lazyBuilder :: M.Builder -> Encoding
bytea_lazyBuilder =
B.bytea_lazyBuilder
{-# INLINE date #-}
date :: Day -> Encoding
date =
B.date
{-# INLINE time_int #-}
time_int :: TimeOfDay -> Encoding
time_int =
B.time_int
{-# INLINE time_float #-}
time_float :: TimeOfDay -> Encoding
time_float =
B.time_float
{-# INLINE timetz_int #-}
timetz_int :: (TimeOfDay, TimeZone) -> Encoding
timetz_int =
B.timetz_int
{-# INLINE timetz_float #-}
timetz_float :: (TimeOfDay, TimeZone) -> Encoding
timetz_float =
B.timetz_float
{-# INLINE timestamp_int #-}
timestamp_int :: LocalTime -> Encoding
timestamp_int =
B.timestamp_int
{-# INLINE timestamp_float #-}
timestamp_float :: LocalTime -> Encoding
timestamp_float =
B.timestamp_float
{-# INLINE timestamptz_int #-}
timestamptz_int :: UTCTime -> Encoding
timestamptz_int =
B.timestamptz_int
{-# INLINE timestamptz_float #-}
timestamptz_float :: UTCTime -> Encoding
timestamptz_float =
B.timestamptz_float
{-# INLINE interval_int #-}
interval_int :: DiffTime -> Encoding
interval_int =
B.interval_int
{-# INLINE interval_float #-}
interval_float :: DiffTime -> Encoding
interval_float =
B.interval_float
{-# INLINE json_bytes #-}
json_bytes :: ByteString -> Encoding
json_bytes =
B.json_bytes
{-# INLINE json_ast #-}
json_ast :: R.Value -> Encoding
json_ast =
B.json_ast
{-# INLINE jsonb_bytes #-}
jsonb_bytes :: ByteString -> Encoding
jsonb_bytes =
B.jsonb_bytes
{-# INLINE jsonb_ast #-}
jsonb_ast :: R.Value -> Encoding
jsonb_ast =
B.jsonb_ast
data Array =
Array !Encoding ![Int32] !Bool
encodingArray :: Encoding -> Array
encodingArray value =
Array (B.sized value) [] False
nullArray :: Array
nullArray =
Array B.null4 [] True
dimensionArray :: (forall b. (b -> a -> b) -> b -> c -> b) -> (a -> Array) -> c -> Array
dimensionArray foldl' elementArray input =
Array builder dimensions nulls
where
dimensions =
foldedLength : foldedDimensions
(builder, foldedDimensions, foldedLength, nulls) =
foldl' step init input
where
init =
(mempty, [], 0, False)
step (!builder, _, !length, !nulls) element =
(builder <> elementBuilder, elementDimensions, succ length, nulls || elementNulls)
where
Array elementBuilder elementDimensions elementNulls =
elementArray element