-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathReport.hs
122 lines (112 loc) · 3.46 KB
/
Report.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# OPTIONS_GHC -fno-warn-name-shadowing #-}
module Main (main) where
import Data.Ord
import Control.DeepSeq
import Data.Function
import Data.List
import System.Environment
import Text.CSV
import Text.Printf
main :: IO ()
main = do
fp : _ <- getArgs
reportFromCsv fp
reportFromCsv :: FilePath -> IO ()
reportFromCsv fp = do
result <- parseCSVFromFile fp
case result of
Right (_:rows) -> do
!readme <- fmap force (readFile "README.md")
let sep = "<!-- RESULTS -->"
before = unlines (takeWhile (/= sep) (lines readme) ++ [sep ++ "\n"])
writeFile
"README.md"
(before ++
unlines
(map
format
(filter
(not . all (all null))
(groupBy
(on (==) (takeWhile (/= '.') . stripAll . concat . take 1))
rows))))
_ -> error "Couldn't parse csv"
format :: [[String]] -> String
format rows =
("## " ++ takeWhile (/= '.') (stripAll (concat (concat (take 1 (drop 1 rows)))))) ++
"\n\n" ++
unlines
[ "|Name|" ++ intercalate "|" scales ++ "|"
, "|" ++ concat (replicate (1 + length scales) "---|")
] ++
unlines
(map
(\name ->
"|" ++ name ++ "|" ++ intercalate "|" (valuesByName name) ++ "|")
(sortBy (comparing totalsByName) names))
where
valuesByName name =
map
(\row@(_ : mean : _) ->
let scale = rowScale row
in float (valuesByScale scale) (read mean))
(filter ((== name) . rowName) rows)
totalsByName :: String -> Double
totalsByName name =
sum (map
(\(_ : mean : _) ->
(read mean))
(filter ((== name) . rowName) rows))
valuesByScale scale =
map (\(_ : mean : _) -> read mean) (filter ((== scale) . rowScale) rows)
names = nub (map rowName rows)
scales = nub (map rowScale rows)
rowName row =
let s =
takeWhile
(/= ':')
(dropWhile (== '.') (dropWhile (/= '.') (stripAll (concat (take 1 row)))))
in s
rowScale row =
let scale = dropWhile (== ':') (dropWhile (/= ':') (concat (take 1 row)))
in scale
-- | Strip "All." prefix.
stripAll :: String -> String
stripAll = drop 4
-- | Inputs are in picoseconds, so scaling by 1e-12.
float :: [Double] -> Double -> String
float others x = let (scale, ext) = secs (mean others * 1e-12)
in with (x * 1e-12 * scale) ext
-- | Convert a number of seconds to a string. The string will consist
-- of four decimal places, followed by a short description of the time
-- units.
secs :: Double -> (Double, String)
secs k
| k >= 1 = 1 `pair` "s"
| k >= 1e-3 = 1e3 `pair` "ms"
| k >= 1e-6 = 1e6 `pair` "μs"
| k >= 1e-9 = 1e9 `pair` "ns"
| k >= 1e-12 = 1e12 `pair` "ps"
| k >= 1e-15 = 1e15 `pair` "fs"
| k >= 1e-18 = 1e18 `pair` "as"
| otherwise = error "Bad scale"
where pair= (,)
with :: Double -> String -> String
with (t :: Double) (u :: String)
| t >= 1e9 = printf "%.4g %s" t u
| t >= 1e3 = printf "%.0f %s" t u
| t >= 1e2 = printf "%.1f %s" t u
| t >= 1e1 = printf "%.2f %s" t u
| otherwise = printf "%.3f %s" t u
-- | Simple rolling average.
mean :: [Double] -> Double
mean =
snd .
foldr
(\x (cnt,avg) ->
( cnt + 1
, (x + avg * cnt) / (cnt + 1)))
(0, 0)