-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy pathArbitrary.hs
406 lines (367 loc) · 18.1 KB
/
Arbitrary.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
{-# OPTIONS_GHC -fno-warn-orphans #-}
{-# LANGUAGE FlexibleInstances, ScopedTypeVariables, OverloadedStrings #-}
-- provides Arbitrary instance for Pandoc types
module Text.Pandoc.Arbitrary ()
where
import Test.QuickCheck
import Control.Applicative (Applicative ((<*>), pure), (<$>))
import Control.Monad (forM)
import Data.Text (Text)
import qualified Data.Text as T
import Text.Pandoc.Definition
import Text.Pandoc.Builder
realString :: Gen Text
realString = fmap T.pack $ resize 8 $ listOf $ frequency [ (9, elements [' '..'\127'])
, (1, elements ['\128'..'\9999']) ]
shrinkText :: Text -> [Text]
shrinkText xs = T.pack <$> shrink (T.unpack xs)
shrinkText2 :: (Text, Text) -> [(Text, Text)]
shrinkText2 = liftShrink2 shrinkText shrinkText
arbAttr :: Gen Attr
arbAttr = do
id' <- elements ["","loc"]
classes' <- elements [[],["haskell"],["c","numberLines"]]
keyvals <- elements [[],[("start","22")],[("a","11"),("b_2","a b c")]]
return (id',classes',keyvals)
shrinkAttr :: Attr -> [Attr]
shrinkAttr (a, b, c)
= [ (a', b', c') | a' <- shrinkText a,
b' <- liftShrink shrinkText b,
c' <- liftShrink shrinkText2 c ]
instance Arbitrary Inlines where
arbitrary = (fromList :: [Inline] -> Inlines) <$> arbitrary
shrink = fmap fromList . ((++) <$> shrink <*> flattenShrinkInlines) . toList
where flattenShrinkInlines (x:xs) =
let x' = flattenInline x
in [x' ++ xs | not (null x')] ++ [x:xs' | xs' <- flattenShrinkInlines xs]
flattenShrinkInlines [] = []
flattenInline :: Inline -> [Inline]
flattenInline (Str _) = []
flattenInline (Emph ils) = ils
flattenInline (Underline ils) = ils
flattenInline (Strong ils) = ils
flattenInline (Strikeout ils) = ils
flattenInline (Superscript ils) = ils
flattenInline (Subscript ils) = ils
flattenInline (SmallCaps ils) = ils
flattenInline (Quoted _ ils) = ils
flattenInline (Cite _ ils) = ils
flattenInline Code{} = []
flattenInline Space = []
flattenInline SoftBreak = []
flattenInline LineBreak = []
flattenInline Math{} = []
flattenInline RawInline{} = []
flattenInline (Link _ ils _) = ils
flattenInline (Image _ ils _) = ils
flattenInline Note{} = []
flattenInline (Span _ ils) = ils
instance Arbitrary Blocks where
arbitrary = (fromList :: [Block] -> Blocks) <$> arbitrary
shrink = fmap fromList . ((++) <$> shrink <*> flattenShrinkBlocks) . toList
where flattenShrinkBlocks (x:xs) =
let x' = flattenBlock x
in [x' ++ xs | not (null x')] ++ [x:xs' | xs' <- flattenShrinkBlocks xs]
flattenShrinkBlocks [] = []
flattenBlock :: Block -> [Block]
flattenBlock Plain{} = []
flattenBlock Para{} = []
flattenBlock (LineBlock lns) = [Para x | x <- lns]
flattenBlock CodeBlock{} = []
flattenBlock RawBlock{} = []
flattenBlock (BlockQuote blks) = blks
flattenBlock (OrderedList _ blksList) = concat blksList
flattenBlock (BulletList blksList) = concat blksList
flattenBlock (DefinitionList defs) = concat [Para ils:concat blks | (ils, blks) <- defs]
flattenBlock (Header _ _ ils) = [Para ils]
flattenBlock HorizontalRule = []
flattenBlock (Table _ capt _ hd bd ft) = flattenCaption capt <>
flattenTableHead hd <>
concatMap flattenTableBody bd <>
flattenTableFoot ft
flattenBlock (Figure _ capt blks) = flattenCaption capt <> blks
flattenBlock (Div _ blks) = blks
flattenCaption (Caption Nothing body) = body
flattenCaption (Caption (Just ils) body) = Para ils : body
flattenTableHead (TableHead _ body) = flattenRows body
flattenTableBody (TableBody _ _ hd bd) = flattenRows hd <> flattenRows bd
flattenTableFoot (TableFoot _ body) = flattenRows body
flattenRows = concatMap flattenRow
flattenRow (Row _ body) = concatMap flattenCell body
flattenCell (Cell _ _ _ _ blks) = blks
shrinkInlineList :: [Inline] -> [[Inline]]
shrinkInlineList = fmap toList . shrink . fromList
shrinkInlinesList :: [[Inline]] -> [[[Inline]]]
shrinkInlinesList = fmap (fmap toList) . shrink . fmap fromList
shrinkBlockList :: [Block] -> [[Block]]
shrinkBlockList = fmap toList . shrink . fromList
shrinkBlocksList :: [[Block]] -> [[[Block]]]
shrinkBlocksList = fmap (fmap toList) . shrink . fmap fromList
instance Arbitrary Inline where
arbitrary = resize 3 $ arbInline 2
shrink (Str s) = Str <$> shrinkText s
shrink (Emph ils) = Emph <$> shrinkInlineList ils
shrink (Underline ils) = Underline <$> shrinkInlineList ils
shrink (Strong ils) = Strong <$> shrinkInlineList ils
shrink (Strikeout ils) = Strikeout <$> shrinkInlineList ils
shrink (Superscript ils) = Superscript <$> shrinkInlineList ils
shrink (Subscript ils) = Subscript <$> shrinkInlineList ils
shrink (SmallCaps ils) = SmallCaps <$> shrinkInlineList ils
shrink (Quoted qtype ils) = Quoted qtype <$> shrinkInlineList ils
shrink (Cite cits ils) = (Cite cits <$> shrinkInlineList ils)
++ (flip Cite ils <$> shrink cits)
shrink (Code attr s) = (Code attr <$> shrinkText s)
++ (flip Code s <$> shrinkAttr attr)
shrink Space = []
shrink SoftBreak = []
shrink LineBreak = []
shrink (Math mtype s) = Math mtype <$> shrinkText s
shrink (RawInline fmt s) = RawInline fmt <$> shrinkText s
shrink (Link attr ils target) = [Link attr ils' target | ils' <- shrinkInlineList ils]
++ [Link attr ils target' | target' <- shrinkText2 target]
++ [Link attr' ils target | attr' <- shrinkAttr attr]
shrink (Image attr ils target) = [Image attr ils' target | ils' <- shrinkInlineList ils]
++ [Image attr ils target' | target' <- shrinkText2 target]
++ [Image attr' ils target | attr' <- shrinkAttr attr]
shrink (Note blks) = Note <$> shrinkBlockList blks
shrink (Span attr s) = (Span attr <$> shrink s)
++ (flip Span s <$> shrinkAttr attr)
arbInlines :: Int -> Gen [Inline]
arbInlines n = listOf1 (arbInline n) `suchThat` (not . startsWithSpace)
where startsWithSpace (Space:_) = True
startsWithSpace (SoftBreak:_) = True
-- Note: no LineBreak, similarly to Text.Pandoc.Builder (trimInlines)
startsWithSpace _ = False
-- restrict to 3 levels of nesting max; otherwise we get
-- bogged down in indefinitely large structures
arbInline :: Int -> Gen Inline
arbInline n = frequency $ [ (60, Str <$> realString)
, (40, pure Space)
, (10, pure SoftBreak)
, (10, pure LineBreak)
, (10, Code <$> arbAttr <*> realString)
, (5, elements [ RawInline (Format "html") "<a id=\"eek\">"
, RawInline (Format "latex") "\\my{command}" ])
] ++ [ x | n > 1, x <- nesters]
where nesters = [ (10, Emph <$> arbInlines (n-1))
, (10, Underline <$> arbInlines (n-1))
, (10, Strong <$> arbInlines (n-1))
, (10, Strikeout <$> arbInlines (n-1))
, (10, Superscript <$> arbInlines (n-1))
, (10, Subscript <$> arbInlines (n-1))
, (10, SmallCaps <$> arbInlines (n-1))
, (10, Span <$> arbAttr <*> arbInlines (n-1))
, (10, Quoted <$> arbitrary <*> arbInlines (n-1))
, (10, Math <$> arbitrary <*> realString)
, (10, Link <$> arbAttr <*> arbInlines (n-1) <*> ((,) <$> realString <*> realString))
, (10, Image <$> arbAttr <*> arbInlines (n-1) <*> ((,) <$> realString <*> realString))
, (2, Cite <$> arbitrary <*> arbInlines 1)
, (2, Note <$> resize 3 (listOf1 $ arbBlock (n-1)))
]
instance Arbitrary Block where
arbitrary = resize 3 $ arbBlock 2
shrink (Plain ils) = Plain <$> shrinkInlineList ils
shrink (Para ils) = Para <$> shrinkInlineList ils
shrink (LineBlock lns) = LineBlock <$> shrinkInlinesList lns
shrink (CodeBlock attr s) = (CodeBlock attr <$> shrinkText s)
++ (flip CodeBlock s <$> shrinkAttr attr)
shrink (RawBlock fmt s) = RawBlock fmt <$> shrinkText s
shrink (BlockQuote blks) = BlockQuote <$> shrinkBlockList blks
shrink (OrderedList listAttrs blksList) = OrderedList listAttrs <$> shrinkBlocksList blksList
shrink (BulletList blksList) = BulletList <$> shrinkBlocksList blksList
shrink (DefinitionList defs) = DefinitionList <$> shrinkDefinitionList defs
where shrinkDefinition (ils, blksList) = [(ils', blksList) | ils' <- shrinkInlineList ils]
++ [(ils, blksList') | blksList' <- shrinkBlocksList blksList]
shrinkDefinitionList (x:xs) = [xs]
++ [x':xs | x' <- shrinkDefinition x]
++ [x:xs' | xs' <- shrinkDefinitionList xs]
shrinkDefinitionList [] = []
shrink (Header n attr ils) = (Header n attr <$> shrinkInlineList ils)
++ (flip (Header n) ils <$> shrinkAttr attr)
shrink HorizontalRule = []
shrink (Table attr capt specs thead tbody tfoot) =
-- TODO: shrink number of columns
[Table attr' capt specs thead tbody tfoot | attr' <- shrinkAttr attr] ++
[Table attr capt specs thead' tbody tfoot | thead' <- shrink thead] ++
[Table attr capt specs thead tbody' tfoot | tbody' <- shrink tbody] ++
[Table attr capt specs thead tbody tfoot' | tfoot' <- shrink tfoot] ++
[Table attr capt' specs thead tbody tfoot | capt' <- shrink capt]
shrink (Figure attr capt blks) =
[Figure attr capt blks' | blks' <- shrinkBlockList blks] ++
[Figure attr capt' blks | capt' <- shrink capt] ++
[Figure attr' capt blks | attr' <- shrinkAttr attr]
shrink (Div attr blks) = (Div attr <$> shrinkBlockList blks)
++ (flip Div blks <$> shrinkAttr attr)
arbBlock :: Int -> Gen Block
arbBlock n = frequency $ [ (10, Plain <$> arbInlines (n-1))
, (15, Para <$> arbInlines (n-1))
, (5, CodeBlock <$> arbAttr <*> realString)
, (3, LineBlock <$>
((:) <$>
arbInlines ((n - 1) `mod` 3) <*>
forM [1..((n - 1) `div` 3)] (const (arbInlines 3))))
, (2, elements [ RawBlock (Format "html")
"<div>\n*&*\n</div>"
, RawBlock (Format "latex")
"\\begin[opt]{env}\nhi\n{\\end{env}"
])
, (5, Header <$> choose (1 :: Int, 6)
<*> pure nullAttr
<*> arbInlines (n-1))
, (2, pure HorizontalRule)
] ++ [x | n > 0, x <- nesters]
where nesters = [ (5, BlockQuote <$> listOf1 (arbBlock (n-1)))
, (5, OrderedList <$> ((,,) <$> (arbitrary `suchThat` (> 0))
<*> arbitrary
<*> arbitrary)
<*> listOf1 (listOf1 $ arbBlock (n-1)))
, (5, BulletList <$> listOf1 (listOf1 $ arbBlock (n-1)))
, (5, DefinitionList <$> listOf1 ((,) <$> arbInlines (n-1)
<*> listOf1 (listOf1 $ arbBlock (n-1))))
, (5, Div <$> arbAttr <*> listOf1 (arbBlock (n-1)))
, (2, do cs <- choose (1 :: Int, 6)
bs <- choose (0 :: Int, 2)
Table <$> arbAttr
<*> arbitrary
<*> vectorOf cs ((,) <$> arbitrary
<*> elements [ ColWidthDefault
, ColWidth (1/3)
, ColWidth 0.25 ])
<*> arbTableHead (n-1)
<*> vectorOf bs (arbTableBody (n-1))
<*> arbTableFoot (n-1))
, (2, Figure <$> arbAttr
<*> arbitrary
<*> listOf1 (arbBlock (n-1)))
]
arbRow :: Int -> Gen Row
arbRow n = do
cs <- choose (0, 5)
Row <$> arbAttr <*> vectorOf cs (arbCell n)
arbTableHead :: Int -> Gen TableHead
arbTableHead n = do
rs <- choose (0, 5)
TableHead <$> arbAttr <*> vectorOf rs (arbRow n)
arbTableBody :: Int -> Gen TableBody
arbTableBody n = do
hrs <- choose (0 :: Int, 2)
rs <- choose (0, 5)
rhc <- choose (0, 5)
TableBody <$> arbAttr
<*> pure (RowHeadColumns rhc)
<*> vectorOf hrs (arbRow n)
<*> vectorOf rs (arbRow n)
arbTableFoot :: Int -> Gen TableFoot
arbTableFoot n = do
rs <- choose (0, 5)
TableFoot <$> arbAttr <*> vectorOf rs (arbRow n)
arbCell :: Int -> Gen Cell
arbCell n = Cell <$> arbAttr
<*> arbitrary
<*> (RowSpan <$> choose (1 :: Int, 2))
<*> (ColSpan <$> choose (1 :: Int, 2))
<*> listOf (arbBlock n)
instance Arbitrary Pandoc where
arbitrary = resize 8 (Pandoc <$> arbitrary <*> arbitrary)
instance Arbitrary CitationMode where
arbitrary
= do x <- choose (0 :: Int, 2)
case x of
0 -> return AuthorInText
1 -> return SuppressAuthor
2 -> return NormalCitation
_ -> error "FATAL ERROR: Arbitrary instance, logic bug"
instance Arbitrary Citation where
arbitrary
= Citation <$> fmap T.pack (listOf $ elements $ ['a'..'z'] ++ ['0'..'9'] ++ ['_'])
<*> arbInlines 1
<*> arbInlines 1
<*> arbitrary
<*> arbitrary
<*> arbitrary
instance Arbitrary Row where
arbitrary = resize 3 $ arbRow 2
shrink (Row attr body)
= [Row attr' body | attr' <- shrinkAttr attr] ++
[Row attr body' | body' <- shrink body]
instance Arbitrary TableHead where
arbitrary = resize 3 $ arbTableHead 2
shrink (TableHead attr body)
= [TableHead attr' body | attr' <- shrinkAttr attr] ++
[TableHead attr body' | body' <- shrink body]
instance Arbitrary TableBody where
arbitrary = resize 3 $ arbTableBody 2
-- TODO: shrink rhc?
shrink (TableBody attr rhc hd bd)
= [TableBody attr' rhc hd bd | attr' <- shrinkAttr attr] ++
[TableBody attr rhc hd' bd | hd' <- shrink hd] ++
[TableBody attr rhc hd bd' | bd' <- shrink bd]
instance Arbitrary TableFoot where
arbitrary = resize 3 $ arbTableFoot 2
shrink (TableFoot attr body)
= [TableFoot attr' body | attr' <- shrinkAttr attr] ++
[TableFoot attr body' | body' <- shrink body]
instance Arbitrary Cell where
arbitrary = resize 3 $ arbCell 2
shrink (Cell attr malign h w body)
= [Cell attr malign h w body' | body' <- shrinkBlockList body] ++
[Cell attr' malign h w body | attr' <- shrinkAttr attr] ++
[Cell attr malign' h w body | malign' <- shrink malign]
instance Arbitrary Caption where
arbitrary = Caption <$> arbitrary <*> arbitrary
shrink (Caption mshort body)
= [Caption mshort' body | mshort' <- shrink mshort] ++
[Caption mshort body' | body' <- shrinkBlockList body]
instance Arbitrary MathType where
arbitrary
= do x <- choose (0 :: Int, 1)
case x of
0 -> return DisplayMath
1 -> return InlineMath
_ -> error "FATAL ERROR: Arbitrary instance, logic bug"
instance Arbitrary QuoteType where
arbitrary
= do x <- choose (0 :: Int, 1)
case x of
0 -> return SingleQuote
1 -> return DoubleQuote
_ -> error "FATAL ERROR: Arbitrary instance, logic bug"
instance Arbitrary Meta where
arbitrary
= do (x1 :: Inlines) <- arbitrary
(x2 :: [Inlines]) <- filter (not . null) <$> arbitrary
(x3 :: Inlines) <- arbitrary
return $ setMeta "title" x1
$ setMeta "author" x2
$ setMeta "date" x3
nullMeta
instance Arbitrary Alignment where
arbitrary
= do x <- choose (0 :: Int, 3)
case x of
0 -> return AlignLeft
1 -> return AlignRight
2 -> return AlignCenter
3 -> return AlignDefault
_ -> error "FATAL ERROR: Arbitrary instance, logic bug"
instance Arbitrary ListNumberStyle where
arbitrary
= do x <- choose (0 :: Int, 6)
case x of
0 -> return DefaultStyle
1 -> return Example
2 -> return Decimal
3 -> return LowerRoman
4 -> return UpperRoman
5 -> return LowerAlpha
6 -> return UpperAlpha
_ -> error "FATAL ERROR: Arbitrary instance, logic bug"
instance Arbitrary ListNumberDelim where
arbitrary
= do x <- choose (0 :: Int, 3)
case x of
0 -> return DefaultDelim
1 -> return Period
2 -> return OneParen
3 -> return TwoParens
_ -> error "FATAL ERROR: Arbitrary instance, logic bug"