265 lines
5.8 KiB
Agda
265 lines
5.8 KiB
Agda
module Prelude
|
||
|
||
|
||
data Bool : U where
|
||
True False : Bool
|
||
|
||
not : Bool → Bool
|
||
not True = False
|
||
not False = True
|
||
|
||
-- In Idris, this is lazy in the second arg, we're not doing
|
||
-- magic laziness for now, it's messy
|
||
infixr 4 _||_
|
||
_||_ : Bool → Bool → Bool
|
||
True || _ = True
|
||
False || b = b
|
||
|
||
data Nat : U where
|
||
Z : Nat
|
||
S : Nat -> Nat
|
||
|
||
data Maybe : U -> U where
|
||
Just : {a : U} -> a -> Maybe a
|
||
Nothing : {a : U} -> Maybe a
|
||
|
||
fromMaybe : {a} → a → Maybe a → a
|
||
fromMaybe a Nothing = a
|
||
fromMaybe _ (Just a) = a
|
||
|
||
data Either : U -> U -> U where
|
||
Left : {a b : U} -> a -> Either a b
|
||
Right : {a b : U} -> b -> Either a b
|
||
|
||
infixr 7 _::_
|
||
data List : U -> U where
|
||
Nil : {A} → List A
|
||
_::_ : {A} → A → List A → List A
|
||
|
||
|
||
infixl 7 _:<_
|
||
data SnocList : U → U where
|
||
Lin : {A} → SnocList A
|
||
_:<_ : {A} → SnocList A → A → SnocList A
|
||
|
||
-- 'chips'
|
||
infixr 6 _<>>_
|
||
_<>>_ : {a} → SnocList a → List a → List a
|
||
Lin <>> ys = ys
|
||
(xs :< x) <>> ys = xs <>> x :: ys
|
||
|
||
-- TODO this is special cased in some languages, maybe for easier
|
||
-- inference? Figure out why.
|
||
-- Currently very noisy in generated code (if nothing else, optimize it out?)
|
||
infixr 0 _$_
|
||
_$_ : {a b : U} -> (a -> b) -> a -> b
|
||
f $ a = f a
|
||
|
||
infixr 8 _×_
|
||
infixr 2 _,_
|
||
data _×_ : U → U → U where
|
||
_,_ : {A B} → A → B → A × B
|
||
|
||
|
||
infixl 6 _<_
|
||
data Ord : U → U where
|
||
MkOrd : {A} → (A → A → Bool) → Ord A
|
||
|
||
_<_ : {A} {{Ord A}} → A → A → Bool
|
||
_<_ {{MkOrd cmp}} a b = cmp a b
|
||
|
||
cmpNat : Nat → Nat → Bool
|
||
cmpNat Z Z = True
|
||
cmpNat Z m = False
|
||
cmpNat n Z = True
|
||
cmpNat (S n) (S m) = True
|
||
|
||
OrdNat : Ord Nat
|
||
OrdNat = MkOrd cmpNat
|
||
|
||
-- Monad
|
||
|
||
-- TODO sugar for if then else (mixfix is too eager)
|
||
|
||
-- TODO stack with Applicative, etc?
|
||
|
||
data Monad : (U -> U) -> U where
|
||
MkMonad : { M : U -> U } ->
|
||
(bind : {A B : U} -> (M A) -> (A -> M B) -> M B) ->
|
||
(pure : {A : U} -> A -> M A) ->
|
||
Monad M
|
||
|
||
infixl 1 _>>=_ _>>_
|
||
_>>=_ : {a b : U} -> {m : U -> U} -> {{x : Monad m}} -> (m a) -> (a -> m b) -> m b
|
||
_>>=_ {a} {b} {m} {{MkMonad bind' _}} ma amb = bind' {a} {b} ma amb
|
||
|
||
_>>_ : {a b : U} -> {m : U -> U} -> {{x : Monad m}} -> m a -> m b -> m b
|
||
ma >> mb = mb
|
||
|
||
pure : {a : U} {m : U -> U} {{_ : Monad m}} -> a -> m a
|
||
pure {_} {_} {{MkMonad _ pure'}} a = pure' a
|
||
|
||
-- Equality
|
||
|
||
infixl 1 _≡_
|
||
data _≡_ : {A : U} -> A -> A -> U where
|
||
Refl : {A : U} -> {a : A} -> a ≡ a
|
||
|
||
replace : {A : U} {a b : A} -> (P : A -> U) -> a ≡ b -> P a -> P b
|
||
replace p Refl x = x
|
||
|
||
cong : {A B : U} {a b : A} -> (f : A -> B) -> a ≡ b -> f a ≡ f b
|
||
|
||
sym : {A : U} -> {a b : A} -> a ≡ b -> b ≡ a
|
||
sym Refl = Refl
|
||
|
||
-- Functor
|
||
data Functor : (U → U) → U where
|
||
MkFunctor : {m : U → U} → ({a b : U} → (a → b) → m a → m b) → Functor m
|
||
|
||
map : {m} {{Functor m}} → {a b} → (a → b) → m a → m b
|
||
map {{MkFunctor f}} ma = f ma
|
||
|
||
infixr 4 _<$>_
|
||
_<$>_ : {f : U → U} {{Functor f}} {a b} → (a → b) → f a → f b
|
||
f <$> ma = map f ma
|
||
|
||
|
||
mapMaybe : {a b} → (a → b) → Maybe a → Maybe b
|
||
mapMaybe f Nothing = Nothing
|
||
mapMaybe f (Just a) = Just (f a)
|
||
|
||
FunctorMaybe : Functor Maybe
|
||
FunctorMaybe = MkFunctor mapMaybe
|
||
|
||
|
||
|
||
-- Idris is lazy in second arg, we don't have that.
|
||
data Alternative : (U → U) → U where
|
||
MkAlt : {m : U → U} →
|
||
({a} → m a → m a → m a) →
|
||
Alternative m
|
||
|
||
infixr 2 _<|>_
|
||
_<|>_ : {m : U → U} {{Alternative m}} → {a} → m a → m a → m a
|
||
_<|>_ {m} {{MkAlt f}} {a} x y = f x y
|
||
|
||
altMaybe : {a} → Maybe a → Maybe a → Maybe a
|
||
altMaybe Nothing x = x
|
||
altMaybe (Just x) _ = Just x
|
||
|
||
AltMaybe : Alternative Maybe
|
||
AltMaybe = MkAlt altMaybe
|
||
|
||
-- Semigroup
|
||
|
||
infixl 8 _<+>_
|
||
data Semigroup : U → U where
|
||
MkSemi : {a} → (a → a → a) → Semigroup a
|
||
|
||
_<+>_ : {a} {{Semigroup a}} → a → a → a
|
||
_<+>_ {{MkSemi op}} x y = op x y
|
||
|
||
|
||
infixl 7 _+_
|
||
data Add : U → U where
|
||
MkAdd : {A} → (A → A → A) → Add A
|
||
|
||
_+_ : {A} {{Add A}} → A → A → A
|
||
_+_ {{MkAdd add}} x y = add x y
|
||
|
||
infixl 8 _*_
|
||
data Mul : U → U where
|
||
MkMul : {A} →
|
||
(A → A → A) →
|
||
Mul A
|
||
|
||
_*_ : {A} {{Mul A}} → A → A → A
|
||
_*_ {{MkMul mul}} x y = mul x y
|
||
|
||
|
||
-- TODO codata/copatterns might be nice here?
|
||
-- AddNat : AddNat
|
||
-- AddNat .add Z m = m
|
||
-- AddNat .add (S n) m = S (self .add n m)
|
||
|
||
addNat : Nat → Nat → Nat
|
||
addNat Z m = m
|
||
addNat (S n) m = S (addNat n m)
|
||
|
||
AddNat : Add Nat
|
||
AddNat = MkAdd addNat
|
||
|
||
mulNat : Nat → Nat → Nat
|
||
mulNat Z _ = Z
|
||
mulNat (S n) m = m + mulNat n m
|
||
|
||
MulNat : Mul Nat
|
||
MulNat = MkMul mulNat
|
||
|
||
|
||
-- TODO Sub
|
||
infixl 7 _-_
|
||
_-_ : Nat -> Nat -> Nat
|
||
Z - m = Z
|
||
n - Z = n
|
||
S n - S m = n - m
|
||
|
||
|
||
ptype String
|
||
ptype Int
|
||
ptype Char
|
||
|
||
-- probably want to switch to Int or implement magic Nat
|
||
pfunc length : String → Nat := "(s) => {
|
||
let rval = Z
|
||
for (let i = 0; i < s.length; s++) rval = S(rval)
|
||
return rval
|
||
}"
|
||
|
||
data Unit : U where
|
||
MkUnit : Unit
|
||
|
||
ptype Array : U → U
|
||
pfunc listToArray : {a : U} -> List a -> Array a := "
|
||
(a, l) => {
|
||
let rval = []
|
||
while (l.tag !== 'Nil') {
|
||
rval.push(l.h1)
|
||
l = l.h2
|
||
}
|
||
return rval
|
||
}
|
||
"
|
||
pfunc alen : {a : U} -> Array a -> Int := "(a,arr) => arr.length"
|
||
pfunc aget : {a : U} -> Array a -> Int -> a := "(a, arr, ix) => arr[ix]"
|
||
pfunc aempty : {a : U} -> Unit -> Array a := "() => []"
|
||
|
||
|
||
pfunc fastConcat : List String → String := "(xs) => listToArray(undefined, xs).join('')"
|
||
pfunc replicate : Nat -> Char → String := "() => abort('FIXME replicate')"
|
||
|
||
|
||
|
||
ptype World
|
||
|
||
data IORes : U -> U where
|
||
MkIORes : {a : U} -> a -> World -> IORes a
|
||
|
||
IO : U -> U
|
||
IO a = World -> IORes a
|
||
|
||
-- TODO - if I move w to the left, I get "extra pattern variable"
|
||
-- because I'm not looking instide the IO b type, probably should force it.
|
||
iobind : {a b : U} -> IO a -> (a -> IO b) -> IO b
|
||
iobind ma mab = \ w => case ma w of
|
||
(MkIORes a w) => mab a w
|
||
|
||
iopure : {a : U} -> a -> IO a
|
||
iopure a = \ w => MkIORes a w
|
||
|
||
IOMonad : Monad IO
|
||
IOMonad = MkMonad iobind iopure
|
||
|
||
pfunc putStrLn : String -> IO Unit := "(s) => (w) => console.log(s)"
|