diff --git a/.github/workflows/docker-publish-on-comment.yml b/.github/workflows/docker-publish-on-comment.yml index 50ea0bf..c0f86e3 100644 --- a/.github/workflows/docker-publish-on-comment.yml +++ b/.github/workflows/docker-publish-on-comment.yml @@ -46,12 +46,12 @@ jobs: # Workaround: https://github.com/docker/build-push-action/issues/461 - name: Setup Docker buildx - uses: docker/setup-buildx-action@edfb0fe6204400c56fbfd3feba3fe9ad1adfa345 + uses: docker/setup-buildx-action@5138f76647652447004da686b2411557eaf65f33 # Login against a Docker registry except on PR # https://github.com/docker/login-action - name: Log into GH registry (ghcr.io) - uses: docker/login-action@3d58c274f17dffee475a5520cbe67f0a882c4dbb + uses: docker/login-action@70fccc794acd729b2b22dd6a326895f286447728 with: registry: ghcr.io username: ${{ github.actor }} @@ -59,7 +59,7 @@ jobs: - name: Log into Docker Hub registry if: env.DOCKERHUB_USERNAME != '' - uses: docker/login-action@3d58c274f17dffee475a5520cbe67f0a882c4dbb + uses: docker/login-action@70fccc794acd729b2b22dd6a326895f286447728 with: username: ${{ env.DOCKERHUB_USERNAME }} password: ${{ secrets.DOCKERHUB_TOKEN }} @@ -70,7 +70,7 @@ jobs: # 2nd image name is for DockerHub image - name: Extract Docker metadata id: meta - uses: docker/metadata-action@dbef88086f6cef02e264edb7dbf63250c17cef6c + uses: docker/metadata-action@f7b4ed12385588c3f9bc252f0a2b520d83b52d48 with: context: git images: | @@ -85,7 +85,7 @@ jobs: # https://github.com/docker/build-push-action - name: Build and push Docker image id: build-and-push - uses: docker/build-push-action@9f6f8c940b91232557f8699b21341a08624a8dce + uses: docker/build-push-action@2a53c6ccda456d31fb62eedc658aae06e238b7bd with: context: . push: true @@ -96,7 +96,7 @@ jobs: - name: Extract Docker metadata - alpine id: meta-alpine - uses: docker/metadata-action@dbef88086f6cef02e264edb7dbf63250c17cef6c + uses: docker/metadata-action@f7b4ed12385588c3f9bc252f0a2b520d83b52d48 with: context: git images: | @@ -108,7 +108,7 @@ jobs: flavor: prefix=alpine-,onlatest=true - name: Build and push Docker image - alpine id: build-and-push-alpine - uses: docker/build-push-action@9f6f8c940b91232557f8699b21341a08624a8dce + uses: docker/build-push-action@2a53c6ccda456d31fb62eedc658aae06e238b7bd with: context: . target: alpine-release diff --git a/.github/workflows/docker-publish.yml b/.github/workflows/docker-publish.yml index 162cd9d..b11a375 100644 --- a/.github/workflows/docker-publish.yml +++ b/.github/workflows/docker-publish.yml @@ -47,13 +47,13 @@ jobs: # Workaround: https://github.com/docker/build-push-action/issues/461 - name: Setup Docker buildx - uses: docker/setup-buildx-action@edfb0fe6204400c56fbfd3feba3fe9ad1adfa345 + uses: docker/setup-buildx-action@5138f76647652447004da686b2411557eaf65f33 # Login against a Docker registry except on PR # https://github.com/docker/login-action - name: Log into GH registry (ghcr.io) if: github.event_name != 'pull_request' - uses: docker/login-action@3d58c274f17dffee475a5520cbe67f0a882c4dbb + uses: docker/login-action@70fccc794acd729b2b22dd6a326895f286447728 with: registry: ghcr.io username: ${{ github.actor }} @@ -61,7 +61,7 @@ jobs: - name: Log into Docker Hub registry if: github.event_name != 'pull_request' && env.DOCKERHUB_USERNAME != '' - uses: docker/login-action@3d58c274f17dffee475a5520cbe67f0a882c4dbb + uses: docker/login-action@70fccc794acd729b2b22dd6a326895f286447728 with: username: ${{ env.DOCKERHUB_USERNAME }} password: ${{ secrets.DOCKERHUB_TOKEN }} @@ -72,7 +72,7 @@ jobs: # 2nd image name is for DockerHub image - name: Extract Docker metadata id: meta - uses: docker/metadata-action@dbef88086f6cef02e264edb7dbf63250c17cef6c + uses: docker/metadata-action@f7b4ed12385588c3f9bc252f0a2b520d83b52d48 with: images: | ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} @@ -82,7 +82,7 @@ jobs: # https://github.com/docker/build-push-action - name: Build and push Docker image id: build-and-push - uses: docker/build-push-action@9f6f8c940b91232557f8699b21341a08624a8dce + uses: docker/build-push-action@2a53c6ccda456d31fb62eedc658aae06e238b7bd with: context: . push: ${{ github.event_name != 'pull_request' }} @@ -97,7 +97,7 @@ jobs: # 2nd image name is for DockerHub image - name: Extract Docker metadata - alpine id: meta-alpine - uses: docker/metadata-action@dbef88086f6cef02e264edb7dbf63250c17cef6c + uses: docker/metadata-action@f7b4ed12385588c3f9bc252f0a2b520d83b52d48 with: images: | ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} @@ -107,7 +107,7 @@ jobs: # https://github.com/docker/build-push-action - name: Build and push Docker image - alpine id: build-and-push-alpine - uses: docker/build-push-action@9f6f8c940b91232557f8699b21341a08624a8dce + uses: docker/build-push-action@2a53c6ccda456d31fb62eedc658aae06e238b7bd with: context: . target: alpine-release diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 8edb446..c011f0c 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -35,7 +35,7 @@ jobs: go-version: 1.21 - uses: actions/checkout@v4 - name: golangci-lint - uses: golangci/golangci-lint-action@v4 + uses: golangci/golangci-lint-action@v6 with: # Optional: version of golangci-lint to use in form of v1.2 or v1.2.3 or `latest` to use the latest version version: v1.57.2 diff --git a/Dockerfile b/Dockerfile index 636a31e..9dd6a8c 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,5 +1,5 @@ -FROM golang:1.22.2 as test +FROM golang:1.22.3 as test ARG GOPROXY ENV GOPATH=/go ENV PATH="$PATH:$GOPATH/bin" diff --git a/go.mod b/go.mod index 2c5e8bb..6907ddd 100644 --- a/go.mod +++ b/go.mod @@ -9,20 +9,24 @@ require ( github.com/argoproj/argo-cd/v2 v2.11.0-rc1 github.com/argoproj/gitops-engine v0.7.1-0.20240411122334-1ade3a199867 github.com/bradleyfalzon/ghinstallation/v2 v2.10.0 + github.com/cenkalti/backoff/v4 v4.2.1 github.com/go-test/deep v1.1.0 - github.com/google/go-github/v52 v52.0.0 + github.com/golang/mock v1.6.0 + github.com/google/go-cmp v0.6.0 + github.com/google/go-github/v62 v62.0.0 github.com/hashicorp/golang-lru/v2 v2.0.7 github.com/hexops/gotextdiff v1.0.3 github.com/migueleliasweb/go-github-mock v0.0.22 github.com/mikefarah/yq/v4 v4.43.1 github.com/prometheus/client_golang v1.19.0 - github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 github.com/shurcooL/githubv4 v0.0.0-20240120211514-18a1ae0e79dc github.com/sirupsen/logrus v1.9.3 github.com/spf13/cobra v1.8.0 golang.org/x/exp v0.0.0-20240409090435-93d18d7e34b8 golang.org/x/oauth2 v0.19.0 + google.golang.org/grpc v1.63.2 gopkg.in/yaml.v2 v2.4.0 + k8s.io/api v0.26.11 k8s.io/apimachinery v0.26.11 ) @@ -81,7 +85,6 @@ require ( github.com/google/btree v1.1.2 // indirect github.com/google/gnostic v0.7.0 // indirect github.com/google/gnostic-models v0.6.9-0.20230804172637-c7be7c783f49 // indirect - github.com/google/go-cmp v0.6.0 // indirect github.com/google/go-github/v56 v56.0.0 // indirect github.com/google/go-github/v60 v60.0.0 // indirect github.com/google/go-querystring v1.1.0 // indirect @@ -134,6 +137,7 @@ require ( github.com/redis/go-redis/v9 v9.5.1 // indirect github.com/robfig/cron/v3 v3.0.1 // indirect github.com/russross/blackfriday/v2 v2.1.0 // indirect + github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 // indirect github.com/shurcooL/graphql v0.0.0-20230722043721-ed46e5a46466 // indirect github.com/skeema/knownhosts v1.2.2 // indirect github.com/spf13/pflag v1.0.5 // indirect @@ -161,13 +165,11 @@ require ( google.golang.org/genproto v0.0.0-20240401170217-c3f982113cda // indirect google.golang.org/genproto/googleapis/api v0.0.0-20240401170217-c3f982113cda // indirect google.golang.org/genproto/googleapis/rpc v0.0.0-20240401170217-c3f982113cda // indirect - google.golang.org/grpc v1.63.2 // indirect google.golang.org/protobuf v1.33.0 // indirect gopkg.in/inf.v0 v0.9.1 // indirect gopkg.in/op/go-logging.v1 v1.0.0-20160211212156-b2cb9fa56473 // indirect gopkg.in/warnings.v0 v0.1.2 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect - k8s.io/api v0.26.11 // indirect k8s.io/apiextensions-apiserver v0.26.10 // indirect k8s.io/apiserver v0.26.11 // indirect k8s.io/cli-runtime v0.26.11 // indirect diff --git a/go.sum b/go.sum index 85c0e88..76f4baa 100644 --- a/go.sum +++ b/go.sum @@ -677,6 +677,8 @@ github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= github.com/buger/jsonparser v1.1.1/go.mod h1:6RYKKt7H4d4+iWqouImQ9R2FZql3VbhNgx27UK13J/0= github.com/bwesterb/go-ristretto v1.2.3/go.mod h1:fUIoIZaG73pV5biE2Blr2xEzDoMj7NFEuV9ekS419A0= +github.com/cenkalti/backoff/v4 v4.2.1 h1:y4OZtCnogmCPw98Zjyt5a6+QwPLGkiQsYW5oUqylYbM= +github.com/cenkalti/backoff/v4 v4.2.1/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE= github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= github.com/census-instrumentation/opencensus-proto v0.3.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= github.com/census-instrumentation/opencensus-proto v0.4.1/go.mod h1:4T9NM4+4Vw91VeyqjLS6ao50K5bOcLKN6Q42XnYaRYw= @@ -922,12 +924,12 @@ github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeN github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= -github.com/google/go-github/v52 v52.0.0 h1:uyGWOY+jMQ8GVGSX8dkSwCzlehU3WfdxQ7GweO/JP7M= -github.com/google/go-github/v52 v52.0.0/go.mod h1:WJV6VEEUPuMo5pXqqa2ZCZEdbQqua4zAk2MZTIo+m+4= github.com/google/go-github/v56 v56.0.0 h1:TysL7dMa/r7wsQi44BjqlwaHvwlFlqkK8CtBWCX3gb4= github.com/google/go-github/v56 v56.0.0/go.mod h1:D8cdcX98YWJvi7TLo7zM4/h8ZTx6u6fwGEkCdisopo0= github.com/google/go-github/v60 v60.0.0 h1:oLG98PsLauFvvu4D/YPxq374jhSxFYdzQGNCyONLfn8= github.com/google/go-github/v60 v60.0.0/go.mod h1:ByhX2dP9XT9o/ll2yXAu2VD8l5eNVg8hD4Cr0S/LmQk= +github.com/google/go-github/v62 v62.0.0 h1:/6mGCaRywZz9MuHyw9gD1CwsbmBX8GWsbFkwMmHdhl4= +github.com/google/go-github/v62 v62.0.0/go.mod h1:EMxeUqGJq2xRu9DYBMwel/mr7kZrzUOfQmmpYrZn2a4= github.com/google/go-querystring v1.1.0 h1:AnCroh3fv4ZBgVIf1Iwtovgjaw/GiKJo8M8yD/fhyJ8= github.com/google/go-querystring v1.1.0/go.mod h1:Kcdr2DB4koayq7X8pmAG4sNG59So17icRSOU623lUBU= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= diff --git a/internal/pkg/githubapi/clients.go b/internal/pkg/githubapi/clients.go index 2f46d4f..a0bb9cb 100644 --- a/internal/pkg/githubapi/clients.go +++ b/internal/pkg/githubapi/clients.go @@ -2,13 +2,14 @@ package githubapi import ( "context" + "fmt" "net/http" "os" "strconv" "time" "github.com/bradleyfalzon/ghinstallation/v2" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" lru "github.com/hashicorp/golang-lru/v2" "github.com/shurcooL/githubv4" log "github.com/sirupsen/logrus" @@ -41,25 +42,17 @@ func getAppInstallationId(githubAppPrivateKeyPath string, githubAppId int64, git if err != nil { panic(err) } - var tempClient *github.Client + tempClient := github.NewClient( + &http.Client{ + Transport: atr, + Timeout: time.Second * 30, + }) if githubRestAltURL != "" { - tempClient, err = github.NewEnterpriseClient( - githubRestAltURL, - githubRestAltURL, - &http.Client{ - Transport: atr, - Timeout: time.Second * 30, - }) + tempClient, err = tempClient.WithEnterpriseURLs(githubRestAltURL, githubRestAltURL) if err != nil { log.Fatalf("failed to create git client for app: %v\n", err) } - } else { - tempClient = github.NewClient( - &http.Client{ - Transport: atr, - Timeout: time.Second * 30, - }) } installations, _, err := tempClient.Apps.ListInstallations(ctx, &github.ListOptions{}) @@ -88,7 +81,7 @@ func createGithubAppRestClient(githubAppPrivateKeyPath string, githubAppId int64 if githubRestAltURL != "" { itr.BaseURL = githubRestAltURL - client, _ = github.NewEnterpriseClient(githubRestAltURL, githubRestAltURL, &http.Client{Transport: itr}) + client, _ = github.NewClient(&http.Client{Transport: itr}).WithEnterpriseURLs(githubRestAltURL, githubRestAltURL) } else { client = github.NewClient(&http.Client{Transport: itr}) } @@ -100,11 +93,9 @@ func createGithubRestClient(githubOauthToken string, githubRestAltURL string, ct &oauth2.Token{AccessToken: githubOauthToken}, ) tc := oauth2.NewClient(ctx, ts) - var client *github.Client + client := github.NewClient(tc) if githubRestAltURL != "" { - client, _ = github.NewEnterpriseClient(githubRestAltURL, githubRestAltURL, tc) - } else { - client = github.NewClient(tc) + client, _ = client.WithEnterpriseURLs(githubRestAltURL, githubRestAltURL) } return client @@ -147,8 +138,8 @@ func createGhAppClientPair(ctx context.Context, githubAppId int64, owner string, githubAppPrivateKeyPath := getCrucialEnv(ghAppPKeyPathEnvVarName) githubHost := getEnv("GITHUB_HOST", "") if githubHost != "" { - githubRestAltURL = "https://" + githubHost + "/api/v3" - githubGraphqlAltURL = "https://" + githubHost + "/api/graphql" + githubRestAltURL = fmt.Sprintf("https://%s/api/v3", githubHost) + githubGraphqlAltURL = fmt.Sprintf("https://%s/api/graphql", githubHost) log.Infof("Github REST API endpoint is configured to %s", githubRestAltURL) log.Infof("Github graphql API endpoint is configured to %s", githubGraphqlAltURL) } else { @@ -174,8 +165,8 @@ func createGhTokenClientPair(ctx context.Context, ghOauthToken string) GhClientP var githubGraphqlAltURL string githubHost := getEnv("GITHUB_HOST", "") if githubHost != "" { - githubRestAltURL = "https://" + githubHost + "/api/v3" - githubGraphqlAltURL = "https://" + githubHost + "/api/graphql" + githubRestAltURL = fmt.Sprintf("https://%s/api/v3", githubHost) + githubGraphqlAltURL = fmt.Sprintf("https://%s/api/graphql", githubHost) log.Infof("Github REST API endpoint is configured to %s", githubRestAltURL) log.Infof("Github graphql API endpoint is configured to %s", githubGraphqlAltURL) } else { diff --git a/internal/pkg/githubapi/drift_detection.go b/internal/pkg/githubapi/drift_detection.go index ee3a3db..8546f57 100644 --- a/internal/pkg/githubapi/drift_detection.go +++ b/internal/pkg/githubapi/drift_detection.go @@ -5,7 +5,7 @@ import ( "fmt" "strings" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" "github.com/hexops/gotextdiff" "github.com/hexops/gotextdiff/myers" "github.com/hexops/gotextdiff/span" diff --git a/internal/pkg/githubapi/drift_detection_test.go b/internal/pkg/githubapi/drift_detection_test.go index 54b8464..41c9367 100644 --- a/internal/pkg/githubapi/drift_detection_test.go +++ b/internal/pkg/githubapi/drift_detection_test.go @@ -5,7 +5,7 @@ import ( "testing" "github.com/go-test/deep" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" "github.com/hexops/gotextdiff" "github.com/hexops/gotextdiff/myers" "github.com/hexops/gotextdiff/span" diff --git a/internal/pkg/githubapi/github.go b/internal/pkg/githubapi/github.go index 8d91a62..951e7ca 100644 --- a/internal/pkg/githubapi/github.go +++ b/internal/pkg/githubapi/github.go @@ -3,7 +3,9 @@ package githubapi import ( "bytes" "context" + "crypto/sha1" //nolint:gosec // G505: Blocklisted import crypto/sha1: weak cryptographic primitive (gosec), this is not a cryptographic use case "encoding/base64" + "encoding/hex" "encoding/json" "fmt" "net/http" @@ -13,7 +15,8 @@ import ( "strings" "text/template" - "github.com/google/go-github/v52/github" + "github.com/cenkalti/backoff/v4" + "github.com/google/go-github/v62/github" lru "github.com/hashicorp/golang-lru/v2" log "github.com/sirupsen/logrus" "github.com/wayfair-incubator/telefonistka/internal/pkg/argocd" @@ -400,7 +403,7 @@ func handleMergedPrEvent(ghPrClientDetails GhPrClientDetails, prApproverGithubCl return err } - newBranchName := fmt.Sprintf("promotions/%v-%v-%v8", ghPrClientDetails.PrNumber, strings.Replace(ghPrClientDetails.Ref, "/", "-", -1), strings.Replace(strings.Join(promotion.Metadata.TargetPaths, "_"), "/", "-", -1)) // TODO max branch name length is 250 - make sure this fit + newBranchName := generateSafePromotionBranchName(ghPrClientDetails.PrNumber, ghPrClientDetails.Ref, promotion.Metadata.TargetPaths) newBranchRef, err := createBranch(ghPrClientDetails, commit, newBranchName) if err != nil { @@ -461,15 +464,60 @@ func handleMergedPrEvent(ghPrClientDetails GhPrClientDetails, prApproverGithubCl return nil } +// Creating a unique branch name based on the PR number, PR ref and the promotion target paths +// Max length of branch name is 250 characters +func generateSafePromotionBranchName(prNumber int, originalBranchName string, targetPaths []string) string { + targetPathsBa := []byte(strings.Join(targetPaths, "_")) + hasher := sha1.New() //nolint:gosec // G505: Blocklisted import crypto/sha1: weak cryptographic primitive (gosec), this is not a cryptographic use case + hasher.Write(targetPathsBa) + uniqBranchNameSuffix := firstN(hex.EncodeToString(hasher.Sum(nil)), 12) + safeOriginalBranchName := firstN(strings.Replace(originalBranchName, "/", "-", -1), 200) + return fmt.Sprintf("promotions/%v-%v-%v", prNumber, safeOriginalBranchName, uniqBranchNameSuffix) +} + +func firstN(str string, n int) string { + v := []rune(str) + if n >= len(v) { + return str + } + return string(v[:n]) +} + func MergePr(details GhPrClientDetails, number *int) error { - _, resp, err := details.GhClientPair.v3Client.PullRequests.Merge(details.Ctx, details.Owner, details.Repo, *number, "Auto-merge", nil) - prom.InstrumentGhCall(resp) + operation := func() error { + err := tryMergePR(details, number) + if err != nil { + if isMergeErrorRetryable(err.Error()) { + if err != nil { + details.PrLogger.Warnf("Failed to merge PR: transient err=%v", err) + } + return err + } + details.PrLogger.Errorf("Failed to merge PR: permanent err=%v", err) + return backoff.Permanent(err) + } + return nil + } + + // Using default values, see https://pkg.go.dev/github.com/cenkalti/backoff#pkg-constants + err := backoff.Retry(operation, backoff.NewExponentialBackOff()) if err != nil { - details.PrLogger.Errorf("Failed to merge PR: err=%v", err) + details.PrLogger.Errorf("Failed to merge PR: backoff err=%v", err) } + + return err +} + +func tryMergePR(details GhPrClientDetails, number *int) error { + _, resp, err := details.GhClientPair.v3Client.PullRequests.Merge(details.Ctx, details.Owner, details.Repo, *number, "Auto-merge", nil) + prom.InstrumentGhCall(resp) return err } +func isMergeErrorRetryable(errMessage string) bool { + return strings.Contains(errMessage, "405") && strings.Contains(errMessage, "try the merge again") +} + func (pm *prMetadata) DeSerialize(s string) error { decoded, err := base64.StdEncoding.DecodeString(s) if err != nil { @@ -759,7 +807,7 @@ func createCommit(ghPrClientDetails GhPrClientDetails, treeEntries []*github.Tre Tree: tree, } - commit, resp, err := ghPrClientDetails.GhClientPair.v3Client.Git.CreateCommit(ghPrClientDetails.Ctx, ghPrClientDetails.Owner, ghPrClientDetails.Repo, newCommitConfig) + commit, resp, err := ghPrClientDetails.GhClientPair.v3Client.Git.CreateCommit(ghPrClientDetails.Ctx, ghPrClientDetails.Owner, ghPrClientDetails.Repo, newCommitConfig, nil) prom.InstrumentGhCall(resp) if err != nil { ghPrClientDetails.PrLogger.Errorf("Failed to create Git commit: err=%s\n", err) // TODO comment this error to PR diff --git a/internal/pkg/githubapi/github_test.go b/internal/pkg/githubapi/github_test.go new file mode 100644 index 0000000..84b5c95 --- /dev/null +++ b/internal/pkg/githubapi/github_test.go @@ -0,0 +1,64 @@ +package githubapi + +import ( + "bytes" + "testing" +) + +func TestGenerateSafePromotionBranchName(t *testing.T) { + t.Parallel() + prNumber := 11 + originBranch := "originBranch" + targetPaths := []string{"targetPath1", "targetPath2"} + result := generateSafePromotionBranchName(prNumber, originBranch, targetPaths) + expectedResult := "promotions/11-originBranch-676f02019f18" + if result != expectedResult { + t.Errorf("Expected %s, got %s", expectedResult, result) + } +} + +// TestGenerateSafePromotionBranchNameLongBranchName tests the case where the original branch name is longer than 250 characters +func TestGenerateSafePromotionBranchNameLongBranchName(t *testing.T) { + t.Parallel() + prNumber := 11 + + originBranch := string(bytes.Repeat([]byte("originBranch"), 100)) + targetPaths := []string{"targetPath1", "targetPath2"} + result := generateSafePromotionBranchName(prNumber, originBranch, targetPaths) + if len(result) > 250 { + t.Errorf("Expected branch name to be less than 250 characters, got %d", len(result)) + } +} + +// TestGenerateSafePromotionBranchNameLongTargets tests the case where the target paths are longer than 250 characters +func TestGenerateSafePromotionBranchNameLongTargets(t *testing.T) { + t.Parallel() + prNumber := 11 + originBranch := "originBranch" + targetPaths := []string{ + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/1", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/2", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/3", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/4", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/5", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/6", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/7", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/8", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/9", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/10", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/11", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/12", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/13", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/14", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/15", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/16", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/17", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/18", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/19", + "loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong/target/path/20", + } + result := generateSafePromotionBranchName(prNumber, originBranch, targetPaths) + if len(result) > 250 { + t.Errorf("Expected branch name to be less than 250 characters, got %d", len(result)) + } +} diff --git a/internal/pkg/githubapi/promotion.go b/internal/pkg/githubapi/promotion.go index 1287860..2255891 100644 --- a/internal/pkg/githubapi/promotion.go +++ b/internal/pkg/githubapi/promotion.go @@ -6,7 +6,7 @@ import ( "sort" "strings" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" log "github.com/sirupsen/logrus" cfg "github.com/wayfair-incubator/telefonistka/internal/pkg/configuration" prom "github.com/wayfair-incubator/telefonistka/internal/pkg/prometheus" @@ -112,11 +112,22 @@ func getComponentConfig(ghPrClientDetails GhPrClientDetails, componentPath strin // This function generates a list of "components" that where changed in the PR and are relevant for promotion) func generateListOfRelevantComponents(ghPrClientDetails GhPrClientDetails, config *cfg.Config) (relevantComponents map[relevantComponent]struct{}, err error) { relevantComponents = make(map[relevantComponent]struct{}) - prFiles, resp, err := ghPrClientDetails.GhClientPair.v3Client.PullRequests.ListFiles(ghPrClientDetails.Ctx, ghPrClientDetails.Owner, ghPrClientDetails.Repo, ghPrClientDetails.PrNumber, &github.ListOptions{}) - prom.InstrumentGhCall(resp) - if err != nil { - ghPrClientDetails.PrLogger.Errorf("could not get file list from GH API: err=%s\nresponse=%v", err, resp) - return nil, err + + // Get the list of files in the PR, with pagination + opts := &github.ListOptions{} + prFiles := []*github.CommitFile{} + for { + perPagePrFiles, resp, err := ghPrClientDetails.GhClientPair.v3Client.PullRequests.ListFiles(ghPrClientDetails.Ctx, ghPrClientDetails.Owner, ghPrClientDetails.Repo, ghPrClientDetails.PrNumber, opts) + prom.InstrumentGhCall(resp) + if err != nil { + ghPrClientDetails.PrLogger.Errorf("could not get file list from GH API: err=%s\nstatus code=%v", err, resp.Response.Status) + return nil, err + } + prFiles = append(prFiles, perPagePrFiles...) + if resp.NextPage == 0 { + break + } + opts.Page = resp.NextPage } for _, changedFile := range prFiles { diff --git a/internal/pkg/githubapi/promotion_test.go b/internal/pkg/githubapi/promotion_test.go index 411171a..081ecc8 100644 --- a/internal/pkg/githubapi/promotion_test.go +++ b/internal/pkg/githubapi/promotion_test.go @@ -6,7 +6,7 @@ import ( "testing" "github.com/go-test/deep" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" "github.com/migueleliasweb/go-github-mock/src/mock" log "github.com/sirupsen/logrus" cfg "github.com/wayfair-incubator/telefonistka/internal/pkg/configuration" @@ -481,3 +481,55 @@ func TestGenerateNestedSourceRegexPromotionPlan(t *testing.T) { ) generatePromotionPlanTestHelper(t, config, expectedPromotion, mockedHTTPClient) } + +func TestGeneratePromotionPlanWithPagination(t *testing.T) { + t.Parallel() + config := &cfg.Config{ + PromotionPaths: []cfg.PromotionPath{ + { + SourcePath: "prod/us-east-4/", + PromotionPrs: []cfg.PromotionPr{ + { + TargetPaths: []string{ + "prod/eu-west-1/", + "prod/eu-east-1/", + }, + }, + }, + }, + }, + } + expectedPromotion := map[string]PromotionInstance{ + "prod/us-east-4/>prod/eu-east-1/|prod/eu-west-1/": { + ComputedSyncPaths: map[string]string{ + "prod/eu-east-1/componentA": "prod/us-east-4/componentA", + "prod/eu-west-1/componentA": "prod/us-east-4/componentA", + }, + }, + } + // Note the "relevant" files are in the second page, to ensure pagination is working + mockedHTTPClient := mock.NewMockedHTTPClient( + mock.WithRequestMatchPages( + mock.GetReposPullsFilesByOwnerByRepoByPullNumber, + []github.CommitFile{ + {Filename: github.String(".ci-config/random-file.json")}, + {Filename: github.String(".ci-config/random-file2.json")}, + }, + []github.CommitFile{ + {Filename: github.String("prod/us-east-4/componentA/file.yaml")}, + {Filename: github.String("prod/us-east-4/componentA/file2.yaml")}, + }, + ), + mock.WithRequestMatchHandler( + mock.GetReposContentsByOwnerByRepoByPath, + http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + mock.WriteError( + w, + http.StatusNotFound, + "no *optional* in-component telefonistka config file", + ) + }), + ), + ) + generatePromotionPlanTestHelper(t, config, expectedPromotion, mockedHTTPClient) +} diff --git a/internal/pkg/githubapi/webhook_proxy.go b/internal/pkg/githubapi/webhook_proxy.go index 33729f2..3008e95 100644 --- a/internal/pkg/githubapi/webhook_proxy.go +++ b/internal/pkg/githubapi/webhook_proxy.go @@ -9,7 +9,7 @@ import ( "regexp" "strings" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" log "github.com/sirupsen/logrus" "github.com/wayfair-incubator/telefonistka/internal/pkg/configuration" prom "github.com/wayfair-incubator/telefonistka/internal/pkg/prometheus" diff --git a/internal/pkg/githubapi/webhook_proxy_test.go b/internal/pkg/githubapi/webhook_proxy_test.go index 2bf0e86..b4523e7 100644 --- a/internal/pkg/githubapi/webhook_proxy_test.go +++ b/internal/pkg/githubapi/webhook_proxy_test.go @@ -9,7 +9,7 @@ import ( "testing" "github.com/go-test/deep" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" cfg "github.com/wayfair-incubator/telefonistka/internal/pkg/configuration" ) diff --git a/internal/pkg/prometheus/prometheus.go b/internal/pkg/prometheus/prometheus.go index 3ef4fc6..6056d49 100644 --- a/internal/pkg/prometheus/prometheus.go +++ b/internal/pkg/prometheus/prometheus.go @@ -5,7 +5,7 @@ import ( "strconv" "strings" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" "github.com/prometheus/client_golang/prometheus" "github.com/prometheus/client_golang/prometheus/promauto" ) diff --git a/internal/pkg/prometheus/prometheus_test.go b/internal/pkg/prometheus/prometheus_test.go index c35e865..28cae93 100644 --- a/internal/pkg/prometheus/prometheus_test.go +++ b/internal/pkg/prometheus/prometheus_test.go @@ -6,7 +6,7 @@ import ( "testing" "github.com/go-test/deep" - "github.com/google/go-github/v52/github" + "github.com/google/go-github/v62/github" "github.com/prometheus/client_golang/prometheus" )