purescript-postgresql-client/src/Database/PostgreSQL/Value.purs
2019-11-10 15:27:24 +01:00

190 lines
7.2 KiB
Haskell

module Database.PostgreSQL.Value where
-- | `node-postgres` is playing guessing game regarding type conversions
-- | so we are overriding its defualt behavior for some types - please
-- | check PostgreSQL.js
import Prelude
import Control.Monad.Error.Class (throwError)
import Control.Monad.Except (ExceptT, except, runExcept, runExceptT)
import Data.Argonaut (Json)
import Data.Argonaut (stringify) as Argonaut
import Data.Array (foldl)
import Data.Array as Array
import Data.Bifunctor (lmap)
import Data.ByteString (ByteString)
import Data.Date (Date, canonicalDate, day, month, year)
import Data.DateTime.Instant (Instant, instant)
import Data.Decimal (Decimal)
import Data.Decimal as Decimal
import Data.Either (Either(..), note)
import Data.Enum (fromEnum, toEnum)
import Data.Identity (Identity)
import Data.Int (fromString)
import Data.JSDate (JSDate)
import Data.List (List)
import Data.List as List
import Data.List.NonEmpty (singleton)
import Data.Maybe (Maybe(..))
import Data.Newtype (class Newtype, unwrap, wrap)
import Data.String (Pattern(..), split)
import Data.Time.Duration (Milliseconds(..))
import Data.Traversable (sequence, traverse)
import Foreign (Foreign, ForeignError(..), MultipleErrors, isNull, readArray, readBoolean, readChar, readInt, readNumber, readString, renderForeignError, unsafeFromForeign, unsafeToForeign)
import Foreign.Generic.Internal (readObject)
import Foreign.Object (Object)
-- | Convert things to SQL values.
class ToSQLValue a where
toSQLValue :: a -> Foreign
-- | Convert things from SQL values.
class FromSQLValue a where
fromSQLValue :: Foreign -> Either String a
instance fromSQLValueBoolean :: FromSQLValue Boolean where
fromSQLValue = lmap show <<< runExcept <<< readBoolean
else instance fromSQLValueChar :: FromSQLValue Char where
fromSQLValue = lmap show <<< runExcept <<< readChar
else instance fromSQLValueInt :: FromSQLValue Int where
fromSQLValue = lmap show <<< runExcept <<< readInt
else instance fromSQLValueNumber :: FromSQLValue Number where
fromSQLValue = lmap show <<< runExcept <<< readNumber
else instance fromSQLValueString :: FromSQLValue String where
fromSQLValue = lmap show <<< runExcept <<< readString
else instance fromSQLValueArray :: (FromSQLValue a) => FromSQLValue (Array a) where
fromSQLValue = traverse fromSQLValue <=< lmap show <<< runExcept <<< readArray
else instance fromSQLValueList :: (FromSQLValue a) => FromSQLValue (List a) where
fromSQLValue = map List.fromFoldable <<< traverse fromSQLValue <=< lmap show <<< runExcept <<< readArray
else instance fromSQLValueByteString :: FromSQLValue ByteString where
fromSQLValue x
| unsafeIsBuffer x = pure $ unsafeFromForeign x
| otherwise = throwError "FromSQLValue ByteString: not a buffer"
else instance fromSQLValueInstant :: FromSQLValue Instant where
fromSQLValue v = do
t <- instantFromString Left Right v
note ("Instant construction failed for given timestamp: " <> show t) $ instant (Milliseconds t)
else instance fromSQLValueDate :: FromSQLValue Date where
fromSQLValue v = do
s <- lmap show $ runExcept (readString v)
let
msg = "Date parsing failed for value: " <> s
case split (Pattern "-") s of
[y, m, d] -> do
let
result = canonicalDate
<$> (toEnum =<< fromString y)
<*> (toEnum =<< fromString m)
<*> (toEnum =<< fromString d)
note msg result
_ -> Left msg
else instance fromSQLValueJSDate :: FromSQLValue JSDate where
fromSQLValue = Right <<< unsafeFromForeign
else instance fromSQLValueMaybe :: (FromSQLValue a) => FromSQLValue (Maybe a) where
fromSQLValue x | isNull x = pure Nothing
| otherwise = Just <$> fromSQLValue x
else instance fromSQLValueForeign :: FromSQLValue Foreign where
fromSQLValue = pure
else instance fromSQLValueObject :: FromSQLValue a FromSQLValue (Object a) where
fromSQLValue sql = lmap showErr $ unwrap $ runExceptT main
where
showErr MultipleErrors String
showErr e = foldl (\a x a <> renderForeignError x <> " ") "" e
main ExceptT MultipleErrors Identity (Object a)
main = do
objF Object Foreign <- readObject sql
let eso = sequence $ map fromSQLValue objF
let emo = lmap (singleton <<< ForeignError) eso
except emo
else instance fromSQLValueDecimal :: FromSQLValue Decimal where
fromSQLValue v = do
s <- lmap show $ runExcept (readString v)
note ("Decimal literal parsing failed: " <> s) (Decimal.fromString s)
else instance fromSQLValueJson :: FromSQLValue Json where
fromSQLValue = Right <<< unsafeFromForeign
newtypeFromSQLValue a b. Newtype a b FromSQLValue b Foreign Either String a
newtypeFromSQLValue = map wrap <<< fromSQLValue
instance toSQLValueBoolean :: ToSQLValue Boolean where
toSQLValue = unsafeToForeign
else instance toSQLValueChar :: ToSQLValue Char where
toSQLValue = unsafeToForeign
else instance toSQLValueInt :: ToSQLValue Int where
toSQLValue = unsafeToForeign
else instance toSQLValueNumber :: ToSQLValue Number where
toSQLValue = unsafeToForeign
else instance toSQLValueString :: ToSQLValue String where
toSQLValue = unsafeToForeign
else instance toSQLValueArray :: (ToSQLValue a) => ToSQLValue (Array a) where
toSQLValue = unsafeToForeign <<< map toSQLValue
else instance toSQLValueList :: (ToSQLValue a) => ToSQLValue (List a) where
toSQLValue = unsafeToForeign <<< Array.fromFoldable <<< map toSQLValue
else instance toSQLValueByteString :: ToSQLValue ByteString where
toSQLValue = unsafeToForeign
else instance toSQLValueInstant :: ToSQLValue Instant where
toSQLValue = instantToString
else instance toSQLValueDate :: ToSQLValue Date where
toSQLValue date =
let
y = fromEnum $ year date
m = fromEnum $ month date
d = fromEnum $ day date
in
unsafeToForeign $ show y <> "-" <> show m <> "-" <> show d
else instance toSQLValueJSDate :: ToSQLValue JSDate where
toSQLValue = unsafeToForeign
else instance toSQLValueMaybe :: (ToSQLValue a) => ToSQLValue (Maybe a) where
toSQLValue Nothing = null
toSQLValue (Just x) = toSQLValue x
else instance toSQLValueForeign :: ToSQLValue Foreign where
toSQLValue = identity
else instance toSQLValueObject ToSQLValue a ToSQLValue (Object a) where
toSQLValue = unsafeToForeign
else instance toSQLValueDecimal :: ToSQLValue Decimal where
toSQLValue = Decimal.toString >>> unsafeToForeign
else instance toSQLValueJson :: ToSQLValue Json where
-- | We are forced to stringify value here to avoid
-- | problems with pg auto conversions - please check for example:
-- | https://github.com/brianc/node-postgres/issues/1383
toSQLValue = Argonaut.stringify >>> unsafeToForeign
newtypeToSQLValue a b. Newtype a b ToSQLValue b a Foreign
newtypeToSQLValue = unwrap >>> toSQLValue
foreign import null :: Foreign
foreign import instantToString :: Instant -> Foreign
foreign import instantFromString :: (String -> Either String Number) -> (Number -> Either String Number) -> Foreign -> Either String Number
foreign import unsafeIsBuffer :: a. a -> Boolean