Improve proof output
This commit is contained in:
parent
15769d58b3
commit
70bef152a4
@ -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]
|
||||||
|
41
src/error.rs
41
src/error.rs
@ -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
|
||||||
|
@ -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 = ", "
|
||||||
}
|
}
|
||||||
|
199
src/problem.rs
199
src/problem.rs
@ -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()))
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user