fix: plot generation #61
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
name: "Run benchmark" | |
on: | |
pull_request_target: | |
types: [assigned, opened, synchronize, reopened, edited] | |
push: | |
branches: | |
- main | |
permissions: | |
contents: write | |
packages: write | |
pull-requests: write | |
issues: write | |
jobs: | |
build: | |
if: github.event.head_commit.message != 'Update performance results in README.md' | |
runs-on: ubuntu-latest | |
env: | |
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | |
steps: | |
- name: Checkout | |
uses: actions/checkout@v4 | |
with: | |
token: ${{ secrets.GITHUB_TOKEN }} | |
ref: ${{ github.event_name == 'pull_request_target' && github.event.pull_request.head.sha || github.sha }} | |
- name: Set up Docker buildx | |
uses: docker/setup-buildx-action@v3 | |
- name: Login to GitHub Container Registry | |
uses: docker/login-action@v3 | |
with: | |
registry: ghcr.io | |
username: ${{ github.repository_owner }} | |
password: ${{ secrets.GITHUB_TOKEN }} | |
- name: Build and push Docker image | |
uses: docker/build-push-action@v6 | |
env: | |
DOCKER_BUILD_RECORD_UPLOAD: false | |
with: | |
push: true | |
tags: ghcr.io/${{ github.repository_owner }}/federated-benchmark:latest | |
cache-from: type=gha | |
cache-to: type=gha,mode=max | |
benchmark: | |
if: github.event.head_commit.message != 'Update performance results in README.md' | |
needs: build | |
runs-on: | |
group: benchmarking-runner | |
env: | |
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | |
strategy: | |
matrix: | |
response: [big, medium, small] | |
service: [ | |
# the theoretical maximum | |
# source_graphql, | |
# source_rest_api, | |
# nginx -> source: the baseline for comparison | |
nginx_graphql, | |
# nginx_rest_api, | |
# tailcall configurations | |
tailcall_default, | |
# tailcall_tweaks, | |
# tailcall_http_cache, | |
# tailcall_cache_dir, | |
# tailcall_dedupe_op, | |
tailcall_full_conf, | |
# wundergraph configurations | |
wundergraph_no_opt, | |
# wundergraph_dedupe, | |
wundergraph_default, | |
# apollo | |
apollo_router, | |
# grafbase | |
grafbase_default, | |
grafbase_cache, | |
] | |
steps: | |
- name: Login to GitHub Container Registry | |
uses: docker/login-action@v3 | |
with: | |
registry: ghcr.io | |
username: ${{ github.repository_owner }} | |
password: ${{ secrets.GITHUB_TOKEN }} | |
- name: Create "output.log" file | |
run: touch ${{ github.workspace }}/output.log | |
- id: benchmark | |
name: Benchmark | |
uses: addnab/docker-run-action@v3 | |
with: | |
username: ${{ github.repository_owner }} | |
password: ${{ secrets.GITHUB_TOKEN }} | |
registry: ghcr.io | |
image: ghcr.io/${{ github.repository_owner }}/federated-benchmark:latest | |
options: -v ${{ github.workspace }}:/work | |
run: ./benchmark.sh ${{ matrix.response }} ${{ matrix.service }} >> /work/output.log | |
- name: Print result | |
run: cat ${{ github.workspace }}/output.log | |
- name: Determine if run had errors | |
id: extract_has_error | |
run: | | |
HAS_ERROR=$((grep -q "Error distribution:" ${{ github.workspace }}/output.log && echo true || echo false) | sed 's/true/❌/; s/false/✅/') | |
echo "error=${HAS_ERROR}" | tee -a "$GITHUB_OUTPUT" | |
- name: Extract RPS | |
id: extract_rps | |
run: | | |
RPS=$(grep "Requests/sec:" ${{ github.workspace }}/output.log | awk '{printf int($2)}' | sed -E ':a;s/([0-9])([0-9]{3})$/\1,\2/;ta') | |
echo "rps=${RPS}" | tee -a "$GITHUB_OUTPUT" | |
- name: Extract P95 Latency | |
id: extract_p95 | |
run: | | |
P95=$(grep "95%" ${{ github.workspace }}/output.log | awk '{print $3}') | |
echo "p95=${P95}" | tee -a "$GITHUB_OUTPUT" | |
- uses: cloudposse/github-action-matrix-outputs-write@v1 | |
id: out | |
with: | |
matrix-step-name: ${{ github.job }} | |
matrix-key: ${{ matrix.response }}_${{ matrix.service }} | |
outputs: |- | |
has_error: ${{ steps.extract_has_error.outputs.error }} | |
rps: ${{ steps.extract_rps.outputs.rps }} | |
p95: ${{ steps.extract_p95.outputs.p95 }} | |
read: | |
runs-on: ubuntu-latest | |
needs: [benchmark] | |
steps: | |
- uses: cloudposse/github-action-matrix-outputs-read@v1 | |
id: read | |
with: | |
matrix-step-name: benchmark | |
outputs: | |
result: "${{ steps.read.outputs.result }}" | |
analyze: | |
if: github.event.head_commit.message != 'Update performance results in README.md' | |
needs: read | |
runs-on: ubuntu-latest | |
steps: | |
- name: Checkout | |
uses: actions/checkout@v4 | |
with: | |
token: ${{ secrets.GITHUB_TOKEN }} | |
ref: ${{ github.event_name == 'pull_request_target' && github.event.pull_request.head.sha || github.sha }} | |
- name: Install gnuplot | |
run: sudo apt-get update && sudo apt-get install -y gnuplot | |
- name: Collect results table | |
run: | | |
echo "### [Small Payload - 362 bytes](./source/small.json)" >> "results.md" | |
echo "| Server | Status | RPS | Latency |" >> "results.md" | |
echo "| ---: | ---: | ---: | ---: |" >> "results.md" | |
echo "| [Nginx](https://nginx.org/en/) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_nginx_graphql }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_nginx_graphql }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_nginx_graphql }} sec\` |" >> "results.md" | |
echo "| **Base** | | | |" >> "results.md" | |
echo "| [Tailcall](https://github.com/tailcallhq/tailcall) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_tailcall_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_tailcall_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_tailcall_default }} sec\` |" >> "results.md" | |
echo "| [Grafbase](https://github.com/grafbase/grafbase) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_grafbase_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_grafbase_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_grafbase_default }} sec\` |" >> "results.md" | |
echo "| [Wundegraph](https://github.com/wundergraph/cosmo) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_wundergraph_no_opt }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_wundergraph_no_opt }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_wundergraph_no_opt }} sec\` |" >> "results.md" | |
echo "| [Apollo](https://github.com/apollographql/router) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_apollo_router }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_apollo_router }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_apollo_router }} sec\` |" >> "results.md" | |
echo "| **Cached** | | | |" >> "results.md" | |
echo "| [Tailcall](https://github.com/tailcallhq/tailcall) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_tailcall_full_conf }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_tailcall_full_conf }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_tailcall_full_conf }} sec\` |" >> "results.md" | |
echo "| [Grafbase](https://github.com/grafbase/grafbase) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_grafbase_cache }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_grafbase_cache }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_grafbase_cache }} sec\` |" >> "results.md" | |
echo "| [Wundegraph](https://github.com/wundergraph/cosmo) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.small_wundergraph_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.small_wundergraph_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.small_wundergraph_default }} sec\` |" >> "results.md" | |
echo "### [Medium Payload - 12,598 bytes](./source/medium.json)" >> "results.md" | |
echo "| Server | Status | RPS | Latency |" >> "results.md" | |
echo "| ---: | ---: | ---: | ---: |" >> "results.md" | |
echo "| [Nginx](https://nginx.org/en/) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_nginx_graphql }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_nginx_graphql }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_nginx_graphql }} sec\` |" >> "results.md" | |
echo "| **Base** | | | |" >> "results.md" | |
echo "| [Tailcall](https://github.com/tailcallhq/tailcall) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_tailcall_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_tailcall_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_tailcall_default }} sec\` |" >> "results.md" | |
echo "| [Grafbase](https://github.com/grafbase/grafbase) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_grafbase_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_grafbase_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_grafbase_default }} sec\` |" >> "results.md" | |
echo "| [Wundegraph](https://github.com/wundergraph/cosmo) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_wundergraph_no_opt }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_wundergraph_no_opt }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_wundergraph_no_opt }} sec\` |" >> "results.md" | |
echo "| [Apollo](https://github.com/apollographql/router) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_apollo_router }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_apollo_router }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_apollo_router }} sec\` |" >> "results.md" | |
echo "| **Cached** | | | |" >> "results.md" | |
echo "| [Tailcall](https://github.com/tailcallhq/tailcall) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_tailcall_full_conf }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_tailcall_full_conf }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_tailcall_full_conf }} sec\` |" >> "results.md" | |
echo "| [Grafbase](https://github.com/grafbase/grafbase) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_grafbase_cache }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_grafbase_cache }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_grafbase_cache }} sec\` |" >> "results.md" | |
echo "| [Wundegraph](https://github.com/wundergraph/cosmo) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.medium_wundergraph_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.medium_wundergraph_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.medium_wundergraph_default }} sec\` |" >> "results.md" | |
echo "### [Big Payload - 112,838 bytes](./source/big.json)" >> "results.md" | |
echo "| Server | Status | RPS | Latency |" >> "results.md" | |
echo "| ---: | ---: | ---: | ---: |" >> "results.md" | |
echo "| [Nginx](https://nginx.org/en/) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_nginx_graphql }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_nginx_graphql }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_nginx_graphql }} sec\` |" >> "results.md" | |
echo "| **Base** | | | |" >> "results.md" | |
echo "| [Tailcall](https://github.com/tailcallhq/tailcall) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_tailcall_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_tailcall_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_tailcall_default }} sec\` |" >> "results.md" | |
echo "| [Grafbase](https://github.com/grafbase/grafbase) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_grafbase_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_grafbase_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_grafbase_default }} sec\` |" >> "results.md" | |
echo "| [Wundegraph](https://github.com/wundergraph/cosmo) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_wundergraph_no_opt }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_wundergraph_no_opt }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_wundergraph_no_opt }} sec\` |" >> "results.md" | |
echo "| [Apollo](https://github.com/apollographql/router) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_apollo_router }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_apollo_router }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_apollo_router }} sec\` |" >> "results.md" | |
echo "| **Cached** | | | |" >> "results.md" | |
echo "| [Tailcall](https://github.com/tailcallhq/tailcall) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_tailcall_full_conf }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_tailcall_full_conf }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_tailcall_full_conf }} sec\` |" >> "results.md" | |
echo "| [Grafbase](https://github.com/grafbase/grafbase) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_grafbase_cache }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_grafbase_cache }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_grafbase_cache }} sec\` |" >> "results.md" | |
echo "| [Wundegraph](https://github.com/wundergraph/cosmo) | \ | |
${{ fromJson(needs.read.outputs.result).has_error.big_wundergraph_default }} | \ | |
\`${{ fromJson(needs.read.outputs.result).rps.big_wundergraph_default }} RPS\` | \ | |
\`${{ fromJson(needs.read.outputs.result).p95.big_wundergraph_default }} sec\` |" >> "results.md" | |
- name: Update `README.md` file with table data | |
run: | | |
sed -i '/<!-- PERFORMANCE_RESULTS_START -->/,/<!-- PERFORMANCE_RESULTS_END -->/ { | |
/<!-- PERFORMANCE_RESULTS_START -->/!{ | |
/<!-- PERFORMANCE_RESULTS_END -->/!d | |
} | |
}' README.md | |
sed -i '/<!-- PERFORMANCE_RESULTS_START -->/r results.md' README.md | |
- name: Summarize | |
run: | | |
echo "### Results" >> "$GITHUB_STEP_SUMMARY" | |
cat "results.md" >> "$GITHUB_STEP_SUMMARY" | |
- name: Run gnuplot script | |
run: | | |
cat << 'EOF' > plot_script.gp | |
set terminal png size 800,500 enhanced font "Arial,20" | |
set output 'files/rps_default.png' | |
small = "#D81B60"; medium = "#1E88E5"; big = "#FFC107"; | |
set yrange [0:20] | |
set style data histogram | |
set style histogram cluster gap 1 | |
set style fill solid | |
set boxwidth 0.9 | |
set xtics format "" | |
set grid ytics | |
set title "RPS Default" | |
plot '-' using 2:xtic(1) title "Small" linecolor rgb small, \ | |
'' using 3 title "Medium" linecolor rgb medium, \ | |
'' using 4 title "Big" linecolor rgb big | |
# Inline data | |
# Small Medium Big | |
Tailcall ${{ fromJson(needs.read.outputs.result).rps.small_tailcall_default }} ${{ fromJson(needs.read.outputs.result).rps.medium_tailcall_default }} ${{ fromJson(needs.read.outputs.result).rps.big_tailcall_default }} | |
Grafbase ${{ fromJson(needs.read.outputs.result).rps.small_grafbase_default }} ${{ fromJson(needs.read.outputs.result).rps.medium_grafbase_default }} ${{ fromJson(needs.read.outputs.result).rps.big_grafbase_default }} | |
Wundergraph ${{ fromJson(needs.read.outputs.result).rps.small_wundergraph_no_opt }} ${{ fromJson(needs.read.outputs.result).rps.medium_wundergraph_no_opt }} ${{ fromJson(needs.read.outputs.result).rps.big_wundergraph_no_opt }} | |
Apollo ${{ fromJson(needs.read.outputs.result).rps.small_apollo_router }} ${{ fromJson(needs.read.outputs.result).rps.medium_apollo_router }} ${{ fromJson(needs.read.outputs.result).rps.big_apollo_router }} | |
e | |
EOF | |
gnuplot plot_script.gp | |
- name: Cleanup files | |
run: | | |
rm "results.md" | |
rm "plot_script.gp" | |
- name: Comment `README.md` file | |
if: github.event_name == 'pull_request_target' | |
uses: peter-evans/commit-comment@v3 | |
with: | |
sha: ${{ github.event.pull_request.head.sha }} | |
body-path: "README.md" | |
reactions: eyes | |
- name: Commit and push changes (on main branch) | |
if: github.event_name == 'push' && github.ref == 'refs/heads/main' | |
uses: stefanzweifel/git-auto-commit-action@v5 | |
with: | |
branch: main | |
commit_author: Author <[email protected]> | |
commit_message: "[ci skip] update performance results in README.md" |