Система кодирования неисправностей омега в финале tagless в idris

Поэтому я пытаюсь сделать систему омега в окончательном стиле без тегов. Я успешно закодировал систему f, как показано в следующем коде (также приведен некоторый пример)(также, пожалуйста, не обращайте внимания на тот факт, что логический вывод не работает и примеры очень уродливые, о них я спрошу в отдельном выпуске):

module F

%access public export

interface F (ty : Type) (val : ty -> Type) where
  double : ty
  mkDouble : Double -> val double
  plus : val double -> val double -> val double
  bool : ty
  mkBool : Bool -> val bool
  useBool : val bool -> val x -> val x -> val x
  forall : (ty -> ty) -> ty
  mkForall : (tf : ty -> ty) -> ((x:ty) -> val (tf x)) -> val (forall tf)
  useForall : val (forall tf) -> (x:ty) -> val (tf x)
  arr : ty -> ty -> ty
  mkArr : (val x -> val y) -> val (arr x y)
  useArr : val (arr x y) -> val x -> val y

[FID] F Type (\x => x) where
  double = Double
  mkDouble x = x
  plus x y = x + y
  bool = Bool
  mkBool x = x
  useBool x y z = if x then y else z
  forall f = (x:Type) -> f x
  mkForall tf f = f
  useForall f x = f x
  arr x y = x -> y
  mkArr f = f
  useArr f x = f x

fIdTy : {f : F ty val} -> ty
fIdTy {f=f} = forall @{f} $ \x => arr @{f} x x

fId : {f : F ty val} -> val (fIdTy {f=f})
fId {f=f} = mkForall @{f} (\x => arr @{f} x x) (\t => mkArr @{f} id)

fIdId : {f : F ty val} -> val (fIdTy {f=f})
fIdId {f=f} = useArr @{f} (useForall @{f} fId fIdTy) fId

useIdId : (x : Type) -> x -> x
useIdId = fIdId {f=FID}

compose : {f:F ty val} -> val (arr @{f} y z) -> val (arr @{f} x y) -> val (arr @{f} x z)
compose {f=f} yz xy = mkArr @{f} $ \x => useArr @{f} yz (useArr @{f} xy x)

composeFull : {f:F ty val} -> val (arr @{f} (arr @{f} y z) (arr @{f} (arr @{f} x y) (arr @{f} x z)))
composeFull {f=f} = mkArr @{f} $ \yz => mkArr @{f} $ \xy => compose yz xy

-- Since currying/uncurrying is only manual typing, as we shown how to convert from arr to ->, we will choose the most convient form freely.

Но у меня проблемы с выполнением F омега.

import F

%default total

-- look like the core calculus isnt very extensible.
-- neverthless, ftg allow us to reuse definition, so we will stick with it.
interface F (ty soty) val => FO (so:Type) (ty:so -> Type) (soty:so) (val:ty soty -> Type) where
  tyArr : so -> so -> so
  mkTyArr : (ty x -> ty y) -> ty (tyArr x y)
  useTyArr : ty (tyArr x y) -> ty x -> ty y
  forall : (s:so) -> (ty s -> ty soty) -> ty soty
  mkForall : (s:so) -> (tf:ty s -> ty soty) -> ((x:ty s) -> val (tf x)) -> val (forall s tf)
  useForall : val (forall s tf) -> (x:ty s) -> val (tf x)
  getF : F (ty soty) val

[FOID] FO Type (\x => x) Type (\x => x) using FID where
  tyArr x y = x -> y
  mkTyArr f = f
  useTyArr f x = f x
  forall s f = (x:s) -> f x
  mkForall s tf f = f
  useForall f x = f x
  getF = FID

-- note that we still cant has a single leb - it require type in type
leb : {fo : FO so ty soty val} -> ty s -> ty s -> ty soty
leb {fo=fo} {s=s} {soty=soty} x y = forall @{fo} (tyArr @{fo} s soty) (\f => arr @{getF @{fo}} (useTyArr @{fo} f x) (useTyArr @{fo} f y))

refl : {fo:FO so ty soty val} -> (x:ty s) -> val (leb {fo=fo} x x)
refl {s=s} {soty=soty} {fo=fo} x =
  mkForall @{fo}
    (tyArr @{fo} s soty)
    (\f => arr @{getF @{fo}} (useTyArr @{fo} f x) (useTyArr @{fo} f x))
    (\x => mkArr @{getF @{fo}} (\z => z))

trans : {fo:FO so ty soty val} -> (x:ty s) -> (y:ty s) -> (z:ty s) -> val (arr @{getF @{fo}} (leb {fo=fo} x y) (arr @{getF @{fo}} (leb {fo=fo} y z) (leb {fo=fo} x z)))
trans {s=s} {soty=soty} {fo=fo} tx ty tz =
  mkArr @{getF @{fo}} {x=leb {fo=fo} tx ty} $ \xy => mkArr @{getF @{fo}} {x=leb {fo=fo} ty tz} {y=leb {fo=fo} tx tz} $ \yz =>
    mkForall @{fo} _ _ $ \tf =>
      compose {f=getF @{fo}}
        (useForall @{fo} yz tf)
        (useForall @{fo} xy tf)

symm : {fo:FO so ty soty val} -> (tx:ty s) -> (ty:ty s) -> val (leb {fo=fo} tx ty) -> val (leb {fo=fo} ty tx)
symm {s=s} {soty=soty} {fo=fo} tx ty xy = ?z $
  useForall @{fo} {s=tyArr @{fo} s soty} {tf=\f => arr @{getF @{fo}} (useTyArr @{fo} f tx) (useTyArr @{fo} f ty)} xy
-- what is ?z ? the thing is, I cant write it out, because the current setup wont reduce mkTyArr and useTyArr.

Итак, мой вопрос, как это исправить? Кажется, что я могу добавить бета-равенство для типа лямбда явно (как Idris Eq), и пользователь может использовать его, чтобы переписать бета-сокращение.

Однако я хочу, чтобы это было сделано автоматически. Какой у меня вариант?

0 ответов

Другие вопросы по тегам