(.module: [library [lux "*" [abstract [monad {"+" [Monad do]}]] [control ["." maybe] ["<>" parser ["<.>" type]]] [data ["." product] ["." bit] ["." text ("#\." monoid) ["%" format {"+" [format]}]] [collection ["." list ("#\." monad)] ["." row] ["." array] ["." queue] ["." set] ["." dictionary {"+" [Dictionary]}] ["." tree]]] [macro [syntax {"+" [syntax:]}] ["." code]] [math [number ["." nat ("#\." decimal)] ["." int] ["." rev] ["." frac]]] [time ["." duration] ["." date] ["." instant] ["." day] ["." month]] ["." type ["." poly {"+" [poly:]}] ["." unit]]]] [\\library ["." /]]) (poly: .public equivalence (`` (do [! <>.monad] [.let [g!_ (code.local_identifier "_____________")] *env* .env inputT .next .let [@Equivalence (: (-> Type Code) (function (_ type) (` ((~! /.Equivalence) (~ (poly.code *env* type))))))]] ($_ <>.either ... Basic types (~~ (template [ ] [(do ! [_ ] (in (` (: (~ (@Equivalence inputT)) ))))] [(.exactly Any) (function ((~ g!_) (~ g!_) (~ g!_)) #1)] [(.sub Bit) (~! bit.equivalence)] [(.sub Nat) (~! nat.equivalence)] [(.sub Int) (~! int.equivalence)] [(.sub Rev) (~! rev.equivalence)] [(.sub Frac) (~! frac.equivalence)] [(.sub Text) (~! text.equivalence)])) ... Composite types (~~ (template [ ] [(do ! [[_ argC] (.applied (<>.and (.exactly ) equivalence))] (in (` (: (~ (@Equivalence inputT)) ( (~ argC))))))] [.Maybe (~! maybe.equivalence)] [.List (~! list.equivalence)] [row.Row (~! row.equivalence)] [array.Array (~! array.equivalence)] [queue.Queue (~! queue.equivalence)] [set.Set (~! set.equivalence)] [tree.Tree (~! tree.equivalence)] )) (do ! [[_ _ valC] (.applied ($_ <>.and (.exactly dictionary.Dictionary) .any equivalence))] (in (` (: (~ (@Equivalence inputT)) ((~! dictionary.equivalence) (~ valC)))))) ... Models (~~ (template [ ] [(do ! [_ (.exactly )] (in (` (: (~ (@Equivalence inputT)) ))))] [duration.Duration duration.equivalence] [instant.Instant instant.equivalence] [date.Date date.equivalence] [day.Day day.equivalence] [month.Month month.equivalence] )) (do ! [_ (.applied (<>.and (.exactly unit.Qty) .any))] (in (` (: (~ (@Equivalence inputT)) unit.equivalence)))) ... Variants (do ! [members (.variant (<>.many equivalence)) .let [last (-- (list.size members)) g!_ (code.local_identifier "_____________") g!left (code.local_identifier "_____________left") g!right (code.local_identifier "_____________right")]] (in (` (: (~ (@Equivalence inputT)) (function ((~ g!_) (~ g!left) (~ g!right)) (case [(~ g!left) (~ g!right)] (~+ (list\conjoint (list\each (function (_ [tag g!eq]) (if (nat.= last tag) (list (` [((~ (code.nat (-- tag))) #1 (~ g!left)) ((~ (code.nat (-- tag))) #1 (~ g!right))]) (` ((~ g!eq) (~ g!left) (~ g!right)))) (list (` [((~ (code.nat tag)) #0 (~ g!left)) ((~ (code.nat tag)) #0 (~ g!right))]) (` ((~ g!eq) (~ g!left) (~ g!right)))))) (list.enumeration members)))) (~ g!_) #0)))))) ... Tuples (do ! [g!eqs (.tuple (<>.many equivalence)) .let [g!_ (code.local_identifier "_____________") indices (list.indices (list.size g!eqs)) g!lefts (list\each (|>> nat\encoded (text\composite "left") code.local_identifier) indices) g!rights (list\each (|>> nat\encoded (text\composite "right") code.local_identifier) indices)]] (in (` (: (~ (@Equivalence inputT)) (function ((~ g!_) [(~+ g!lefts)] [(~+ g!rights)]) (and (~+ (|> (list.zipped/3 g!eqs g!lefts g!rights) (list\each (function (_ [g!eq g!left g!right]) (` ((~ g!eq) (~ g!left) (~ g!right))))))))))))) ... Type recursion (do ! [[g!self bodyC] (.recursive equivalence) .let [g!_ (code.local_identifier "_____________")]] (in (` (: (~ (@Equivalence inputT)) ((~! /.rec) (.function ((~ g!_) (~ g!self)) (~ bodyC))))))) .recursive_self ... Type applications (do ! [[funcC argsC] (.applied (<>.and equivalence (<>.many equivalence)))] (in (` ((~ funcC) (~+ argsC))))) ... Parameters .parameter ... Polymorphism (do ! [[funcC varsC bodyC] (.polymorphic equivalence)] (in (` (: (All ((~ g!_) (~+ varsC)) (-> (~+ (list\each (|>> (~) ((~! /.Equivalence)) (`)) varsC)) ((~! /.Equivalence) ((~ (poly.code *env* inputT)) (~+ varsC))))) (function ((~ funcC) (~+ varsC)) (~ bodyC)))))) .recursive_call ... If all else fails... (|> .any (\ ! each (|>> %.type (format "Cannot create Equivalence for: ") <>.failure)) (\ ! conjoint)) ))))