Improve proof output

This commit is contained in:
Patrick Lühne 2020-05-06 21:38:48 +02:00
parent 15769d58b3
commit 70bef152a4
4 changed files with 217 additions and 49 deletions

View File

@ -8,6 +8,7 @@ edition = "2018"
foliage = {git = "https://github.com/pluehne/foliage", branch = "parser-new"} foliage = {git = "https://github.com/pluehne/foliage", branch = "parser-new"}
log = "0.4" log = "0.4"
pretty_env_logger = "0.4" pretty_env_logger = "0.4"
regex = "1"
structopt = "0.3" structopt = "0.3"
[dependencies.clingo] [dependencies.clingo]

View File

@ -21,6 +21,10 @@ pub enum Kind
UnknownDomainIdentifier(String), UnknownDomainIdentifier(String),
VariableNameNotAllowed(String), VariableNameNotAllowed(String),
WriteTPTPProgram, WriteTPTPProgram,
RunVampire,
// TODO: rename to something Vampire-specific
ProveProgram(Option<i32>, String, String),
ParseVampireOutput(String, String),
} }
pub struct Error pub struct Error
@ -135,6 +139,21 @@ impl Error
{ {
Self::new(Kind::WriteTPTPProgram).with(source) Self::new(Kind::WriteTPTPProgram).with(source)
} }
pub(crate) fn new_run_vampire<S: Into<Source>>(source: S) -> Self
{
Self::new(Kind::RunVampire).with(source)
}
pub(crate) fn new_prove_program(exit_code: Option<i32>, stdout: String, stderr: String) -> Self
{
Self::new(Kind::ProveProgram(exit_code, stdout, stderr))
}
pub(crate) fn new_parse_vampire_output(stdout: String, stderr: String) -> Self
{
Self::new(Kind::ParseVampireOutput(stdout, stderr))
}
} }
impl std::fmt::Debug for Error impl std::fmt::Debug for Error
@ -175,6 +194,28 @@ impl std::fmt::Debug for Error
"variable name “{}” not allowed (program variables must start with X, Y, or Z and integer variables with I, J, K, L, M, or N)", "variable name “{}” not allowed (program variables must start with X, Y, or Z and integer variables with I, J, K, L, M, or N)",
variable_name), variable_name),
Kind::WriteTPTPProgram => write!(formatter, "error writing TPTP program"), Kind::WriteTPTPProgram => write!(formatter, "error writing TPTP program"),
Kind::RunVampire => write!(formatter, "could not run Vampire"),
Kind::ProveProgram(exit_code, ref stdout, ref stderr) =>
{
let exit_code_output = match exit_code
{
None => "no exit code".to_string(),
Some(exit_code) => format!("exit code: {}", exit_code),
};
write!(formatter,
"error proving program ({})\n\
==== stdout ===========================================================\n\
{}\
==== stderr ===========================================================\n\
{}", exit_code_output, stdout, stderr)
},
Kind::ParseVampireOutput(ref stdout, ref stderr) => write!(formatter,
"could not parse Vampire output\n\
==== stdout ===========================================================\n\
{}\
==== stderr ===========================================================\n\
{}", stdout, stderr),
}?; }?;
if let Some(source) = &self.source if let Some(source) = &self.source

View File

