Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Develop a common proof format and export proofs #458

Draft
wants to merge 67 commits into
base: master
Choose a base branch
from

Conversation

gcarpio21
Copy link
Member

@gcarpio21 gcarpio21 commented Mar 19, 2025

Objective

This PR aims to expose proofs produced by supported solvers through the API, primarily via a getProof() method.

Key Changes

  • Introduced ProofNode, ProofDag, and ProofRule as the main components for handling proofs.
  • Implemented proof retrieval for the Z3 solver.
  • Explored a standardized format for proofs across different solvers.

Proof Format Conversion

To unify proof representation, this PR investigates converting solver-specific proofs into a common format, with RESOLUTE (used by SMTInterpol) as a candidate.

  • The proposed approach models proof rules in CNF and transforms them into resolution steps.
  • The Z3ToResoluteProofConverter provides an example of this transformation.

Task List

  • Implement getProof() for Z3.

  • Introduce ProofNode, ProofDag, and ProofRule.

  • Implement getProof() for other solvers.

    • Z3
    • MathSAT5
    • CVC5
    • SMTInterpol
    • Princess
    • Opensmt
  • Test getProof more robustly.

Next Steps

  • More flexibility while handling proofs could be provided through the ProofDag interface.

…aviour of getProof to return unsupported operation.
…ts from SMTInterpol to a ResolutionProofDAG. Not working. Class SmtInterpolProofNode should help transform from the internal SMTInterpol proof representation to ResolutionProofDAG. Not working. Added some tests to evaluate proofs given back by SMTInterpol and the ResolutionProofDAG objects that should be returned.
Option requireProofs must be set to true through Configuration.

Added a class for testing the proof generation of Z3.

SMTInterpol: some documentation.
Various changes to the ProofDAG interface and all classes inheriting it.
Various changes to the ProofDAG interface and all classes inheriting it.
…n now be converted to resolution proof rules.
- Renamed some classes for clarity
- Z3: Made proof traversal non-recursive for efficiency.Added tests for proof-related functionality
- CVC5: Implemented necessary classes for proof production
- Improved documentation
- Added test for retrieving proofs in ProverEnvironmentTest.java
@gcarpio21 gcarpio21 requested a review from baierd March 19, 2025 12:29
@gcarpio21 gcarpio21 self-assigned this Mar 20, 2025
Copy link
Collaborator

@baierd baierd left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some general things that i noticed (some may be mentioned in change requests, but apply generally):

  • Tests should be generalized and moved into the test package if they are not using native API. You can still have tests for a single solver in the test package by using the assume API and only allowing a single one. The name of the solver should be mentioned in the test name then.

  • Please add documentation to the new classes and interfaces. Especially mention which ones are used internally and which ones are public/usable by users and the purpose/background of the classes/interfaces.

  • You can have classes within classes, which is especially helpful to structure smaller components, e.g. nodes of a graph, within the context you are using them in. That makes the code much easier to read and structures it better.

  • Please avoid pushing print-outs or commented out code. If you are working on a feature thats not yet working, thats fine. Document it and don't use the component. Please add some documentation to the commented out code. Print-outs should never be pushed. Try using the logger or exceptions. In tests please use assertion instead of printouts. (It is OK to expect the current static output of a solver. If the test fails in the future, we devs are notified about changes that we might need to consider.)

  • Native API tests that just test that proofs work are good and can be added for all solvers supporting proofs.

