module Data.Bson (
Document, (!?), look, lookup, valueAt, at, include, exclude, merge,
Field(..), (=:), (=?),
Label,
Value(..), Val(..), fval, cast, typed, typeOfVal,
Binary(..), Function(..), UUID(..), MD5(..), UserDefined(..),
Regex(..), Javascript(..), Symbol(..), MongoStamp(..), MinMaxKey(..),
ObjectId(..), timestamp, genObjectId, showHexLen
) where
import Prelude hiding (lookup)
#if !MIN_VERSION_base(4,8,0)
import Control.Applicative ((<$>))
#endif
import Control.Monad (foldM)
import Data.Bits (shift, (.|.))
import Data.Int (Int32, Int64)
import Data.IORef (IORef, newIORef, atomicModifyIORef)
import Data.List (find, findIndex)
import Data.Maybe (maybeToList, mapMaybe, fromMaybe)
import Data.Time.Clock (UTCTime)
import Data.Time.Clock.POSIX (POSIXTime, posixSecondsToUTCTime,
utcTimeToPOSIXSeconds, getPOSIXTime)
import Data.Time.Format ()
import Data.Typeable hiding (cast)
import Data.Word (Word8, Word16, Word32, Word64)
import Numeric (readHex, showHex)
import System.IO.Unsafe (unsafePerformIO)
import Text.Read (Read(..))
import qualified Data.ByteString as S
import qualified Data.ByteString.Char8 as SC
import qualified Text.ParserCombinators.ReadP as R
import qualified Text.ParserCombinators.ReadPrec as R (lift, readS_to_Prec)
import Control.Monad.Identity (runIdentity)
import Network.BSD (getHostName)
import Data.Text (Text)
import qualified Data.Text as T
import qualified Crypto.Hash.MD5 as MD5
getProcessID :: IO Int
getProcessID = c_getpid
foreign import ccall unsafe "getpid"
c_getpid :: IO Int
roundTo :: (RealFrac a) => a -> a -> a
roundTo mult n = fromIntegral (round (n / mult) :: Integer) * mult
showHexLen :: (Show n, Integral n) => Int -> n -> ShowS
showHexLen d n = showString (replicate (d sigDigits n) '0') . showHex n where
sigDigits 0 = 1
sigDigits n' = truncate (logBase 16 $ fromIntegral n' :: Double) + 1
type Document = [Field]
(!?) :: Val a => Document -> Label -> Maybe a
doc !? l = foldM (flip lookup) doc (init chunks) >>= lookup (last chunks)
where chunks = T.split (== '.') l
look :: (Monad m) => Label -> Document -> m Value
look k doc = maybe notFound (return . value) (find ((k ==) . label) doc)
where notFound = fail $ "expected " ++ show k ++ " in " ++ show doc
lookup :: (Val v, Monad m) => Label -> Document -> m v
lookup k doc = cast =<< look k doc
valueAt :: Label -> Document -> Value
valueAt k = runIdentity . look k
at :: (Val v) => Label -> Document -> v
at k doc = result
where
result = fromMaybe err (lookup k doc)
err = error $ "expected (" ++ show k ++ " :: " ++ show (typeOf result) ++ ") in " ++ show doc
include :: [Label] -> Document -> Document
include keys doc = mapMaybe (\k -> find ((k ==) . label) doc) keys
exclude :: [Label] -> Document -> Document
exclude keys = filter (\(k := _) -> notElem k keys)
merge :: Document -> Document -> Document
merge es docInitial = foldl f docInitial es
where f doc (k := v) = case findIndex ((k ==) . label) doc of
Nothing -> doc ++ [k := v]
Just i -> let (x, _ : y) = splitAt i doc in x ++ [k := v] ++ y
infix 0 :=, =:, =?
data Field = (:=) {label :: !Label, value :: Value} deriving (Typeable, Eq, Ord)
(=:) :: (Val v) => Label -> v -> Field
k =: v = k := val v
(=?) :: (Val a) => Label -> Maybe a -> Document
k =? ma = maybeToList (fmap (k =:) ma)
instance Show Field where
showsPrec d (k := v) = showParen (d > 0) $ showString (' ' : T.unpack k) . showString ": " . showsPrec 1 v
type Label = Text
data Value = Float Double
| String Text
| Doc Document
| Array [Value]
| Bin Binary
| Fun Function
| Uuid UUID
| Md5 MD5
| UserDef UserDefined
| ObjId ObjectId
| Bool Bool
| UTC UTCTime
| Null
| RegEx Regex
| JavaScr Javascript
| Sym Symbol
| Int32 Int32
| Int64 Int64
| Stamp MongoStamp
| MinMax MinMaxKey
deriving (Typeable, Eq, Ord)
instance Show Value where
showsPrec d = fval (showsPrec d)
fval :: (forall a . (Val a) => a -> b) -> Value -> b
fval f v = case v of
Float x -> f x
String x -> f x
Doc x -> f x
Array x -> f x
Bin x -> f x
Fun x -> f x
Uuid x -> f x
Md5 x -> f x
UserDef x -> f x
ObjId x -> f x
Bool x -> f x
UTC x -> f x
Null -> f (Nothing :: Maybe Value)
RegEx x -> f x
JavaScr x -> f x
Sym x -> f x
Int32 x -> f x
Int64 x -> f x
Stamp x -> f x
MinMax x -> f x
cast :: (Val a, Monad m) => Value -> m a
cast v = maybe notType return castingResult
where
castingResult = cast' v
unMaybe :: Maybe a -> a
unMaybe = undefined
notType = fail $ "expected " ++ show (typeOf $ unMaybe castingResult) ++ ": " ++ show v
typed :: (Val a) => Value -> a
typed = runIdentity . cast
typeOfVal :: Value -> TypeRep
typeOfVal = fval typeOf
class (Typeable a, Show a, Eq a) => Val a where
val :: a -> Value
valList :: [a] -> Value
valList = Array . map val
valMaybe :: Maybe a -> Value
valMaybe = maybe Null val
cast' :: Value -> Maybe a
cast'List :: Value -> Maybe [a]
cast'List (Array x) = mapM cast x
cast'List _ = Nothing
cast'Maybe :: Value -> Maybe (Maybe a)
cast'Maybe Null = Just Nothing
cast'Maybe v = fmap Just (cast' v)
instance Val Double where
val = Float
cast' (Float x) = Just x
cast' (Int32 x) = Just (fromIntegral x)
cast' (Int64 x) = Just (fromIntegral x)
cast' _ = Nothing
instance Val Float where
val = Float . realToFrac
cast' (Float x) = Just (realToFrac x)
cast' (Int32 x) = Just (fromIntegral x)
cast' (Int64 x) = Just (fromIntegral x)
cast' _ = Nothing
instance Val Text where
val = String
cast' (String x) = Just x
cast' (Sym (Symbol x)) = Just x
cast' _ = Nothing
instance Val Char where
val x = valList [x]
valList = String . T.pack
cast' v = cast'List v >>= safeHead
where safeHead list = case list of
x:_ -> Just x
_ -> Nothing
cast'List (String x) = Just $ T.unpack x
cast'List (Sym (Symbol x)) = Just $ T.unpack x
cast'List _ = Nothing
instance Val Field where
val x = valList [x]
valList = Doc
cast' _ = Nothing
cast'List v = case v of
Doc x -> Just x
_ -> Nothing
instance Val Value where
val = id
cast' = Just
instance (Val a) => Val [a] where
val = valList
cast' = cast'List
instance Val Binary where
val = Bin
cast' (Bin x) = Just x
cast' _ = Nothing
instance Val Function where
val = Fun
cast' (Fun x) = Just x
cast' _ = Nothing
instance Val UUID where
val = Uuid
cast' (Uuid x) = Just x
cast' _ = Nothing
instance Val MD5 where
val = Md5
cast' (Md5 x) = Just x
cast' _ = Nothing
instance Val UserDefined where
val = UserDef
cast' (UserDef x) = Just x
cast' _ = Nothing
instance Val ObjectId where
val = ObjId
cast' (ObjId x) = Just x
cast' _ = Nothing
instance Val Bool where
val = Bool
cast' (Bool x) = Just x
cast' _ = Nothing
instance Val UTCTime where
val = UTC
cast' (UTC x) = Just x
cast' _ = Nothing
instance Val POSIXTime where
val = UTC . posixSecondsToUTCTime . roundTo (1/1000)
cast' (UTC x) = Just (utcTimeToPOSIXSeconds x)
cast' _ = Nothing
instance (Val a) => Val (Maybe a) where
val = valMaybe
cast' = cast'Maybe
instance Val Regex where
val = RegEx
cast' (RegEx x) = Just x
cast' _ = Nothing
instance Val Javascript where
val = JavaScr
cast' (JavaScr x) = Just x
cast' _ = Nothing
instance Val Symbol where
val = Sym
cast' (Sym x) = Just x
cast' (String x) = Just (Symbol x)
cast' _ = Nothing
instance Val Int32 where
val = Int32
cast' (Int32 x) = Just x
cast' (Int64 x) = fitInt x
cast' (Float x) = Just (round x)
cast' _ = Nothing
instance Val Int64 where
val = Int64
cast' (Int64 x) = Just x
cast' (Int32 x) = Just (fromIntegral x)
cast' (Float x) = Just (round x)
cast' _ = Nothing
instance Val Int where
val n = maybe (Int64 $ fromIntegral n) Int32 (fitInt n)
cast' (Int32 x) = Just (fromIntegral x)
cast' (Int64 x) = Just (fromEnum x)
cast' (Float x) = Just (round x)
cast' _ = Nothing
instance Val Integer where
val n = maybe (maybe err Int64 $ fitInt n) Int32 (fitInt n)
where err = error $ show n ++ " is too large for Bson Int Value"
cast' (Int32 x) = Just (fromIntegral x)
cast' (Int64 x) = Just (fromIntegral x)
cast' (Float x) = Just (round x)
cast' _ = Nothing
instance Val MongoStamp where
val = Stamp
cast' (Stamp x) = Just x
cast' _ = Nothing
instance Val MinMaxKey where
val = MinMax
cast' (MinMax x) = Just x
cast' _ = Nothing
fitInt :: (Integral n, Integral m, Bounded m) => n -> Maybe m
fitInt n =
if fromIntegral (minBound `asTypeOf` result) <= n && n <= fromIntegral (maxBound `asTypeOf` result)
then Just result
else Nothing
where result = fromIntegral n
newtype Binary = Binary S.ByteString deriving (Typeable, Show, Read, Eq, Ord)
newtype Function = Function S.ByteString deriving (Typeable, Show, Read, Eq, Ord)
newtype UUID = UUID S.ByteString deriving (Typeable, Show, Read, Eq, Ord)
newtype MD5 = MD5 S.ByteString deriving (Typeable, Show, Read, Eq, Ord)
newtype UserDefined = UserDefined S.ByteString deriving (Typeable, Show, Read, Eq, Ord)
data Regex = Regex Text Text deriving (Typeable, Show, Read, Eq, Ord)
data Javascript = Javascript Document Text deriving (Typeable, Show, Eq, Ord)
newtype Symbol = Symbol Text deriving (Typeable, Show, Read, Eq, Ord)
newtype MongoStamp = MongoStamp Int64 deriving (Typeable, Show, Read, Eq, Ord)
data MinMaxKey = MinKey | MaxKey deriving (Typeable, Show, Read, Eq, Ord)
data ObjectId = Oid !Word32 !Word64 deriving (Typeable, Eq, Ord)
instance Show ObjectId where
showsPrec _ (Oid x y) = showHexLen 8 x . showHexLen 16 y
instance Read ObjectId where
readPrec = do
[(x, "")] <- readHex <$> R.lift (R.count 8 R.get)
y <- R.readS_to_Prec $ const readHex
return (Oid x y)
timestamp :: ObjectId -> UTCTime
timestamp (Oid time _) = posixSecondsToUTCTime (fromIntegral time)
genObjectId :: IO ObjectId
genObjectId = do
time <- truncate <$> getPOSIXTime
pid <- fromIntegral <$> getProcessID
inc <- nextCount
return $ Oid time (composite machineId pid inc)
where
machineId :: Word24
machineId = unsafePerformIO (makeWord24 . S.unpack . S.take 3 . MD5.hash . SC.pack <$> getHostName)
counter :: IORef Word24
counter = unsafePerformIO (newIORef 0)
nextCount :: IO Word24
nextCount = atomicModifyIORef counter $ \n -> (wrap24 (n + 1), n)
composite :: Word24 -> Word16 -> Word24 -> Word64
composite mid pid inc = fromIntegral mid `shift` 40 .|. fromIntegral pid `shift` 24 .|. fromIntegral inc
type Word24 = Word32
wrap24 :: Word24 -> Word24
wrap24 n = n `mod` 0x1000000
makeWord24 :: [Word8] -> Word24
makeWord24 = foldl (\a b -> a `shift` 8 .|. fromIntegral b) 0