module Database.Neo4j.Types where
import Data.Hashable (Hashable)
import Data.Int (Int64)
import Data.Maybe (fromMaybe)
import Data.Monoid (Monoid, mappend)
import Data.String (fromString)
import Data.Typeable (Typeable)
import Control.Exception (throw)
import Control.Exception.Base (Exception)
import Control.Applicative
import Control.Monad (mzero, ap, liftM)
import Control.Monad.Base (MonadBase, liftBase)
import Control.Monad.IO.Class (MonadIO, liftIO)
import Control.Monad.Trans.Resource (MonadThrow, throwM)
import GHC.Generics (Generic)
import Data.Aeson ((.:))
import qualified Data.Aeson as J
import qualified Data.ByteString as S
import qualified Data.ByteString.Lazy as L
import qualified Data.HashMap.Lazy as M
import qualified Data.Scientific as Sci
import qualified Data.Text as T
import qualified Data.Text.Encoding as TE
import qualified Data.Vector as V
import qualified Network.HTTP.Conduit as HC
import qualified Network.HTTP.Types as HT
(<>) :: (Monoid a) => a -> a -> a
(<>) = mappend
data Val = IntVal Int64 | BoolVal Bool | TextVal T.Text | DoubleVal Double deriving (Show, Eq)
data PropertyValue = ValueProperty Val | ArrayProperty [Val] deriving (Show, Eq)
class PropertyValueConstructor a where
newval :: a -> PropertyValue
instance PropertyValueConstructor Int64 where
newval v = ValueProperty $ IntVal v
instance PropertyValueConstructor Bool where
newval v = ValueProperty $ BoolVal v
instance PropertyValueConstructor T.Text where
newval v = ValueProperty $ TextVal v
instance PropertyValueConstructor Double where
newval v = ValueProperty $ DoubleVal v
instance PropertyValueConstructor [Int64] where
newval v = ArrayProperty $ map IntVal v
instance PropertyValueConstructor [Bool] where
newval v = ArrayProperty $ map BoolVal v
instance PropertyValueConstructor [T.Text] where
newval v = ArrayProperty $ map TextVal v
instance PropertyValueConstructor [Double] where
newval v = ArrayProperty $ map DoubleVal v
(|:) :: PropertyValueConstructor a => T.Text -> a -> (T.Text, PropertyValue)
name |: v = (name, newval v)
instance J.ToJSON Val where
toJSON (IntVal v) = J.Number $ fromIntegral v
toJSON (BoolVal v) = J.Bool v
toJSON (TextVal v) = J.String v
toJSON (DoubleVal v) = J.Number $ Sci.fromFloatDigits v
instance J.ToJSON PropertyValue where
toJSON (ValueProperty v) = J.toJSON v
toJSON (ArrayProperty vs) = J.Array (V.fromList $ map J.toJSON vs)
instance J.FromJSON Val where
parseJSON (J.Number v) = let parsedNum = Sci.floatingOrInteger v in
return $ case parsedNum of
Left d -> DoubleVal d
Right i -> IntVal i
parseJSON (J.Bool v) = return $ BoolVal v
parseJSON (J.String v) = return $ TextVal v
parseJSON _ = mzero
instance J.FromJSON PropertyValue where
parseJSON (J.Array v) = ArrayProperty <$> mapM J.parseJSON (V.toList v)
parseJSON v = ValueProperty <$> J.parseJSON v
type Properties = M.HashMap T.Text PropertyValue
emptyProperties :: M.HashMap T.Text PropertyValue
emptyProperties = M.empty
urlPath :: T.Text -> T.Text
urlPath url = fromMaybe url $ T.dropWhile (/='/') <$> T.stripPrefix "http://" url
urlMinPath :: T.Text -> T.Text
urlMinPath url = fromMaybe url $ T.stripPrefix "/db/data" (urlPath url)
data EntityObj = EntityNode Node | EntityRel Relationship deriving (Eq, Show)
class Entity a where
entityPath :: a -> S.ByteString
propertyPath :: a -> S.ByteString
getEntityProperties :: a -> Properties
setEntityProperties :: a -> Properties -> a
entityObj :: a -> EntityObj
newtype NodeUrl = NodeUrl {runNodeUrl :: T.Text} deriving (Show, Eq, Generic)
instance Hashable NodeUrl
data Node = Node {nodePath :: NodePath, nodeProperties :: Properties} deriving (Show, Eq)
instance Ord Node where
a <= b = nodePath a <= nodePath b
getNodeProperties :: Node -> Properties
getNodeProperties = nodeProperties
instance J.FromJSON Node where
parseJSON (J.Object v) = Node <$> (getNodePath . NodeUrl <$> (v .: "self")) <*> (v .: "data" >>= J.parseJSON)
parseJSON _ = mzero
instance Entity Node where
entityPath = runNodeIdentifier
propertyPath n = runNodeIdentifier n <> "/properties"
getEntityProperties = nodeProperties
setEntityProperties n props = n {nodeProperties = props}
entityObj = EntityNode
nodeAPI :: S.ByteString
nodeAPI = "/db/data/node"
nodeAPITxt :: T.Text
nodeAPITxt = "/db/data/node"
newtype NodePath = NodePath {runNodePath :: T.Text} deriving (Show, Ord, Eq, Generic)
instance Hashable NodePath
class NodeIdentifier a where
getNodePath :: a -> NodePath
instance NodeIdentifier Node where
getNodePath = nodePath
instance NodeIdentifier S.ByteString where
getNodePath t = NodePath $ TE.decodeUtf8 $ nodeAPI <> "/" <> t
instance NodeIdentifier Integer where
getNodePath n = NodePath $ nodeAPITxt <> "/" <> (fromString . show) n
instance NodeIdentifier NodePath where
getNodePath = id
instance NodeIdentifier NodeUrl where
getNodePath n = NodePath $ (urlPath . runNodeUrl) n
runNodeIdentifier :: NodeIdentifier a => a -> S.ByteString
runNodeIdentifier = TE.encodeUtf8 . runNodePath . getNodePath
type RelationshipType = T.Text
data Direction = Outgoing | Incoming | Any deriving (Eq, Show)
newtype RelUrl = RelUrl {runRelUrl :: T.Text} deriving (Show, Eq, Generic)
instance Hashable RelUrl
data Relationship = Relationship {relPath :: RelPath,
relType :: RelationshipType,
relProperties :: Properties,
relFrom :: NodePath,
relTo :: NodePath} deriving (Show, Eq)
instance Ord Relationship where
a <= b = relPath a <= relPath b
getRelProperties :: Relationship -> Properties
getRelProperties = relProperties
getRelType :: Relationship -> RelationshipType
getRelType = relType
instance J.FromJSON Relationship where
parseJSON (J.Object v) = Relationship <$> (getRelPath . RelUrl <$> v .: "self") <*> v .: "type" <*>
(v .: "data" >>= J.parseJSON) <*> (getNodePath . NodeUrl <$> v .: "start") <*>
(getNodePath . NodeUrl <$> v .: "end")
parseJSON _ = mzero
instance Entity Relationship where
entityPath = runRelIdentifier
propertyPath r = runRelIdentifier r <> "/properties"
getEntityProperties = relProperties
setEntityProperties r props = r {relProperties = props}
entityObj = EntityRel
instance Entity EntityObj where
entityPath (EntityNode n) = runNodeIdentifier n
entityPath (EntityRel n) = runRelIdentifier n
propertyPath (EntityNode n) = runNodeIdentifier n <> "/properties"
propertyPath (EntityRel n) = runRelIdentifier n <> "/properties"
getEntityProperties (EntityNode n) = nodeProperties n
getEntityProperties (EntityRel n) = relProperties n
setEntityProperties (EntityNode n) props = EntityNode $ n {nodeProperties = props}
setEntityProperties (EntityRel n) props = EntityRel $ n {relProperties = props}
entityObj = id
relationshipAPI :: S.ByteString
relationshipAPI = "/db/data/relationship"
relationshipAPITxt :: T.Text
relationshipAPITxt = "/db/data/relationship"
newtype RelPath = RelPath {runRelPath :: T.Text} deriving (Show, Eq, Ord, Generic)
instance Hashable RelPath
class RelIdentifier a where
getRelPath :: a -> RelPath
instance RelIdentifier Relationship where
getRelPath = relPath
instance RelIdentifier RelPath where
getRelPath = id
instance RelIdentifier S.ByteString where
getRelPath t = RelPath $ TE.decodeUtf8 $ relationshipAPI <> "/" <> t
instance RelIdentifier RelUrl where
getRelPath = RelPath . urlPath . runRelUrl
instance RelIdentifier Integer where
getRelPath n = RelPath $ relationshipAPITxt <> "/" <> (fromString . show) n
runRelIdentifier :: RelIdentifier a => a -> S.ByteString
runRelIdentifier = TE.encodeUtf8 . runRelPath . getRelPath
data EntityPath = EntityRelPath RelPath | EntityNodePath NodePath deriving (Show, Eq)
class EntityIdentifier a where
getEntityPath :: a -> EntityPath
instance EntityIdentifier Node where
getEntityPath = EntityNodePath . getNodePath
instance EntityIdentifier NodePath where
getEntityPath = EntityNodePath . getNodePath
instance EntityIdentifier NodeUrl where
getEntityPath = EntityNodePath . getNodePath
instance EntityIdentifier Relationship where
getEntityPath = EntityRelPath . getRelPath
instance EntityIdentifier RelPath where
getEntityPath = EntityRelPath . getRelPath
instance EntityIdentifier RelUrl where
getEntityPath = EntityRelPath . getRelPath
instance EntityIdentifier T.Text where
getEntityPath i = (if T.count "/node" p > 0 then EntityNodePath . NodePath else EntityRelPath . RelPath) p
where p = urlPath i
type Label = T.Text
data Index = Index {indexLabel :: Label, indexProperties :: [T.Text]} deriving (Show, Eq)
instance J.FromJSON Index where
parseJSON (J.Object v) = Index <$> v .: "label" <*> (v .: "property_keys" >>= J.parseJSON)
parseJSON _ = mzero
data Neo4jException = Neo4jHttpException String |
Neo4jNonOrphanNodeDeletionException S.ByteString |
Neo4jNoEntityException S.ByteString |
Neo4jUnexpectedResponseException HT.Status |
Neo4jNoSuchProperty T.Text |
Neo4jBatchException L.ByteString |
Neo4jParseException String |
TransactionEndedExc deriving (Show, Typeable, Eq)
instance Exception Neo4jException
data Connection = Connection {dbHostname :: Hostname, dbPort :: Port, manager :: HC.Manager}
type Hostname = S.ByteString
type Port = Int
newtype Neo4j a = Neo4j { runNeo4j :: Connection -> IO a }
instance Monad Neo4j where
return x = Neo4j (const (return x))
(Neo4j cmd) >>= f = Neo4j $ \con -> do
a <- cmd con
runNeo4j (f a) con
instance Functor Neo4j where
fmap = liftM
instance Applicative Neo4j where
pure = return
(<*>) = ap
instance MonadIO Neo4j where
liftIO f = Neo4j $ const (liftIO f)
instance MonadThrow Neo4j where
throwM e = Neo4j $ const (throw e)
instance MonadBase (Neo4j) (Neo4j) where
liftBase = id