summaryrefslogtreecommitdiff
path: root/lib
diff options
context:
space:
mode:
authorstuebinm2021-12-30 23:35:10 +0100
committerstuebinm2021-12-30 23:35:10 +0100
commit8f292660630b3154a3441cc673d6aa605f668e5b (patch)
tree183d57f89c9dd8811d2cf7141d63a13d7df18074 /lib
parent09f6bbdad52e58a3d128c9806078d359832f07e1 (diff)
add dot language output for repository map graphs
Diffstat (limited to '')
-rw-r--r--lib/CheckDir.hs9
-rw-r--r--lib/Dirgraph.hs60
2 files changed, 48 insertions, 21 deletions
diff --git a/lib/CheckDir.hs b/lib/CheckDir.hs
index cb4f886..1f69abf 100644
--- a/lib/CheckDir.hs
+++ b/lib/CheckDir.hs
@@ -25,7 +25,7 @@ import Data.Maybe (isJust, mapMaybe)
import Data.Text (Text, isInfixOf)
import qualified Data.Text as T
import Dirgraph (graphToDot, invertGraph, resultToGraph,
- unreachableFrom)
+ takeSubGraph, unreachableFrom)
import GHC.Generics (Generic)
import LintConfig (LintConfig', configMaxLintLevel)
import Paths (normalise, normaliseWithFrag)
@@ -107,7 +107,7 @@ instance ToJSON DirResult where
. M.toList
$ dirresultMaps res)
-- unused in the hub, temporarily removed to make the output smaller
- -- , "exitGraph" .= showDot (dirresultGraph res)
+ , "exitGraph" .= showDot (dirresultGraph res)
]
, "severity" .= maximumLintLevel res
, "mapInfo" .= fmap (\tm -> A.object [ "badges" .= mapresultBadges tm ])
@@ -178,7 +178,10 @@ recursiveCheckDir config prefix root = do
pure $ DirResult { dirresultDeps = missingDeps root maps'
, dirresultMissingAssets = mAssets
, dirresultMaps = maps'
- , dirresultGraph = graphToDot exitGraph
+ , dirresultGraph =
+ graphToDot
+ . takeSubGraph 7 root
+ $ exitGraph
}
diff --git a/lib/Dirgraph.hs b/lib/Dirgraph.hs
index b97a644..8d4a5f2 100644
--- a/lib/Dirgraph.hs
+++ b/lib/Dirgraph.hs
@@ -5,19 +5,22 @@
module Dirgraph where
-import CheckMap (MapResult (mapresultDepends))
-import Control.Monad (forM_)
-import Data.Functor ((<&>))
-import Data.Map.Strict (Map, mapMaybeWithKey, mapWithKey,
- traverseWithKey)
-import qualified Data.Map.Strict as M
-import Data.Set (Set, (\\))
-import qualified Data.Set as S
-import Paths (normalise)
-import Text.Dot (Dot, (.->.))
-import qualified Text.Dot as D
-import Types (Dep (LocalMap))
-import Witherable (mapMaybe)
+import CheckMap (MapResult (mapresultDepends))
+import Control.Monad (forM_, unless)
+import Data.Functor ((<&>))
+import Data.Map.Strict (Map, mapMaybeWithKey, mapWithKey,
+ traverseMaybeWithKey, traverseWithKey)
+import qualified Data.Map.Strict as M
+import Data.Maybe (fromMaybe)
+import Data.Set (Set, (\\))
+import qualified Data.Set as S
+import Paths (normalise)
+import qualified System.FilePath as FP
+import System.FilePath.Posix (takeDirectory, (</>))
+import Text.Dot (Dot, (.->.))
+import qualified Text.Dot as D
+import Types (Dep (LocalMap))
+import Witherable (mapMaybe)
-- | a simple directed graph
type Graph a = Map a (Set a)
@@ -27,9 +30,11 @@ nodes = M.keysSet
-- | simple directed graph of exits
resultToGraph :: Map FilePath MapResult -> Graph FilePath
-resultToGraph = fmap (S.fromList . mapMaybe onlyLocalMaps . mapresultDepends)
- where onlyLocalMaps = \case
- LocalMap path -> Just (normalise "" path)
+resultToGraph = mapWithKey (\p r -> S.fromList
+ . mapMaybe (onlyLocalMaps (takeDirectory p))
+ . mapresultDepends $ r)
+ where onlyLocalMaps prefix = \case
+ LocalMap path -> Just (FP.normalise (prefix </> normalise "" path))
_ -> Nothing
-- | invert edges of a directed graph
@@ -54,12 +59,31 @@ unreachableFrom :: Ord a => a -> Graph a -> Set a
unreachableFrom entrypoint graph =
nodes graph \\ reachableFrom entrypoint graph
+takeSubGraph :: (Eq a, Ord a) => Int -> a -> Graph a -> Graph a
+takeSubGraph i start graph
+ | i <= 0 = mempty
+ | i == 1 =
+ M.singleton start reachable
+ `M.union` M.fromList ((,mempty) <$> S.toList reachable)
+ | otherwise =
+ M.singleton start reachable
+ `M.union` (M.unionsWith S.union
+ . S.map (flip (takeSubGraph (i-1)) graph)
+ $ reachable)
+ where reachable = fromMaybe mempty (M.lookup start graph)
+
graphToDot :: Graph FilePath -> Dot ()
graphToDot graph = do
- nodes <- traverseWithKey
- (\name edges -> D.node [("label",name)] <&> (,edges))
+ main <- D.node [("label","main.json")]
+ nodes' <- traverseMaybeWithKey
+ (\name edges -> if name /= "main.json"
+ then D.node [("label",name)] <&> (, edges) <&> Just
+ else pure Nothing
+ )
graph
+ let reachable = fromMaybe mempty (M.lookup "main.json" graph)
+ let nodes = M.insert "main.json" (main,reachable) nodes'
forM_ nodes $ \(node, edges) ->
forM_ edges $ \key ->
case M.lookup key nodes of