-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathBase.hs
107 lines (88 loc) · 2.73 KB
/
Base.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
{-# OPTIONS -fno-warn-orphans #-}
module BotPlutusInterface.Config.Base (
maybeSpec,
customRationalSpec,
portSpec,
pathSpec,
filepathSpec,
toValueTextViaJSON,
textSpecViaJSON,
) where
import BotPlutusInterface.Config.Types
import BotPlutusInterface.Types ()
import Config
import Config.Schema
import Data.Aeson (FromJSON, ToJSON, eitherDecode, encode)
import Data.Ratio ((%))
import Data.String
import Data.String.ToString
import Data.Text (Text)
import Data.Text qualified as Text
import Network.Wai.Handler.Warp (Port)
import Numeric.Natural (Natural)
import Servant.Client.Core (BaseUrl (..), parseBaseUrl, showBaseUrl)
import Text.Regex
import Prelude
instance ToValue Bool where
toValue = Atom () . MkAtom . Text.toLower . Text.pack . show
instance ToValue Natural where
toValue x = Number () $ integerToNumber $ toInteger x
instance ToValue Integer where
toValue x = Number () $ integerToNumber x
instance ToValue Text where
toValue = Text ()
instance (ToValue a) => ToValue (Maybe a) where
toValue = maybe (Atom () "nothing") toValue
maybeSpec :: ValueSpec a -> ValueSpec (Maybe a)
maybeSpec spec =
Nothing <$ atomSpec "nothing"
<!> Just <$> spec
instance ToValue Rational where
toValue x = Text () $ Text.pack $ show x
customRationalSpec :: ValueSpec Rational
customRationalSpec =
customSpec
"Ratio number (\"1 % 2\") in"
stringSpec
( \x -> case matchRegex ratioRE x of
Just [n, d] ->
let n' = read n
d' = read d
in if d' == 0
then Left "denominator should not be zero"
else Right $ n' % d'
_ -> Left $ Text.pack "Ratio format: '1 % 2'"
)
where
ratioRE = mkRegex "^ *([0-9]+) *% *([0-9]+) *$"
pathSpec :: ValueSpec Text
pathSpec = withNamePrefixSpec "path" anySpec
filepathSpec :: ValueSpec Text
filepathSpec = withNamePrefixSpec "filepath" anySpec
toValueTextViaJSON :: (ToJSON a) => a -> Value ()
toValueTextViaJSON = Text () . Text.pack . filter (/= '"') . toString . encode
textSpecViaJSON :: (FromJSON a) => Text -> ValueSpec a
textSpecViaJSON name =
customSpec
name
textSpec
( \s -> case eitherDecode $ fromString $ wrap $ toString s of
Left err -> Left $ "parse error: " <> fromString err
Right res -> Right res
)
where
wrap s = "\"" <> s <> "\""
instance ToValue BaseUrl where
toValue = Text () . Text.pack . showBaseUrl
instance HasSpec BaseUrl where
anySpec = baseUrlSpec
baseUrlSpec :: ValueSpec BaseUrl
baseUrlSpec =
customSpec
"url"
anySpec
(first Text.showText . parseBaseUrl . Text.unpack)
instance ToValue Port where
toValue = Number () . integerToNumber . toInteger
portSpec :: ValueSpec Port
portSpec = fromEnum <$> customSpec "port" naturalSpec Right