This commit is contained in:
Joey Hess 2011-02-03 22:44:17 -04:00
parent 17829be0fd
commit 1b1a37b7b1
2 changed files with 57 additions and 42 deletions

View file

@ -21,6 +21,7 @@ import Messages
import Types import Types
import Utility import Utility
import UUID import UUID
import qualified Dot
-- a link from the first repository to the second (its remote) -- a link from the first repository to the second (its remote)
data Link = Link Git.Repo Git.Repo data Link = Link Git.Repo Git.Repo
@ -50,41 +51,41 @@ start = do
- displayed as a node, and each of its remotes is represented as an edge - displayed as a node, and each of its remotes is represented as an edge
- pointing at the node for the remote. -} - pointing at the node for the remote. -}
drawMap :: [Git.Repo] -> (M.Map UUID String) -> String drawMap :: [Git.Repo] -> (M.Map UUID String) -> String
drawMap rs umap = dotGraph $ repos ++ others drawMap rs umap = Dot.graph $ repos ++ others
where where
repos = map (dotGraphRepo umap rs) rs repos = map (dotGraphRepo umap rs) rs
others = map uuidnode (M.keys umap) others = map uuidnode (M.keys umap)
uuidnode u = dotGraphNode u $ M.findWithDefault "" u umap uuidnode u = Dot.graphNode u $ M.findWithDefault "" u umap
dotGraphRepo :: (M.Map UUID String) -> [Git.Repo] -> Git.Repo -> String dotGraphRepo :: (M.Map UUID String) -> [Git.Repo] -> Git.Repo -> String
dotGraphRepo umap fullinfo r = unlines $ node:edges dotGraphRepo umap fullinfo r = unlines $ node:edges
where where
node = inhost $ dotGraphNode (nodeid r) (repoName umap r) node = inhost $ Dot.graphNode (nodeid r) (repoName umap r)
edges = map edge (Git.remotes r) edges = map edge (Git.remotes r)
inhost a inhost a
| Git.repoIsUrl r = dotSubGraph hostname a | Git.repoIsUrl r = Dot.subGraph (Git.urlHost r) (hostname r) a
| otherwise = a | otherwise = a
hostname = head $ split "." $ Git.urlHost r hostname n = head $ split "." $ Git.urlHost n
edge to = edge to =
-- get the full info for the repo since its UUID -- get the full info for the repo since its UUID
-- is in there -- is in there
let to' = findfullinfo to let to' = findfullinfo to
in dotGraphEdge in Dot.graphEdge
(nodeid r) (nodeid r)
(nodeid $ makeabs r to') (nodeid $ makeabs r to')
(edgename to to') (edgename to to')
-- Only name an edge if the name is different than the name -- Only name an edge if the name is different than the name
-- that will be used for the destination node. (This -- that will be used for the destination node, and is
-- reduces visual clutter.) -- different from its hostname. (This reduces visual clutter.)
edgename to to' = edgename to to' =
case (Git.repoRemoteName to) of case (Git.repoRemoteName to) of
Nothing -> Nothing Nothing -> Nothing
Just n -> Just n ->
if (n == repoName umap to') if (n == repoName umap to' || n == hostname to')
then Nothing then Nothing
else Just n else Just n
@ -108,39 +109,6 @@ repoName umap r
Just n -> n Just n -> n
Nothing -> "unknown" Nothing -> "unknown"
dotGraphNode :: String -> String -> String
dotGraphNode nodeid desc = dotLineLabeled desc $ dotQuote nodeid
dotGraphEdge :: String -> String -> Maybe String -> String
dotGraphEdge fromid toid d =
case d of
Nothing -> dotLine edge
Just desc -> dotLineLabeled desc edge
where
edge = dotQuote fromid ++ " -> " ++ dotQuote toid
dotGraph :: [String] -> String
dotGraph s = unlines $ [header] ++ s ++ [footer]
where
header = "digraph map {"
footer= "}"
dotQuote :: String -> String
dotQuote s = "\"" ++ s ++ "\""
dotLine :: String -> String
dotLine s = "\t" ++ s ++ ";"
dotLineLabeled :: String -> String -> String
dotLineLabeled label s = dotLine $ s ++ " [ label=" ++ dotQuote label ++ " ]"
dotSubGraph :: String -> String -> String
dotSubGraph label s = "subgraph " ++ name ++ "{ " ++ setlabel ++ s ++ " }"
where
-- the "cluster_" makes dot draw a box
name = dotQuote ("cluster_ " ++ label)
setlabel = dotLine $ "label=" ++ dotQuote label
{- Recursively searches out remotes starting with the specified repo. -} {- Recursively searches out remotes starting with the specified repo. -}
spider :: Git.Repo -> Annex [Git.Repo] spider :: Git.Repo -> Annex [Git.Repo]
spider r = spider' [r] [] spider r = spider' [r] []

47
Dot.hs Normal file
View file

@ -0,0 +1,47 @@
{- a simple graphviz / dot(1) digraph description generator library
-
- Copyright 2010 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Dot where -- import qualified
{- generates a graph description from a list of lines -}
graph :: [String] -> String
graph s = unlines $ [header] ++ s ++ [footer]
where
header = "digraph map {"
footer= "}"
{- a node in the graph -}
graphNode :: String -> String -> String
graphNode nodeid desc = lineLabeled desc $ quote nodeid
{- an edge between two nodes -}
graphEdge :: String -> String -> Maybe String -> String
graphEdge fromid toid d =
case d of
Nothing -> line edge
Just desc -> lineLabeled desc edge
where
edge = quote fromid ++ " -> " ++ quote toid
quote :: String -> String
quote s = "\"" ++ s ++ "\""
line :: String -> String
line s = "\t" ++ s ++ ";"
{- a line with a label -}
lineLabeled :: String -> String -> String
lineLabeled label s = line $ s ++ " [ label=" ++ quote label ++ " ]"
{- apply to graphNode to put the node in a labeled box -}
subGraph :: String -> String -> String -> String
subGraph subid label s = line $
"subgraph " ++ name ++ "{\n" ++ setlabel ++ "\n" ++ s ++ "\n}"
where
-- the "cluster_" makes dot draw a box
name = quote ("cluster_" ++ subid)
setlabel = line $ "label=" ++ quote label