generated from tpl/purs
190 lines
7.2 KiB
Haskell
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
|