DB actions for selecting nodes and edges of single graph

This commit is contained in:
fr33domlover 2016-08-03 21:26:39 +00:00
parent 6220c78f74
commit 7d6ef47e05
3 changed files with 69 additions and 6 deletions

View file

@ -22,6 +22,7 @@ where
import Prelude import Prelude
import Data.Proxy (Proxy)
import Database.Persist import Database.Persist
class (PersistEntity n, PersistEntity e) => PersistEntityGraph n e where class (PersistEntity n, PersistEntity e) => PersistEntityGraph n e where
@ -30,10 +31,13 @@ class (PersistEntity n, PersistEntity e) => PersistEntityGraph n e where
destParam :: e -> Key n destParam :: e -> Key n
destField :: EntityField e (Key n) destField :: EntityField e (Key n)
class PersistEntityGraph n e => PersistEntityGraphSelect n e where class (PersistEntityGraph n e, PersistField (PersistEntityGraphSelector n e))
=> PersistEntityGraphSelect n e where
type PersistEntityGraphSelector n e type PersistEntityGraphSelector n e
selectorParam :: n -> PersistEntityGraphSelector n e selectorParam
selectorField :: EntityField n (PersistEntityGraphSelector n e) :: Proxy (n, e) -> n -> PersistEntityGraphSelector n e
selectorField
:: Proxy (n, e) -> EntityField n (PersistEntityGraphSelector n e)
class PersistEntityGraphSelect n e => PersistEntityGraphNumbered n e where class PersistEntityGraphSelect n e => PersistEntityGraphNumbered n e where
numberParam :: n -> Int numberParam :: n -> Int

View file

@ -32,6 +32,8 @@ module Database.Persist.Local.Sql
, tcontains , tcontains
, sqlUEdge , sqlUEdge
, FollowDirection (..) , FollowDirection (..)
, selectGraphNodesList
, selectGraphEdgesList
) )
where where
@ -218,3 +220,60 @@ sqlUEdge dbname filt tEdge bwd fwd =
in if T.null filt in if T.null filt
then sqlEdge $ entityDB tEdge then sqlEdge $ entityDB tEdge
else sqlBase <> sqlEdge ubase else sqlBase <> sqlEdge ubase
selectGraphNodesList
:: ( MonadIO m
, PersistEntityGraphSelect node edge
, backend ~ PersistEntityBackend node
, backend ~ PersistEntityBackend edge
, PersistQuery backend
)
=> PersistEntityGraphSelector node edge
-> [Filter node]
-> [SelectOpt node]
-> Proxy (node, edge)
-> ReaderT backend m [Entity node]
selectGraphNodesList sel filt opts proxy =
selectList ((selectorField proxy ==. sel) : filt) opts
selectGraphEdgesList
:: ( MonadIO m
, PersistEntityGraphSelect node edge
, SqlBackend ~ PersistEntityBackend node
, SqlBackend ~ PersistEntityBackend edge
)
=> PersistEntityGraphSelector node edge
-> [Filter edge]
-> [SelectOpt edge]
-> Proxy (node, edge)
-> ReaderT SqlBackend m [Entity edge]
selectGraphEdgesList sel filt opts proxy = do
conn <- ask
let tNode = entityDef $ dummyFromFst proxy
tEdge = entityDef $ dummyFromSnd proxy
dbname = connEscapeName conn
t ^* f = dbname t <> "." <> dbname f
t <#> f = dbname t <> " INNER JOIN " <> dbname f
(limit, offset, orders) = limitOffsetOrder opts
applyLimitOffset =
connLimitOffset conn (limit, offset) (not $ null orders)
sql = applyLimitOffset $ mconcat
[ "SELECT ?? FROM ", entityDB tNode <#> entityDB tEdge, " ON "
, entityDB tNode ^* (fieldDB $ entityId tNode)
, " = "
, entityDB tEdge ^*
(fieldDB $ persistFieldDef $ sourceFieldFromProxy proxy)
, let flt = filterClause True conn filt
in if T.null flt
then " WHERE"
else flt
, " AND "
, entityDB tNode ^*
(fieldDB $ persistFieldDef $ selectorField proxy)
, " = ? "
, case map (orderClause True conn) orders of
[] -> ""
ords -> " ORDER BY " <> T.intercalate ", " ords
]
vals = getFiltsValues conn filt ++ [toPersistValue sel]
rawSql sql vals

View file

@ -63,8 +63,8 @@ instance PersistEntityGraph Ticket TicketDependency where
instance PersistEntityGraphSelect Ticket TicketDependency where instance PersistEntityGraphSelect Ticket TicketDependency where
type PersistEntityGraphSelector Ticket TicketDependency = ProjectId type PersistEntityGraphSelector Ticket TicketDependency = ProjectId
selectorParam = ticketProject selectorParam _ = ticketProject
selectorField = TicketProject selectorField _ = TicketProject
instance PersistEntityGraphNumbered Ticket TicketDependency where instance PersistEntityGraphNumbered Ticket TicketDependency where
numberParam = ticketNumber numberParam = ticketNumber