Improve proof readability
parent
323b87189c
commit
0612e320b0
138
proof.py
138
proof.py
|
|
@ -15,6 +15,38 @@ class App(Term):
|
||||||
f : Term
|
f : Term
|
||||||
x : Term
|
x : Term
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
match self.f:
|
||||||
|
case App(f=Const("Nat.add"), x=a):
|
||||||
|
return f"{a} + {self.x}"
|
||||||
|
case App(f=Const("List.cons"), x=a):
|
||||||
|
return f"{a} :: {self.x}"
|
||||||
|
case Const("Nat.Succ"):
|
||||||
|
cursor, i = self, 0
|
||||||
|
|
||||||
|
while True:
|
||||||
|
match cursor:
|
||||||
|
case App(f=Const("Nat.Succ")):
|
||||||
|
cursor = cursor.x
|
||||||
|
i += 1
|
||||||
|
case Const("Nat.Zero"):
|
||||||
|
return str(i)
|
||||||
|
case _:
|
||||||
|
xs = str(cursor)
|
||||||
|
if " " in xs:
|
||||||
|
xs = "(" + xs + ")"
|
||||||
|
return "{xs} + {i}"
|
||||||
|
case _:
|
||||||
|
fs = str(self.f)
|
||||||
|
xs = str(self.x)
|
||||||
|
|
||||||
|
if " " in fs:
|
||||||
|
fs = "(" + fs + ")"
|
||||||
|
if " " in xs:
|
||||||
|
xs = "(" + xs + ")"
|
||||||
|
|
||||||
|
return f"{fs} xs"
|
||||||
|
|
||||||
A0 = lambda fn : Const(fn)
|
A0 = lambda fn : Const(fn)
|
||||||
A1 = lambda fn, a : App(f=A0(fn), x=a)
|
A1 = lambda fn, a : App(f=A0(fn), x=a)
|
||||||
A2 = lambda fn, a, b : App(f=A1(fn, a), x=b)
|
A2 = lambda fn, a, b : App(f=A1(fn, a), x=b)
|
||||||
|
|
@ -29,6 +61,15 @@ A8 = lambda fn, a, b, c, d, e, f, g, h : App(f=A7(fn, a, b, c, d, e, f, g), x=h)
|
||||||
class Const(Term):
|
class Const(Term):
|
||||||
name : str
|
name : str
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
match self.name:
|
||||||
|
case "List.Nil":
|
||||||
|
return "[]"
|
||||||
|
case "Nat.Zero":
|
||||||
|
return "0"
|
||||||
|
case _:
|
||||||
|
return self.name
|
||||||
|
|
||||||
# --- Bool constructors ---
|
# --- Bool constructors ---
|
||||||
|
|
||||||
def Truth() -> Term:
|
def Truth() -> Term:
|
||||||
|
|
@ -68,21 +109,84 @@ def kernel_from_int(n : int) -> Term:
|
||||||
class Prop:
|
class Prop:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class Contradict(Prop):
|
||||||
|
s : str
|
||||||
|
|
||||||
@dataclass(frozen=True)
|
@dataclass(frozen=True)
|
||||||
class Eq(Prop):
|
class Eq(Prop):
|
||||||
lhs : Term
|
lhs : Term
|
||||||
rhs : Term
|
rhs : Term
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class Trivial(Prop):
|
||||||
|
pass
|
||||||
|
# def __repr__(self) -> str:
|
||||||
|
# return "<trivial>"
|
||||||
|
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
# Proofs
|
# Proofs
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
|
|
||||||
class Proof:
|
class Proof:
|
||||||
pass
|
def apply(self, goal : Prop, ctx : Context) -> Prop:
|
||||||
|
return (
|
||||||
|
Contradict("Cannot prove using the base class")
|
||||||
|
)
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class Assumption(Proof):
|
||||||
|
def apply(self, goal : Prop, ctx : Context) -> Prop:
|
||||||
|
return (
|
||||||
|
Trivial() if goal in ctx.props else
|
||||||
|
Contradict(f"Couldn't find proposition `{goal}` in assumptions")
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@dataclass(frozen=True)
|
@dataclass(frozen=True)
|
||||||
class Refl(Proof):
|
class Refl(Proof):
|
||||||
pass
|
def apply(self, goal : Prop, ctx : Context) -> Prop:
|
||||||
|
match goal:
|
||||||
|
case Eq():
|
||||||
|
l = normalize(goal.lhs)
|
||||||
|
r = normalize(goal.rhs)
|
||||||
|
|
||||||
|
if l == r:
|
||||||
|
return Trivial()
|
||||||
|
else:
|
||||||
|
# Reduce to differing parts
|
||||||
|
while True:
|
||||||
|
match ( l, r ):
|
||||||
|
case ( App(), App() ):
|
||||||
|
if l.x == r.x:
|
||||||
|
l, r = l.f, r.f
|
||||||
|
elif l.f == r.f:
|
||||||
|
l, r = l.x, r.x
|
||||||
|
else:
|
||||||
|
break
|
||||||
|
case _:
|
||||||
|
break
|
||||||
|
|
||||||
|
return Contradict(
|
||||||
|
f"Couldn't normalize {goal.lhs} = {goal.rhs} any further than to {l} = {r}"
|
||||||
|
)
|
||||||
|
|
||||||
|
case Prop():
|
||||||
|
return Contradict(
|
||||||
|
"Cannot prove base proposition"
|
||||||
|
)
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------------
|
||||||
|
# Context
|
||||||
|
# -----------------------------------------------------------------------------
|
||||||
|
# The context is a set of information you already know.
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class Context:
|
||||||
|
props : list[Prop]
|
||||||
|
|
||||||
|
def with_prop(self, prop : Prop):
|
||||||
|
return Context([ p for p in self.props ] + [ prop ])
|
||||||
|
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
# Normalization function
|
# Normalization function
|
||||||
|
|
@ -188,13 +292,18 @@ def normalize(term : Term) -> Term:
|
||||||
# Proof checker
|
# Proof checker
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
|
|
||||||
def check(goal: Prop, proof : Proof) -> bool:
|
ProofCheck = Union["Proven", "ProveFailure"]
|
||||||
match ( goal, proof ):
|
|
||||||
case ( Eq(), Refl() ):
|
|
||||||
return normalize(goal.lhs) == normalize(goal.rhs)
|
|
||||||
|
|
||||||
case _:
|
@dataclass(frozen=True)
|
||||||
return False
|
class Proven:
|
||||||
|
pass
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class ProveFailure:
|
||||||
|
s : str
|
||||||
|
|
||||||
|
def check(goal: Prop, proof : Proof, ctx : Context) -> Prop:
|
||||||
|
return proof.apply(goal, ctx)
|
||||||
|
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
# -----------------------------------------------------------------------------
|
# -----------------------------------------------------------------------------
|
||||||
|
|
@ -208,6 +317,17 @@ if __name__ == "__main__":
|
||||||
rhs=kernel_from_int(5),
|
rhs=kernel_from_int(5),
|
||||||
),
|
),
|
||||||
Refl(),
|
Refl(),
|
||||||
|
Context(props=[]),
|
||||||
|
))
|
||||||
|
|
||||||
|
# 2 + 3 == 7
|
||||||
|
print(check(
|
||||||
|
Eq(
|
||||||
|
lhs=A2("Nat.add", kernel_from_int(2), kernel_from_int(3)),
|
||||||
|
rhs=kernel_from_int(7),
|
||||||
|
),
|
||||||
|
Refl(),
|
||||||
|
Context(props=[]),
|
||||||
))
|
))
|
||||||
|
|
||||||
# 0 + x == x
|
# 0 + x == x
|
||||||
|
|
@ -217,6 +337,7 @@ if __name__ == "__main__":
|
||||||
rhs=Const("x")
|
rhs=Const("x")
|
||||||
),
|
),
|
||||||
Refl(),
|
Refl(),
|
||||||
|
Context(props=[]),
|
||||||
))
|
))
|
||||||
|
|
||||||
# x + 0 == x
|
# x + 0 == x
|
||||||
|
|
@ -226,4 +347,5 @@ if __name__ == "__main__":
|
||||||
rhs=Const("x")
|
rhs=Const("x")
|
||||||
),
|
),
|
||||||
Refl(),
|
Refl(),
|
||||||
|
Context(props=[]),
|
||||||
))
|
))
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue