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

Rerun claims when the claim body or those of dependent claims changes #2099

Merged
merged 24 commits into from
Oct 31, 2023
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
14f2ff4
Rerun claims based on digest, computed from claim body + dependent cl…
nwatson22 Sep 30, 2023
c0a6147
Merge branch 'master' into noah/claims-invalidate
nwatson22 Sep 30, 2023
382420d
Set Version: 1.0.309
Sep 30, 2023
b1c5c74
Make KClaimJob a frozen class and cache digest
nwatson22 Oct 2, 2023
f64639e
Set Version: 1.0.310
Oct 2, 2023
f7c9df1
Change to using frozenset
nwatson22 Oct 2, 2023
8b228ce
Merge branch 'noah/claims-invalidate' of https://github.com/runtimeve…
nwatson22 Oct 2, 2023
fb6cc81
Raise exception when label is not found
nwatson22 Oct 3, 2023
89f2c06
Merge branch 'master' into noah/claims-invalidate
nwatson22 Oct 3, 2023
6392d20
Set Version: 1.0.310
Oct 3, 2023
569160c
Merge branch 'master' into noah/claims-invalidate
nwatson22 Oct 4, 2023
7446c6a
Set Version: 1.0.312
Oct 4, 2023
f8362ba
Merge branch 'master' into noah/claims-invalidate
nwatson22 Oct 7, 2023
80fd6b8
Set Version: 1.0.314
Oct 7, 2023
d1ffb8b
Merge branch 'master' into noah/claims-invalidate
nwatson22 Oct 26, 2023
f4ba050
Set Version: 1.0.329
Oct 26, 2023
4c5bb31
Update kevm-pyk/src/kevm_pyk/__main__.py
nwatson22 Oct 27, 2023
d007502
Fix formatting
nwatson22 Oct 27, 2023
0a1af44
Merge branch 'master' into noah/claims-invalidate
rv-jenkins Oct 27, 2023
faff6b4
Set Version: 1.0.330
Oct 27, 2023
97991a1
Merge branch 'master' into noah/claims-invalidate
nwatson22 Oct 30, 2023
0703fe2
Set Version: 1.0.331
Oct 30, 2023
c610bb2
Merge branch 'master' into noah/claims-invalidate
nwatson22 Oct 30, 2023
dded76e
Set Version: 1.0.332
Oct 30, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion kevm-pyk/pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ build-backend = "poetry.core.masonry.api"

