feat(library/tactic): add pretty printer for goal and proof_state objects, add solve method for tactics, add trivial example

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-11-21 16:44:31 -08:00
parent 28a56e3acf
commit 41062fdf9f
7 changed files with 60 additions and 6 deletions

View file

@ -6,6 +6,7 @@ Author: Leonardo de Moura
*/ */
#include <utility> #include <utility>
#include <vector> #include <vector>
#include <algorithm>
#include "util/name_set.h" #include "util/name_set.h"
#include "util/buffer.h" #include "util/buffer.h"
#include "kernel/for_each_fn.h" #include "kernel/for_each_fn.h"
@ -18,6 +19,28 @@ namespace lean {
goal::goal(list<std::pair<name, expr>> const & h, expr const & c):m_hypotheses(h), m_conclusion(c) {} goal::goal(list<std::pair<name, expr>> const & h, expr const & c):m_hypotheses(h), m_conclusion(c) {}
format goal::pp(formatter const & fmt, options const & opts) const {
unsigned indent = get_pp_indent(opts);
bool unicode = get_pp_unicode(opts);
format turnstile = unicode ? format("\u22A2") /* ⊢ */ : format("|-");
buffer<std::pair<name, expr>> tmp;
to_buffer(m_hypotheses, tmp);
bool first = true;
format r;
for (auto const & p : tmp) {
if (first) {
first = false;
} else {
r += compose(comma(), line());
}
r += format{format(p.first), space(), colon(), nest(indent, compose(line(), fmt(p.second, opts)))};
}
r = group(r);
r += format{line(), turnstile, space(), nest(indent, fmt(m_conclusion, opts))};
return group(r);
}
goal_proof_fn::goal_proof_fn(std::vector<expr> && consts): goal_proof_fn::goal_proof_fn(std::vector<expr> && consts):
m_constants(consts) { m_constants(consts) {
} }
@ -56,7 +79,7 @@ static name mk_unique_name(name_set & s, name const & suggestion) {
std::pair<goal, goal_proof_fn> to_goal(environment const & env, context const & ctx, expr const & t) { std::pair<goal, goal_proof_fn> to_goal(environment const & env, context const & ctx, expr const & t) {
type_inferer inferer(env); type_inferer inferer(env);
if (!inferer.is_proposition(t)) if (!inferer.is_proposition(t, ctx))
throw exception("to goal failed, type is not a proposition"); throw exception("to goal failed, type is not a proposition");
name_set used_names = collect_used_names(ctx, t); name_set used_names = collect_used_names(ctx, t);
buffer<context_entry> entries; buffer<context_entry> entries;
@ -103,6 +126,7 @@ std::pair<goal, goal_proof_fn> to_goal(environment const & env, context const &
} }
} }
expr conclusion = replacer(t); expr conclusion = replacer(t);
std::reverse(consts.begin(), consts.end());
return mk_pair(goal(reverse_to_list(hypotheses.begin(), hypotheses.end()), conclusion), return mk_pair(goal(reverse_to_list(hypotheses.begin(), hypotheses.end()), conclusion),
goal_proof_fn(std::move(consts))); goal_proof_fn(std::move(consts)));
} }

View file

