aboutsummaryrefslogtreecommitdiff
path: root/stdlib/source/lux/abstract/equivalence.lux
blob: eacb4a48f9c699f973e6120f68eb4b1b9e6297b7 (plain)
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
(.module:
  [lux #*]
  [//
   [functor (#+ Contravariant)]])

(signature: #export (Equivalence a)
  {#.doc "Equivalence for a type's instances."}
  (: (-> a a Bit)
     =))

(def: #export (product left right)
  (All [l r] (-> (Equivalence l) (Equivalence r) (Equivalence [l r])))
  (structure
   (def: (= [a b] [x y])
     (and (:: left = a x)
          (:: right = b y)))))

(def: #export (sum left right)
  (All [l r] (-> (Equivalence l) (Equivalence r) (Equivalence (| l r))))
  (structure
   (def: (= reference sample)
     (case [reference sample]
       [(#.Left reference) (#.Left sample)]
       (:: left = reference sample)

       [(#.Right reference) (#.Right sample)]
       (:: right = reference sample)

       _
       false))))

(def: #export (rec sub)
  (All [a] (-> (-> (Equivalence a) (Equivalence a)) (Equivalence a)))
  (structure
   (def: (= left right)
     (sub (rec sub) left right))))

(structure: #export contravariant
  (Contravariant Equivalence)
  
  (def: (map-1 f equivalence)
    (structure
     (def: (= reference sample)
       (:: equivalence = (f reference) (f sample))))))