module Debug.Hoed.Pure.DemoGUI (guiMain,treeFilePath)
where
import qualified Prelude
import Prelude hiding(Right)
import Debug.Hoed.Pure.Render
import Debug.Hoed.Pure.CompTree
import Debug.Hoed.Pure.EventForest
import Debug.Hoed.Pure.Observe
import Debug.Hoed.Pure.Serialize
import qualified Debug.Hoed.Pure.Prop as Prop
import Debug.Hoed.Pure.Prop(Propositions,lookupPropositions,UnevalHandler(..),Judge(..),treeFilePath)
import Paths_Hoed (version)
import Data.Version (showVersion)
import Data.Graph.Libgraph
import qualified Graphics.UI.Threepenny as UI
import Graphics.UI.Threepenny (startGUI,defaultConfig, Window, UI, (#), (#+), (#.), string, on,get,set)
import System.Process(system)
import Data.IORef
import Text.Regex.Posix
import Text.Regex.Posix.String
import Data.IntMap (IntMap)
import qualified Data.IntMap as IntMap
import Data.List(findIndex,intersperse,nub,sort,sortBy
#if __GLASGOW_HASKELL__ >= 710
, sortOn
#endif
)
#if __GLASGOW_HASKELL__ < 710
sortOn :: Ord b => (a -> b) -> [a] -> [a]
sortOn f = map snd . sortOn' fst . map (\x -> (f x, x))
sortOn' :: Ord b => (a -> b) -> [a] -> [a]
sortOn' f = sortBy (\x y -> compare (f x) (f y))
#endif
guiMain :: Trace -> CompTree -> EventForest -> [Propositions] -> Window -> UI ()
guiMain trace tree frt propositions window
= do return window # set UI.title "Hoed debugging session"
compTreeRef <- UI.liftIO $ newIORef tree
traceRef <- UI.liftIO $ newIORef trace
let ns = filter (not . isRootVertex) (preorder tree)
currentVertexRef <- UI.liftIO $ newIORef (vertexUID . head $ ns)
regexRef <- UI.liftIO $ newIORef ""
imgCountRef <- UI.liftIO $ newIORef (0 :: Int)
tab1 <- UI.button # set UI.text "About" # set UI.style activeTab
tab2 <- UI.button # set UI.text "Observe" # set UI.style otherTab
tab3 <- UI.button # set UI.text "Algorithmic Debugging" # set UI.style otherTab
tab4 <- UI.button # set UI.text "Assisted Debugging" # set UI.style otherTab
tab5 <- UI.button # set UI.text "Explore" # set UI.style otherTab
tab6 <- UI.button # set UI.text "Stats" # set UI.style otherTab
logo <- UI.img # set UI.src "static/hoed-logo.png" # set UI.style [("float","right"), ("height","2.2em")]
tabs <- UI.div # set UI.style [("background-color","#D3D3D3")] #+ (map return [tab1,tab2,tab3,tab4,tab5,tab6,logo])
let coloActive tab = do mapM_ (\t -> (return t) # set UI.style otherTab) [tab1,tab2,tab3,tab4,tab5,tab6]; return tab # set UI.style activeTab
help <- guiHelp # set UI.style [("margin-top","0.5em")]
on UI.click tab1 $ \_ -> do
coloActive tab1
UI.getBody window # set UI.children [tabs,help]
on UI.click tab2 $ \_ -> do
coloActive tab2
pane <- guiObserve compTreeRef currentVertexRef # set UI.style [("margin-top","0.5em")]
UI.getBody window # set UI.children [tabs,pane]
on UI.click tab3 $ \_ -> do
coloActive tab3
pane <- guiAlgoDebug compTreeRef currentVertexRef regexRef imgCountRef # set UI.style [("margin-top","0.5em")]
UI.getBody window # set UI.children [tabs,pane]
on UI.click tab4 $ \_ -> do
coloActive tab4
pane <- guiAssisted traceRef propositions compTreeRef currentVertexRef regexRef imgCountRef # set UI.style [("margin-top","0.5em")]
UI.getBody window # set UI.children [tabs,pane]
on UI.click tab5 $ \_ -> do
coloActive tab5
pane <- guiExplore compTreeRef currentVertexRef regexRef imgCountRef # set UI.style [("margin-top","0.5em")]
UI.getBody window # set UI.children [tabs,pane]
on UI.click tab6 $ \_ -> do
coloActive tab6
pane <- guiStats compTreeRef
UI.getBody window # set UI.children [tabs,pane]
UI.getBody window # set UI.style [("margin","0")] #+ (map return [tabs,help])
return ()
activeTab = ("background-color", "white") : tabstyle
otherTab = ("background-color", "#f0f0f0") : tabstyle
tabstyle = [("-webkit-border-top-left-radius", "19"), ("-moz-border-top-left-radius", "19"), ("border-top-left-radius", "0.5em"),("-webkit-border-top-right-radius", "19"), ("-moz-border-top-right-radius", "19"), ("border-top-right-radius", "0.5em"), ("border-width", "medium medium 0px"),("margin-top","1em")]
guiHelp :: UI UI.Element
guiHelp = UI.div # set UI.style [("margin-left", "20%"),("margin-right", "20%")] #+
[ UI.h1 # set UI.text ("Welcome to Hoed " ++ showVersion version)
, UI.p # set UI.text "Hoed is a tracer and debugger for the language Haskell. You can trace a program by annotating functions in suspected modules. After running the program the trace can be viewed in different ways using a web browser. Use the tabs at the top of this page to select the view you want to use. Below we give a short explenation of each view."
, UI.h2 # set UI.text "Observe"
, UI.p # set UI.text "The observe view is useful to get a first impression of what is happening in your program, or to get an overview of the computation statements of a particular slice or pattern. At the top the list of slices and for each slice how many times it was reduced. Below the line a list of computation statements."
, UI.h2 # set UI.text "Algorithmic Debugging"
, UI.p # set UI.text "The algorithmic debugger shows you recorded computation statements, that is a function applied to an argument and its result. You judge these statements as right or wrong. When enough statements are judged the debugger tells you the location of the fault in your code."
, UI.h2 # set UI.text "Explore"
, UI.p # set UI.text "The trace is translated into a tree of computation statements for the algorithmic debugging view. In the explore view you can freely browse this tree to get a better understanding of your program. You can decide yourself in which order you want to judge statements. When enough statements are judged the debugger tells you the location of the fault in your code."
]
guiStats :: IORef CompTree -> UI UI.Element
guiStats compTreeRef = do
(Graph _ vs as) <- UI.liftIO $ readIORef compTreeRef
sec1 <- UI.h3 # set UI.text "Computation Tree Statistics"
vcount <- UI.div # set UI.text (show ((length vs) 1) ++ " computation statements")
acount <- UI.div # set UI.text (show (length as) ++ " dependencies")
sec2 <- UI.h3 # set UI.text "Judgements"
rightCount <- UI.div # set UI.text ((show $ (length . filter isRight $ vs) 1) ++ " Right")
wrongCount <- UI.div # set UI.text ((show . length . filter isWrong $ vs) ++ " Wrong")
assCount <- UI.div # set UI.text ((show . length . filter isAssisted $ vs) ++ " Assisted")
incon <- UI.div # set UI.text ((show . length . filter isInconclusive $ vs) ++ " Inconclusive") # set UI.style [("margin-left","2em")]
passing <- UI.div # set UI.text ((show . length . filter isPassing $ vs) ++ " Passing") # set UI.style [("margin-left","2em")]
unassCount <- UI.div # set UI.text ((show . length . filter isUnassessed $ vs) ++ " Unassessed")
UI.div # set UI.style [("margin-left", "20%"),("margin-right", "20%")] #+ (map return [sec1,vcount,acount, sec2, rightCount, wrongCount, assCount, incon, passing, unassCount])
guiObserve :: IORef CompTree -> IORef Int -> UI UI.Element
guiObserve compTreeRef currentVertexRef = do
(Graph _ vs _) <- UI.liftIO $ readIORef compTreeRef
let slices' = sort $ map (stmtLabel . vertexStmt) . filter (not . isRootVertex) $ vs
slices = nub slices'
count slice = length (filter (==slice) slices')
span s = UI.span # set UI.text s # set UI.style [("margin-right","1em")]
spans = map (\(c,lbl) -> span $ show c ++ " " ++ lbl) $ zip (map count slices) slices
let vs_sorted = sortOn (vertexRes) . filter (not . isRootVertex) $ vs
stmtDiv <- UI.form # set UI.style [("margin-left","2em")]
updateRegEx currentVertexRef vs_sorted stmtDiv ""
regexRef <- UI.liftIO $ newIORef ""
matchField <- UI.input
matchButton <- UI.button # UI.set UI.text "search"
on UI.valueChange matchField $ \s -> UI.liftIO $ writeIORef regexRef s
on UI.click matchButton $ \_ -> do
r <- UI.liftIO $ readIORef regexRef
updateRegEx currentVertexRef vs_sorted stmtDiv r
UI.div #+ (spans ++ [UI.hr, UI.span # set UI.text "regex filter: ", return matchField, return matchButton, UI.hr, return stmtDiv])
updateRegEx :: IORef Int -> [Vertex] -> UI.Element -> String -> UI ()
updateRegEx currentVertexRef vs stmtDiv r = do
(return stmtDiv) # set UI.text "Applying filter ..."
rComp <- UI.liftIO $ compile defaultCompOpt defaultExecOpt r
case rComp of Prelude.Right _ -> drawR
Prelude.Left (_, errorMessage) -> drawL errorMessage
return ()
where
drawL m = do (return stmtDiv) # set UI.text m
drawR
| vs_filtered == [] = drawL $ "There are no computation statements matching \"" ++ r ++ "\"."
| otherwise = (return stmtDiv) # set UI.children [] #+ csDivs
vs_filtered = if r == "" then vs else filter (\v -> (noNewlines . vertexRes $ v) =~ r) vs
csDivs = map stmtToDiv vs_filtered
stmtToDiv v = do
i <- UI.liftIO $ readIORef currentVertexRef
s <- UI.span # set UI.text (vertexRes v)
r <- UI.input # set UI.type_ "radio" # set UI.checked (i == vertexUID v)
let src | isRight v = "static/right.png"
| isWrong v = "static/wrong.png"
| otherwise = "static/unassessed.png"
j <- UI.img # set UI.src src # set UI.height 20 # set UI.style [("margin-left","1em")]
on UI.checkedChange r $ \_ -> checked v
UI.div #+ map return [r,s,j]
checked v = do
UI.liftIO $ writeIORef currentVertexRef (vertexUID v)
drawR
guiAssisted :: IORef Trace -> [Propositions] -> IORef CompTree -> IORef Int -> IORef String -> IORef Int -> UI UI.Element
guiAssisted traceRef ps compTreeRef currentVertexRef regexRef imgCountRef = do
handlerRef <- UI.liftIO $ newIORef Bottom
tree <- UI.liftIO $ readIORef compTreeRef
status <- UI.span
updateStatus status compTreeRef
compStmt <- UI.pre # set UI.style [("margin","0 2em")]
showStmt compStmt compTreeRef currentVertexRef
right <- UI.button # UI.set UI.text "right " #+ [UI.img # set UI.src "static/right.png" # set UI.height 30] # set UI.style [("margin-right","1em")]
wrong <- UI.button # set UI.text "wrong " #+ [UI.img # set UI.src "static/wrong.png" # set UI.height 30] # set UI.style [("margin-right","1em")]
let j = judge AdvanceToNext status compStmt Nothing Nothing currentVertexRef compTreeRef
j right Right
j wrong Wrong
testB <- UI.button # set UI.text "test" # set UI.height 30 # set UI.style [("margin-right","1em")]
resetB <- UI.button # set UI.text "clear all judgements" # set UI.height 30
on UI.click testB $ \_ -> testCurrent compTreeRef traceRef ps status currentVertexRef compStmt handlerRef
on UI.click resetB $ \_ -> resetTree compTreeRef ps status currentVertexRef compStmt
r1 <- UI.input # set UI.type_ "radio" # set UI.checked True
r2 <- UI.input # set UI.type_ "radio" # set UI.checked False
r3 <- UI.input # set UI.type_ "radio" # set UI.checked False
d1 <- UI.div #+ [return r1, UI.span # set UI.text "Abort when property depends on an unevaluated expression."]
d2 <- UI.div #+ [return r2, UI.span # set UI.text "Try to find counterexamples using randomly generated values for unevaluated parts of a computation statement."]
d3 <- UI.div #+ [return r3, UI.span # set UI.text "Accept specification without counter examples as enough to judge as right (use with caution)."]
radioButtons <- UI.div #+ map return [d1,d2,d3]
let onCheck r a b h = on UI.checkedChange r $ \_ -> do
UI.liftIO $ writeIORef handlerRef h
return a # set UI.checked False
return b # set UI.checked False
onCheck r1 r2 r3 Bottom
onCheck r2 r1 r3 Forall
top <- UI.center #+ [return status, UI.br, return right, return wrong, return testB, return resetB]
UI.div #+ [return top, return radioButtons, UI.hr, return compStmt]
resetTree compTreeRef ps status currentVertexRef compStmt = do
t <- UI.liftIO $ readIORef compTreeRef
UI.liftIO $ writeIORef compTreeRef (mapGraph resetVertex t)
advance AdvanceToNext status compStmt Nothing Nothing currentVertexRef compTreeRef
where resetVertex = flip setJudgement Unassessed
testCurrent compTreeRef traceRef ps status currentVertexRef compStmt handlerRef = do
return status # UI.set UI.text "Evaluating propositions ..." #+ [UI.img # set UI.src "static/loading.gif" # set UI.height 30]
mcv <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
case mcv of
(Just cv) -> do
case lookupPropositions ps cv of
Nothing -> updateStatus status compTreeRef
(Just p) -> do
compTree <- UI.liftIO $ readIORef compTreeRef
trace <- UI.liftIO $ readIORef traceRef
j <- UI.liftIO $ Prop.judge trace p cv unjudgedCharacterCount compTree
case j of
(Judge jmt) ->
UI.liftIO $ writeIORef compTreeRef (replaceVertex compTree (setJudgement cv jmt))
(AlternativeTree compTree' trace') -> do
UI.liftIO $ writeIORef compTreeRef compTree'
UI.liftIO $ writeIORef traceRef trace'
UI.liftIO $ putStrLn $ "Switched to a simpler computation tree!"
advance AdvanceToNext status compStmt Nothing Nothing currentVertexRef compTreeRef
Nothing -> updateStatus status compTreeRef
guiAlgoDebug :: IORef CompTree -> IORef Int -> IORef String -> IORef Int -> UI UI.Element
guiAlgoDebug compTreeRef currentVertexRef regexRef imgCountRef = do
tree <- UI.liftIO $ readIORef compTreeRef
status <- UI.span
updateStatus status compTreeRef
compStmt <- UI.pre # set UI.style [("margin","0 2em")]
showStmt compStmt compTreeRef currentVertexRef
right <- UI.button # UI.set UI.text "right " #+ [UI.img # set UI.src "static/right.png" # set UI.height 30] # set UI.style [("margin-right","1em")]
wrong <- UI.button # set UI.text "wrong " #+ [UI.img # set UI.src "static/wrong.png" # set UI.height 30]
let j = judge AdvanceToNext status compStmt Nothing Nothing currentVertexRef compTreeRef
j right Right
j wrong Wrong
top <- UI.center #+ [return status, UI.br, return right, return wrong]
UI.div #+ [return top, UI.hr, return compStmt]
data Advance = AdvanceToNext | DoNotAdvance
judge :: Advance -> UI.Element -> UI.Element -> Maybe UI.Element -> Maybe (UI.Element,IORef Int)
-> IORef UID -> IORef CompTree -> UI.Element -> Judgement -> UI ()
judge adv status compStmt mMenu mImg currentVertexRef compTreeRef b j =
on UI.click b $ \_ -> do
mv <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
case mv of
(Just v) -> judge' v
Nothing -> return ()
where
judge' :: Vertex -> UI ()
judge' v = do
t' <- UI.liftIO $ readIORef compTreeRef
UI.liftIO $ writeIORef compTreeRef (markNode t' v j)
advance adv status compStmt mMenu mImg currentVertexRef compTreeRef
advance :: Advance -> UI.Element -> UI.Element -> Maybe UI.Element -> Maybe (UI.Element,IORef Int)
-> IORef UID -> IORef CompTree -> UI ()
advance adv status compStmt mMenu mImg currentVertexRef compTreeRef = do
mv <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
case mv of
Nothing -> do
t <- UI.liftIO $ readIORef compTreeRef
case next_step t getJudgement of
RootVertex -> return ()
w -> advanceTo w status compStmt mMenu mImg currentVertexRef compTreeRef
(Just v) -> do
t <- UI.liftIO $ readIORef compTreeRef
case (adv, next_step t getJudgement) of
(DoNotAdvance,_) -> advanceTo v status compStmt mMenu mImg currentVertexRef compTreeRef
(AdvanceToNext,RootVertex) -> advanceTo v status compStmt mMenu mImg currentVertexRef compTreeRef
(AdvanceToNext,w) -> advanceTo w status compStmt mMenu mImg currentVertexRef compTreeRef
updateStatus status compTreeRef
advanceTo w status compStmt mMenu mImg currentVertexRef compTreeRef = do
UI.liftIO $ writeIORef currentVertexRef (vertexUID w)
showStmt compStmt compTreeRef currentVertexRef
case mMenu of Nothing -> return ()
(Just menu) -> updateMenu menu compTreeRef currentVertexRef
case mImg of Nothing -> return ()
(Just (img,imgCountRef)) -> redraw img imgCountRef compTreeRef (Just w)
guiExplore :: IORef CompTree -> IORef Int -> IORef String -> IORef Int -> UI UI.Element
guiExplore compTreeRef currentVertexRef regexRef imgCountRef = do
tree <- UI.liftIO $ readIORef compTreeRef
img <- UI.img
redrawWith img imgCountRef compTreeRef currentVertexRef
img' <- UI.center #+ [UI.element img]
compStmt <- UI.pre
menu <- UI.select # set UI.style [("margin-right","1em")]
showStmt compStmt compTreeRef currentVertexRef
updateMenu menu compTreeRef currentVertexRef
let selectVertex' = selectVertex compStmt menu compTreeRef currentVertexRef (redrawWith img imgCountRef compTreeRef)
on UI.selectionChange menu selectVertex'
status <- UI.span
updateStatus status compTreeRef
right <- UI.button # UI.set UI.text "right " #+ [UI.img # set UI.src "static/right.png" # set UI.height 20] # set UI.style [("margin-right","1em")]
wrong <- UI.button # set UI.text "wrong " #+ [UI.img # set UI.src "static/wrong.png" # set UI.height 20] # set UI.style [("margin-right","1em")]
let j = judge DoNotAdvance status compStmt (Just menu) (Just (img, imgCountRef)) currentVertexRef compTreeRef
j right Right
j wrong Wrong
storeb <- UI.button # UI.set UI.text "store" # set UI.style [("margin-right","1em")]
restoreb <- UI.button # UI.set UI.text "restore"
on UI.click storeb $ \_ -> UI.liftIO $ store compTreeRef
on UI.click restoreb $ \_ -> do UI.liftIO $ restore compTreeRef
updateStatus status compTreeRef
v <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
redraw img imgCountRef compTreeRef v
hr <- UI.hr
br <- UI.br
UI.div #+ (map UI.element [menu, right, wrong, storeb, restoreb, br, hr, img', br, status, hr, compStmt])
judgementFilePath :: FilePath
judgementFilePath = ".Hoed/savedJudgements"
store :: IORef CompTree -> IO ()
store compTreeRef = do
t <- readIORef compTreeRef
storeJudgements judgementFilePath t
restore :: IORef CompTree -> IO ()
restore compTreeRef = do
t <- readIORef compTreeRef
t' <- restoreJudgements judgementFilePath t
writeIORef compTreeRef t'
preorder :: CompTree -> [Vertex]
preorder = getPreorder . getDfs
showStmt :: UI.Element -> IORef CompTree -> IORef Int -> UI ()
showStmt e compTreeRef currentVertexRef = do
mv <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
let s = case mv of
Nothing -> "No computation statement selected."
(Just v) -> let s = show . vertexStmt $ v in case getJudgement v of
(Assisted s') -> "Results from testing computation statement:\n" ++ (getMessage s') ++ "\n---\n\n" ++ s
_ -> s
UI.element e # set UI.text s
return ()
getMessage :: [AssistedMessage] -> String
getMessage = foldl (\acc m -> acc ++ getMessage' m) ""
getMessage' (InconclusiveProperty msg) = msg
getMessage' (PassingProperty msg) = msg ++ " holds for this statement's values"
updateMenu :: UI.Element -> IORef CompTree -> IORef Int -> UI ()
updateMenu menu compTreeRef currentVertexRef = do
vs <- menuVertices compTreeRef currentVertexRef
i <- UI.liftIO $ readIORef currentVertexRef
let j = case findIndex (\v -> vertexUID v == i) vs of
(Just j') -> j'
Nothing -> 0
t <- UI.liftIO $ readIORef compTreeRef
let fs = faultyVertices t
ops <- mapM (\s->UI.option # set UI.text s) $ map (summarizeVertex fs) vs
(UI.element menu) # set UI.children []
UI.element menu #+ (map UI.element ops)
(UI.element menu) # set UI.selection (Just j)
return ()
selectVertex :: UI.Element -> UI.Element -> IORef CompTree -> IORef Int
-> (IORef Int -> UI ()) -> Maybe Int -> UI ()
selectVertex compStmt menu compTreeRef currentVertexRef myRedraw mi = case mi of
Just j -> do vs <- menuVertices compTreeRef currentVertexRef
mcv <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
let v = vs !! j
UI.liftIO $ writeIORef currentVertexRef (vertexUID v)
showStmt compStmt compTreeRef currentVertexRef
myRedraw currentVertexRef
updateMenu menu compTreeRef currentVertexRef
return ()
Nothing -> do UI.liftIO $ putStrLn "selectVertex: Nothing selected"
return ()
menuVertices :: IORef CompTree -> IORef Int -> UI [Vertex]
menuVertices compTreeRef currentVertexRef = do
t <- UI.liftIO $ readIORef compTreeRef
i <- UI.liftIO $ readIORef currentVertexRef
mcv <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
let cv = case mcv of (Just v) -> v; Nothing -> RootVertex
ps = preds t cv
sibl = if RootVertex `elem` ps then succs t RootVertex else [cv]
return $ filter (/= RootVertex) $ ps ++ sibl ++ (succs t cv)
lookupCurrentVertex :: IORef Int -> IORef CompTree -> IO (Maybe Vertex)
lookupCurrentVertex currentVertexRef compTree = do
i <- readIORef currentVertexRef
t <- readIORef compTree
return $ case filter (\v->vertexUID v==i) (vertices t) of
[] -> Nothing
[v] -> Just v
vs -> error $ "lookupCurrentVertex: UID " ++ show i ++ " identifies "
++ (show . length $ vs) ++ " computation statements"
markNode :: CompTree -> Vertex -> Judgement -> CompTree
markNode g v s = mapGraph f g
where f RootVertex = RootVertex
f v' = if v' === v then setJudgement v s else v'
(===) :: Vertex -> Vertex -> Bool
v1 === v2 = (vertexUID v1) == (vertexUID v2)
data MaxStringLength = ShorterThan Int | Unlimited
shorten :: MaxStringLength -> String -> String
shorten Unlimited s = s
shorten (ShorterThan l) s
| length s < l = s
| l > 3 = take (l 3) s ++ "..."
| otherwise = take l s
shorterThan60 = shorten (ShorterThan 60)
summarizeVertex :: [Vertex] -> Vertex -> String
summarizeVertex fs v = shorterThan60 (noNewlines . show . vertexStmt $ v) ++ s
where s = if v `elem` fs then " !!" else case getJudgement v of
Wrong -> " :("
Right -> " :)"
_ -> " ??"
vertexGraphvizLabel :: [Vertex] -> Vertex -> String
vertexGraphvizLabel fs v =
"<<TABLE BORDER=\"0\" CELLBORDER=\"0\"><TR><TD HEIGHT=\"30\" WIDTH=\"30\" FIXEDSIZE=\"true\"><IMG SCALE=\"true\" SRC=\"" ++ (vertexImg fs v) ++ "\"/></TD><TD><FONT POINT-SIZE=\"30\">" ++ (htmlEscape . shorterThan60 . noNewlines . show . vertexStmt $ v) ++ "</FONT></TD></TR></TABLE>>"
htmlEscape :: String -> String
htmlEscape = foldr (\c acc -> replace c ++ acc) ""
where
replace :: Char -> String
replace '"' = """
replace '{' = "{"
replace '\\' = "\"
replace '>' = ">"
replace '<' = "<"
replace '}' = "}"
replace c = [c]
vertexImg :: [Vertex] -> Vertex -> String
vertexImg fs v = if v `elem` fs then ".Hoed/wwwroot/faulty.png" else case vertexJmt v of
Unassessed -> ".Hoed/wwwroot/unassessed.png"
(Assisted _) -> ".Hoed/wwwroot/unassessed.png"
Wrong -> ".Hoed/wwwroot/wrong.png"
Right -> ".Hoed/wwwroot/right.png"
updateStatus :: UI.Element -> IORef CompTree -> UI ()
updateStatus e compGraphRef = do
g <- UI.liftIO $ readIORef compGraphRef
let isJudged v = case getJudgement v of
Right -> True
Wrong -> True
_ -> False
slen = show . length
ns = filter (not . isRootVertex) (preorder g)
js = filter isJudged ns
fs = faultyVertices g
txt = if length fs > 0 then " Fault detected in: " ++ (vertexRes . head) fs
else " Judged " ++ slen js ++ "/" ++ slen ns
UI.element e # set UI.text txt
return ()
redrawWith :: UI.Element -> IORef Int -> IORef CompTree -> IORef Int -> UI ()
redrawWith img imgCountRef compTreeRef currentVertexRef = do
mv <- UI.liftIO $ lookupCurrentVertex currentVertexRef compTreeRef
redraw img imgCountRef compTreeRef mv
redraw :: UI.Element -> IORef Int -> IORef CompTree -> (Maybe Vertex) -> UI ()
redraw img imgCountRef compTreeRef mcv
= do tree <- UI.liftIO $ readIORef compTreeRef
UI.liftIO $ writeFile ".Hoed/debugTree.dot" $ shw (faultyVertices tree) (summarize tree mcv)
UI.liftIO $ system $ "cat .Hoed/debugTree.dot | unflatten -l 5| dot -Tpng -Gsize=15,15 -Gdpi=100"
++ "> .Hoed/wwwroot/debugTree.png"
i <- UI.liftIO $ readIORef imgCountRef
UI.liftIO $ writeIORef imgCountRef (i+1)
UI.element img # set UI.src ("static/debugTree.png#" ++ show i)
return ()
where shw fs t = showWith t (coloVertex $ fs) showArc
coloVertex _ RootVertex = ("\".\"", "shape=none")
coloVertex fs v = ( vertexGraphvizLabel fs v
, if isCurrentVertex mcv v then "shape=none fontcolor=blue"
else "shape=none"
)
showArc _ = ""
summarize :: CompTree -> Maybe Vertex -> CompTree
summarize tree (Just cv) = Graph r vs as'
where
i = vertexUID cv
ps = preds tree cv
ps' = if RootVertex `elem` ps then ps ++ (succs tree RootVertex) else ps
cs = succs tree cv
vs = nub (ps' ++ cv : cs)
as = filter (\a -> isCV (source a) || isCV (target a)) (arcs tree)
as' = if RootVertex `elem` ps then nub (as ++ filter (\a -> isRV (source a) || isRV (target a)) (arcs tree)) else as
r = if RootVertex `elem` vs then RootVertex else head vs
isCV = (==i) . vertexUID
isRV = (==) RootVertex
summarize tree Nothing = tree
isCurrentVertex :: Maybe Vertex -> Vertex -> Bool
isCurrentVertex mcv v = case v of
RootVertex -> False
_ -> case mcv of
Nothing -> False
(Just RootVertex) -> False
(Just w) -> vertexStmt v == vertexStmt w
faultyVertices :: CompTree -> [Vertex]
faultyVertices = findFaulty_dag getJudgement