*
* @author Gabriel Carpio
*/
public class ResProofRule {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You could add sources (publication(s), reference sites etc.), the solvers that support this natively, as well as the usage/logic.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added some documentation and will wait for further information on the format form Mathsat5 as it seems to be very similar.

public class ResolutionProofDag extends AbstractProofDag {

/*
public ResolutionProofDAG() {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this class being worked on? If yes, in what form?

Please try to not push commented out code if possible. Also, please document this if it is supposed to be used in the future.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Made some adjustments to this.

try {
throw new UnsupportedOperationException("Proof generation isn't enabled.");
} catch (UnsupportedOperationException e) {
System.out.println(e.getMessage());
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You can check if the prover is closed first, making sure that the prover is available.

It does not make much sense to throw an exception and catch it immediately again ;D
And please don't push print-outs. Try using the logger, exceptions if you want to notify the user.

Also, "Proof generation is not available for the chosen solver" is a better message for the exception type.

Copy link
Member Author

@gcarpio21 gcarpio21 Mar 25, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To check the state I will move this method to AbstractProver. There, there also a method checkGenerateProofs that mimics the behavior of the other check methods, I understand this should throw an Exception already. Aside from this, there is still the issue that, for example, for Z3 proof generation must be set as an Option in the configuration, so the solver should retrieve this information from the context in this case.

import org.sosy_lab.java_smt.api.proofs.visitors.ProofVisitor;

public interface ProofDag {
void addNode(ProofNode node);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The traversal of the proofs should be public API. Is this interface exposed to users?

Also, this is missing some comments and formatting.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is supposed to be exposed to user, at least the get methods. However, this is still in the works and right now traversal should be only possible through the methods in ProofNode.

import org.sosy_lab.java_smt.ResProofRule.ResAxiom;

public class ProofFactory {
public static ProofNode createSourceNode(ResAxiom rule, Formula formula) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Building a proof should not be exposed to the user. Is ProofFactory available for users?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm currently trying to utilize this class to eliminate classes like Mathsat5ProofProcessor and Z3ToResoluteProofConverter. However, I'm facing an issue: ideally, the user shouldn't have direct access to these methods and should instead retrieve proofs via getProof from a ProverEnvironment instance. The challenge is that making the class non-public complicates its use within the solvers' packages.



//String proofStr = proof.toString();
System.out.println(invokeGetProofMode(smtinterpol).toString());
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please use assertions instead of printouts.

ShutdownManager shutdown = ShutdownManager.create();

// Create new context with SMTInterpol
context = Z3SolverContext.create(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why not use the SolverContextFactory?

}

private static Solver createEnvironment() throws CVC5ApiException {
Solver newSolver = new Solver();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This tests nothing currently. Also, native proof tests are good, but should be part of the CVC5 native test-class.

@@ -79,6 +85,7 @@ public void createEnvironment() {
long cfg = msat_create_config();

msat_set_option_checked(cfg, "model_generation", "true");
msat_set_option_checked(cfg, "proof_generation", "true");
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why enable proofs for all tests and then build a new, dedicated solver environment for the proof test?

*/

@SuppressWarnings({"unchecked", "rawtypes", "unused", "static-access","ModifiedButNotUsed"})
public class Z3ToResoluteProofConverter {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This class tackles a hard problem and should be documented with the used ideas and possible sources behind it.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added further documentation

@baierd
Copy link
Collaborator

baierd commented Mar 21, 2025

@leventeBajczi could you spare a look at this PR and maybe give us some feedback on the current proof interface?

@gcarpio21
Copy link
Member Author

@leventeBajczi could you spare a look at this PR and maybe give us some feedback on the current proof interface?

Definitely!

So far, I really like the relatively clean API for accessing the proofs, and I'm especially happy to see that the Z3 proofs are transformed to a 'simple' proof tree/DAG -- however, I'm uncertain how complete that transformation is based on the code, I think some information about that is missing (maybe as a comment?) to state how well Z3 proofs can be transformed into the common format.

Other than that, great work! I'm looking forward to integrating this with Theta. I'm really surprised at the number of solvers supporting proof generation, I only knew about MathSAT and Z3 beforehand. They will definitely be handy for us.

Have you maybe done some performance impact tests with the solvers when proof generation is enabled? I'd be interested to see how much performance is lost when it's active.

Right now, the Z3ToResoluteProofConverter class is still a work in progress. The goal is to transform Z3 proofs into a format similar to SMTInterpol’s (RESOLUTE), following the same proof rules.

Currently, the main functionality is to make Z3 proofs accessible through JavaSMT by simplifying Z3’s proof structures and implementing the ProofNode interface, which should capture all relevant proof information.

I haven’t run any performance tests yet. From what I know, enabling proof generation in a solver itself is usually quite costly. Hopefully, retrieving proofs through JavaSMT won’t introduce significant additional overhead.

I’ve also added more documentation in the code to clarify some of these points.

…ion if proof generation is not enabled. The default method is now in `AbstractProver`instead of `BasicProverEnvironment`
Fixed `Map`s in `ProofRuleResgistry` being empty
…gain in interface `BasicProverEnvironment`
…gain in interface `BasicProverEnvironment`
…roofs

# Conflicts:
#	src/org/sosy_lab/java_smt/solvers/cvc5/CVC5AbstractProver.java
…f node for processing the proof terms. Renamed `getResAxiomRuleByName` to `getFromName` in `ResProofRule`
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

3 participants