From 2c511991a8b3b3533e24bf485629c38fc638de1f Mon Sep 17 00:00:00 2001 From: 0marperez <60363173+0marperez@users.noreply.github.com> Date: Mon, 29 Apr 2024 13:59:49 -0600 Subject: [PATCH] feat: artifact size metrics (#1290) --- .github/actions/setup-build/action.yml | 2 +- .github/scripts/run-codebuild-batch-job.sh | 46 +++++-- .github/workflows/codebuild-ci.yml | 152 ++++++++++++++++++++- build.gradle.kts | 8 ++ gradle/libs.versions.toml | 3 +- 5 files changed, 195 insertions(+), 16 deletions(-) diff --git a/.github/actions/setup-build/action.yml b/.github/actions/setup-build/action.yml index 9a01b590a6a..8201ca1f38f 100644 --- a/.github/actions/setup-build/action.yml +++ b/.github/actions/setup-build/action.yml @@ -10,7 +10,7 @@ runs: with: path: 'aws-kotlin-repo-tools' repository: 'awslabs/aws-kotlin-repo-tools' - ref: '0.2.3' + ref: '0.4.2' sparse-checkout: | .github - name: Checkout smithy-kotlin diff --git a/.github/scripts/run-codebuild-batch-job.sh b/.github/scripts/run-codebuild-batch-job.sh index 357427a6a4a..46d585d96b4 100755 --- a/.github/scripts/run-codebuild-batch-job.sh +++ b/.github/scripts/run-codebuild-batch-job.sh @@ -8,8 +8,11 @@ error_exit() { PROJECT_NAME=gh-aws-sdk-kotlin-svc-check-batch # get the source version to be built (defaults to main branch if not specified) SOURCE_VERSION=main -SDK_PR="" -SMITHY_PR="" +GITHUB_RELEASE="" +GITHUB_PULL_REQUEST_NUMBER="" +GITHUB_REPOSITORY_NO_ORG="" +EXTERNAL_CONTRIBUTOR_SDK_PR="" +EXTERNAL_CONTRIBUTOR_SMITHY_PR="" while [[ $# -gt 0 ]]; do case "$1" in @@ -29,19 +32,43 @@ while [[ $# -gt 0 ]]; do shift 2 fi ;; - --sdk-pr) + --release) if [[ "$2" == --* ]]; then shift 1 else - SDK_PR="$2" + GITHUB_RELEASE="$2" shift 2 fi ;; - --smithy-pr) + --pr-number) + if [[ "$2" == --* ]]; then + shift 1 + else + GITHUB_PULL_REQUEST_NUMBER="$2" + shift 2 + fi + ;; + --repository) + if [[ "$2" == --* ]]; then + shift 1 + else + GITHUB_REPOSITORY_NO_ORG="$2" + shift 2 + fi + ;; + --external-contributor-sdk-pr-number) + if [[ "$2" == --* ]]; then + shift 1 + else + EXTERNAL_CONTRIBUTOR_SDK_PR="$2" + shift 2 + fi + ;; + --external-contributor-smithy-pr-number) if [[ "$2" == --* || -z "$2" ]]; then shift 1 else - SMITHY_PR="$2" + EXTERNAL_CONTRIBUTOR_SMITHY_PR="$2" shift 2 fi ;; @@ -52,8 +79,11 @@ while [[ $# -gt 0 ]]; do esac done -export SDK_PR -export SMITHY_PR +export GITHUB_RELEASE +export GITHUB_PULL_REQUEST_NUMBER +export GITHUB_REPOSITORY_NO_ORG +export EXTERNAL_CONTRIBUTOR_SDK_PR +export EXTERNAL_CONTRIBUTOR_SMITHY_PR echo "Starting CodeBuild project ${PROJECT_NAME}" diff --git a/.github/workflows/codebuild-ci.yml b/.github/workflows/codebuild-ci.yml index 201399c6568..48a5b88634a 100644 --- a/.github/workflows/codebuild-ci.yml +++ b/.github/workflows/codebuild-ci.yml @@ -1,9 +1,11 @@ name: AWS CodeBuild CI on: + release: + types: [ published ] pull_request: - branches: - - main + types: [ opened, synchronize, reopened, labeled, unlabeled ] + branches: [ main ] workflow_dispatch: inputs: aws-sdk-kotlin-pr: @@ -27,6 +29,7 @@ env: permissions: id-token: write contents: read + pull-requests: write # Allow one instance of this workflow per pull request, and cancel older runs when new changes are pushed concurrency: @@ -35,6 +38,7 @@ concurrency: jobs: e2e-tests: + if: github.event_name == 'pull_request' || github.event_name == 'workflow_dispatch' runs-on: ubuntu-latest steps: - name: Verify PRs are not running malicious code @@ -63,7 +67,8 @@ jobs: aws codebuild stop-build --id $BUILD_ID fi - service-check-batch: + service-check-batch-and-artifact-size-metrics: + if: github.event_name == 'pull_request' || github.event_name == 'workflow_dispatch' runs-on: ubuntu-latest steps: - name: Verify PRs are not running malicious code @@ -78,14 +83,143 @@ jobs: with: role-to-assume: ${{ secrets.CI_AWS_ROLE_ARN }} aws-region: us-west-2 - - name: Run Service Check Batch + - name: Run Service Check Batch and Calculate Artifact Size Metrics + id: svc-check-batch + run: | + REPOSITORY=$(echo ${{ github.repository }} | cut -d '/' -f 2) + PULL_REQUEST="" + INTERNAL=${{ github.event.number }} + EXTERNAL=${{ inputs.aws-sdk-kotlin-pr }} + + if [ ! -z "$INTERNAL" ]; then + PULL_REQUEST=${{ github.event.number }} + echo "Identified this an internal pull request build: $PULL_REQUEST" + elif [ ! -z "$EXTERNAL" ]; then + PULL_REQUEST=${{ inputs.aws-sdk-kotlin-pr }} + echo "Identified this as an external pull request build: $PULL_REQUEST" + else + echo "Couldn't identify this as an internal or external pull request" + exit 1 + fi + + .github/scripts/run-codebuild-batch-job.sh \ + --project gh-aws-sdk-kotlin-svc-check-batch \ + --source ${{ github.event.pull_request.head.sha }} \ + --pr-number $PULL_REQUEST \ + --repository $REPOSITORY \ + --external-contributor-sdk-pr-number ${{ inputs.aws-sdk-kotlin-pr }} \ + --external-contributor-smithy-pr-number ${{ inputs.smithy-kotlin-pr }} + - name: Cancel build + if: ${{ cancelled() }} + env: + BUILD_ID: ${{ steps.svc-check-batch.outputs.aws-build-id }} + run: | + if [ ! -z "$BUILD_ID" ]; then + echo "cancelling in-progress batch build: id=$BUILD_ID" + aws codebuild stop-build --id $BUILD_ID + fi + - name: Collect Artifact Size Metrics + run: | + PULL_REQUEST="" + INTERNAL=${{ github.event.number }} + EXTERNAL=${{ inputs.aws-sdk-kotlin-pr }} + + if [ ! -z "$INTERNAL" ]; then + PULL_REQUEST=${{ github.event.number }} + echo "Identified this an internal pull request build: $PULL_REQUEST" + elif [ ! -z "$EXTERNAL" ]; then + PULL_REQUEST=${{ inputs.aws-sdk-kotlin-pr }} + echo "Identified this as an external pull request build: $PULL_REQUEST" + else + echo "Couldn't identify this as an internal or external pull request" + exit 1 + fi + + ./gradlew collectDelegatedArtifactSizeMetrics -PpullRequest=$PULL_REQUEST + - name: Analyze Artifact Size Metrics + run: ./gradlew analyzeArtifactSizeMetrics + - name: Show Results + uses: actions/github-script@v7 + with: + script: | + const getComments = + `query { + repository(owner:"${context.repo.owner}", name:"${context.repo.repo}"){ + pullRequest(number: ${context.issue.number}) { + id + comments(last:100) { + nodes { + id + body + author { + login + } + isMinimized + } + } + } + } + }` + + const response = await github.graphql(getComments) + const comments = response.repository.pullRequest.comments.nodes + + const mutations = comments + .filter(comment => comment.author.login == 'github-actions' && !comment.isMinimized && comment.body.startsWith('Affected Artifacts')) + .map(comment => + github.graphql( + `mutation { + minimizeComment(input:{subjectId:"${comment.id}", classifier:OUTDATED}){ + clientMutationId + } + }` + ) + ) + await Promise.all(mutations) + + const fs = require('node:fs') + const comment = fs.readFileSync('build/reports/metrics/artifact-analysis.md', 'utf8') + + const writeComment = + `mutation { + addComment(input:{body:"""${comment}""", subjectId:"${response.repository.pullRequest.id}"}){ + clientMutationId + } + }` + + await github.graphql(writeComment) + + - name: Evaluate Result + if: ${{ !contains(github.event.pull_request.labels.*.name, 'acknowledge-artifact-size-increase') }} + run: | + cd build/reports/metrics + cat has-significant-change.txt | grep false || { + echo An artifact increased in size by more than allowed or a new artifact was created. + echo If this is expected please add the 'acknowledge-artifact-size-increase' label to this pull request. + exit 1 + } + + release-artifact-size-metrics: + if: github.event_name == 'release' + runs-on: ubuntu-latest + steps: + - name: Checkout sources + uses: actions/checkout@v2 + - name: Configure AWS Credentials + uses: aws-actions/configure-aws-credentials@v4 + with: + role-to-assume: ${{ secrets.CI_AWS_ROLE_ARN }} + aws-region: us-west-2 + - name: Calculate Artifact Size Metrics id: svc-check-batch run: | + REPOSITORY=$(echo ${{ github.repository }} | cut -d '/' -f 2) + .github/scripts/run-codebuild-batch-job.sh \ --project gh-aws-sdk-kotlin-svc-check-batch \ --source ${{ github.event.pull_request.head.sha }} \ - --sdk-pr ${{ inputs.aws-sdk-kotlin-pr }} \ - --smithy-pr ${{ inputs.smithy-kotlin-pr }} + --repository $REPOSITORY \ + --release ${{ github.event.release.tag_name }} - name: Cancel build if: ${{ cancelled() }} env: @@ -95,3 +229,9 @@ jobs: echo "cancelling in-progress batch build: id=$BUILD_ID" aws codebuild stop-build --id $BUILD_ID fi + - name: Collect Artifact Size Metrics + run: ./gradlew collectDelegatedArtifactSizeMetrics -Prelease=${{ github.event.release.tag_name }} + - name: Save Artifact Size Metrics + run: ./gradlew saveArtifactSizeMetrics + - name: Put Artifact Size Metrics in CloudWatch + run: ./gradlew putArtifactSizeMetricsInCloudWatch -Prelease=${{ github.event.release.tag_name }} diff --git a/build.gradle.kts b/build.gradle.kts index 11f19cc45d0..f2f6e5accb5 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -22,6 +22,14 @@ plugins { // ensure the correct version of KGP ends up on our buildscript classpath alias(libs.plugins.kotlin.multiplatform) apply false alias(libs.plugins.kotlin.jvm) apply false + alias(libs.plugins.aws.kotlin.repo.tools.artifactsizemetrics) +} + +artifactSizeMetrics { + artifactPrefixes = setOf(":services", ":aws-runtime") + closurePrefixes = setOf(":services") + significantChangeThresholdPercentage = 5.0 + projectRepositoryName = "aws-sdk-kotlin" } val testJavaVersion = typedProp("test.java.version")?.let { diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 0c7e57bcf8b..f8b76dd2cf1 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -2,7 +2,7 @@ kotlin-version = "1.9.23" dokka-version = "1.9.10" -aws-kotlin-repo-tools-version = "0.4.0" +aws-kotlin-repo-tools-version = "0.4.4" # libs coroutines-version = "1.7.3" @@ -135,3 +135,4 @@ kotlinx-binary-compatibility-validator = { id = "org.jetbrains.kotlinx.binary-co kotlinx-serialization = { id = "org.jetbrains.kotlin.plugin.serialization", version.ref = "kotlin-version"} aws-kotlin-repo-tools-kmp = { id = "aws.sdk.kotlin.gradle.kmp", version.ref = "aws-kotlin-repo-tools-version" } aws-kotlin-repo-tools-smithybuild = { id = "aws.sdk.kotlin.gradle.smithybuild", version.ref = "aws-kotlin-repo-tools-version" } +aws-kotlin-repo-tools-artifactsizemetrics = { id = "aws.sdk.kotlin.gradle.artifactsizemetrics", version.ref = "aws-kotlin-repo-tools-version" }