[tool.poetry]
name = "kevm-pyk"
version = "1.0.308"
version = "1.0.309"
description = ""
authors = [
"Runtime Verification, Inc. <[email protected]>",
Expand Down
2 changes: 1 addition & 1 deletion kevm-pyk/src/kevm_pyk/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,4 +6,4 @@
from typing import Final


VERSION: Final = '1.0.308'
VERSION: Final = '1.0.309'
90 changes: 83 additions & 7 deletions kevm-pyk/src/kevm_pyk/__main__.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
import sys
import time
from argparse import ArgumentParser
from dataclasses import dataclass
from pathlib import Path
from typing import TYPE_CHECKING

Expand All @@ -24,7 +25,7 @@
from pyk.proof.equality import EqualityProof
from pyk.proof.show import APRProofShow
from pyk.proof.tui import APRProofViewer
from pyk.utils import single
from pyk.utils import FrozenDict, hash_str, single

from . import VERSION, config
from .cli import KEVMCLIArgs, node_id_like
Expand Down Expand Up @@ -201,6 +202,69 @@ def wrap_process_pool(workers: int) -> Iterator[ZeroProcessPool | ProcessPool]:
yield pp


class JSONEncoder(json.JSONEncoder):
def default(self, obj: Any) -> Any:
if isinstance(obj, FrozenDict):
return json.JSONEncoder.encode(self, dict(obj))
return json.JSONEncoder.default(self, obj)


@dataclass
class KClaimJob:
claim: KClaim
dependencies: list[KClaimJob]

@property
def digest(self) -> str:
anvacaru marked this conversation as resolved.
Show resolved Hide resolved
deps_digest = ''.join([dep.digest for dep in self.dependencies])
claim_hash = hash_str(json.dumps(self.claim.to_dict(), sort_keys=True, cls=JSONEncoder))
return hash_str(f'{claim_hash}{deps_digest}')

def up_to_date(self, digest_file: Path | None) -> bool:
if digest_file is None:
return False
if not digest_file.exists():
return False
nwatson22 marked this conversation as resolved.
Show resolved Hide resolved
digest_dict = json.loads(digest_file.read_text())
if 'claims' not in digest_dict:
digest_dict['claims'] = {}
digest_file.write_text(json.dumps(digest_dict, indent=4))
if self.claim.label not in digest_dict['claims']:
return False
return digest_dict['claims'][self.claim.label] == self.digest

def update_digest(self, digest_file: Path | None) -> None:
if digest_file is None:
return
digest_dict = {}
if digest_file.exists():
digest_dict = json.loads(digest_file.read_text())
if 'claims' not in digest_dict:
digest_dict['claims'] = {}
digest_dict['claims'][self.claim.label] = self.digest
digest_file.write_text(json.dumps(digest_dict, indent=4))

_LOGGER.info(f'Updated claim {self.claim.label} in digest file: {digest_file}')


def init_claim_jobs(spec_module_name: str, claims: list[KClaim]) -> list[KClaimJob]:
labels_to_claims = {claim.label: claim for claim in claims}
labels_to_claim_jobs: dict[str, KClaimJob] = {}

def get_or_load_claim_job(claim_label: str) -> KClaimJob:
if claim_label not in labels_to_claim_jobs:
if claim_label in labels_to_claims:
claim = labels_to_claims[claim_label]
if f'{spec_module_name}.{claim_label}' in labels_to_claims:
anvacaru marked this conversation as resolved.
Show resolved Hide resolved
claim = labels_to_claims[f'{spec_module_name}.{claim_label}']
deps = [get_or_load_claim_job(dep_label) for dep_label in claim.dependencies]
claim_job = KClaimJob(claim, deps)
labels_to_claim_jobs[claim_label] = claim_job
return labels_to_claim_jobs[claim_label]

return [get_or_load_claim_job(claim.label) for claim in claims]


def exec_prove(
spec_file: Path,
includes: Iterable[str],
Expand Down Expand Up @@ -230,6 +294,8 @@ def exec_prove(
_ignore_arg(kwargs, 'md_selector', f'--md-selector: {kwargs["md_selector"]}')
md_selector = 'k'

digest_file = save_directory / 'digest' if save_directory is not None else None

if definition_dir is None:
definition_dir = DistTarget.HASKELL.get()

Expand Down Expand Up @@ -267,11 +333,19 @@ def is_functional(claim: KClaim) -> bool:
)
if all_claims is None:
raise ValueError(f'No claims found in file: {spec_file}')
all_claims_by_label: dict[str, KClaim] = {c.label: c for c in all_claims}
spec_module_name = spec_module if spec_module is not None else os.path.splitext(spec_file.name)[0].upper()
all_claim_jobs = init_claim_jobs(spec_module_name, all_claims)
all_claim_jobs_by_label = {c.claim.label: c for c in all_claim_jobs}
claims_graph = claim_dependency_dict(all_claims, spec_module_name=spec_module_name)

def _init_and_run_proof(claim: KClaim) -> tuple[bool, list[str] | None]:
def _init_and_run_proof(claim_job: KClaimJob) -> tuple[bool, list[str] | None]:
claim = claim_job.claim
up_to_date = claim_job.up_to_date(digest_file)
if up_to_date:
_LOGGER.info(f'Claim {claim.label} is up to date.')
else:
_LOGGER.info(f'Claim {claim.label} reinitialized because it is out of date.')
claim_job.update_digest(digest_file)
with legacy_explore(
kevm,
kcfg_semantics=KEVMSemantics(auto_abstract_gas=auto_abstract_gas),
Expand All @@ -288,6 +362,7 @@ def _init_and_run_proof(claim: KClaim) -> tuple[bool, list[str] | None]:
if (
save_directory is not None
and not reinit
and up_to_date
and EqualityProof.proof_exists(claim.label, save_directory)
):
proof_problem = EqualityProof.read_proof_data(save_directory, claim.label)
Expand All @@ -297,6 +372,7 @@ def _init_and_run_proof(claim: KClaim) -> tuple[bool, list[str] | None]:
if (
save_directory is not None
and not reinit
and up_to_date
and APRProof.proof_data_exists(claim.label, save_directory)
):
proof_problem = APRProof.read_proof_data(save_directory, claim.label)
Expand Down Expand Up @@ -365,21 +441,21 @@ def _init_and_run_proof(claim: KClaim) -> tuple[bool, list[str] | None]:
while topological_sorter.is_active():
ready = topological_sorter.get_ready()
_LOGGER.info(f'Discharging proof obligations: {ready}')
curr_claim_list = [all_claims_by_label[label] for label in ready]
curr_claim_list = [all_claim_jobs_by_label[label] for label in ready]
results: list[tuple[bool, list[str] | None]] = process_pool.map(_init_and_run_proof, curr_claim_list)
for label in ready:
topological_sorter.done(label)
selected_results.extend(results)
selected_claims.extend(curr_claim_list)

failed = 0
for claim, r in zip(selected_claims, selected_results, strict=True):
for job, r in zip(selected_claims, selected_results, strict=True):
passed, failure_log = r
if passed:
print(f'PROOF PASSED: {claim.label}')
print(f'PROOF PASSED: {job.claim.label}')
else:
failed += 1
print(f'PROOF FAILED: {claim.label}')
print(f'PROOF FAILED: {job.claim.label}')
if failure_info and failure_log is not None:
for line in failure_log:
print(line)
Expand Down
2 changes: 1 addition & 1 deletion package/version
Original file line number Diff line number Diff line change
@@ -1 +1 @@
1.0.308
1.0.309