@ -321,6 +321,7 @@ where
+ crate::traits::VariableDeclarationDomain + crate::traits::VariableDeclarationDomain
+ crate::traits::VariableDeclarationID + crate::traits::VariableDeclarationID
{ {
// TODO: rename format to formatter
fn fmt(&self, format: &mut std::fmt::Formatter) -> std::fmt::Result fn fmt(&self, format: &mut std::fmt::Formatter) -> std::fmt::Result
{ {
let display_variable_declaration = |variable_declaration| let display_variable_declaration = |variable_declaration|
@ -400,7 +401,7 @@ where
separator = ", " separator = ", "
} }
write!(format, "]: {:?}", display_formula(&exists.argument))?; write!(format, "]: ({:?})", display_formula(&exists.argument))?;
}, },
foliage::Formula::ForAll(for_all) => foliage::Formula::ForAll(for_all) =>
{ {
@ -419,7 +420,7 @@ where
separator = ", " separator = ", "
} }
write!(format, "]: {:?}", display_formula(&for_all.argument))?; write!(format, "]: ({:?})", display_formula(&for_all.argument))?;
}, },
foliage::Formula::Not(argument) => write!(format, "~{:?}", display_formula(argument))?, foliage::Formula::Not(argument) => write!(format, "~{:?}", display_formula(argument))?,
foliage::Formula::And(arguments) => foliage::Formula::And(arguments) =>
@ -529,7 +530,17 @@ where
for argument in &predicate.arguments for argument in &predicate.arguments
{ {
write!(format, "{}{:?}", separator, display_term(argument))?; write!(format, "{}", separator)?;
let is_argument_arithmetic =
crate::is_term_arithmetic(argument, self.context)
.expect("could not determine whether term is arithmetic");
match is_argument_arithmetic
{
true => write!(format, "f__integer__({})", display_term(argument))?,
false => write!(format, "{}", display_term(argument))?,
}
separator = ", " separator = ", "
} }

View File

