- Notifications
You must be signed in to change notification settings - Fork 3
feat: add benchmark #101
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
feat: add benchmark #101
Changes from 6 commits
89d7faa
c21a796
e2859e2
60eea57
dd0a13a
2e6797a
447bd7f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
name: Benchmark Hooks | ||
| ||
on: | ||
workflow_dispatch: | ||
| ||
jobs: | ||
benchmark: | ||
runs-on: ubuntu-latest | ||
steps: | ||
- name: Checkout code | ||
uses: actions/checkout@v2 | ||
| ||
- name: Set up Python | ||
uses: actions/setup-python@v2 | ||
with: | ||
python-version: '3.8' | ||
| ||
- name: Install dependencies | ||
run: | | ||
python -m pip install --upgrade pip | ||
pip install pre-commit | ||
| ||
- name: Run benchmarks | ||
run: | | ||
python testing/benchmark_hooks.py |
Original file line number | Diff line number | Diff line change | ||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,20 @@ | ||||||||||||||||||
# Benchmarking | ||||||||||||||||||
| ||||||||||||||||||
[](https://codspeed.io/cpp-linter/cpp-linter-hooks) | ||||||||||||||||||
| ||||||||||||||||||
This document outlines the benchmarking process for comparing the performance of cpp-linter-hooks and mirrors-clang-format. | ||||||||||||||||||
| ||||||||||||||||||
## Running the Benchmark | ||||||||||||||||||
| ||||||||||||||||||
```bash | ||||||||||||||||||
python3 testing/benchmark_hooks.py | ||||||||||||||||||
``` | ||||||||||||||||||
| ||||||||||||||||||
## Results | ||||||||||||||||||
| ||||||||||||||||||
The results of the benchmarking process will be saved to `testing/benchmark_results.txt`. | ||||||||||||||||||
Comment on lines +13 to +15 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Call out cache-warming to ensure fair, stable timings. Make it explicit that cold-cache runs will dominate timings unless caches are warmed. ## Results -The results of the benchmarking process will be saved to `testing/benchmark_results.txt`. +The results of the benchmarking process will be saved to `testing/benchmark_results.txt`. + +Note: For fair comparisons, warm pre-commit caches once per hook (do not clean between repeats). If you need cold-cache numbers, run a separate pass that cleans caches before the first run only. 📝 Committable suggestion
Suggested change
🤖 Prompt for AI Agents
| ||||||||||||||||||
| ||||||||||||||||||
## To Do | ||||||||||||||||||
| ||||||||||||||||||
- Run benchmark against a larger codebase, such as [TheAlgorithms/C-Plus-Plus](https://github.com/TheAlgorithms/C-Plus-Plus). | ||||||||||||||||||
- Run benchmark with GitHub Actions for continuous integration. |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
repos: | ||
- repo: https://github.com/pre-commit/pre-commit-hooks | ||
rev: v1.1.0 | ||
hooks: | ||
- id: clang-format | ||
args: [--style=file, --version=21] |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
repos: | ||
- repo: https://github.com/pre-commit/mirrors-clang-format | ||
rev: v21.1.0 | ||
hooks: | ||
- id: clang-format |
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,154 @@ | ||||||||||||||||||||||||||||||||||||||||||||||||
#!/usr/bin/env python3 | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
Benchmark script to compare performance of cpp-linter-hooks vs mirrors-clang-format. | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
Usage: | ||||||||||||||||||||||||||||||||||||||||||||||||
python benchmark_hooks.py | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
Requirements: | ||||||||||||||||||||||||||||||||||||||||||||||||
- pre-commit must be installed and available in PATH | ||||||||||||||||||||||||||||||||||||||||||||||||
- Two config files: | ||||||||||||||||||||||||||||||||||||||||||||||||
- testing/pre-commit-config-cpp-linter-hooks.yaml | ||||||||||||||||||||||||||||||||||||||||||||||||
- testing/pre-commit-config-mirrors-clang-format.yaml | ||||||||||||||||||||||||||||||||||||||||||||||||
- Target files: testing/main.c (or adjust as needed) | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
Comment on lines +5 to +14 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Fix docstring: wrong paths and usage. The guidance doesn’t match the actual file path and config filenames. -Usage: - python benchmark_hooks.py +Usage: + python testing/benchmark_hooks.py @@ -- Two config files: - - testing/pre-commit-config-cpp-linter-hooks.yaml - - testing/pre-commit-config-mirrors-clang-format.yaml -- Target files: testing/main.c (or adjust as needed) +- Two config files: + - testing/benchmark_hook_1.yaml + - testing/benchmark_hook_2.yaml +- Target files are auto-discovered under testing/test-examples/ after cloning.
🤖 Prompt for AI Agents
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
import os | ||||||||||||||||||||||||||||||||||||||||||||||||
import subprocess | ||||||||||||||||||||||||||||||||||||||||||||||||
import time | ||||||||||||||||||||||||||||||||||||||||||||||||
import statistics | ||||||||||||||||||||||||||||||||||||||||||||||||
import glob | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
HOOKS = [ | ||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||
"name": "cpp-linter-hooks", | ||||||||||||||||||||||||||||||||||||||||||||||||
"config": "testing/benchmark_hook_1.yaml", | ||||||||||||||||||||||||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||||||||||||||||||||||||
{ | ||||||||||||||||||||||||||||||||||||||||||||||||
"name": "mirrors-clang-format", | ||||||||||||||||||||||||||||||||||||||||||||||||
"config": "testing/benchmark_hook_2.yaml", | ||||||||||||||||||||||||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||||||||||||||||||||||||
] | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
# Automatically find all C/C++ files in testing/ (and optionally src/, include/) | ||||||||||||||||||||||||||||||||||||||||||||||||
TARGET_FILES = glob.glob("testing/test-examples/*.c", recursive=True) | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
REPEATS = 5 | ||||||||||||||||||||||||||||||||||||||||||||||||
RESULTS_FILE = "testing/benchmark_results.txt" | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
def git_clone(): | ||||||||||||||||||||||||||||||||||||||||||||||||
try: | ||||||||||||||||||||||||||||||||||||||||||||||||
subprocess.run( | ||||||||||||||||||||||||||||||||||||||||||||||||
[ | ||||||||||||||||||||||||||||||||||||||||||||||||
"git", | ||||||||||||||||||||||||||||||||||||||||||||||||
"clone", | ||||||||||||||||||||||||||||||||||||||||||||||||
"--depth", | ||||||||||||||||||||||||||||||||||||||||||||||||
"1", | ||||||||||||||||||||||||||||||||||||||||||||||||
"https://github.com/gouravthakur39/beginners-C-program-examples.git", | ||||||||||||||||||||||||||||||||||||||||||||||||
"testing/test-examples", | ||||||||||||||||||||||||||||||||||||||||||||||||
], | ||||||||||||||||||||||||||||||||||||||||||||||||
check=True, | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
except subprocess.CalledProcessError: | ||||||||||||||||||||||||||||||||||||||||||||||||
pass | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
def run_hook(config, files): | ||||||||||||||||||||||||||||||||||||||||||||||||
cmd = ["pre-commit", "run", "--config", config, "--files"] + files | ||||||||||||||||||||||||||||||||||||||||||||||||
start = time.perf_counter() | ||||||||||||||||||||||||||||||||||||||||||||||||
try: | ||||||||||||||||||||||||||||||||||||||||||||||||
subprocess.run(cmd, check=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) | ||||||||||||||||||||||||||||||||||||||||||||||||
except subprocess.CalledProcessError: | ||||||||||||||||||||||||||||||||||||||||||||||||
# Still record time even if hook fails | ||||||||||||||||||||||||||||||||||||||||||||||||
pass | ||||||||||||||||||||||||||||||||||||||||||||||||
end = time.perf_counter() | ||||||||||||||||||||||||||||||||||||||||||||||||
return end - start | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
Comment on lines +57 to +67 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Harden subprocess call: use iterable unpacking, capture_output, and a timeout. Prevents hangs from blocking hooks and addresses style nits. -def run_hook(config, files): - cmd = ["pre-commit", "run", "--config", config, "--files"] + files +def run_hook(config, files): + cmd = ["pre-commit", "run", "--config", config, "--files", *files] start = time.perf_counter() try: - subprocess.run(cmd, check=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + subprocess.run(cmd, check=True, capture_output=True, timeout=300) except subprocess.CalledProcessError: # Still record time even if hook fails pass + except subprocess.TimeoutExpired: + # Record as a timeout-run; caller still gets elapsed wall time + pass end = time.perf_counter() return end - start 📝 Committable suggestion
Suggested change
🧰 Tools🪛 Ruff (0.12.2)45-45: Consider iterable unpacking instead of concatenation Replace with iterable unpacking (RUF005) 48-48: (S603) 🤖 Prompt for AI Agents
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
def safe_git_restore(files): | ||||||||||||||||||||||||||||||||||||||||||||||||
# Only restore files tracked by git | ||||||||||||||||||||||||||||||||||||||||||||||||
tracked = [] | ||||||||||||||||||||||||||||||||||||||||||||||||
for f in files: | ||||||||||||||||||||||||||||||||||||||||||||||||
result = subprocess.run( | ||||||||||||||||||||||||||||||||||||||||||||||||
["git", "ls-files", "--error-unmatch", f], | ||||||||||||||||||||||||||||||||||||||||||||||||
stdout=subprocess.PIPE, | ||||||||||||||||||||||||||||||||||||||||||||||||
stderr=subprocess.PIPE, | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
if result.returncode == 0: | ||||||||||||||||||||||||||||||||||||||||||||||||
tracked.append(f) | ||||||||||||||||||||||||||||||||||||||||||||||||
if tracked: | ||||||||||||||||||||||||||||||||||||||||||||||||
subprocess.run(["git", "restore"] + tracked) | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
Comment on lines +69 to +82 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Restoring via top-level Git misses cloned examples (nested repo). Files in testing/test-examples are not tracked by the top-level repo, so no restore happens and runs are not independent. Reset the nested repo to HEAD between runs. -def safe_git_restore(files): - # Only restore files tracked by git - tracked = [] - for f in files: - result = subprocess.run( - ["git", "ls-files", "--error-unmatch", f], - stdout=subprocess.PIPE, - stderr=subprocess.PIPE, - ) - if result.returncode == 0: - tracked.append(f) - if tracked: - subprocess.run(["git", "restore"] + tracked) +def safe_git_restore(_files): + # Reset the cloned examples repository (nested Git repo) to a clean state. + examples_repo = Path("testing/test-examples") + if (examples_repo / ".git").exists(): + subprocess.run( + ["git", "-C", str(examples_repo), "reset", "--hard", "HEAD"], + check=False, + capture_output=True, + )
🧰 Tools🪛 Ruff (0.12.2)72-72: (S603) 73-73: Starting a process with a partial executable path (S607) 80-80: (S603) 80-80: Consider Replace with (RUF005) 🤖 Prompt for AI Agents
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
def benchmark(): | ||||||||||||||||||||||||||||||||||||||||||||||||
results = {} | ||||||||||||||||||||||||||||||||||||||||||||||||
for hook in HOOKS: | ||||||||||||||||||||||||||||||||||||||||||||||||
times = [] | ||||||||||||||||||||||||||||||||||||||||||||||||
print(f"\nBenchmarking {hook['name']}...") | ||||||||||||||||||||||||||||||||||||||||||||||||
for i in range(REPEATS): | ||||||||||||||||||||||||||||||||||||||||||||||||
safe_git_restore(TARGET_FILES) | ||||||||||||||||||||||||||||||||||||||||||||||||
subprocess.run(["pre-commit", "clean"]) | ||||||||||||||||||||||||||||||||||||||||||||||||
t = run_hook(hook["config"], TARGET_FILES) | ||||||||||||||||||||||||||||||||||||||||||||||||
print(f" Run {i + 1}: {t:.3f} seconds") | ||||||||||||||||||||||||||||||||||||||||||||||||
times.append(t) | ||||||||||||||||||||||||||||||||||||||||||||||||
results[hook["name"]] = times | ||||||||||||||||||||||||||||||||||||||||||||||||
return results | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
def report(results): | ||||||||||||||||||||||||||||||||||||||||||||||||
headers = ["Hook", "Avg (s)", "Std (s)", "Min (s)", "Max (s)", "Runs"] | ||||||||||||||||||||||||||||||||||||||||||||||||
col_widths = [max(len(h), 16) for h in headers] | ||||||||||||||||||||||||||||||||||||||||||||||||
# Calculate max width for each column | ||||||||||||||||||||||||||||||||||||||||||||||||
for name, times in results.items(): | ||||||||||||||||||||||||||||||||||||||||||||||||
col_widths[0] = max(col_widths[0], len(name)) | ||||||||||||||||||||||||||||||||||||||||||||||||
print("\nBenchmark Results:\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
# Print header | ||||||||||||||||||||||||||||||||||||||||||||||||
header_row = " | ".join(h.ljust(w) for h, w in zip(headers, col_widths)) | ||||||||||||||||||||||||||||||||||||||||||||||||
print(header_row) | ||||||||||||||||||||||||||||||||||||||||||||||||
print("-+-".join("-" * w for w in col_widths)) | ||||||||||||||||||||||||||||||||||||||||||||||||
# Print rows | ||||||||||||||||||||||||||||||||||||||||||||||||
lines = [] | ||||||||||||||||||||||||||||||||||||||||||||||||
for name, times in results.items(): | ||||||||||||||||||||||||||||||||||||||||||||||||
avg = statistics.mean(times) | ||||||||||||||||||||||||||||||||||||||||||||||||
std = statistics.stdev(times) if len(times) > 1 else 0.0 | ||||||||||||||||||||||||||||||||||||||||||||||||
min_t = min(times) | ||||||||||||||||||||||||||||||||||||||||||||||||
max_t = max(times) | ||||||||||||||||||||||||||||||||||||||||||||||||
row = [ | ||||||||||||||||||||||||||||||||||||||||||||||||
name.ljust(col_widths[0]), | ||||||||||||||||||||||||||||||||||||||||||||||||
f"{avg:.3f}".ljust(col_widths[1]), | ||||||||||||||||||||||||||||||||||||||||||||||||
f"{std:.3f}".ljust(col_widths[2]), | ||||||||||||||||||||||||||||||||||||||||||||||||
f"{min_t:.3f}".ljust(col_widths[3]), | ||||||||||||||||||||||||||||||||||||||||||||||||
f"{max_t:.3f}".ljust(col_widths[4]), | ||||||||||||||||||||||||||||||||||||||||||||||||
str(len(times)).ljust(col_widths[5]), | ||||||||||||||||||||||||||||||||||||||||||||||||
] | ||||||||||||||||||||||||||||||||||||||||||||||||
print(" | ".join(row)) | ||||||||||||||||||||||||||||||||||||||||||||||||
lines.append(" | ".join(row)) | ||||||||||||||||||||||||||||||||||||||||||||||||
# Save to file | ||||||||||||||||||||||||||||||||||||||||||||||||
with open(RESULTS_FILE, "w") as f: | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write(header_row + "\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write("-+-".join("-" * w for w in col_widths) + "\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
for line in lines: | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write(line + "\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
print(f"\nResults saved to {RESULTS_FILE}") | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
# Write to GitHub Actions summary if available | ||||||||||||||||||||||||||||||||||||||||||||||||
summary_path = os.environ.get("GITHUB_STEP_SUMMARY") | ||||||||||||||||||||||||||||||||||||||||||||||||
if summary_path: | ||||||||||||||||||||||||||||||||||||||||||||||||
with open(summary_path, "a") as f: | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write("## Benchmark Results\n\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write(header_row + "\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write("-+-".join("-" * w for w in col_widths) + "\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
for line in lines: | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write(line + "\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
f.write("\n") | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
def main(): | ||||||||||||||||||||||||||||||||||||||||||||||||
git_clone() | ||||||||||||||||||||||||||||||||||||||||||||||||
results = benchmark() | ||||||||||||||||||||||||||||||||||||||||||||||||
report(results) | ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
| ||||||||||||||||||||||||||||||||||||||||||||||||
if __name__ == "__main__": | ||||||||||||||||||||||||||||||||||||||||||||||||
main() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Update Actions to supported versions; bump Python.
actionlint flags v2 runners as too old. Upgrade to current majors and avoid EOL Python 3.8.
Apply this diff:
Also applies to: 14-16
🧰 Tools
🪛 actionlint (1.7.7)
11-11: the runner of "actions/checkout@v2" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
🤖 Prompt for AI Agents