@ -9,6 +9,7 @@ Author: Leonardo de Moura
#include <vector> #include <vector>
#include "util/list.h" #include "util/list.h"
#include "util/name.h" #include "util/name.h"
#include "kernel/formatter.h"
#include "kernel/expr.h" #include "kernel/expr.h"
#include "kernel/context.h" #include "kernel/context.h"
#include "kernel/environment.h" #include "kernel/environment.h"
@ -23,6 +24,7 @@ public:
list<std::pair<name, expr>> const & get_hypotheses() const { return m_hypotheses; } list<std::pair<name, expr>> const & get_hypotheses() const { return m_hypotheses; }
expr const & get_conclusion() const { return m_conclusion; } expr const & get_conclusion() const { return m_conclusion; }
operator bool() const { return m_conclusion; } operator bool() const { return m_conclusion; }
format pp(formatter const & fmt, options const & opts) const;
}; };
/** /**

View file

@ -9,6 +9,19 @@ Author: Leonardo de Moura
#include "library/tactic/proof_state.h" #include "library/tactic/proof_state.h"
namespace lean { namespace lean {
format proof_state::pp(formatter const & fmt, options const & opts) const {
format r;
bool first = true;
for (auto const & p : m_goals) {
if (first)
first = false;
else
r += line();
r += group(format{format(p.first), colon(), line(), p.second.pp(fmt, opts)});
}
return r;
}
void swap(proof_state & s1, proof_state & s2) { void swap(proof_state & s1, proof_state & s2) {
swap(s1.m_goals, s2.m_goals); swap(s1.m_goals, s2.m_goals);
swap(s1.m_menv, s2.m_menv); swap(s1.m_menv, s2.m_menv);

View file

@ -28,6 +28,7 @@ public:
metavar_env const & get_menv() const { return m_menv; } metavar_env const & get_menv() const { return m_menv; }
proof_builder const & get_proof_builder() const { return m_proof_builder; } proof_builder const & get_proof_builder() const { return m_proof_builder; }
justification_builder const & get_justification_builder() const { return m_justification_builder; } justification_builder const & get_justification_builder() const { return m_justification_builder; }
format pp(formatter const & fmt, options const & opts) const;
}; };
void swap(proof_state & s1, proof_state & s2); void swap(proof_state & s1, proof_state & s2);

View file

@ -48,6 +48,18 @@ tactic & tactic::operator=(tactic && s) {
LEAN_MOVE_REF(tactic, s); LEAN_MOVE_REF(tactic, s);
} }
expr tactic::solve(environment const & env, io_state const & io, proof_state const & s) {
tactic_result_ref r = operator()(s);
proof_state_ref final = r->next(env, io);
if (!final)
throw tactic_exception("tactic did not produce any result");
if (!empty(final->get_goals()))
throw tactic_exception("tactic did not solve all goals");
assignment a(final->get_menv());
proof_map m;
return final->get_proof_builder()(m, env, a);
}
class then_tactic : public tactic_cell { class then_tactic : public tactic_cell {
tactic m_t1; tactic m_t1;
tactic m_t2; tactic m_t2;

View file

@ -56,6 +56,8 @@ public:
tactic & operator=(tactic && s); tactic & operator=(tactic && s);
tactic_result_ref operator()(proof_state const & s) { return m_ptr->operator()(s); } tactic_result_ref operator()(proof_state const & s) { return m_ptr->operator()(s); }
expr solve(environment const & env, io_state const & io, proof_state const & s);
}; };
template<typename F> template<typename F>
@ -89,7 +91,6 @@ public:
} }
}; };
template<typename F> template<typename F>
tactic mk_tactic(F && f) { return tactic(new simple_tactic_cell<F>(std::forward<F>(f))); } tactic mk_tactic(F && f) { return tactic(new simple_tactic_cell<F>(std::forward<F>(f))); }

View file

@ -16,6 +16,7 @@ using namespace lean;
static void tst1() { static void tst1() {
environment env; environment env;
io_state io(options(), mk_simple_formatter());
import_all(env); import_all(env);
env.add_var("p", Bool); env.add_var("p", Bool);
env.add_var("q", Bool); env.add_var("q", Bool);
@ -24,10 +25,10 @@ static void tst1() {
context ctx; context ctx;
ctx = extend(ctx, "H1", p); ctx = extend(ctx, "H1", p);
ctx = extend(ctx, "H2", q); ctx = extend(ctx, "H2", q);
// proof_state s1 = to_proof_state(env, ctx, mk_var(0)); proof_state s = to_proof_state(env, ctx, p);
// tactic t = then(assumption_tactic(), now_tactic()); std::cout << s.pp(mk_simple_formatter(), options()) << "\n";
// tactic_result_ref r = t(s1); tactic t = then(assumption_tactic(), now_tactic());
// proof_state_ref s2 = r->next(); std::cout << "proof: " << t.solve(env, io, s) << "\n";
} }
int main() { int main() {