Skip to content

Commit

Permalink
feat: allow outputting indirect dependencies
Browse files Browse the repository at this point in the history
  • Loading branch information
Otto-AA committed Aug 9, 2024
1 parent ce28501 commit d32fba5
Show file tree
Hide file tree
Showing 3 changed files with 79 additions and 22 deletions.
54 changes: 33 additions & 21 deletions tod_attack_miner/cli.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
from tod_attack_miner.db.filters import (
get_filters_duplicate_limits,
get_filters_except_duplicate_limits,
get_filters_up_to_indirect_dependencies,
)
from tod_attack_miner.miner.miner import Miner

Expand All @@ -25,7 +26,7 @@ def main():
)
parser.add_argument("--archive-node-provider", default="http://localhost:8124/eth")
parser.add_argument("--from-block", default=19895500)
parser.add_argument("--to-block", default=19895500 + 100 - 1)
parser.add_argument("--to-block", default=19895504)
parser.add_argument(
"--window-size",
type=int,
Expand All @@ -48,6 +49,11 @@ def main():
default=Path("evaluations.csv"),
help="Path, where evaluation results should be stored",
)
parser.add_argument(
"--extract-indirect-dependencies",
action="store_true",
help="When evaluating candidates, stop before the indirect dependencies filter and output indirect dependency paths",
)
parser.add_argument("--postgres-user", type=str, default="postgres")
parser.add_argument("--postgres-password", type=str, default="password")
parser.add_argument("--postgres-host", type=str, default="localhost")
Expand All @@ -63,6 +69,7 @@ def main():
args = parser.parse_args()
evaluate_candidates_csv: Path | None = args.evaluate_candidates_csv
evaluation_results_csv: Path = args.evaluation_result_csv
extract_indirect_dependencies: bool = args.extract_indirect_dependencies

with psycopg.connect(
f"user={args.postgres_user} password={args.postgres_password} host={args.postgres_host} port={args.postgres_port}"
Expand All @@ -79,30 +86,35 @@ def main():
) as results_csv_file:
csv_reader = csv.DictReader(csv_file)
candidates = [(c["tx_a"], c["tx_b"]) for c in csv_reader]
if args.reset_db:
miner.reset_db()
miner.fetch(int(args.from_block), int(args.to_block))
miner.find_collisions()
results = miner.evaluate_candidates(
get_filters_except_duplicate_limits(25)
+ get_filters_duplicate_limits(10),
candidates,
)
if extract_indirect_dependencies:
filters = get_filters_up_to_indirect_dependencies(25)
results = miner.get_indirect_dependencies(filters, candidates, 1)
csv_writer = csv.writer(results_csv_file)
csv_writer.writerow(("tx_a", "tx_b", "dependency_path"))
csv_writer.writerows(results)
else:
filters = get_filters_except_duplicate_limits(
25
) + get_filters_duplicate_limits(10)
results = miner.evaluate_candidates(filters, candidates)

csv_writer = csv.DictWriter(
results_csv_file, ["tx_a", "tx_b", "filtered_by"]
)
csv_writer.writeheader()
rows = [
{
"tx_a": c["tx_a"],
"tx_b": c["tx_b"],
"filtered_by": c["filter"] or "",
}
for c in results
]
csv_writer.writerows(rows)
csv_writer = csv.DictWriter(
results_csv_file, ["tx_a", "tx_b", "filtered_by"]
)
csv_writer.writeheader()
rows = [
{
"tx_a": c["tx_a"],
"tx_b": c["tx_b"],
"filtered_by": c["filter"] or "",
}
for c in results
]
csv_writer.writerows(rows)
print(f"Saved results to {evaluation_results_csv}")

else:
if args.reset_db:
miner.reset_db()
Expand Down
40 changes: 39 additions & 1 deletion tod_attack_miner/db/filters.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
from typing import Callable
from typing import Callable, Iterable
import psycopg
import psycopg.sql
from tod_attack_miner.db.db import DB
Expand Down Expand Up @@ -126,6 +126,33 @@ def filter_indirect_dependencies_recursive(db: DB):
return deleted


def get_evaluation_indirect_dependencies_recursive(
db: DB, max_depth: int
) -> Iterable[tuple[str, str, str]]:
sql = psycopg.sql.SQL("""
WITH RECURSIVE depends_on(tx_a, tx_b, min_block_number, min_tx_index, tx_x, path, depth) AS (
SELECT tx_write_hash, tx_access_hash, block_number, tx_index, tx_access_hash, tx_access_hash, 1
FROM evaluation_candidates
INNER JOIN transactions ON hash = tx_write_hash
UNION
SELECT tx_a, tx_b, min_block_number, min_tx_index, tx_write_hash, tx_write_hash || '|' || path, depth + 1
FROM depends_on, candidates
INNER JOIN transactions ON hash = tx_write_hash
WHERE depends_on.tx_x = tx_access_hash
AND depth <= {}
AND (block_number > min_block_number
OR block_number = min_block_number AND tx_index > min_tx_index)
)
SELECT tx_a, tx_b, tx_a || '|' || path
FROM depends_on
INNER JOIN evaluation_candidates
ON tx_a = tx_write_hash AND tx_b = tx_access_hash
WHERE tx_b != tx_x
""").format(max_depth)
with db._con.cursor() as cursor:
return cursor.execute(sql).fetchall()


def create_limit_collisions_per_address(limit: int):
def limit_collisions_per_address(db: DB):
sql = f"""
Expand Down Expand Up @@ -209,6 +236,17 @@ def get_filters_except_duplicate_limits(
]


def get_filters_up_to_indirect_dependencies(
window_size: int | None,
) -> list[tuple[str, Callable[[DB], int]]]:
return [
("block_window", create_block_window_filter(window_size)),
("block_producers", filter_block_producers),
("nonces", filter_nonces),
("codes", filter_codes),
]


def get_filters_duplicate_limits(limit: int) -> list[tuple[str, Callable[[DB], int]]]:
return [
("limited_collisions_per_address", create_limit_collisions_per_address(limit)),
Expand Down
7 changes: 7 additions & 0 deletions tod_attack_miner/miner/miner.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
from typing import Callable, Iterable, Sequence
from tod_attack_miner.db.db import DB, Candidate, EvaluationCandidate
from tod_attack_miner.db.filters import get_evaluation_indirect_dependencies_recursive
from tod_attack_miner.fetcher.fetcher import BlockRange, fetch_block_range
from tod_attack_miner.rpc.rpc import RPC

Expand Down Expand Up @@ -48,6 +49,12 @@ def evaluate_candidates(
self._filter_stats["candidates"]["final"] = self.db.count_candidates()
return self.db.get_evaluation_candidates()

def get_indirect_dependencies(
self, filters: Filters, candidates: Iterable[tuple[str, str]], max_depth: int
) -> Iterable[tuple[str, str, str]]:
self.evaluate_candidates(filters, candidates)
return get_evaluation_indirect_dependencies_recursive(self.db, max_depth)

def count_candidates(self) -> int:
return self.db.count_candidates()

Expand Down

0 comments on commit d32fba5

Please sign in to comment.