Compare Performance #604
  
    
      This file contains hidden or 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: Compare Performance | |
| env: | |
| REPORTS_ARTIFACT_PREFIX: reports- | |
| # See: https://docs.github.com/en/free-pro-team@latest/actions/reference/events-that-trigger-workflows | |
| on: | |
| push: | |
| paths: | |
| - ".github/workflows/compare-performance.ya?ml" | |
| - "**/go.mod" | |
| - "**/go.sum" | |
| - "Taskfile.ya?ml" | |
| - "**.go" | |
| pull_request: | |
| paths: | |
| - ".github/workflows/compare-performance.ya?ml" | |
| - "**/go.mod" | |
| - "**/go.sum" | |
| - "Taskfile.ya?ml" | |
| - "**.go" | |
| schedule: | |
| # Run periodically to catch breakage caused by external changes. | |
| - cron: "0 9 * * THU" | |
| workflow_dispatch: | |
| inputs: | |
| comparison-ref: | |
| description: Comparison ref | |
| jobs: | |
| init: | |
| runs-on: ubuntu-latest | |
| permissions: {} | |
| outputs: | |
| base-ref: ${{ steps.base-ref.outputs.ref }} | |
| steps: | |
| # Repo is required to get the previous tag ref that is the base of the comparison on tag push triggered run. | |
| - name: Checkout repository | |
| if: github.event_name == 'push' | |
| uses: actions/checkout@v5 | |
| with: | |
| # Parent commit is needed for determining the base ref on commit push trigg. | |
| fetch-depth: 2 | |
| - name: Determine comparison ref | |
| id: base-ref | |
| run: | | |
| if [[ "${{ github.event_name }}" == "workflow_dispatch" ]]; then | |
| COMPARISON_REF="${{ github.event.inputs.comparison-ref }}" | |
| elif [[ "${{ github.event_name }}" == "pull_request" ]]; then | |
| COMPARISON_REF="${{ github.base_ref }}" | |
| elif [[ "${{ startsWith(github.ref, 'refs/tags/') }}" == "true" ]]; then | |
| COMPARISON_REF="$( \ | |
| git ls-remote \ | |
| --quiet \ | |
| --tags \ | |
| --refs \ | |
| --sort=version:refname | \ | |
| cut \ | |
| --delimiter='/' \ | |
| --fields=3 | \ | |
| tail -2 | \ | |
| head -1 | |
| )" | |
| else | |
| COMPARISON_REF="$(git rev-parse ${{ github.sha }}^)" | |
| fi | |
| if [[ "$COMPARISON_REF" == "" ]]; then | |
| echo "::error::Unable to determine comparison ref" | |
| exit 1 | |
| fi | |
| echo "ref=$COMPARISON_REF" >> $GITHUB_OUTPUT | |
| run: | |
| name: Run at ${{ matrix.data.ref }} (${{ matrix.data.description }}) | |
| needs: init | |
| runs-on: ubuntu-latest | |
| permissions: {} | |
| strategy: | |
| matrix: | |
| data: | |
| # Use two copies of each job to catch job-specific anomalous durations. | |
| - artifact-suffix: tip-run-1 | |
| ref: ${{ github.ref }} # The tip of the branch selected in the workflow dispatch dialog's "Use workflow from" menu | |
| description: tip run 1 | |
| position: after | |
| - artifact-suffix: tip-run-2 | |
| ref: ${{ github.ref }} | |
| description: tip run 2 | |
| position: after | |
| - artifact-suffix: comparison-run-1 | |
| ref: ${{ needs.init.outputs.base-ref }} | |
| description: comparison run 1 | |
| position: before | |
| - artifact-suffix: comparison-run-2 | |
| ref: ${{ needs.init.outputs.base-ref }} | |
| description: comparison run 2 | |
| position: before | |
| steps: | |
| - name: Set environment variables | |
| run: | | |
| # See: https://docs.github.com/en/actions/reference/workflow-commands-for-github-actions#setting-an-environment-variable | |
| ENGINE_DATA_PATH="${{ runner.temp }}/engine" | |
| mkdir --parents "$ENGINE_DATA_PATH" | |
| echo "ENGINE_DATA_PATH=${ENGINE_DATA_PATH}" >> "$GITHUB_ENV" | |
| echo "GIT_CLONES_PATH=${ENGINE_DATA_PATH}/gitclones" >> "$GITHUB_ENV" | |
| echo "LIBRARY_ARCHIVES_PATH=${ENGINE_DATA_PATH}/libraries" >> "$GITHUB_ENV" | |
| echo "LOGS_PATH=${ENGINE_DATA_PATH}/logs" >> "$GITHUB_ENV" | |
| echo "CONFIG_PATH=${ENGINE_DATA_PATH}/config.json" >> "$GITHUB_ENV" | |
| echo "REGISTRY_PATH=${ENGINE_DATA_PATH}/registry.txt" >> "$GITHUB_ENV" | |
| echo "REPORTS_PATH=${ENGINE_DATA_PATH}/reports" >> "$GITHUB_ENV" | |
| - name: Checkout repository | |
| uses: actions/checkout@v5 | |
| with: | |
| ref: ${{ matrix.data.ref }} | |
| - name: Determine appropriate Go version | |
| id: go-version | |
| run: | | |
| if [[ -f "go.mod" ]]; then | |
| # This will use the runner's pre-installed Go | |
| USE_GO_VERSION="$(go mod edit -json | jq --raw-output '.Go')" | |
| else | |
| # Dependency installation for old engine versions fails when not in GOPATH mode. Go <1.16 uses | |
| # GO111MODULE=auto by default, meaning it will use GOPATH mode. Old Go versions were used by the old engine | |
| # anyway. | |
| USE_GO_VERSION="1.14" | |
| fi | |
| echo "version=$USE_GO_VERSION" >> $GITHUB_OUTPUT | |
| - name: Install Go | |
| uses: actions/setup-go@v6 | |
| with: | |
| go-version: ${{ steps.go-version.outputs.version }} | |
| - name: Install Task | |
| uses: arduino/setup-task@v2 | |
| with: | |
| repo-token: ${{ secrets.GITHUB_TOKEN }} | |
| version: 3.x | |
| - name: Install latest release of Arduino Lint | |
| run: | | |
| ARDUINO_LINT_INSTALLATION_PATH="${{ runner.temp }}/arduino-lint" | |
| mkdir --parents "$ARDUINO_LINT_INSTALLATION_PATH" | |
| curl \ | |
| -fsSL \ | |
| https://raw.githubusercontent.com/arduino/arduino-lint/main/etc/install.sh \ | |
| | \ | |
| BINDIR="$ARDUINO_LINT_INSTALLATION_PATH" \ | |
| sh | |
| # Add installation folder to path | |
| echo "$ARDUINO_LINT_INSTALLATION_PATH" >> "$GITHUB_PATH" | |
| - name: Configure Git for `go get` access to private repo | |
| run: | | |
| if ! [[ -f "go.mod" ]]; then | |
| # engine versions prior to 7dd8f69282232919955c82c143fefb14e50d0889 had a dependency that is hosted in a | |
| # private repo. The `go.mod` file was added at the same time the dependency was removed, so its presence can | |
| # be used as the indicator. | |
| git config \ | |
| --global \ | |
| url."https://${{ secrets.REPO_SCOPE_TOKEN }}:[email protected]/".insteadOf "https://github.com/" | |
| fi | |
| - name: Build engine | |
| run: | | |
| task go:build | |
| - name: Generate configuration file | |
| run: | | |
| cat > "${{ env.CONFIG_PATH }}" << EOF | |
| { | |
| "BaseDownloadUrl": "https://downloads.arduino.cc/libraries/", | |
| "LibrariesFolder": "${{ env.LIBRARY_ARCHIVES_PATH }}", | |
| "LibrariesIndex": "${{ env.ENGINE_DATA_PATH }}/library_index.json", | |
| "LogsFolder": "${{ env.ENGINE_DATA_PATH }}/logs", | |
| "LibrariesDB": "${{ env.ENGINE_DATA_PATH }}/db.json", | |
| "GitClonesFolder": "${{ env.GIT_CLONES_PATH }}", | |
| "DoNotRunClamav": true | |
| } | |
| EOF | |
| - name: Generate registry file | |
| run: | | |
| FULL_REGISTRY_PATH="${{ runner.temp }}/registry.txt" | |
| curl \ | |
| --output "$FULL_REGISTRY_PATH" \ | |
| https://raw.githubusercontent.com/arduino/library-registry/1c3f73b279d2845ff139883c78e733e2954437b8/registry.txt | |
| # Only use the first part of the file for the test | |
| head \ | |
| -300 \ | |
| "$FULL_REGISTRY_PATH" > \ | |
| "${{ env.REGISTRY_PATH }}" | |
| - name: Run sync on empty environment | |
| id: fresh | |
| run: | | |
| SECONDS=0 | |
| ./libraries-repository-engine "${{ env.CONFIG_PATH }}" "${{ env.REGISTRY_PATH }}" | |
| # Define step outputs with the performance data | |
| echo "Type=fresh" >> $GITHUB_OUTPUT | |
| echo "Duration=$SECONDS" >> $GITHUB_OUTPUT | |
| echo "GitClonesSize=$( \ | |
| du \ | |
| --apparent-size \ | |
| --bytes \ | |
| --summarize \ | |
| "${{ env.GIT_CLONES_PATH }}" \ | |
| | \ | |
| cut \ | |
| --fields=1 \ | |
| )" >> $GITHUB_OUTPUT | |
| echo "LibraryArchivesSize=$( \ | |
| du \ | |
| --apparent-size \ | |
| --bytes \ | |
| --summarize \ | |
| "${{ env.LIBRARY_ARCHIVES_PATH }}" \ | |
| | \ | |
| cut \ | |
| --fields=1 \ | |
| )" >> $GITHUB_OUTPUT | |
| echo "LogsSize=$( \ | |
| du \ | |
| --apparent-size \ | |
| --bytes \ | |
| --summarize \ | |
| "${{ env.LOGS_PATH }}" \ | |
| | \ | |
| cut \ | |
| --fields=1 \ | |
| )" >> $GITHUB_OUTPUT | |
| - name: Run sync on populated database | |
| id: populated | |
| run: | | |
| SECONDS=0 | |
| ./libraries-repository-engine "${{ env.CONFIG_PATH }}" "${{ env.REGISTRY_PATH }}" | |
| # Define step outputs with the performance data | |
| echo "Type=populated" >> $GITHUB_OUTPUT | |
| echo "Duration=$SECONDS" >> $GITHUB_OUTPUT | |
| echo "GitClonesSize=$( \ | |
| du \ | |
| --apparent-size \ | |
| --bytes \ | |
| --summarize \ | |
| "${{ env.GIT_CLONES_PATH }}" \ | |
| | \ | |
| cut \ | |
| --fields=1 \ | |
| )" >> $GITHUB_OUTPUT | |
| echo "LibraryArchivesSize=$( \ | |
| du \ | |
| --apparent-size \ | |
| --bytes \ | |
| --summarize \ | |
| "${{ env.LIBRARY_ARCHIVES_PATH }}" \ | |
| | \ | |
| cut \ | |
| --fields=1 \ | |
| )" >> $GITHUB_OUTPUT | |
| echo "LogsSize=$( \ | |
| du \ | |
| --apparent-size \ | |
| --bytes \ | |
| --summarize \ | |
| "${{ env.LOGS_PATH }}" \ | |
| | \ | |
| cut \ | |
| --fields=1 \ | |
| )" >> $GITHUB_OUTPUT | |
| - name: Create report | |
| run: | | |
| mkdir --parents "${{ env.REPORTS_PATH }}" | |
| cat > "${{ env.REPORTS_PATH }}/$RANDOM.json" << EOF | |
| { | |
| "Ref": "${{ matrix.data.ref }}", | |
| "Description": "${{ matrix.data.description }}", | |
| "Position": "${{ matrix.data.position }}", | |
| "Results": [ | |
| ${{ toJSON(steps.fresh.outputs) }}, | |
| ${{ toJSON(steps.populated.outputs) }} | |
| ] | |
| } | |
| EOF | |
| - name: Upload report to a workflow artifact | |
| uses: actions/upload-artifact@v4 | |
| with: | |
| if-no-files-found: error | |
| path: ${{ env.REPORTS_PATH }} | |
| name: ${{ env.REPORTS_ARTIFACT_PREFIX }}${{ matrix.data.artifact-suffix }} | |
| results: | |
| needs: run | |
| runs-on: ubuntu-latest | |
| permissions: {} | |
| env: | |
| REPORTS_PATH: reports | |
| steps: | |
| - name: Download reports | |
| uses: actions/download-artifact@v5 | |
| with: | |
| merge-multiple: true | |
| path: ${{ env.REPORTS_PATH }} | |
| pattern: ${{ env.REPORTS_ARTIFACT_PREFIX }}* | |
| - name: Print results | |
| shell: python | |
| run: | | |
| import json | |
| import pathlib | |
| reports_path = pathlib.Path("${{ env.REPORTS_PATH }}") | |
| reports = [] | |
| for report_path in reports_path.iterdir(): | |
| with report_path.open() as report_file: | |
| reports.append(json.load(fp=report_file)) | |
| sample_size = 0 | |
| summary_data = { | |
| "Duration": [], | |
| "GitClonesSize": [], | |
| "LibraryArchivesSize": [], | |
| "LogsSize": [], | |
| } | |
| for report in reports: | |
| if report["Position"] == "before": | |
| sample_size += 1 | |
| for result in report["Results"]: | |
| for key in list(summary_data): | |
| type_index = None | |
| for index, summary_item in enumerate(summary_data[key]): | |
| if summary_item["type"] == result["Type"]: | |
| type_index = index | |
| break | |
| if type_index is None: | |
| summary_data[key].append( | |
| {"type": result["Type"], "before": 0, "after": 0} | |
| ) | |
| type_index = len(summary_data[key]) - 1 | |
| summary_data[key][type_index][report["Position"]] += int(result[key]) | |
| print("% change:") | |
| for key in list(summary_data): | |
| for type_data in summary_data[key]: | |
| print( | |
| "{key} ({type}): {value}".format( | |
| key=key, | |
| type=type_data["type"], | |
| value=round( | |
| 100 | |
| * (type_data["after"] - type_data["before"]) | |
| / type_data["before"] | |
| ), | |
| ) | |
| ) | |
| print("::group::Full results") | |
| print(json.dumps(obj=reports, indent=2)) | |
| print("::endgroup::") |