mirror of
https://code.naskya.net/repos/ndqEd
synced 2025-01-10 16:26:46 +09:00
Layering of intransitive DAGs
This commit is contained in:
parent
5340cf23f1
commit
7886d9a515
3 changed files with 241 additions and 0 deletions
207
src/Data/Graph/Inductive/Query/Layer.hs
Normal file
207
src/Data/Graph/Inductive/Query/Layer.hs
Normal file
|
@ -0,0 +1,207 @@
|
||||||
|
{- This file is part of Vervis.
|
||||||
|
-
|
||||||
|
- Written in 2016 by fr33domlover <fr33domlover@riseup.net>.
|
||||||
|
-
|
||||||
|
- ♡ Copying is an act of love. Please copy, reuse and share.
|
||||||
|
-
|
||||||
|
- The author(s) have dedicated all copyright and related and neighboring
|
||||||
|
- rights to this software to the public domain worldwide. This software is
|
||||||
|
- distributed without any warranty.
|
||||||
|
-
|
||||||
|
- You should have received a copy of the CC0 Public Domain Dedication along
|
||||||
|
- with this software. If not, see
|
||||||
|
- <http://creativecommons.org/publicdomain/zero/1.0/>.
|
||||||
|
-}
|
||||||
|
|
||||||
|
-- | Layering of directed acyclic graphs
|
||||||
|
module Data.Graph.Inductive.Query.Layer
|
||||||
|
( -- * Intro
|
||||||
|
-- $into
|
||||||
|
|
||||||
|
-- * Forward Layer
|
||||||
|
-- $forward
|
||||||
|
layer
|
||||||
|
, layern
|
||||||
|
, layerWith
|
||||||
|
, layernWith
|
||||||
|
|
||||||
|
-- * Backward Layer
|
||||||
|
-- $backward
|
||||||
|
, rlayer
|
||||||
|
, rlayern
|
||||||
|
, rlayerWith
|
||||||
|
, rlayernWith
|
||||||
|
|
||||||
|
-- * Custom Layer
|
||||||
|
-- $custom
|
||||||
|
, xlayern
|
||||||
|
, xlayernWith
|
||||||
|
)
|
||||||
|
where
|
||||||
|
|
||||||
|
import Prelude
|
||||||
|
|
||||||
|
import Data.Graph.Inductive.Basic (gsel)
|
||||||
|
import Data.Graph.Inductive.Graph
|
||||||
|
import Data.Graph.Inductive.Internal.Queue
|
||||||
|
import Data.List (sortOn)
|
||||||
|
|
||||||
|
import qualified Data.HashMap.Lazy as M
|
||||||
|
|
||||||
|
import qualified Data.HashMap.Lazy.Local as ML
|
||||||
|
|
||||||
|
noIn :: Graph g => g a b -> [Node]
|
||||||
|
noIn = map node' . gsel (null . pre')
|
||||||
|
|
||||||
|
noOut :: Graph g => g a b -> [Node]
|
||||||
|
noOut = map node' . gsel (null . suc')
|
||||||
|
|
||||||
|
-- $intro
|
||||||
|
-- Layering a directed acyclic graph basically means to partition its nodes
|
||||||
|
-- such that all the edges point in the same direction. Layering is often used
|
||||||
|
-- for graph visualization, an therefore requires that the result has certain
|
||||||
|
-- human-friendly properties.
|
||||||
|
--
|
||||||
|
-- This module currently offers a very simple algorithm meant for DAGs that are
|
||||||
|
-- transitively reduced, i.e. if edges AB and BC exist, an edge AC shouldn't
|
||||||
|
-- exist in the graph. In other words, assuming the edges represent partial
|
||||||
|
-- ordering of the nodes, no edge should be possible to deduce from other
|
||||||
|
-- edges.
|
||||||
|
|
||||||
|
-- $forward
|
||||||
|
-- Forward layering starts from a set of nodes, usually the nodes which don't
|
||||||
|
-- have in-edges, and builds the layers by traversing the out-edges
|
||||||
|
-- recursively. The initial nodes are the first layer, their children are the
|
||||||
|
-- second layer, the children's children are the third layer, and so on.
|
||||||
|
|
||||||
|
-- | The initial nodes are the nodes which don't have in-edges.
|
||||||
|
layer :: Graph g => g a b -> [[Node]]
|
||||||
|
layer = layerWith node'
|
||||||
|
|
||||||
|
-- | Specify the initial nodes.
|
||||||
|
layern :: Graph g => [Node] -> g a b -> [[Node]]
|
||||||
|
layern = layernWith node'
|
||||||
|
|
||||||
|
-- | Specify function to apply to nodes whose result will be in the result
|
||||||
|
-- list. The initial nodes are the nodes which don't have in-edges.
|
||||||
|
layerWith :: Graph g => (Context a b -> c) -> g a b -> [[c]]
|
||||||
|
layerWith result graph = layernWith result (noIn graph) graph
|
||||||
|
|
||||||
|
-- | Specify function to apply to nodes whose result will be in the result
|
||||||
|
-- list, and specify initial nodes.
|
||||||
|
layernWith :: Graph g => (Context a b -> c) -> [Node] -> g a b -> [[c]]
|
||||||
|
layernWith = xlayernWith suc' (not . null . pre')
|
||||||
|
|
||||||
|
-- $backward
|
||||||
|
-- Backward layering starts from a set of nodes, usually the nodes which don't
|
||||||
|
-- have out-edges, and builds the layers by traversing the in-edges
|
||||||
|
-- recursively. The initial nodes are the first layer, their parents are the
|
||||||
|
-- second layer, the parents' parents are the third layer, and so on.
|
||||||
|
|
||||||
|
-- | The initial nodes are the nodes which don't have out-edges.
|
||||||
|
rlayer :: Graph g => g a b -> [[Node]]
|
||||||
|
rlayer = rlayerWith node'
|
||||||
|
|
||||||
|
-- | Specify the initial nodes.
|
||||||
|
rlayern :: Graph g => [Node] -> g a b -> [[Node]]
|
||||||
|
rlayern = rlayernWith node'
|
||||||
|
|
||||||
|
-- | Specify function to apply to nodes whose result will be in the result
|
||||||
|
-- list. The initial nodes are the nodes which don't have out-edges.
|
||||||
|
rlayerWith :: Graph g => (Context a b -> c) -> g a b -> [[c]]
|
||||||
|
rlayerWith result graph = rlayernWith result (noOut graph) graph
|
||||||
|
|
||||||
|
-- | Specify function to apply to nodes whose result will be in the result
|
||||||
|
-- list, and specify initial nodes.
|
||||||
|
rlayernWith :: Graph g => (Context a b -> c) -> [Node] -> g a b -> [[c]]
|
||||||
|
rlayernWith = xlayernWith pre' (not . null . suc')
|
||||||
|
|
||||||
|
-- $custom
|
||||||
|
-- Custom layering starts from a set of nodes, and builds the layers by
|
||||||
|
-- traversing edges recursively. A user-specified function determines which
|
||||||
|
-- edges are traversed, and another functions is used for checking whether
|
||||||
|
-- there are edges through which a given node can be reached. For example, if
|
||||||
|
-- you follow just out-edges that point from red-colored nodes, the second
|
||||||
|
-- function would check whether the given nodes has red-colored nodes pointing
|
||||||
|
-- to it. The initial nodes are the first layer, the nodes reached from them
|
||||||
|
-- are the second layer and so on.
|
||||||
|
|
||||||
|
-- | Specify which paths to follow, and the initial nodes.
|
||||||
|
xlayern
|
||||||
|
:: Graph g
|
||||||
|
=> (Context a b -> [Node])
|
||||||
|
-> (Context a b -> Bool)
|
||||||
|
-> [Node]
|
||||||
|
-> g a b
|
||||||
|
-> [[Node]]
|
||||||
|
xlayern follow back = xlayernWith follow back node'
|
||||||
|
|
||||||
|
-- (1) All nodes have unspecified layer
|
||||||
|
-- (2) Mark all child-less nodes with layer 1 and place in a queue
|
||||||
|
-- (3) Dequeue a node N and remove N from the graph
|
||||||
|
-- (4) For each parent of N, P:
|
||||||
|
-- (5) layer(P) = max (layer(P), layer(N)+1)
|
||||||
|
-- (6) If N was P's only child, enqueue P
|
||||||
|
-- (7) Jump back to 3
|
||||||
|
depths
|
||||||
|
:: Graph g
|
||||||
|
=> (Context a b -> [Node])
|
||||||
|
-> (Context a b -> Bool)
|
||||||
|
-> g a b
|
||||||
|
-> Queue Node
|
||||||
|
-> M.HashMap Node Int
|
||||||
|
-> M.HashMap Node Int
|
||||||
|
depths follow back = go
|
||||||
|
where
|
||||||
|
depth n m =
|
||||||
|
case M.lookup n m of
|
||||||
|
Nothing -> error "Layer of node not found, should never happen"
|
||||||
|
Just d -> d
|
||||||
|
visit g l p (m, q) =
|
||||||
|
( case M.lookup p m of
|
||||||
|
Nothing -> M.insert p l m
|
||||||
|
Just d ->
|
||||||
|
if l > d
|
||||||
|
then M.insert p l m
|
||||||
|
else m
|
||||||
|
, if back $ context g p
|
||||||
|
then q
|
||||||
|
else queuePut p q
|
||||||
|
)
|
||||||
|
go g q m =
|
||||||
|
if queueEmpty q
|
||||||
|
then m
|
||||||
|
else
|
||||||
|
let (n, q') = queueGet q
|
||||||
|
in case match n g of
|
||||||
|
(Nothing, g') -> go g' q' m
|
||||||
|
(Just c, g') ->
|
||||||
|
let ps = follow c
|
||||||
|
l = depth n m + 1
|
||||||
|
(m', q'') = foldr (visit g' l) (m, q') ps
|
||||||
|
in go g' q'' m'
|
||||||
|
|
||||||
|
-- | Specify which paths to follow, a function to apply to nodes whose result
|
||||||
|
-- will be in the result list, and the initial nodes.
|
||||||
|
xlayernWith
|
||||||
|
:: Graph g
|
||||||
|
=> (Context a b -> [Node])
|
||||||
|
-> (Context a b -> Bool)
|
||||||
|
-> (Context a b -> c)
|
||||||
|
-> [Node]
|
||||||
|
-> g a b
|
||||||
|
-> [[c]]
|
||||||
|
xlayernWith follow back result initials graph =
|
||||||
|
-- Sort by layer number and drop the layer numbers, leaving just nodes
|
||||||
|
map snd $ sortOn fst $ M.toList $
|
||||||
|
-- Map nodes to results according to user specified function
|
||||||
|
M.map (map $ result . context graph) $
|
||||||
|
-- Turn node-to-layer map into layer-to-nodes map
|
||||||
|
ML.flip $
|
||||||
|
-- Determine the layer number for each node
|
||||||
|
depths
|
||||||
|
follow
|
||||||
|
back
|
||||||
|
graph
|
||||||
|
(queuePutList initials mkQueue)
|
||||||
|
(M.fromList $ zip initials (repeat 1))
|
32
src/Data/HashMap/Lazy/Local.hs
Normal file
32
src/Data/HashMap/Lazy/Local.hs
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
{- This file is part of Vervis.
|
||||||
|
-
|
||||||
|
- Written in 2016 by fr33domlover <fr33domlover@riseup.net>.
|
||||||
|
-
|
||||||
|
- ♡ Copying is an act of love. Please copy, reuse and share.
|
||||||
|
-
|
||||||
|
- The author(s) have dedicated all copyright and related and neighboring
|
||||||
|
- rights to this software to the public domain worldwide. This software is
|
||||||
|
- distributed without any warranty.
|
||||||
|
-
|
||||||
|
- You should have received a copy of the CC0 Public Domain Dedication along
|
||||||
|
- with this software. If not, see
|
||||||
|
- <http://creativecommons.org/publicdomain/zero/1.0/>.
|
||||||
|
-}
|
||||||
|
|
||||||
|
module Data.HashMap.Lazy.Local
|
||||||
|
( flip
|
||||||
|
)
|
||||||
|
where
|
||||||
|
|
||||||
|
import Prelude hiding (flip)
|
||||||
|
|
||||||
|
import Data.Hashable (Hashable)
|
||||||
|
|
||||||
|
import qualified Data.HashMap.Lazy as M
|
||||||
|
|
||||||
|
-- | Build a 'M.HashMap' which maps each value in the original HashMap to the
|
||||||
|
-- keys under which it appears there.
|
||||||
|
flip :: (Eq b, Hashable b) => M.HashMap a b -> M.HashMap b [a]
|
||||||
|
flip = M.foldrWithKey collect M.empty
|
||||||
|
where
|
||||||
|
collect k v = M.insertWith (\ _new old -> k : old) v [k]
|
|
@ -53,6 +53,8 @@ library
|
||||||
Data.EventTime.Local
|
Data.EventTime.Local
|
||||||
Data.Functor.Local
|
Data.Functor.Local
|
||||||
Data.Git.Local
|
Data.Git.Local
|
||||||
|
Data.Graph.Inductive.Query.Layer
|
||||||
|
Data.HashMap.Lazy.Local
|
||||||
Data.Hourglass.Local
|
Data.Hourglass.Local
|
||||||
Data.List.Local
|
Data.List.Local
|
||||||
Data.Paginate.Local
|
Data.Paginate.Local
|
||||||
|
|
Loading…
Reference in a new issue