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

feat(@libp2p/protocol-perf): Implement perf protocol #1604

Merged
merged 38 commits into from
Aug 11, 2023
Merged
Show file tree
Hide file tree
Changes from 32 commits
Commits
Show all changes
38 commits
Select commit Hold shift + click to select a range
c524755
feat: implement perf protocol
maschad Jun 19, 2023
0898dd0
test: only runs test in node environment
maschad Jun 21, 2023
2458fbc
test: use CI to output test results
maschad Jun 21, 2023
8aee800
chore: linting fixes + remove unused deps
maschad Jun 21, 2023
2ec297d
Merge branch 'master' into marco/perf
maschad Jun 21, 2023
8ea1870
test: updated workflow
maschad Jun 21, 2023
4de5595
ci: updated workflow to fail if bandwidth drops below 20%
maschad Jun 23, 2023
9a7f9d8
ci: updated job to generate report for performance data
maschad Jun 27, 2023
65bfaf3
Merge branch 'master' into marco/perf
maschad Jun 27, 2023
9bed517
chore: linting + updated ci job
maschad Jun 28, 2023
c1b262f
ci: updated saving performance data
maschad Jun 28, 2023
fe79a72
ci: update rendering
maschad Jun 28, 2023
d28dd92
Merge branch 'master' into marco/perf
maschad Jun 28, 2023
9dd0248
test: update perf test to take an average of 5 runs + standardize bytes
maschad Jun 29, 2023
6661424
ci: Added units to output matrix
maschad Jun 29, 2023
79cbe19
test: updated previous bandwidths
maschad Jun 29, 2023
118703a
ci: updated save performance action
maschad Jun 29, 2023
7eb1a79
wip
maschad Jul 26, 2023
e21cda8
Merge branch 'master' into marco/perf
maschad Jul 26, 2023
b1d89bd
feat: added perf test and output to JSON
maschad Jul 26, 2023
ccd5f26
chore: linting fixes
maschad Jul 26, 2023
cf0981b
Merge branch 'master' into marco/perf
maschad Jul 28, 2023
7f9c716
deps(dev): upgrade aegir
maschad Jul 28, 2023
35c79a4
deps: install missing deps
maschad Jul 28, 2023
ec98ddf
Merge branch 'master' into marco/perf
maschad Jul 28, 2023
8ba8db3
refactor: refactored main runner
maschad Jul 30, 2023
59dc3aa
Merge branch 'master' into marco/perf
maschad Jul 30, 2023
f9b91cb
feat: updated main to use fake peer id for client mode perf
maschad Aug 2, 2023
68dfd68
feat: fixed issue with run server + added stable peer id
maschad Aug 4, 2023
0bff620
refactor: Updated tests + perf signature + linting
maschad Aug 4, 2023
94caca8
Merge branch 'master' into marco/perf
maschad Aug 4, 2023
fb914fd
deps: updated deps
maschad Aug 4, 2023
a28ff46
feat: address perf review comments
maschad Aug 5, 2023
8b056fb
Merge branch 'master' into marco/perf
maschad Aug 5, 2023
5e62a7b
refactor: removed maxInbound and outbound streams on handler + bandwi…
maschad Aug 8, 2023
3ad9ea3
Merge branch 'master' into marco/perf
maschad Aug 8, 2023
0a4eae5
refactor: rename to protocol-perf + update package documentation
maschad Aug 10, 2023
b2da056
Merge branch 'master' into marco/perf
maschad Aug 10, 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
158 changes: 158 additions & 0 deletions .github/workflows/perf-test.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,158 @@
name: Performance Test
maschad marked this conversation as resolved.
Show resolved Hide resolved
on:
push:
branches:
- master
pull_request:
branches:
- "**"

jobs:
performance_test:
name: Run performance tests
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: ipfs/aegir/actions/cache-node-modules@master
with:
directories: ./packages/perf/node_modules

