1 {-# LANGUAGE OverloadedLists #-}
2 {-# LANGUAGE TypeFamilies #-}
3 {-# OPTIONS_GHC -fno-warn-orphans #-}
7 import Test.Tasty.HUnit
9 import Control.Arrow (second)
10 import Data.Hashable (Hashable)
11 import Data.Ratio ((%))
12 import Data.Tree (Tree(..))
13 import GHC.Exts (IsList(..))
15 import qualified Data.HashMap.Strict as HM
17 import Majority.Judgment
23 [ testGroup "MajorityValue" $
25 (majorityValue $ Merit [(3,15), (2,7), (1,3), (0::Int,2)])
26 (majorityValue $ Merit [(3,16), (2,6), (1,2), (0,3)])
28 [ let m = mkMerit ['A'..'F'] in
29 testMajorityValueOfMerits
30 [ (The, m [136,307,251,148,84,74])
33 [ Middle ( 57 % 1) 'C' 'C'
34 , Middle (137 % 1) 'B' 'C'
35 , Middle (148 % 1) 'B' 'D'
36 , Middle ( 22 % 1) 'B' 'E'
37 , Middle ( 62 % 1) 'A' 'E'
38 , Middle ( 74 % 1) 'A' 'F'
41 , let m = mkMerit [ToReject .. TooGood] in
42 testMajorityValueOfMerits
43 [ (This, m [12,10,21,5,5,5,2])
44 , (That, m [12,16,22,3,3,3,1])
46 [ (This, [ Middle (8 % 1) Acceptable Acceptable
47 , Middle (5 % 1) Insufficient Acceptable
48 , Middle (5 % 1) Insufficient Good
49 , Middle (5 % 1) ToReject VeryGood
50 , Middle (5 % 1) ToReject Perfect
51 , Middle (2 % 1) ToReject TooGood
53 , (That, [ Middle ( 2 % 1) Acceptable Acceptable
54 , Middle (16 % 1) Insufficient Acceptable
55 , Middle ( 2 % 1) ToReject Acceptable
56 , Middle ( 3 % 1) ToReject Good
57 , Middle ( 3 % 1) ToReject VeryGood
58 , Middle ( 3 % 1) ToReject Perfect
59 , Middle ( 1 % 1) ToReject TooGood
63 , testGroup "MajorityRanking"
64 [ testMajorityValueOfOpinions
65 [ (The, [No,No,No,No,Yes,Yes]) ]
66 [ (The, [ Middle (1 % 1) No No
67 , Middle (2 % 1) No Yes
69 , testMajorityValueOfOpinions
70 [ (The, [No,No,No,Yes,Yes,Yes]) ]
71 [ (The, [ Middle (3 % 1) No Yes ]) ]
72 , testMajorityValueOfOpinions
73 [ (The, [No,No,No,No,Yes,Yes,Yes]) ]
74 [ (The, [ Middle (1 % 2) No No
75 , Middle (3 % 1) No Yes ]) ]
76 , testMajorityValueOfOpinions
77 [ (This, [No,No,No,No,Yes,Yes])
78 , (That, [No,Yes,Yes,Yes,Yes,Yes])
80 [ (This, [ Middle (1 % 1) No No
81 , Middle (2 % 1) No Yes
83 , (That, [ Middle (2 % 1) Yes Yes
84 , Middle (1 % 1) No Yes
87 , testMajorityValueOfOpinions
88 [ (This, [No,No,No,No,No,No])
89 , (That, [No,No,No,Yes,Yes,Yes])
91 [ (This, [Middle (3 % 1) No No])
92 , (That, [Middle (3 % 1) No Yes])
94 , testMajorityValueOfOpinions
95 [ (This, [Yes,Yes,Yes,Yes,Yes,Yes])
96 , (That, [No,No,No,Yes,Yes,Yes])
98 [ (This, [Middle (3 % 1) Yes Yes])
99 , (That, [Middle (3 % 1) No Yes])
101 , testMajorityValueOfOpinions
102 [ (This, [No,No,Yes,Yes,Yes,Yes])
103 , (That, [No,No,No,Yes,Yes,Yes])
105 [ (This, [ Middle (1 % 1) Yes Yes
106 , Middle (2 % 1) No Yes
108 , (That, [ Middle (3 % 1) No Yes ])
110 , testMajorityValueOfOpinions
111 [ (1::Int, [Perfect,Perfect,VeryGood,Perfect,Perfect,Perfect])
112 , (2, [Perfect,VeryGood,VeryGood,VeryGood,Good,VeryGood])
113 , (3, [Acceptable,Perfect,Good,VeryGood,VeryGood,Perfect])
114 , (4, [VeryGood,Good,Acceptable,Good,Good,Good])
115 , (5, [Good,Acceptable,VeryGood,Good,Good,Good])
116 , (6, [VeryGood,Acceptable,Insufficient,Acceptable,Acceptable,Good])
118 [ (1, [ Middle (2 % 1) Perfect Perfect
119 , Middle (1 % 1) VeryGood Perfect
121 , (2, [ Middle (2 % 1) VeryGood VeryGood
122 , Middle (1 % 1) Good Perfect
124 , (3, [ Middle (1 % 1) VeryGood VeryGood
125 , Middle (1 % 1) Good Perfect
126 , Middle (1 % 1) Acceptable Perfect
128 , (4, [ Middle (2 % 1) Good Good
129 , Middle (1 % 1) Acceptable VeryGood
131 , (5, [ Middle (2 % 1) Good Good
132 , Middle (1 % 1) Acceptable VeryGood
134 , (6, [ Middle (1 % 1) Acceptable Acceptable
135 , Middle (1 % 1) Acceptable Good
136 , Middle (1 % 1) Insufficient VeryGood
140 , testGroup "Section"
141 [ testSection "0 judge"
146 , testSection "1 judge, default grade"
150 (Right $ node0 [(This, [(1,[(ToReject,1%1)])])])
151 , testSection "1 judge, default grade, 2 choices"
155 (Right $ node0 [ (This, [(1,[(ToReject,1%1)])])
156 , (That, [(1,[(ToReject,1%1)])])
158 , testSection "1 judge, default grade"
161 (node0 [(This,[(1,Section Nothing Nothing)])])
162 (Right $ node0 [(This,[(1,[(ToReject,1%1)])])])
163 , testSection "2 judges, default grade"
165 [(1::Int,ToReject), (2::Int,ToReject)]
167 [ (This, [(1,Section Nothing Nothing)])
170 [ (This, [ (1,[(ToReject,1%1)])
171 , (2,[(ToReject,1%1)])
174 , testSection "ErrorSection_unknown_choices"
178 (Left $ ErrorSection_unknown_choices [This])
179 , testSection "ErrorSection_unknown_choices"
182 (node0 [(This,[(2,Section Nothing Nothing)])])
183 (Left $ ErrorSection_unknown_choices [This])
184 , testSection "ErrorSection_unknown_choices"
187 (node0 [ (This,[(1,Section Nothing Nothing)])
188 , (That,[(2,Section Nothing Nothing)])
190 (Left $ ErrorSection_unknown_choices [That])
191 , testSection "ErrorSection_unknown_judges"
194 (node0 [(This,[(2,Section Nothing Nothing)])])
195 (Left $ ErrorSection_unknown_judges [(This,[2])])
196 , testSection "1 judge, 1 grade"
199 (node0 [(This,[(1,Section Nothing (Just Acceptable))])])
200 (Right $ node0 [(This,[(1,[(Acceptable,1%1)])])])
201 , testSection "1 judge, 1 grade, 2 sections"
205 [ (This, [(1,Section Nothing (Just Acceptable))]) ]
206 [ node0 [(This, [(1,Section (Just $ 1%2) Nothing)])]
207 , node0 [(This, [(1,Section (Just $ 1%2) Nothing)])]
210 [ (This, [(1,[(Acceptable,1%1)])]) ]
211 [ node0 [(This, [(1,[(Acceptable,1%1)])])]
212 , node0 [(This, [(1,[(Acceptable,1%1)])])]
214 , testSection "sectionNodeShare with judge"
216 [(1::Int,ToReject), (2,Insufficient)]
218 [(This, [(1,Section Nothing (Just Acceptable))])]
219 [ node0 $ SectionNode (Just $ 1%3) [(This, [ (1,Section (Just $ 1%2) Nothing)
220 , (2,Section Nothing Nothing)
222 , node0 [(This, [ (1,Section (Just $ 1%2) Nothing)
223 , (2,Section Nothing (Just Good))
227 [ (This, [ (1,[(Acceptable,1%2 + 1%2)])
228 , (2,[(Insufficient,1%3), (Good,2%3)])
230 [ node0 [(This, [ (1,[(Acceptable,1%1)])
231 , (2,[(Insufficient,1%1)])
233 , node0 [(This, [ (1,[(Acceptable,1%1)])
237 , testSection "sectionNodeShare without judge"
239 [(1::Int,ToReject), (2,Insufficient)]
241 [(This, [(1,Section Nothing (Just Acceptable))])]
242 [ node0 $ SectionNode (Just $ 1%3) [(This, [ (1,Section (Just $ 1%2) Nothing) ])]
243 , node0 [(This, [ (1,Section (Just $ 1%2) Nothing)
244 , (2,Section Nothing (Just Good))
248 [ (This, [ (1,[(Acceptable,1%2 + 1%2)])
249 , (2,[(Insufficient,1%3), (Good,2%3)])
251 [ node0 [(This, [ (1,[(Acceptable,1%1)])
252 , (2,[(Insufficient,1%1)])
254 , node0 [(This, [ (1,[(Acceptable,1%1)])
258 , testSection "1 judge, 2 grades, 2 sections"
262 [ (This, [(1,Section Nothing (Just Acceptable))]) ]
263 [ node0 [(This, [(1,Section (Just $ 1%2) Nothing)])]
264 , node0 [(This, [(1,Section (Just $ 1%2) (Just Good))])]
267 [(This, [(1,[(Acceptable,1%2), (Good,1%2)])])]
268 [ node0 [(This, [(1,[(Acceptable,1%1)])])]
269 , node0 [(This, [(1,[(Good,1%1)])])]
271 , testSection "1 judge, 2 grades, 2 sections (1 default)"
275 [ (This, [(1,Section Nothing (Just Acceptable))]) ]
276 [ node0 [(This, [(1,Section Nothing Nothing)])]
277 , node0 [(This, [(1,Section (Just $ 1%2) (Just Good))])]
280 [(This, [(1,[(Acceptable,1%2), (Good,1%2)])])]
281 [ node0 [(This, [(1,[(Acceptable,1%1)])])]
282 , node0 [(This, [(1,[(Good,1%1)])])]
284 , testSection "1 judge, 3 grades, 3 sections (2 default)"
288 [ (This, [(1,Section Nothing (Just Acceptable))]) ]
289 [ node0 [(This, [(1,Section Nothing Nothing)])]
290 , node0 [(This, [(1,Section (Just $ 1%2) (Just Good))])]
291 , node0 [(This, [(1,Section Nothing (Just VeryGood))])]
294 [(This, [(1,[(Acceptable,1%4), (Good,1%2), (VeryGood,1%4)])])]
295 [ node0 [(This, [(1,[(Acceptable,1%1)])])]
296 , node0 [(This, [(1,[(Good,1%1)])])]
297 , node0 [(This, [(1,[(VeryGood,1%1)])])]
299 , testSection "ErrorSection_invalid_shares sum not 1"
303 [ (This, [(1,Section Nothing (Just Acceptable))]) ]
304 [ node0 [(This, [(1,Section (Just $ 1%2) Nothing)])]
305 , node0 [(This, [(1,Section (Just $ 1%3) (Just Good))])]
307 (Left $ ErrorSection_invalid_shares [(This, [(1,[1%2,1%3])])])
308 , testSection "ErrorSection_invalid_shares negative share"
312 [ (This, [(1,Section Nothing (Just Acceptable))]) ]
313 [ node0 [(This, [(1,Section (Just $ 1%2) Nothing)])]
314 , node0 [(This, [(1,Section (Just $ -1%2) (Just Good))])]
316 (Left $ ErrorSection_invalid_shares [(This, [(1,[1%2,-1%2])])])
317 , testSection "2 judges, 3 grade, 3 sections (1 default)"
319 [(1::Int,ToReject), (2::Int,ToReject)]
321 [ (This, [(1,Section Nothing (Just Acceptable))])
324 [ (This, [(1,Section Nothing Nothing)])
327 [ (This, [(1,Section (Just $ 1%2) (Just Good))])
331 [ (This, [ (1,[(Acceptable,1%2), (Good,1%2)])
332 , (2,[(ToReject,1%1)])
336 [ (This, [ (1,[(Acceptable,1%1)])
337 , (2,[(ToReject,1%1)])
341 [ (This, [ (1,[(Good,1%1)])
342 , (2,[(ToReject,1%1)])
346 , testSection "2 judges, 4 grades, 5 sections (2 defaults)"
348 [(1::Int,ToReject), (2::Int,ToReject)]
350 [ (This, [(1,Section Nothing (Just Acceptable))])
353 [ (This, [(1,Section Nothing Nothing)])
356 [ (This, [(1,Section (Just $ 1%2) (Just Good))])
359 [ (This, [(1,Section Nothing (Just Good))])
362 [ (This, [ (1,Section Nothing (Just VeryGood))
363 , (2,Section Nothing (Just Insufficient))
369 [ (This, [ (1,[(Acceptable,1%4), (Good,1%2), (VeryGood,1%4)])
370 , (2,[(ToReject,2%3), (Insufficient,1%3)])
374 [ (This, [ (1,[(Acceptable,1%1)])
375 , (2,[(ToReject,1%1)])
379 [ (This, [ (1,[(Good,1%1)])
380 , (2,[(ToReject,1%1)])
384 [ (This, [ (1,[(VeryGood,1%1)])
385 , (2,[(Insufficient,1%1)])
389 [ (This, [ (1,[(VeryGood,1%1)])
390 , (2,[(Insufficient,1%1)])
395 , testSection "1 judge, default grade, 2 choices"
399 (Right $ node0 [ (This,[(1,[(ToReject,1%1)])])
400 , (That,[(1,[(ToReject,1%1)])])
402 , testSection "2 judges, 2 choices"
404 [(1::Int,ToReject), (2::Int,ToReject)]
408 [ (This, [(1,Section Nothing (Just Good))])
409 , (That, [(2,Section Nothing (Just Insufficient))])
412 [ (This, [(1,Section Nothing (Just Acceptable))])
413 , (That, [(2,Section Nothing (Just VeryGood))])
417 [ (This, [ (1,[(Good,1%2), (Acceptable,1%2)])
418 , (2,[(ToReject,1%1)])
420 , (That, [ (1,[(ToReject,1%1)])
421 , (2,[(Insufficient,1%2), (VeryGood,1%2)])
424 [ node0 [ (This, [ (1,[(Good,1%1)])
425 , (2,[(ToReject,1%1)])
427 , (That, [ (1,[(ToReject,1%1)])
428 , (2,[(Insufficient,1%1)])
431 , node0 [ (This, [ (1,[(Acceptable,1%1)])
432 , (2,[(ToReject,1%1)])
434 , (That, [ (1,[(ToReject,1%1)])
435 , (2,[(VeryGood,1%1)])
439 , testSection "1 judge, 1 choice"
443 [ node0 [ (This, [(1,Section (Just $ 1%8) (Just Acceptable))])
445 , node0 [ (This, [(1,Section Nothing Nothing)])
449 [ (This, [(1,[(Acceptable,1%8), (ToReject,7%8)])])
451 [ node0 [ (This, [(1,[(Acceptable, 1%1)])])
453 , node0 [ (This, [(1,[(ToReject, 1%1)])])
456 , testSection "1 judge, 1 choice (missing judge)"
460 [ node0 [ (This, [(1,Section (Just $ 1%8) (Just Acceptable))])
466 [ (This, [(1,[(Acceptable,1%8), (ToReject,7%8)])])
468 [ node0 [ (This, [(1,[(Acceptable, 1%1)])])
470 , node0 [ (This, [(1,[(ToReject, 1%1)])])
473 , testSection "1 judge, 1 choice (missing judge)"
479 , node0 [ (This, [(1,Section (Just $ 1%8) (Just Acceptable))])
483 [ (This, [(1,[(Acceptable,1%8), (ToReject,7%8)])])
485 [ node0 [ (This, [(1,[(ToReject, 1%1)])])
487 , node0 [ (This, [(1,[(Acceptable, 1%1)])])
490 , testSection "1 judge, 1 choice (missing choice)"
494 [ node0 [ (This, [(1,Section (Just $ 1%8) (Just Acceptable))])
500 [ (This, [(1,[(Acceptable,1%8), (ToReject,7%8)])])
502 [ node0 [ (This, [(1,[(Acceptable, 1%1)])])
504 , node0 [ (This, [(1,[(ToReject, 1%1)])])
507 , testSection "1 judge, 1 choice (missing choice)"
513 , node0 [ (This, [(1,Section (Just $ 1%8) (Just Acceptable))])
517 [ (This, [(1,[(Acceptable,1%8), (ToReject,7%8)])])
519 [ node0 [ (This, [(1,[(ToReject, 1%1)])])
521 , node0 [ (This, [(1,[(Acceptable, 1%1)])])
524 , testSection "2 judges, 2 choices"
526 [(1::Int,ToReject), (2::Int,ToReject)]
528 [ (This, [(1,Section Nothing (Just Acceptable))])
529 , (That, [(2,Section (Just $ 1%8) (Just VeryGood))])
532 [ (This, [ (1,[(Acceptable,1%1)])
533 , (2,[(ToReject,1%1)])
535 , (That, [ (1,[(ToReject,1%1)])
536 , (2,[(VeryGood,1%1)])
539 , testSection "2 judges, 2 choices"
541 [(1::Int,ToReject), (2::Int,ToReject)]
545 [ (This, [(1,Section Nothing (Just Good))])
546 , (That, [(2,Section Nothing (Just Insufficient))])
549 [ (This, [(1,Section Nothing (Just Acceptable))])
550 , (That, [(2,Section (Just $ 1%8) (Just VeryGood))])
554 [ (This, [ (1,[(Good,1%2), (Acceptable,1%2)])
555 , (2,[(ToReject,1%1)])
557 , (That, [ (1,[(ToReject,1%1)])
558 , (2,[(Insufficient,7%8), (VeryGood,1%8)])
561 [ node0 [ (This, [ (1,[(Good,1%1)])
562 , (2,[(ToReject,1%1)])
564 , (That, [ (1,[(ToReject,1%1)])
565 , (2,[(Insufficient,1%1)])
568 , node0 [ (This, [ (1,[(Acceptable,1%1)])
569 , (2,[(ToReject,1%1)])
571 , (That, [ (1,[(ToReject,1%1)])
572 , (2,[(VeryGood,1%1)])
576 , testSection "2 judges, 2 choices"
578 [(1::Int,ToReject), (2::Int,ToReject)]
579 (Node [ (This, [(1,Section Nothing (Just Acceptable))])
581 [ node0 [ (This, [(1,Section Nothing Nothing)])
583 , node0 [ (This, [ (1,Section (Just $ 1%2) (Just Good)) ])
584 , (That, [ (1,Section (Just $ 1%3) Nothing)
585 , (2,Section (Just $ 1%5) (Just Insufficient))
588 , Node [ (This, [(1,Section Nothing (Just Good))])
589 , (That, [(2,Section Nothing (Just VeryGood))])
591 [ node0 [ (This, [ (1,Section Nothing (Just VeryGood))
592 , (2,Section Nothing (Just Insufficient))
594 , (That, [ (1,Section Nothing (Just Acceptable)) ])
596 , node0 [ (This, [ (1,Section Nothing (Just Acceptable))
598 , (That, [ (1,Section Nothing (Just VeryGood))
599 , (2,Section Nothing (Just Good))
605 Node [ (This, [ (1,[(Acceptable,1%4 + 1%8), (Good,1%2), (VeryGood,1%8)])
606 , (2,[(ToReject,1%3 + 1%3 + 1%6), (Insufficient,1%6)])
608 , (That, [ (1,[(ToReject,1%3 + 1%3), (Acceptable,1%6), (VeryGood,1%6)])
609 , (2,[(ToReject,4%10), (Insufficient,1%5), (VeryGood,4%20), (Good,4%20)])
612 [ node0 [ (This, [ (1,[(Acceptable,1%1)]) -- 1%4
613 , (2,[(ToReject,1%1)]) -- 1%3
615 , (That, [ (1,[(ToReject,1%1)]) -- 1%3
616 , (2,[(ToReject,1%1)]) -- 4%10
619 , node0 [ (This, [ (1,[(Good,1%1)]) -- 1%2
620 , (2,[(ToReject,1%1)]) -- 1%3
622 , (That, [ (1,[(ToReject,1%1)]) -- 1%3
623 , (2,[(Insufficient,1%1)]) -- 1%5
626 , Node [ (This, [ (1,[(VeryGood,1%2), (Acceptable,1%2)]) -- 1%4
627 , (2,[(Insufficient,1%2), (ToReject,1%2)]) -- 1%3
629 , (That, [ (1,[(Acceptable,1%2), (VeryGood,1%2)]) -- 1%3
630 , (2,[(VeryGood,1%2), (Good,1%2)]) -- 4%10
633 [ node0 [ (This, [ (1,[(VeryGood,1%1)])
634 , (2,[(Insufficient,1%1)])
636 , (That, [ (1,[(Acceptable,1%1)])
637 , (2,[(VeryGood,1%1)])
640 , node0 [ (This, [ (1,[(Acceptable,1%1)])
641 , (2,[(ToReject,1%1)])
643 , (That, [ (1,[(VeryGood,1%1)])
654 elide :: String -> String
655 elide s | length s > 42 = take 42 s ++ ['…']
658 mkMerit :: (Ord grade, Show grade) => [grade] -> [Share] -> Merit grade
659 mkMerit gs = fromList . (gs`zip`)
662 (Eq choice, Hashable choice, Ord grade) =>
663 [(choice,[grade])] ->
664 MeritByChoice choice grade
666 meritByChoice $ fromList $
667 second (fromList . zip [1::Int ..] . (singleGrade <$>)) <$> os
669 testCompareValue :: (Ord grade, Show grade) =>
670 MajorityValue grade -> MajorityValue grade -> TestTree
671 testCompareValue x y =
672 testGroup (elide $ show (unMajorityValue x, unMajorityValue y))
673 [ testCase "x == x" $ x`compare`x @?= EQ
674 , testCase "y == y" $ y`compare`y @?= EQ
675 , testCase "x < y" $ x`compare`y @?= LT
676 , testCase "y > x" $ y`compare`x @?= GT
679 testMajorityRanking ::
680 (Eq choice, Hashable choice, Ord grade, Show grade, Show choice) =>
681 [(choice, [grade])] ->
682 MajorityRanking choice grade -> TestTree
683 testMajorityRanking os expect =
684 testCase (elide $ show os) $
685 majorityRanking (mkMeritByChoice os) @?= expect
687 testMajorityValueOfOpinions ::
688 (Show grade, Show choice, Ord grade, Eq choice, Hashable choice) =>
689 [(choice, [grade])] ->
690 [(choice, [Middle grade])] -> TestTree
691 testMajorityValueOfOpinions os expect =
692 testCase (elide $ show os) $
693 majorityValueByChoice (mkMeritByChoice os)
694 @?= (MajorityValue<$>HM.fromList expect)
696 testMajorityValueOfMerits ::
697 (Show grade, Show choice, Ord grade, Eq choice, Hashable choice) =>
698 MeritByChoice choice grade ->
699 [(choice, [Middle grade])] -> TestTree
700 testMajorityValueOfMerits ms expect =
701 testCase (elide $ show $ unMeritByChoice ms) $
702 majorityValueByChoice ms
703 @?= (MajorityValue<$>HM.fromList expect)
716 Judges judge grade ->
717 Tree (SectionNode choice judge grade) ->
718 Either (ErrorSection choice judge grade)
719 (Tree (OpinionsByChoice choice judge grade)) ->
721 testSection msg cs js ss expect =
722 testCase (elide msg) $
723 opinionsBySection cs js ss @?= expect
728 instance (Eq choice, Hashable choice) => IsList (SectionNode choice judge grade) where
729 type Item (SectionNode choice judge grade) = (choice, SectionByJudge judge grade)
730 fromList = SectionNode Nothing . fromList
731 toList = GHC.Exts.toList . sectionByJudgeByChoice