@ -1,3 +1,4 @@
#[derive(Copy, Clone, Eq, PartialEq)]
pub enum StatementKind pub enum StatementKind
{ {
Axiom, Axiom,
@ -10,9 +11,12 @@ pub enum StatementKind
enum ProofStatus enum ProofStatus
{ {
AssumedProven, AssumedProven,
ToProve, ToProveNow,
ToProveLater,
Ignored,
} }
#[derive(Copy, Clone, Eq, PartialEq)]
pub enum ProofResult pub enum ProofResult
{ {
Proven, Proven,
@ -50,7 +54,7 @@ impl Statement
name: None, name: None,
description: None, description: None,
formula, formula,
proof_status: ProofStatus::ToProve, proof_status: ProofStatus::ToProveLater,
} }
} }
@ -126,7 +130,7 @@ impl Problem
section.push(statement); section.push(statement);
} }
pub fn prove(&self, proof_direction: crate::ProofDirection) pub fn prove(&self, proof_direction: crate::ProofDirection) -> Result<(), crate::Error>
{ {
if proof_direction == crate::ProofDirection::Forward if proof_direction == crate::ProofDirection::Forward
|| proof_direction == crate::ProofDirection::Both || proof_direction == crate::ProofDirection::Both
@ -145,14 +149,16 @@ impl Problem
StatementKind::Axiom StatementKind::Axiom
| StatementKind::Assumption | StatementKind::Assumption
=> statement.proof_status = ProofStatus::AssumedProven, => statement.proof_status = ProofStatus::AssumedProven,
_ => statement.proof_status = ProofStatus::ToProve, StatementKind::Lemma(crate::ProofDirection::Backward)
=> statement.proof_status = ProofStatus::Ignored,
_ => statement.proof_status = ProofStatus::ToProveLater,
} }
} }
} }
drop(statements); drop(statements);
self.prove_unproven_statements(); self.prove_unproven_statements()?;
log::info!("finished forward proof"); log::info!("finished forward proof");
} }
@ -174,63 +180,113 @@ impl Problem
StatementKind::Axiom StatementKind::Axiom
| StatementKind::Assertion | StatementKind::Assertion
=> statement.proof_status = ProofStatus::AssumedProven, => statement.proof_status = ProofStatus::AssumedProven,
_ => statement.proof_status = ProofStatus::ToProve, StatementKind::Lemma(crate::ProofDirection::Forward)
=> statement.proof_status = ProofStatus::Ignored,
_ => statement.proof_status = ProofStatus::ToProveLater,
} }
} }
} }
drop(statements); drop(statements);
self.prove_unproven_statements(); self.prove_unproven_statements()?;
log::info!("finished backward proof"); log::info!("finished backward proof");
} }
Ok(())
}
fn next_unproven_statement_do_mut<F, G>(&self, mut functor: F) -> Option<G>
where
F: FnMut(SectionKind, &mut Statement) -> G,
{
for section in self.statements.borrow_mut().iter_mut()
{
for statement in section.1.iter_mut()
{
if statement.proof_status == ProofStatus::ToProveNow
|| statement.proof_status == ProofStatus::ToProveLater
{
return Some(functor(*section.0, statement));
}
}
}
None
} }
fn prove_unproven_statements(&self) -> Result<(), crate::Error> fn prove_unproven_statements(&self) -> Result<(), crate::Error>
{ {
while self.statements.borrow() loop
.iter()
.find(|section|
section.1.iter().find(|x| x.proof_status == ProofStatus::ToProve).is_some())
.is_some()
{ {
match self.next_unproven_statement_do_mut(
|section_kind, statement|
{
statement.proof_status = ProofStatus::ToProveNow;
let statement_type_output = match section_kind
{
SectionKind::CompletedDefinitions => "completed definition",
SectionKind::IntegrityConstraints => "integrity constraint",
SectionKind::Axioms => "axiom",
SectionKind::Assumptions => "assumption",
SectionKind::Lemmas => "lemma",
SectionKind::Assertions => "assertion",
};
println!("verifying {}: {}", statement_type_output, statement.formula);
})
{
Some(_) => (),
None => break,
}
let mut tptp_problem_to_prove_next_statement = "".to_string(); let mut tptp_problem_to_prove_next_statement = "".to_string();
self.write_tptp_problem_to_prove_next_statement( self.write_tptp_problem_to_prove_next_statement(
&mut tptp_problem_to_prove_next_statement) &mut tptp_problem_to_prove_next_statement)
.map_err(|error| crate::Error::new_write_tptp_program(error))?; .map_err(|error| crate::Error::new_write_tptp_program(error))?;
log::info!("proving program:\n{}", &tptp_problem_to_prove_next_statement); log::trace!("TPTP program :\n{}", &tptp_problem_to_prove_next_statement);
// TODO: refactor // TODO: make configurable again
let mut conjecture_found = false; let (proof_result, proof_time_seconds) =
run_vampire(&tptp_problem_to_prove_next_statement,
Some(&["--mode", "casc", "--cores", "8", "--time_limit", "300"]))?;
for section in self.statements.borrow_mut().iter_mut() match proof_result
{ {
for statement in section.1.iter_mut() ProofResult::NotProven =>
{ {
if statement.proof_status == ProofStatus::ToProve println!("could not prove statement");
return Ok(());
},
ProofResult::Disproven =>
{ {
conjecture_found = true; println!("statement disproven");
statement.proof_status = ProofStatus::AssumedProven;
return Ok(());
},
_ => (),
} }
if conjecture_found match self.next_unproven_statement_do_mut(
|_, statement| statement.proof_status = ProofStatus::AssumedProven)
{ {
break; Some(_) => (),
} None => unreachable!("could not set the statement to proven"),
} }
if conjecture_found match proof_time_seconds
{ {
break; None => log::info!("statement proven"),
Some(proof_time_seconds) =>
log::info!("statement proven in {} seconds", proof_time_seconds),
} }
} }
log::info!("program proven!");
}
Ok(()) Ok(())
} }
@ -402,7 +458,6 @@ impl Problem
fn write_tptp_problem_to_prove_next_statement(&self, formatter: &mut String) -> std::fmt::Result fn write_tptp_problem_to_prove_next_statement(&self, formatter: &mut String) -> std::fmt::Result
{ {
use std::fmt::Write as _; use std::fmt::Write as _;
use std::io::Write as _;
let format_context = FormatContext let format_context = FormatContext
{ {
@ -495,8 +550,6 @@ impl Problem
last_symbolic_constant = Some(std::rc::Rc::clone(symbolic_constant)); last_symbolic_constant = Some(std::rc::Rc::clone(symbolic_constant));
} }
let mut conjecture_found = false;
for (section_kind, statements) in self.statements.borrow().iter() for (section_kind, statements) in self.statements.borrow().iter()
{ {
if statements.is_empty() if statements.is_empty()
@ -521,12 +574,6 @@ impl Problem
for statement in statements.iter() for statement in statements.iter()
{ {
// Only prove one statement at a time
if conjecture_found && statement.proof_status == ProofStatus::ToProve
{
continue;
}
if let Some(ref description) = statement.description if let Some(ref description) = statement.description
{ {
writeln!(formatter, "% {}", description)?; writeln!(formatter, "% {}", description)?;
@ -546,11 +593,11 @@ impl Problem
let statement_type = match statement.proof_status let statement_type = match statement.proof_status
{ {
ProofStatus::AssumedProven => "axiom", ProofStatus::AssumedProven => "axiom",
ProofStatus::ToProve => ProofStatus::ToProveNow => "conjecture",
{ // Skip statements that will be proven later
conjecture_found = true; ProofStatus::ToProveLater => continue,
"conjecture" // Skip statements that are not part of this proof
}, ProofStatus::Ignored => continue,
}; };
// TODO: add proper statement type // TODO: add proper statement type
@ -741,3 +788,71 @@ impl<'a, 'b> foliage::format::Format for FormatContext<'a, 'b>
write!(formatter, "{}{}", prefix, id + 1) write!(formatter, "{}{}", prefix, id + 1)
} }
} }
fn run_vampire<I, S>(input: &str, arguments: Option<I>)
-> Result<(ProofResult, Option<f32>), crate::Error>
where
I: IntoIterator<Item = S>, S: AsRef<std::ffi::OsStr>,
{
let mut vampire = std::process::Command::new("vampire");
let vampire = match arguments
{
Some(arguments) => vampire.args(arguments),
None => &mut vampire,
};
let mut vampire = vampire
.stdin(std::process::Stdio::piped())
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.spawn()
.map_err(|error| crate::Error::new_run_vampire(error))?;
{
use std::io::Write as _;
let vampire_stdin = vampire.stdin.as_mut().unwrap();
vampire_stdin.write_all(input.as_bytes())
.map_err(|error| crate::Error::new_run_vampire(error))?;
}
let output = vampire.wait_with_output()
.map_err(|error| crate::Error::new_run_vampire(error))?;
let stdout = std::str::from_utf8(&output.stdout)
.map_err(|error| crate::Error::new_run_vampire(error))?;
let stderr = std::str::from_utf8(&output.stderr)
.map_err(|error| crate::Error::new_run_vampire(error))?;
if !output.status.success()
{
let proof_not_found_regex = regex::Regex::new(r"% \(\d+\)Proof not found in time").unwrap();
if proof_not_found_regex.is_match(stdout)
{
return Ok((ProofResult::NotProven, None));
}
return Err(crate::Error::new_prove_program(output.status.code(), stdout.to_string(),
stderr.to_string()));
}
let proof_time_regex = regex::Regex::new(r"% \(\d+\)Success in time (\d+(?:\.\d+)?) s").unwrap();
let proof_time = proof_time_regex.captures(stdout)
.map(|captures| captures.get(1).unwrap().as_str().parse::<f32>().ok())
.unwrap_or(None);
let refutation_regex = regex::Regex::new(r"% \(\d+\)Termination reason: Refutation").unwrap();
if refutation_regex.is_match(stdout)
{
return Ok((ProofResult::Proven, proof_time));
}
// TODO: support disproven result
Err(crate::Error::new_parse_vampire_output(stdout.to_string(), stderr.to_string()))
}