-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathFileLocation.hs
94 lines (81 loc) · 2.8 KB
/
FileLocation.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
{-# LANGUAGE TemplateHaskell #-}
-- | see Debug.FileLocation module for more definitions
module FileLocation
( err, err', undef, fromJst, fromRht, indx, indxShow
, debug, debugM, debugMsg, debugMsgIf, dbg, dbgMsg, trc, ltrace, ltraceM, strace
, locationToString
, thrwIO, thrwsIO
, reThrow
)
where
import FileLocation.LocationString (locationToString)
import Debug.FileLocation (debug, debugM, debugMsg, dbg, dbgMsg, trc, ltrace, ltraceM, strace)
import Debug.Util (debugMsgIf)
import Control.Exception.FileLocation (thrwIO, thrwsIO, reThrow)
import Debug.Trace (trace)
import Language.Haskell.TH.Syntax
import Language.Haskell.TH(varE)
import Data.Maybe(fromMaybe)
import qualified Data.Map as M (lookup)
-- | Like Prelude.error, but gives the file location.
--
-- > $(err "OH NO!")
-- > main:Main main.hs:4:10 OH NO!
err :: String -> Q Exp
err str = do
loc <- qLocation
let prefix = (locationToString loc) ++ " "
[|error (prefix ++ str)|]
-- | Like 'err', but the error message (to be appended to the location) is an argument of the generated expression.
--
-- > $(err) "OH NO!"
-- > main:Main main.hs:4:10 OH NO!
err' :: Q Exp
err' = do
loc <- qLocation
let prefix = (locationToString loc) ++ " "
[| error . (prefix ++) |]
-- | Like Prelude.undefined, but gives the file location.
--
-- Uses trace to output the location (this way we still use undefined instead of calling error).
--
-- > $(undef)
-- > main:Main main.hs:4:10 undefined
-- > err: Prelude.undefined
undef :: Q Exp
undef = do
loc <- qLocation
let prefix = (locationToString loc) ++ " "
[|trace (prefix ++ "undefined") undefined|]
-- | Like 'fromJust', but also shows the file location.
fromJst :: Q Exp
fromJst = do
loc <- qLocation
let msg = (locationToString loc) ++ " fromJst: Nothing"
[|\_m -> case _m of
Just _v -> _v
Nothing -> error msg|]
-- | Like 'fromRight', but also show the file location.
fromRht :: Q Exp
fromRht = do
loc <- qLocation
let msg = (locationToString loc) ++ " fromRht: Left: "
[|\_m -> case _m of
Right _v -> _v
Left _e -> error (msg ++ show _e)|]
-- | Like @(flip ('Data.Map.!')@, but also shows the file location in case the element isn't found.
indx :: Q Exp
indx = indx_common False
-- | Like 'indx', but also 'show's the looked-up element in case it isn't found.
indxShow :: Q Exp
indxShow = indx_common True
indx_common :: Bool -> Q Exp
indx_common = indxWith_common [| M.lookup |]
indxWith_common :: Q Exp -> Bool -> Q Exp
indxWith_common lookupE showElt = do
loc <- qLocation
let msg = (locationToString loc) ++ " indx: Element not in the map"
msgE varName = if showElt
then [| msg ++ ": " ++ show $(varE varName) |]
else [| msg |]
[| \_x _m -> fromMaybe (error $(msgE '_x)) ($(lookupE) _x _m) |]