diff --git a/.github/workflows/bench.yml b/.github/workflows/bench.yml index 84b7b283..540ffb67 100644 --- a/.github/workflows/bench.yml +++ b/.github/workflows/bench.yml @@ -35,15 +35,19 @@ jobs: token: ${{ secrets.GITHUB_TOKEN }} ref: ${{ github.event_name == 'pull_request_target' && github.event.pull_request.head.sha || github.sha }} - - name: Build devcontainer and run setup and benchmark + - name: Build devcontainer uses: devcontainers/ci@v0.3 with: imageName: graphql-benchmarks push: never - runCmd: | - bash ./graphql/${{ matrix.service }}/setup.sh - bash run_benchmarks.sh ${{ matrix.service }} + - name: Setup k6 + uses: grafana/setup-k6-action@v1 + + - name: Run benchmarks + run: | + bash ./graphql/${{ matrix.service }}/setup.sh + bash run_benchmarks.sh ${{ matrix.service }} - name: List benchmark files run: | ls -la bench*.txt || echo "No matching files found" @@ -57,10 +61,28 @@ jobs: analyze: needs: build runs-on: benchmarking-runner + env: + GRAFANA_API_KEY: ${{ secrets.GRAFANA_API_KEY }} + INFLUXDB_TOKEN: ${{ secrets.INFLUXDB_TOKEN }} + INFLUXDB_ORG: ${{ secrets.INFLUXDB_ORG }} + INFLUXDB_URL: ${{ secrets.INFLUXDB_URL }} steps: - name: Checkout (GitHub) uses: actions/checkout@v4 + - name: Setup influxdb-cli + if: github.event_name == 'push' && github.ref == 'refs/heads/main' + run: | + echo "UPLOAD_TO_CLOUD=true" >> $GITHUB_ENV + mkdir -p influxdb + cd influxdb + wget https://download.influxdata.com/influxdb/releases/influxdb2-client-2.7.5-linux-amd64.tar.gz + tar xvzf influxdb2-client-2.7.5-linux-amd64.tar.gz + cp influx /usr/local/bin/ + influx config create --config-name benchmark --host-url $INFLUXDB_URL --org $INFLUXDB_ORG --token $INFLUXDB_TOKEN --active + cd .. + rm -rf influxdb + - name: Download all benchmark results uses: actions/download-artifact@v3 with: @@ -73,7 +95,6 @@ jobs: - name: Analyze results run: | bash run_analyze_script.sh - - name: Print benchmark results run: cat ./results.md diff --git a/README.md b/README.md index 81e679f2..8f4647e3 100644 --- a/README.md +++ b/README.md @@ -7,10 +7,8 @@ Explore and compare the performance of the fastest GraphQL frameworks through ou - [Introduction](#introduction) - [Quick Start](#quick-start) - [Benchmark Results](#benchmark-results) - - [Throughput (Higher is better)](#throughput-higher-is-better) - - [Latency (Lower is better)](#latency-lower-is-better) - [Architecture](#architecture) - - [WRK](#wrk) + - [K6](#k6) - [GraphQL](#graphql) - [Nginx](#nginx) - [Jsonplaceholder](#jsonplaceholder) @@ -46,6 +44,15 @@ Get started with the benchmarks: ## Benchmark Results +| Throughput (Higher is better) | Latency (Lower is better) | +|-------:|--------:| +| `{{ posts { id userId title user { id name email }}}}` | +| ![](assets/posts_users_req.png) | ![](assets/posts_users_latency.png) | +| `{ posts { title }}` | +| ![](assets/posts_req.png) | ![](assets/posts_latency.png) | +| `{greet}` | +| ![](assets/greet_req.png) | ![](assets/greet_latency.png) | + | Query | Server | Requests/sec | Latency (ms) | Relative | @@ -80,44 +87,15 @@ Get started with the benchmarks: - - -### 1. `{posts {title body user {name}}}` -#### Throughput (Higher is better) - -![Throughput Histogram](assets/req_sec_histogram1.png) - -#### Latency (Lower is better) - -![Latency Histogram](assets/latency_histogram1.png) - -### 2. `{posts {title body}}` -#### Throughput (Higher is better) - -![Throughput Histogram](assets/req_sec_histogram2.png) - -#### Latency (Lower is better) - -![Latency Histogram](assets/latency_histogram2.png) - -### 3. `{greet}` -#### Throughput (Higher is better) - -![Throughput Histogram](assets/req_sec_histogram3.png) - -#### Latency (Lower is better) - -![Latency Histogram](assets/latency_histogram3.png) - ## Architecture ![Architecture Diagram](assets/architecture.png) -A client (`wrk`) sends requests to a GraphQL server to fetch post titles. The GraphQL server, in turn, retrieves data from an external source, `jsonplaceholder.typicode.com`, routed through the `nginx` reverse proxy. +A client (`k6`) sends requests to a GraphQL server to fetch post titles. The GraphQL server, in turn, retrieves data from an external source, `jsonplaceholder.typicode.com`, routed through the `nginx` reverse proxy. -### WRK +### K6 -`wrk` serves as our test client, sending GraphQL requests at a high rate. +`k6` serves as our test client, sending GraphQL requests at a high rate. ### GraphQL diff --git a/analyze.sh b/analyze.sh index a52e8471..172bc49a 100755 --- a/analyze.sh +++ b/analyze.sh @@ -44,20 +44,6 @@ for idx in "${!servers[@]}"; do avgLatencies[${servers[$idx]}]=$(average "${latencyVals[@]}") done -# Generating data files for gnuplot -reqSecData="/tmp/reqSec.dat" -latencyData="/tmp/latency.dat" - -echo "Server Value" >"$reqSecData" -for server in "${servers[@]}"; do - echo "$server ${avgReqSecs[$server]}" >>"$reqSecData" -done - -echo "Server Value" >"$latencyData" -for server in "${servers[@]}"; do - echo "$server ${avgLatencies[$server]}" >>"$latencyData" -done - whichBench=1 if [[ $1 == bench2* ]]; then whichBench=2 @@ -65,36 +51,6 @@ elif [[ $1 == bench3* ]]; then whichBench=3 fi -reqSecHistogramFile="req_sec_histogram${whichBench}.png" -latencyHistogramFile="latency_histogram${whichBench}.png" - -# Plotting using gnuplot -gnuplot <<-EOF - set term pngcairo size 1280,720 enhanced font "Courier,12" - set output "$reqSecHistogramFile" - set style data histograms - set style histogram cluster gap 1 - set style fill solid border -1 - set xtics rotate by -45 - set boxwidth 0.9 - set title "Requests/Sec" - stats "$reqSecData" using 2 nooutput - set yrange [0:STATS_max*1.2] - set key outside right top - plot "$reqSecData" using 2:xtic(1) title "Req/Sec" - - set output "$latencyHistogramFile" - set title "Latency (in ms)" - stats "$latencyData" using 2 nooutput - set yrange [0:STATS_max*1.2] - plot "$latencyData" using 2:xtic(1) title "Latency" -EOF - -# Move PNGs to assets -mkdir -p assets -mv $reqSecHistogramFile assets/ -mv $latencyHistogramFile assets/ - # Declare an associative array for server RPS declare -A serverRPS @@ -123,6 +79,11 @@ fi for server in "${sortedServers[@]}"; do formattedReqSecs=$(printf "%.2f" ${avgReqSecs[$server]} | perl -pe 's/(?<=\d)(?=(\d{3})+(\.\d*)?$)/,/g') formattedLatencies=$(printf "%.2f" ${avgLatencies[$server]} | perl -pe 's/(?<=\d)(?=(\d{3})+(\.\d*)?$)/,/g') + echo "Writing to influx for $server and benchmark $whichBench with ${avgReqSecs[$server]} and ${avgLatencies[$server]}" + influx write -b bench " + http_reqs,test_name=$server,benchmark=$whichBench value=${avgReqSecs[$server]} + latency,test_name=$server,benchmark=$whichBench value=${avgLatencies[$server]} + " # Calculate the relative performance relativePerformance=$(echo "${avgReqSecs[$server]} $lastServerReqSecs" | awk '{printf "%.2f", $1 / $2}') @@ -157,10 +118,6 @@ if [[ $whichBench == 3 ]]; then fi fi -# Move the generated images to the assets folder -mv $reqSecHistogramFile assets/ -mv $latencyHistogramFile assets/ - # Delete the result TXT files for file in "${resultFiles[@]}"; do rm "$file" diff --git a/assets/architecture.png b/assets/architecture.png index 843a5153..daf361dc 100644 Binary files a/assets/architecture.png and b/assets/architecture.png differ diff --git a/assets/canvas.excalidraw b/assets/canvas.excalidraw deleted file mode 100644 index 666000e6..00000000 --- a/assets/canvas.excalidraw +++ /dev/null @@ -1,531 +0,0 @@ -{ - "type": "excalidraw", - "version": 2, - "source": "https://marketplace.visualstudio.com/items?itemName=pomdtr.excalidraw-editor", - "elements": [ - { - "id": "Rsw1rvlpsMVnF6AZxWIkW", - "type": "image", - "x": 1100, - "y": 400, - "width": 96.40234375000007, - "height": 96.40234375000007, - "angle": 0, - "strokeColor": "transparent", - "backgroundColor": "transparent", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 1, - "opacity": 100, - "groupIds": [ - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": null, - "seed": 1699644304, - "version": 636, - "versionNonce": 447823728, - "isDeleted": false, - "boundElements": [ - { - "id": "XXbH3ykZ79DT8eWIm84op", - "type": "arrow" - }, - { - "id": "g5NWEgTJkbmvyDrLogkTK", - "type": "arrow" - } - ], - "updated": 1695740447494, - "link": null, - "locked": false, - "status": "pending", - "fileId": "882593600714d96f8245fd3c5a0f2a9f3554d842", - "scale": [ - 1, - 1 - ] - }, - { - "id": "Etr7dDR8itdEWLgRJvltG", - "type": "text", - "x": 1310.7732633072605, - "y": 420, - "width": 157.22515402305373, - "height": 53.666185906535674, - "angle": 0, - "strokeColor": "#1e1e1e", - "backgroundColor": "transparent", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 1, - "opacity": 100, - "groupIds": [ - "CZveppB0W1fKMrZxAl6M3", - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": null, - "seed": 761551248, - "version": 83, - "versionNonce": 328773520, - "isDeleted": false, - "boundElements": null, - "updated": 1695740447494, - "link": null, - "locked": false, - "text": "{JSON}", - "fontSize": 44.72182158877973, - "fontFamily": 3, - "textAlign": "left", - "verticalAlign": "top", - "baseline": 42.99999999999999, - "containerId": null, - "originalText": "{JSON}", - "lineHeight": 1.2 - }, - { - "id": "9G9G6EhMhBEwYVGnpEWwH", - "type": "text", - "x": 1220, - "y": 492.73701477414687, - "width": 349.3892311623417, - "height": 29.814547725853153, - "angle": 0, - "strokeColor": "#1e1e1e", - "backgroundColor": "transparent", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 1, - "opacity": 100, - "groupIds": [ - "CZveppB0W1fKMrZxAl6M3", - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": null, - "seed": 1163199344, - "version": 157, - "versionNonce": 322425200, - "isDeleted": false, - "boundElements": null, - "updated": 1695740447495, - "link": null, - "locked": false, - "text": "placeholder.typicode.com", - "fontSize": 24.845456438210963, - "fontFamily": 3, - "textAlign": "left", - "verticalAlign": "top", - "baseline": 24, - "containerId": null, - "originalText": "placeholder.typicode.com", - "lineHeight": 1.2 - }, - { - "id": "sq33Pa8q5QOuLhEloCSmK", - "type": "image", - "x": 900, - "y": 400, - "width": 100.00000000000001, - "height": 100.00000000000001, - "angle": 0, - "strokeColor": "transparent", - "backgroundColor": "transparent", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 1, - "opacity": 100, - "groupIds": [ - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": null, - "seed": 1638037872, - "version": 378, - "versionNonce": 972832144, - "isDeleted": false, - "boundElements": [ - { - "id": "XINLfJxqeAEckPm99t5rz", - "type": "arrow" - }, - { - "id": "XXbH3ykZ79DT8eWIm84op", - "type": "arrow" - }, - { - "id": "UzCtCJDfIl-h1RCrG_Y_5", - "type": "arrow" - }, - { - "id": "2vVqwcIk_kRI7H1_R9QpR", - "type": "arrow" - } - ], - "updated": 1695740447495, - "link": null, - "locked": false, - "status": "pending", - "fileId": "c30cf7a84df4d4ae7518d24f694f86a8aa4b22f6", - "scale": [ - 1, - 1 - ] - }, - { - "id": "NuIejDWxUniTTJfkcDkW1", - "type": "rectangle", - "x": 700, - "y": 420, - "width": 100, - "height": 80, - "angle": 0, - "strokeColor": "#343a40", - "backgroundColor": "#ffec99", - "fillStyle": "solid", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 0, - "opacity": 100, - "groupIds": [ - "MZaR2cvEAUpN82_3SSsp3", - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": { - "type": 3 - }, - "seed": 1413848944, - "version": 45, - "versionNonce": 555492208, - "isDeleted": false, - "boundElements": null, - "updated": 1695740447495, - "link": null, - "locked": false - }, - { - "id": "Mfi6c7eDmzW3SD7KY6ADr", - "type": "text", - "x": 720, - "y": 437.0320251177394, - "width": 60, - "height": 44.3679748822606, - "angle": 0, - "strokeColor": "#343a40", - "backgroundColor": "transparent", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 1, - "opacity": 100, - "groupIds": [ - "MZaR2cvEAUpN82_3SSsp3", - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": null, - "seed": 744145808, - "version": 173, - "versionNonce": 683543440, - "isDeleted": false, - "boundElements": [], - "updated": 1695740447495, - "link": null, - "locked": false, - "text": "wrk", - "fontSize": 38.58084772370487, - "fontFamily": 2, - "textAlign": "center", - "verticalAlign": "top", - "baseline": 35, - "containerId": null, - "originalText": "wrk", - "lineHeight": 1.15 - }, - { - "type": "arrow", - "version": 319, - "versionNonce": 1386624368, - "isDeleted": false, - "id": "XXbH3ykZ79DT8eWIm84op", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 0, - "opacity": 100, - "angle": 0, - "x": 1020, - "y": 459.9999999999635, - "strokeColor": "#343a40", - "backgroundColor": "transparent", - "width": 60.65234375, - "height": 1.8985701899509877e-11, - "seed": 2085269360, - "groupIds": [ - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": { - "type": 2 - }, - "boundElements": [], - "updated": 1695740447495, - "link": null, - "locked": false, - "startBinding": { - "elementId": "sq33Pa8q5QOuLhEloCSmK", - "focus": 0.2, - "gap": 20 - }, - "endBinding": { - "elementId": "Rsw1rvlpsMVnF6AZxWIkW", - "focus": -0.24478301389845603, - "gap": 19.34765625 - }, - "lastCommittedPoint": null, - "startArrowhead": null, - "endArrowhead": "triangle", - "points": [ - [ - 0, - 0 - ], - [ - 60.65234375, - 1.8985701899509877e-11 - ] - ] - }, - { - "type": "arrow", - "version": 191, - "versionNonce": 974526864, - "isDeleted": false, - "id": "g5NWEgTJkbmvyDrLogkTK", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 0, - "opacity": 100, - "angle": 0, - "x": 1220, - "y": 460, - "strokeColor": "#343a40", - "backgroundColor": "transparent", - "width": 60.65234375, - "height": 0, - "seed": 2081439600, - "groupIds": [ - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": { - "type": 2 - }, - "boundElements": [], - "updated": 1695740447495, - "link": null, - "locked": false, - "startBinding": { - "elementId": "Rsw1rvlpsMVnF6AZxWIkW", - "focus": 0.24478301389845603, - "gap": 23.59765625 - }, - "endBinding": null, - "lastCommittedPoint": null, - "startArrowhead": null, - "endArrowhead": "triangle", - "points": [ - [ - 0, - 0 - ], - [ - 60.65234375, - 0 - ] - ] - }, - { - "id": "XINLfJxqeAEckPm99t5rz", - "type": "arrow", - "x": 820, - "y": 460, - "width": 60.652343759225914, - "height": 0.0005974943387059284, - "angle": 0, - "strokeColor": "#343a40", - "backgroundColor": "transparent", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 0, - "opacity": 100, - "groupIds": [ - "ssZidudtdkgYlNmYDYucQ", - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": { - "type": 2 - }, - "seed": 373341072, - "version": 242, - "versionNonce": 756942704, - "isDeleted": false, - "boundElements": null, - "updated": 1695740447495, - "link": null, - "locked": false, - "points": [ - [ - 0, - 0 - ], - [ - 60.652343759225914, - 0.0005974943387059284 - ] - ], - "lastCommittedPoint": null, - "startBinding": null, - "endBinding": { - "elementId": "sq33Pa8q5QOuLhEloCSmK", - "focus": -0.20002364248762314, - "gap": 19.347656240774086 - }, - "startArrowhead": null, - "endArrowhead": "triangle" - }, - { - "type": "arrow", - "version": 273, - "versionNonce": 1094477712, - "isDeleted": false, - "id": "2vVqwcIk_kRI7H1_R9QpR", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 0, - "opacity": 100, - "angle": 0, - "x": 820, - "y": 440, - "strokeColor": "#343a40", - "backgroundColor": "transparent", - "width": 60, - "height": 0.00001876576482118253, - "seed": 1380811632, - "groupIds": [ - "I_azqQXfVWz-oycklVOyg", - "ssZidudtdkgYlNmYDYucQ", - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": { - "type": 2 - }, - "boundElements": [], - "updated": 1695740447495, - "link": null, - "locked": false, - "startBinding": null, - "endBinding": { - "elementId": "sq33Pa8q5QOuLhEloCSmK", - "focus": 0.19999912426458225, - "gap": 20 - }, - "lastCommittedPoint": null, - "startArrowhead": null, - "endArrowhead": "triangle", - "points": [ - [ - 0, - 0 - ], - [ - 60, - 0.00001876576482118253 - ] - ] - }, - { - "type": "arrow", - "version": 179, - "versionNonce": 1288411504, - "isDeleted": false, - "id": "UzCtCJDfIl-h1RCrG_Y_5", - "fillStyle": "hachure", - "strokeWidth": 4, - "strokeStyle": "solid", - "roughness": 0, - "opacity": 100, - "angle": 0, - "x": 820, - "y": 480, - "strokeColor": "#343a40", - "backgroundColor": "transparent", - "width": 60, - "height": 0, - "seed": 158820208, - "groupIds": [ - "13ZV-TJqzrAOQRakletZR", - "ssZidudtdkgYlNmYDYucQ", - "FaBNXvUjSPnNDD11T70FA" - ], - "frameId": null, - "roundness": { - "type": 2 - }, - "boundElements": [], - "updated": 1695740447495, - "link": null, - "locked": false, - "startBinding": null, - "endBinding": { - "elementId": "sq33Pa8q5QOuLhEloCSmK", - "focus": -0.5999999999999999, - "gap": 20 - }, - "lastCommittedPoint": null, - "startArrowhead": null, - "endArrowhead": "triangle", - "points": [ - [ - 0, - 0 - ], - [ - 60, - 0 - ] - ] - } - ], - "appState": { - "gridSize": 20, - "viewBackgroundColor": "#ffffff" - }, - "files": { - "882593600714d96f8245fd3c5a0f2a9f3554d842": { - "mimeType": "image/png", - "id": "882593600714d96f8245fd3c5a0f2a9f3554d842", - "dataURL": "", - "created": 1695739935557, - "lastRetrieved": 1695739935557 - }, - "c30cf7a84df4d4ae7518d24f694f86a8aa4b22f6": { - "mimeType": "image/svg+xml", - "id": "c30cf7a84df4d4ae7518d24f694f86a8aa4b22f6", - "dataURL": "", - "created": 1695740054415, - "lastRetrieved": 1695740054415 - } - } -} \ No newline at end of file diff --git a/assets/latency_histogram1.png b/assets/latency_histogram1.png deleted file mode 100644 index 74bae03b..00000000 Binary files a/assets/latency_histogram1.png and /dev/null differ diff --git a/assets/latency_histogram2.png b/assets/latency_histogram2.png deleted file mode 100644 index f4593830..00000000 Binary files a/assets/latency_histogram2.png and /dev/null differ diff --git a/assets/latency_histogram3.png b/assets/latency_histogram3.png deleted file mode 100644 index f59dd4ef..00000000 Binary files a/assets/latency_histogram3.png and /dev/null differ diff --git a/assets/req_sec_histogram1.png b/assets/req_sec_histogram1.png deleted file mode 100644 index 11375dbc..00000000 Binary files a/assets/req_sec_histogram1.png and /dev/null differ diff --git a/assets/req_sec_histogram2.png b/assets/req_sec_histogram2.png deleted file mode 100644 index 57223b25..00000000 Binary files a/assets/req_sec_histogram2.png and /dev/null differ diff --git a/assets/req_sec_histogram3.png b/assets/req_sec_histogram3.png deleted file mode 100644 index 7e886b92..00000000 Binary files a/assets/req_sec_histogram3.png and /dev/null differ diff --git a/fetch_panels.sh b/fetch_panels.sh new file mode 100644 index 00000000..6f391075 --- /dev/null +++ b/fetch_panels.sh @@ -0,0 +1,9 @@ + # Get rendered panels from grafana + from=$(date -u -d "-30 minutes" +"%Y-%m-%dT%H:%M:%S.%3NZ") + now=$(date -u +"%Y-%m-%dT%H:%M:%S.%3NZ") + curl -o assets/posts_users_req.png -H "Authorization: Bearer $GRAFANA_API_KEY" "https://tailcall.grafana.net/render/d-solo/cdqucydulbfggb?tab=queries&from=$from&to=$now&panelId=panel-1&__feature.dashboardSceneSolo&width=1000&height=500&tz=Asia%2FCalcutta" --connect-timeout 120 + curl -o assets/posts_users_latency.png -H "Authorization: Bearer $GRAFANA_API_KEY" "https://tailcall.grafana.net/render/d-solo/cdqucydulbfggb?tab=queries&from=$from&to=$now&panelId=panel-2&__feature.dashboardSceneSolo&width=1000&height=500&tz=Asia%2FCalcutta" --connect-timeout 120 + curl -o assets/posts_req.png -H "Authorization: Bearer $GRAFANA_API_KEY" "https://tailcall.grafana.net/render/d-solo/cdqucydulbfggb?tab=queries&from=$from&to=$now&panelId=panel-5&__feature.dashboardSceneSolo&width=1000&height=500&tz=Asia%2FCalcutta" --connect-timeout 120 + curl -o assets/posts_latency.png -H "Authorization: Bearer $GRAFANA_API_KEY" "https://tailcall.grafana.net/render/d-solo/cdqucydulbfggb?tab=queries&from=$from&to=$now&panelId=panel-6&__feature.dashboardSceneSolo&width=1000&height=500&tz=Asia%2FCalcutta" --connect-timeout 120 + curl -o assets/greet_req.png -H "Authorization: Bearer $GRAFANA_API_KEY" "https://tailcall.grafana.net/render/d-solo/cdqucydulbfggb?tab=queries&from=$from&to=$now&panelId=panel-8&__feature.dashboardSceneSolo&width=1000&height=500&tz=Asia%2FCalcutta" --connect-timeout 120 + curl -o assets/greet_latency.png -H "Authorization: Bearer $GRAFANA_API_KEY" "https://tailcall.grafana.net/render/d-solo/cdqucydulbfggb?tab=queries&from=$from&to=$now&panelId=panel-9&__feature.dashboardSceneSolo&width=1000&height=500&tz=Asia%2FCalcutta" --connect-timeout 120 diff --git a/k6/bench.js b/k6/bench.js new file mode 100644 index 00000000..d923ea2a --- /dev/null +++ b/k6/bench.js @@ -0,0 +1,50 @@ +import http from 'k6/http'; +import { check } from 'k6'; + +const whichBenchmark = Number(__ENV.BENCHMARK); +const benchmarkName = whichBenchmark === 1 ? 'posts_users' : 2 ? 'posts' : 'greet'; +const duration = whichBenchmark === 1 ? '30' : '10'; + +export const options = { + scenarios: { + [__ENV.TEST_NAME + '-' + benchmarkName]: { + executor: 'constant-vus', + duration: duration+'s', + gracefulStop: '0s', + vus: 100, + }, + }, +}; + +const url = __ENV.GRAPHQL_ENDPOINT || 'http://localhost:8000/graphql'; +const params = { + headers: { + 'Connection': 'keep-alive', + 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36', + 'Content-Type': 'application/json', + }, +}; + +export default function() { + const payload = JSON.stringify({ + operationName: null, + variables: {}, + query: whichBenchmark === 1 ? '{posts{id,userId,title,user{id,name,email}}}' : 2 ? '{posts{title}}' : '{greet}', + }); + + const res = http.post(url, payload, params); + check(res, { + 'status is 200': (r) => r.status === 200, + }); +} + +export function handleSummary(data) { + const requestCount = data.metrics.http_reqs.values.count/duration; + const avgLatency = Math.round(data.metrics.http_req_duration.values.avg * 100) / 100; + const requestCountMessage = `Requests/sec: ${requestCount}\n`; + const latencyMessage = `Latency: ${avgLatency} ms\n`; + + return { + stdout: requestCountMessage + latencyMessage, + }; +} diff --git a/k6/bench.sh b/k6/bench.sh new file mode 100644 index 00000000..f82dbe0e --- /dev/null +++ b/k6/bench.sh @@ -0,0 +1,5 @@ +graphql_endpoint=$1 +test_name=$2 +benchmark=$3 + +k6 run k6/bench.js --env TEST_NAME=$test_name --env BENCHMARK=$benchmark --env GRAPHQL_ENDPOINT=$graphql_endpoint --quiet diff --git a/run_analyze_script.sh b/run_analyze_script.sh index 84f7206a..a133a7c6 100755 --- a/run_analyze_script.sh +++ b/run_analyze_script.sh @@ -33,4 +33,8 @@ for bench in 1 2 3; do # Execute the command echo "Executing: $cmd" eval $cmd -done \ No newline at end of file +done + +if [[ "$UPLOAD_TO_CLOUD" == "true" ]]; then + bash fetch_panels.sh +fi \ No newline at end of file diff --git a/run_benchmarks.sh b/run_benchmarks.sh index 9c1b1aa8..976df80f 100755 --- a/run_benchmarks.sh +++ b/run_benchmarks.sh @@ -22,10 +22,11 @@ sh nginx/run.sh function runBenchmark() { killServerOnPort 8000 sleep 5 - local serviceScript="$1" + local service="$1" + local serviceScript="graphql/${service}/run.sh" local benchmarks=(1 2 3) - if [[ "$serviceScript" == *"hasura"* ]]; then + if [[ "$service" == "hasura" ]]; then bash "$serviceScript" # Run synchronously without background process else bash "$serviceScript" & # Run in daemon mode @@ -34,12 +35,12 @@ function runBenchmark() { sleep 15 # Give some time for the service to start up local graphqlEndpoint="http://localhost:8000/graphql" - if [[ "$serviceScript" == *"hasura"* ]]; then + if [[ "$service" == "hasura" ]]; then graphqlEndpoint=http://127.0.0.1:8080/v1/graphql fi for bench in "${benchmarks[@]}"; do - local benchmarkScript="wrk/bench.sh" + local benchmarkScript="k6/bench.sh" # Replace / with _ local sanitizedServiceScriptName=$(echo "$serviceScript" | tr '/' '_') @@ -59,7 +60,7 @@ function runBenchmark() { # 3 benchmark runs for resultFile in "${resultFiles[@]}"; do echo "Running benchmark $bench for $serviceScript" - bash "$benchmarkScript" "$graphqlEndpoint" "$bench" >"bench${bench}_${resultFile}" + bash "$benchmarkScript" "$graphqlEndpoint" "$service" "$bench" > "bench${bench}_${resultFile}" if [ "$bench" == "1" ]; then bench1Results+=("bench1_${resultFile}") elif [ "$bench" == "2" ]; then @@ -88,9 +89,7 @@ if [[ ! " ${valid_services[@]} " =~ " ${service} " ]]; then exit 1 fi - - -runBenchmark "graphql/${service}/run.sh" +runBenchmark "${service}" if [ "$service" == "apollo_server" ]; then cd graphql/apollo_server/ diff --git a/wrk/bench.sh b/wrk/bench.sh deleted file mode 100755 index a271a4b7..00000000 --- a/wrk/bench.sh +++ /dev/null @@ -1,8 +0,0 @@ -graphqlEndpoint="${1:-http://localhost:8000/graphql}" -whichBench=$2 - -if [ "$whichBench" == "2" ]; then - wrk -d 30 -t 4 -c 100 -s $(pwd)/wrk/wrk2.lua "$graphqlEndpoint" -else - wrk -d 10 -t 4 -c 100 -s "$(pwd)/wrk/wrk${whichBench}.lua" "$graphqlEndpoint" -fi diff --git a/wrk/wrk1.lua b/wrk/wrk1.lua deleted file mode 100644 index 2c8ad791..00000000 --- a/wrk/wrk1.lua +++ /dev/null @@ -1,6 +0,0 @@ -wrk.method = "POST" -wrk.body = '{"operationName":null,"variables":{},"query":"{posts{id,userId,title,user{id,name,email}}}"}' -wrk.headers["Connection"] = "keep-alive" -wrk.headers["User-Agent"] = - "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36" -wrk.headers["Content-Type"] = "application/json" diff --git a/wrk/wrk2.lua b/wrk/wrk2.lua deleted file mode 100644 index ba4bc632..00000000 --- a/wrk/wrk2.lua +++ /dev/null @@ -1,6 +0,0 @@ -wrk.method = "POST" -wrk.body = '{"operationName":null,"variables":{},"query":"{posts{title}}"}' -wrk.headers["Connection"] = "keep-alive" -wrk.headers["User-Agent"] = - "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36" -wrk.headers["Content-Type"] = "application/json" diff --git a/wrk/wrk3.lua b/wrk/wrk3.lua deleted file mode 100644 index 362cf4fe..00000000 --- a/wrk/wrk3.lua +++ /dev/null @@ -1,6 +0,0 @@ -wrk.method = "POST" -wrk.body = '{"operationName":null,"variables":{},"query":"{greet}"}' -wrk.headers["Connection"] = "keep-alive" -wrk.headers["User-Agent"] = - "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36" -wrk.headers["Content-Type"] = "application/json"