| // Copyright 2020 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| // relui is a web interface for managing the release process of Go. |
| package main |
| |
| import ( |
| "bytes" |
| "context" |
| "crypto/hmac" |
| "crypto/md5" |
| "encoding/json" |
| "flag" |
| "fmt" |
| "io" |
| "log" |
| "net/http" |
| "net/mail" |
| "net/url" |
| "strings" |
| |
| cloudbuild "cloud.google.com/go/cloudbuild/apiv1/v2" |
| "cloud.google.com/go/compute/metadata" |
| "cloud.google.com/go/storage" |
| "github.com/google/go-github/v48/github" |
| "github.com/jackc/pgx/v4/pgxpool" |
| "github.com/shurcooL/githubv4" |
| "go.chromium.org/luci/auth" |
| pb "go.chromium.org/luci/buildbucket/proto" |
| "go.chromium.org/luci/grpc/prpc" |
| "go.chromium.org/luci/swarming/client/swarming" |
| "go.opencensus.io/plugin/ochttp" |
| "golang.org/x/build/buildlet" |
| "golang.org/x/build/gerrit" |
| "golang.org/x/build/internal/access" |
| "golang.org/x/build/internal/https" |
| "golang.org/x/build/internal/metrics" |
| "golang.org/x/build/internal/relui" |
| "golang.org/x/build/internal/relui/db" |
| "golang.org/x/build/internal/relui/protos" |
| "golang.org/x/build/internal/relui/sign" |
| "golang.org/x/build/internal/secret" |
| "golang.org/x/build/internal/task" |
| "golang.org/x/build/repos" |
| "golang.org/x/oauth2" |
| "golang.org/x/oauth2/google" |
| "google.golang.org/grpc" |
| ) |
| |
| var ( |
| baseURL = flag.String("base-url", "", "Prefix URL for routing and links.") |
| siteTitle = flag.String("site-title", "Go Releases", "Site title.") |
| siteHeaderCSS = flag.String("site-header-css", "", "Site header CSS class name. Can be used to pick a look for the header.") |
| |
| downUp = flag.Bool("migrate-down-up", false, "Run all Up migration steps, then the last down migration step, followed by the final up migration. Exits after completion.") |
| migrateOnly = flag.Bool("migrate-only", false, "Exit after running migrations. Migrations are run by default.") |
| pgConnect = flag.String("pg-connect", "", "Postgres connection string or URI. If empty, libpq connection defaults are used.") |
| |
| scratchFilesBase = flag.String("scratch-files-base", "", "Storage for scratch files. gs://bucket/path or file:///path/to/scratch.") |
| signedFilesBase = flag.String("signed-files-base", "", "Storage for signed files. gs://bucket/path or file:///path/to/signed.") |
| servingFilesBase = flag.String("serving-files-base", "", "Storage for serving files. gs://bucket/path or file:///path/to/serving.") |
| edgeCacheURL = flag.String("edge-cache-url", "", "URL release files appear at when published to the CDN, e.g. https://dl.google.com/go.") |
| websiteUploadURL = flag.String("website-upload-url", "", "URL to POST website file data to, e.g. https://go.dev/dl/upload.") |
| |
| cloudBuildProject = flag.String("cloud-build-project", "", "GCP project to run miscellaneous Cloud Build tasks") |
| cloudBuildAccount = flag.String("cloud-build-account", "", "Service account to run miscellaneous Cloud Build tasks") |
| |
| swarmingURL = flag.String("swarming-url", "", "Swarming service to use for tasks") |
| swarmingAccount = flag.String("swarming-account", "", "Service account to use for Swarming tasks") |
| swarmingPool = flag.String("swarming-pool", "", "Swarming pool to run tasks in") |
| swarmingRealm = flag.String("swarming-realm", "", "Swarming realm to run tasks in") |
| |
| buildbucketHost = flag.String("buildbucket-host", "", "Buildbucket host to use for tasks") |
| ) |
| |
| func main() { |
| if err := secret.InitFlagSupport(context.Background()); err != nil { |
| log.Fatalln(err) |
| } |
| sendgridAPIKey := secret.Flag("sendgrid-api-key", "SendGrid API key for workflows involving sending email.") |
| var annMail task.MailHeader |
| addressVarFlag(&annMail.From, "announce-mail-from", "The From address to use for the (pre-)announcement mail.") |
| addressVarFlag(&annMail.To, "announce-mail-to", "The To address to use for the (pre-)announcement mail.") |
| addressListVarFlag(&annMail.BCC, "announce-mail-bcc", "The BCC address list to use for the (pre-)announcement mail.") |
| var schedMail task.MailHeader |
| addressVarFlag(&schedMail.From, "schedule-mail-from", "The From address to use for the scheduled workflow failure mail.") |
| addressVarFlag(&schedMail.To, "schedule-mail-to", "The To address to use for the scheduled workflow failure mail.") |
| addressListVarFlag(&schedMail.BCC, "schedule-mail-bcc", "The BCC address list to use for the scheduled workflow failure mail.") |
| var twitterAPI secret.TwitterCredentials |
| secret.JSONVarFlag(&twitterAPI, "twitter-api-secret", "Twitter API secret to use for workflows involving tweeting.") |
| var mastodonAPI secret.MastodonCredentials |
| secret.JSONVarFlag(&mastodonAPI, "mastodon-api-secret", "Mastodon API secret to use for workflows involving posting.") |
| masterKey := secret.Flag("builder-master-key", "Builder master key") |
| githubToken := secret.Flag("github-token", "GitHub API token") |
| https.RegisterFlags(flag.CommandLine) |
| flag.Parse() |
| |
| ctx := context.Background() |
| if err := relui.InitDB(ctx, *pgConnect); err != nil { |
| log.Fatalf("relui.InitDB() = %v", err) |
| } |
| if *migrateOnly { |
| return |
| } |
| if *downUp { |
| if err := relui.MigrateDB(*pgConnect, true); err != nil { |
| log.Fatalf("relui.MigrateDB() = %v", err) |
| } |
| return |
| } |
| |
| // Define the site header and external service configuration. |
| // The site header communicates to humans what will happen |
| // when workflows run. |
| // Keep these appropriately in sync. |
| siteHeader := relui.SiteHeader{ |
| Title: *siteTitle, |
| CSSClass: *siteHeaderCSS, |
| } |
| creds, err := google.FindDefaultCredentials(ctx, gerrit.OAuth2Scopes...) |
| if err != nil { |
| log.Fatalf("reading GCP credentials: %v", err) |
| } |
| gerritClient := &task.RealGerritClient{ |
| Gitiles: "https://go.googlesource.com", |
| Client: gerrit.NewClient("https://go-review.googlesource.com", gerrit.OAuth2Auth(creds.TokenSource)), |
| } |
| privateGerritClient := &task.RealGerritClient{ |
| Gitiles: "https://go-internal.googlesource.com", |
| Client: gerrit.NewClient("https://go-internal-review.googlesource.com", gerrit.OAuth2Auth(creds.TokenSource)), |
| } |
| gitClient := &task.Git{} |
| gitClient.UseOAuth2Auth(creds.TokenSource) |
| mailFunc := task.NewSendGridMailClient(*sendgridAPIKey).SendMail |
| var mastodonClient task.Poster |
| if mastodonAPI != (secret.MastodonCredentials{}) { |
| var err error |
| mastodonClient, err = task.NewMastodonClient(mastodonAPI) |
| if err != nil { |
| log.Fatalln("task.NewMastodonClient:", err) |
| } |
| } |
| commTasks := task.CommunicationTasks{ |
| AnnounceMailTasks: task.AnnounceMailTasks{ |
| SendMail: mailFunc, |
| AnnounceMailHeader: annMail, |
| }, |
| SocialMediaTasks: task.SocialMediaTasks{ |
| TwitterClient: task.NewTwitterClient(twitterAPI), |
| MastodonClient: mastodonClient, |
| }, |
| } |
| dh := relui.NewDefinitionHolder() |
| userPassAuth := buildlet.UserPass{ |
| Username: "user-relui", |
| Password: key(*masterKey, "user-relui"), |
| } |
| gcsClient, err := storage.NewClient(ctx) |
| if err != nil { |
| log.Fatalf("Could not connect to GCS: %v", err) |
| } |
| cbClient, err := cloudbuild.NewClient(ctx) |
| if err != nil { |
| log.Fatalf("Could not connect to Cloud Build: %v", err) |
| } |
| cloudBuildClient := &task.RealCloudBuildClient{ |
| BuildClient: cbClient, |
| StorageClient: gcsClient, |
| ScriptProject: *cloudBuildProject, |
| ScriptAccount: *cloudBuildAccount, |
| ScratchURL: *scratchFilesBase + "/build-outputs", |
| } |
| var swarmingClient swarming.Client |
| if *swarmingURL != "" { |
| var err error |
| swarmingClient, err = swarming.NewClient(ctx, swarming.ClientOptions{ |
| ServiceURL: *swarmingURL, |
| Auth: auth.Options{ |
| GCEAllowAsDefault: true, |
| }, |
| }) |
| if err != nil { |
| log.Fatalln("swarming.NewClient:", err) |
| } |
| } |
| var buildBucketClient task.BuildBucketClient |
| if *buildbucketHost != "" { |
| luciHTTPClient, err := auth.NewAuthenticator(ctx, auth.SilentLogin, auth.Options{GCEAllowAsDefault: true}).Client() |
| if err != nil { |
| log.Fatalln("auth.NewAuthenticator:", err) |
| } |
| buildBucketClient = &task.RealBuildBucketClient{ |
| BuildersClient: pb.NewBuildersClient(&prpc.Client{ |
| C: luciHTTPClient, |
| Host: *buildbucketHost, |
| }), |
| BuildsClient: pb.NewBuildsClient(&prpc.Client{ |
| C: luciHTTPClient, |
| Host: *buildbucketHost, |
| }), |
| } |
| } |
| |
| var dbPool db.PGDBTX |
| dbPool, err = pgxpool.Connect(ctx, *pgConnect) |
| if err != nil { |
| log.Fatalln("pgxpool.Connect:", err) |
| } |
| defer dbPool.Close() |
| dbPool = &relui.MetricsDB{dbPool} |
| |
| var gr *metrics.MonitoredResource |
| if metadata.OnGCE() { |
| gr, err = metrics.GKEResource("relui-deployment") |
| if err != nil { |
| log.Println("metrics.GKEResource:", err) |
| } |
| } |
| ms, err := metrics.NewService(gr, relui.Views) |
| if err != nil { |
| log.Println("failed to initialize metrics:", err) |
| } else { |
| defer ms.Stop() |
| } |
| grpcServer := grpc.NewServer(grpc.UnaryInterceptor(access.RequireIAPAuthUnaryInterceptor(access.IAPSkipAudienceValidation)), |
| grpc.StreamInterceptor(access.RequireIAPAuthStreamInterceptor(access.IAPSkipAudienceValidation))) |
| signServer := sign.NewServer() |
| protos.RegisterReleaseServiceServer(grpcServer, signServer) |
| buildTasks := &relui.BuildReleaseTasks{ |
| GerritClient: gerritClient, |
| GerritProject: "go", |
| GerritHTTPClient: oauth2.NewClient(ctx, creds.TokenSource), |
| PrivateGerritClient: privateGerritClient, |
| PrivateGerritProject: "go", |
| SignService: signServer, |
| GCSClient: gcsClient, |
| ScratchFS: &task.ScratchFS{ |
| BaseURL: *scratchFilesBase, |
| GCS: gcsClient, |
| }, |
| SignedURL: *signedFilesBase, |
| ServingURL: *servingFilesBase, |
| DownloadURL: *edgeCacheURL, |
| ProxyPrefix: "https://proxy.golang.org/golang.org/toolchain/@v", |
| CloudBuildClient: cloudBuildClient, |
| BuildBucketClient: buildBucketClient, |
| SwarmingClient: &task.RealSwarmingClient{ |
| SwarmingClient: swarmingClient, |
| SwarmingURL: *swarmingURL, |
| ServiceAccount: *swarmingAccount, |
| Realm: *swarmingRealm, |
| Pool: *swarmingPool, |
| }, |
| GoogleDockerBuildProject: "symbolic-datum-552", |
| GoogleDockerBuildTrigger: "golang-publish-internal-boringcrypto", |
| PublishFile: func(f task.WebsiteFile) error { |
| return publishFile(*websiteUploadURL, userPassAuth, f) |
| }, |
| ApproveAction: relui.ApproveActionDep(dbPool), |
| } |
| githubHTTPClient := oauth2.NewClient(ctx, oauth2.StaticTokenSource(&oauth2.Token{AccessToken: *githubToken})) |
| milestoneTasks := &task.MilestoneTasks{ |
| Client: &task.GitHubClient{ |
| V3: github.NewClient(githubHTTPClient), |
| V4: githubv4.NewClient(githubHTTPClient), |
| }, |
| RepoOwner: "golang", |
| RepoName: "go", |
| ApproveAction: relui.ApproveActionDep(dbPool), |
| } |
| versionTasks := &task.VersionTasks{ |
| Gerrit: gerritClient, |
| CloudBuild: cloudBuildClient, |
| GoProject: "go", |
| UpdateProxyTestRepoTasks: task.UpdateProxyTestRepoTasks{ |
| Git: gitClient, |
| GerritURL: "https://golang-modproxy-test.googlesource.com/latest-go-version", |
| Branch: "main", |
| }, |
| } |
| if err := relui.RegisterReleaseWorkflows(ctx, dh, buildTasks, milestoneTasks, versionTasks, commTasks); err != nil { |
| log.Fatalf("RegisterReleaseWorkflows: %v", err) |
| } |
| |
| ignoreProjects := map[string]bool{} |
| for p, r := range repos.ByGerritProject { |
| ignoreProjects[p] = !r.ShowOnDashboard() |
| } |
| tagTasks := &task.TagXReposTasks{ |
| IgnoreProjects: ignoreProjects, |
| Gerrit: gerritClient, |
| CloudBuild: cloudBuildClient, |
| BuildBucket: buildBucketClient, |
| } |
| dh.RegisterDefinition("Tag x/ repos", tagTasks.NewDefinition()) |
| dh.RegisterDefinition("Tag a single x/ repo", tagTasks.NewSingleDefinition()) |
| |
| bundleTasks := &task.BundleNSSRootsTask{ |
| Gerrit: gerritClient, |
| CloudBuild: cloudBuildClient, |
| } |
| dh.RegisterDefinition("Update x/crypto NSS root bundle", bundleTasks.NewDefinition()) |
| |
| tagTelemetryTasks := &task.TagTelemetryTasks{ |
| Gerrit: gerritClient, |
| CloudBuild: cloudBuildClient, |
| } |
| dh.RegisterDefinition("Tag a new version of x/telemetry/config (if necessary)", tagTelemetryTasks.NewDefinition()) |
| |
| releaseGoplsTasks := task.ReleaseGoplsTasks{ |
| Gerrit: gerritClient, |
| } |
| dh.RegisterDefinition("Release a new version of gopls", releaseGoplsTasks.NewDefinition()) |
| |
| privateSyncTask := &task.PrivateMasterSyncTask{ |
| Git: gitClient, |
| PrivateGerritURL: "https://go-internal.googlesource.com/golang/go-private", |
| Ref: "public", |
| } |
| dh.RegisterDefinition("Sync go-private master branch with public", privateSyncTask.NewDefinition()) |
| |
| privateXPatchTask := &task.PrivXPatch{ |
| Git: gitClient, |
| PublicGerrit: gerritClient, |
| PrivateGerrit: privateGerritClient, |
| PublicRepoURL: func(repo string) string { |
| return "https://go.googlesource.com/" + repo |
| }, |
| ApproveAction: relui.ApproveActionDep(dbPool), |
| SendMail: mailFunc, |
| AnnounceMailHeader: annMail, |
| } |
| dh.RegisterDefinition("Publish a private patch to a x/ repo", privateXPatchTask.NewDefinition(tagTasks)) |
| |
| var base *url.URL |
| if *baseURL != "" { |
| base, err = url.Parse(*baseURL) |
| if err != nil { |
| log.Fatalf("url.Parse(%q) = %v, %v", *baseURL, base, err) |
| } |
| } |
| l := &relui.PGListener{ |
| DB: dbPool, |
| BaseURL: base, |
| ScheduleFailureMailHeader: schedMail, |
| SendMail: mailFunc, |
| } |
| w := relui.NewWorker(dh, dbPool, l) |
| go w.Run(ctx) |
| if err := w.ResumeAll(ctx); err != nil { |
| log.Printf("w.ResumeAll() = %v", err) |
| } |
| var h http.Handler = relui.NewServer(dbPool, w, base, siteHeader, ms) |
| if metadata.OnGCE() { |
| project, err := metadata.ProjectID() |
| if err != nil { |
| log.Fatalln("failed to read project ID from metadata server") |
| } |
| if project == "symbolic-datum-552" { |
| h = access.RequireIAPAuthHandler(h, access.IAPSkipAudienceValidation) |
| } |
| } |
| log.Fatalln(https.ListenAndServe(ctx, &ochttp.Handler{Handler: GRPCHandler(grpcServer, h)})) |
| } |
| |
| // GRPCHandler creates handler which intercepts requests intended for a GRPC server and directs the calls to the server. |
| // All other requests are directed toward the passed in handler. |
| func GRPCHandler(gs *grpc.Server, h http.Handler) http.Handler { |
| return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { |
| if r.ProtoMajor == 2 && strings.HasPrefix(r.Header.Get("Content-Type"), "application/grpc") { |
| gs.ServeHTTP(w, r) |
| return |
| } |
| h.ServeHTTP(w, r) |
| }) |
| } |
| |
| func key(masterKey, principal string) string { |
| h := hmac.New(md5.New, []byte(masterKey)) |
| io.WriteString(h, principal) |
| return fmt.Sprintf("%x", h.Sum(nil)) |
| } |
| |
| func publishFile(uploadURL string, auth buildlet.UserPass, f task.WebsiteFile) error { |
| req, err := json.Marshal(f) |
| if err != nil { |
| return err |
| } |
| u, err := url.Parse(uploadURL) |
| if err != nil { |
| return fmt.Errorf("invalid website upload URL %q: %v", *websiteUploadURL, err) |
| } |
| q := u.Query() |
| q.Set("user", strings.TrimPrefix(auth.Username, "user-")) |
| q.Set("key", auth.Password) |
| u.RawQuery = q.Encode() |
| resp, err := http.Post(u.String(), "application/json", bytes.NewReader(req)) |
| if err != nil { |
| return err |
| } |
| defer resp.Body.Close() |
| if resp.StatusCode != http.StatusOK { |
| b, _ := io.ReadAll(resp.Body) |
| return fmt.Errorf("upload failed to %q: %v\n%s", uploadURL, resp.Status, b) |
| } |
| return nil |
| } |
| |
| // addressVarFlag defines an address flag with specified name and usage string. |
| // The argument p points to a mail.Address variable in which to store the value of the flag. |
| func addressVarFlag(p *mail.Address, name, usage string) { |
| flag.Func(name, usage, func(s string) error { |
| a, err := mail.ParseAddress(s) |
| if err != nil { |
| return err |
| } |
| *p = *a |
| return nil |
| }) |
| } |
| |
| // addressListVarFlag defines an address list flag with specified name and usage string. |
| // The argument p points to a []mail.Address variable in which to store the value of the flag. |
| func addressListVarFlag(p *[]mail.Address, name, usage string) { |
| flag.Func(name, usage, func(s string) error { |
| as, err := mail.ParseAddressList(s) |
| if err != nil { |
| return err |
| } |
| *p = nil // Clear out the list before appending. |
| for _, a := range as { |
| *p = append(*p, *a) |
| } |
| return nil |
| }) |
| } |