- name: Install playwright dependencies
uses: ipfs/aegir/actions/cache-node-modules@master
- run: npx playwright install-deps
maschad marked this conversation as resolved.
Show resolved Hide resolved

- name: Build perf
working-directory: packages/perf
run: (npm i && npm run build)
maschad marked this conversation as resolved.
Show resolved Hide resolved

- name: Run tests and store in test_output
working-directory: packages/perf
run: |
npm run test > test_output.txt

- name: Extract bandwidth from test output
id: extract_bandwidths
working-directory: packages/perf
shell: bash
run: |
maschad marked this conversation as resolved.
Show resolved Hide resolved
bandwidth_file="test_output.txt"
bandwidth_object="{"

# Function to assign bandwidth values to the object
assign_bandwidth() {
local environment=$1
local current_upload_bandwidth=$2
local current_download_bandwidth=$3

if [[ ${#bandwidth_object} -gt 1 ]]; then
bandwidth_object+=", "
fi

bandwidth_object+="\"$environment\": { \"currentUploadBandwidth\": $current_upload_bandwidth, \"currentDownloadBandwidth\": $current_download_bandwidth }"
}

# Read the bandwidth values from the file
upload_count=0
download_count=0

upload_bandwidths=()
download_bandwidths=()

while IFS= read -r line; do
if [[ $line == *"Upload bandwidth"* ]]; then
current_upload_bandwidth=$(grep -o 'Upload bandwidth: [0-9]*' <<< "$line" | grep -o '[0-9]*')
upload_count=$((upload_count + 1))
elif [[ $line == *"Download bandwidth"* ]]; then
current_download_bandwidth=$(grep -o 'Download bandwidth: [0-9]*' <<< "$line" | grep -o '[0-9]*')
download_count=$((download_count + 1))
fi

if [[ $upload_count -eq 1 && $download_count -eq 1 ]]; then
upload_bandwidths[0]=$current_upload_bandwidth
download_bandwidths[0]=$current_download_bandwidth
elif [[ $upload_count -eq 2 && $download_count -eq 2 ]]; then
upload_bandwidths[1]=$current_upload_bandwidth
download_bandwidths[1]=$current_download_bandwidth
elif [[ $upload_count -eq 3 && $download_count -eq 3 ]]; then
upload_bandwidths[2]=$current_upload_bandwidth
download_bandwidths[2]=$current_download_bandwidth
break # Exit the loop after processing the third instance
fi
done < "$bandwidth_file"

# Construct the bandwidth object
assign_bandwidth "node" "${upload_bandwidths[0]}" "${download_bandwidths[0]}"
assign_bandwidth "browser" "${upload_bandwidths[1]}" "${download_bandwidths[1]}"
assign_bandwidth "webworker" "${upload_bandwidths[2]}" "${download_bandwidths[2]}"

# Remove trailing comma if there are values in the object
if [[ ${#bandwidth_object} -gt 1 ]]; then
bandwidth_object="${bandwidth_object#, }"
fi

bandwidth_object+="}"

current_bandwidths=$bandwidth_object

echo "::set-output name=current_bandwidths::$current_bandwidths"

# Read the previous bandwidth values from the file
previous_bandwidths=$(cat previousPerf.json | jq -r '.')

# Iterate over the current bandwidth object and merge with previous values
merged_object=$(jq --argjson previous "$previous_bandwidths" '. as $current | reduce keys[] as $key ({}; .[$key] = ($current[$key] + $previous[$key]))' <<< "$current_bandwidths")

# Set the merged bandwidth object as environment variable
EOF=$(dd if=/dev/urandom bs=15 count=1 status=none | base64)
echo "BANDWIDTH_TEST_RESULTS<<$EOF" >> "$GITHUB_ENV"
echo $merged_object >> "$GITHUB_ENV"
echo "$EOF" >> "$GITHUB_ENV"

- name: Generate Test Report
id: generate_report
working-directory: packages/perf
run: npm run renderResults > test_report.md

- name: Upload Test Report
id: upload_report
uses: actions/upload-artifact@v3
with:
name: test_report
path: packages/perf/test_report.md

- name: Show Report Output
working-directory: packages/perf
run: cat test_report.md >> $GITHUB_STEP_SUMMARY

- name: Save current performance data
id: save_performance_data
working-directory: packages/perf
shell: bash
run: |
current_bandwidths=$(echo '${{ steps.extract_bandwidths.outputs.current_bandwidths }}' | jq -r '.')

# Parse the input object using jq, modify the keys, and store the result in a new variable
updated_object=$(echo "$current_bandwidths" | jq 'map_values(.previousUploadBandwidth = .currentUploadBandwidth | del(.currentUploadBandwidth) | .previousDownloadBandwidth = .currentDownloadBandwidth | del(.currentDownloadBandwidth))')

# Output the updated object
echo $updated_object > previousPerf.json

- name: Commit and push changes
uses: stefanzweifel/git-auto-commit-action@v4
with:
commit_message: Update performance data
file_pattern: "*.json"
branch: ${{ github.ref }}

- name: Check CI job status
id: check_ci
working-directory: packages/perf
run: |
# Check if the file contains the "❌" symbol
if grep -q "❌" test_report.md; then
echo "Failure: There was a performance degradation of over 5%."
maschad marked this conversation as resolved.
Show resolved Hide resolved
exit 1 # Exit with a non-zero status code for failure
else
echo "Success: Performance benchmarks passed."
exit 0 # Exit with a zero status code for pass
fi
12 changes: 12 additions & 0 deletions packages/libp2p/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -52,10 +52,22 @@
"types": "./dist/src/autonat/index.d.ts",
"import": "./dist/src/autonat/index.js"
},
"./address-manager": {
maschad marked this conversation as resolved.
Show resolved Hide resolved
"types": "./dist/src/address-manager/index.d.ts",
"import": "./dist/src/address-manager/index.js"
},
"./circuit-relay": {
"types": "./dist/src/circuit-relay/index.d.ts",
"import": "./dist/src/circuit-relay/index.js"
},
"./components": {
maschad marked this conversation as resolved.
Show resolved Hide resolved
"types": "./dist/src/components.d.ts",
"import": "./dist/src/components.js"
},
"./connection-manager": {
maschad marked this conversation as resolved.
Show resolved Hide resolved
"types": "./dist/src/connection-manager/index.d.ts",
"import": "./dist/src/connection-manager/index.js"
},
"./fetch": {
"types": "./dist/src/fetch/index.d.ts",
"import": "./dist/src/fetch/index.js"
Expand Down
3 changes: 3 additions & 0 deletions packages/perf/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
node_modules/
maschad marked this conversation as resolved.
Show resolved Hide resolved

dist/
4 changes: 4 additions & 0 deletions packages/perf/LICENSE
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
This project is dual licensed under MIT and Apache-2.0.

MIT: https://www.opensource.org/licenses/mit
Apache-2.0: https://www.apache.org/licenses/license-2.0
5 changes: 5 additions & 0 deletions packages/perf/LICENSE-APACHE
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
19 changes: 19 additions & 0 deletions packages/perf/LICENSE-MIT
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
The MIT License (MIT)

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
24 changes: 24 additions & 0 deletions packages/perf/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
# @libp2p/perf

[![libp2p.io](https://img.shields.io/badge/project-libp2p-yellow.svg?style=flat-square)](http://libp2p.io/)
[![Discuss](https://img.shields.io/discourse/https/discuss.libp2p.io/posts.svg?style=flat-square)](https://discuss.libp2p.io)
[![codecov](https://img.shields.io/codecov/c/github/libp2p/js-libp2p.svg?style=flat-square)](https://codecov.io/gh/libp2p/js-libp2p)
[![CI](https://img.shields.io/github/actions/workflow/status/libp2p/js-libp2p/main.yml?branch=master\&style=flat-square)](https://github.com/libp2p/js-libp2p/actions/workflows/main.yml?query=branch%3Amaster)

> Implementation of the [Perf Protocol](https://github.com/libp2p/specs/blob/master/perf/perf.md)
## API Docs

- <https://libp2p.github.io/js-libp2p/modules/_libp2p_perf.html>


## License

Licensed under either of

- Apache 2.0, ([LICENSE-APACHE](LICENSE-APACHE) / <http://www.apache.org/licenses/LICENSE-2.0>)
- MIT ([LICENSE-MIT](LICENSE-MIT) / <http://opensource.org/licenses/MIT>)

## Contribution

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.
68 changes: 68 additions & 0 deletions packages/perf/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
{
"name": "@libp2p/perf",
"version": "1.0.0",
"description": "Implementation of Perf Protocol",
"license": "Apache-2.0 OR MIT",
"homepage": "https://github.com/libp2p/js-libp2p/tree/master/packages/perf#readme",
"repository": {
"type": "git",
"url": "git+https://github.com/libp2p/js-libp2p.git"
},
"bugs": {
"url": "https://github.com/libp2p/js-libp2p/issues"
},
"type": "module",
"author": "@maschad / @marcopolo",
"files": [
"src",
"dist",
"!dist/test",
"!**/*.tsbuildinfo"
],
"eslintConfig": {
"extends": "ipfs",
"parserOptions": {
"sourceType": "module"
}
},
"scripts": {
"start": "node dist/src/main.js",
"build": "aegir build",
"test": "aegir test",
"clean": "aegir clean",
"lint": "aegir lint",
"test:chrome": "aegir test -t browser --cov",
"test:chrome-webworker": "aegir test -t webworker",
"test:firefox": "aegir test -t browser -- --browser firefox",
"test:firefox-webworker": "aegir test -t webworker -- --browser firefox",
"test:node": "aegir test -t node --cov",
"dep-check": "aegir dep-check",
"renderResults": "node dist/src/renderResults.js"
},
"dependencies": {
"@chainsafe/libp2p-yamux": "^5.0.0",
"@libp2p/crypto": "^2.0.0",
"@libp2p/interface": "^0.1.0",
"@libp2p/interface-compliance-tests": "^4.0.0",
"@libp2p/interface-internal": "^0.1.0",
"@libp2p/interfaces": "3.3.2",
"@libp2p/logger": "^3.0.0",
"@libp2p/peer-id-factory": "3.0.0",
"@libp2p/peer-store": "9.0.0",
"@libp2p/tcp": "^8.0.0",
"@multiformats/multiaddr": "^12.1.5",
"any-signal": "^4.1.1",
"datastore-core": "9.2.2",
"libp2p": "^0.46.1",
"p-wait-for": "^5.0.2",
"uint8arrays": "^4.0.6",
"yargs": "^17.7.2"
},
"devDependencies": {
"aegir": "^40.0.8",
"sinon-ts": "1.0.0"
maschad marked this conversation as resolved.
Show resolved Hide resolved
},
"typedoc": {
"entryPoint": "./src/index.ts"
}
maschad marked this conversation as resolved.
Show resolved Hide resolved
}
5 changes: 5 additions & 0 deletions packages/perf/previousPerf.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
maschad marked this conversation as resolved.
Show resolved Hide resolved
"browser": { "previousUploadBandwidth": 87900055, "previousDownloadBandwidth": 87736073 },
"node": { "previousUploadBandwidth": 93373758, "previousDownloadBandwidth": 93798254 },
"webworker": { "previousUploadBandwidth": 85991114, "previousDownloadBandwidth": 85401884 }
}
6 changes: 6 additions & 0 deletions packages/perf/src/constants.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
export const MAX_INBOUND_STREAMS = 1 << 10
export const MAX_OUTBOUND_STREAMS = 1 << 10
export const PROTOCOL_NAME = '/perf/1.0.0'

export const TIMEOUT = 10000
maschad marked this conversation as resolved.
Show resolved Hide resolved
export const WRITE_BLOCK_SIZE = BigInt(64 << 10)
Loading