| // Copyright 2015 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. |
| |
| package main |
| |
| import ( |
| "context" |
| "encoding/json" |
| "flag" |
| "fmt" |
| "io" |
| "log" |
| "net/http" |
| "os" |
| "path/filepath" |
| "sort" |
| "strings" |
| "sync" |
| "time" |
| |
| "golang.org/x/build/internal/gomote/protos" |
| "golang.org/x/sync/errgroup" |
| ) |
| |
| type builderType struct { |
| Name string |
| IsReverse bool |
| ExpectNum int |
| } |
| |
| func builders() (bt []builderType) { |
| type builderInfo struct { |
| HostType string |
| } |
| type hostInfo struct { |
| IsReverse bool |
| ExpectNum int |
| ContainerImage string |
| VMImage string |
| } |
| // resj is the response JSON from the builders. |
| var resj struct { |
| Builders map[string]builderInfo |
| Hosts map[string]hostInfo |
| } |
| res, err := http.Get("https://farmer.golang.org/builders?mode=json") |
| if err != nil { |
| log.Fatal(err) |
| } |
| defer res.Body.Close() |
| if res.StatusCode != 200 { |
| log.Fatalf("fetching builder types: %s", res.Status) |
| } |
| if err := json.NewDecoder(res.Body).Decode(&resj); err != nil { |
| log.Fatalf("decoding builder types: %v", err) |
| } |
| for b, bi := range resj.Builders { |
| if strings.HasPrefix(b, "misc-compile") { |
| continue |
| } |
| hi, ok := resj.Hosts[bi.HostType] |
| if !ok { |
| continue |
| } |
| if !hi.IsReverse && hi.ContainerImage == "" && hi.VMImage == "" { |
| continue |
| } |
| bt = append(bt, builderType{ |
| Name: b, |
| IsReverse: hi.IsReverse, |
| ExpectNum: hi.ExpectNum, |
| }) |
| } |
| sort.Slice(bt, func(i, j int) bool { |
| return bt[i].Name < bt[j].Name |
| }) |
| return |
| } |
| |
| func swarmingBuilders() ([]string, error) { |
| ctx := context.Background() |
| client := gomoteServerClient(ctx) |
| |
| ctx, cancel := context.WithTimeout(ctx, 5*time.Second) |
| defer cancel() |
| resp, err := client.ListSwarmingBuilders(ctx, &protos.ListSwarmingBuildersRequest{}) |
| if err != nil { |
| return nil, fmt.Errorf("unable to retrieve swarming builders: %s", err) |
| } |
| return resp.Builders, nil |
| } |
| |
| func create(args []string) error { |
| fs := flag.NewFlagSet("create", flag.ContinueOnError) |
| fs.Usage = func() { |
| log := usageLogger |
| log.Print("create usage: gomote create [create-opts] <type>") |
| log.Print() |
| log.Print("If there's a valid group specified, new instances are") |
| log.Print("automatically added to the group. If the group in") |
| log.Print("$GOMOTE_GROUP doesn't exist, and there's no other group") |
| log.Print("specified, it will be created and new instances will be") |
| log.Print("added to that group.") |
| log.Print() |
| log.Print("Run 'gomote create -list' to see a list of valid builder") |
| log.Print("types.") |
| log.Print() |
| log.Print("Builder types are structured according to the following") |
| log.Print("format, where the bracketed parts are optional:") |
| log.Print() |
| log.Print(" [<subrepo>-]<go branch>-<goos>-<goarch>[_<host>][-<mods>*]") |
| log.Print() |
| log.Print("Subrepo names always start with 'x_'. Go branch names are") |
| log.Print("either 'gotip' or 'go<version>' like 'go1.23'. goos and goarch") |
| log.Print("are the same as the values you'd use in build tags and all") |
| log.Print("lower-case. The host suffix is optional and you likely do not") |
| log.Print("need to specify it, but see the full list for what's available.") |
| log.Print("It's usually just an indicator of the OS version, like '13' to") |
| log.Print("indicate macOS 13 for darwin/amd64 builders. Mods are specifiers") |
| log.Print("like 'race' and 'longtest'.") |
| log.Print() |
| log.Print("gomotes are set up with the same code used to set up the") |
| log.Print("environment on the builder except without a Go toolchain.") |
| log.Print("Subrepo gomotes set up a copy of the subrepo in the workdir,") |
| log.Print("a full git checkout sync'd to tip-of-tree.") |
| log.Print() |
| log.Print("Flags:") |
| fs.PrintDefaults() |
| if luciDisabled() { |
| log.Print("\nValid types:") |
| for _, bt := range builders() { |
| var warn string |
| if bt.IsReverse { |
| if bt.ExpectNum > 0 { |
| warn = fmt.Sprintf(" [limited capacity: %d machines]", bt.ExpectNum) |
| } else { |
| warn = " [limited capacity]" |
| } |
| } |
| log.Printf(" * %s%s\n", bt.Name, warn) |
| } |
| } |
| os.Exit(1) |
| } |
| var listBuilders bool |
| fs.BoolVar(&listBuilders, "list", false, "list builder types and exit") |
| var cfg createConfig |
| fs.BoolVar(&cfg.printStatus, "status", true, "print regular status updates while waiting") |
| fs.IntVar(&cfg.count, "count", 1, "number of instances to create") |
| fs.BoolVar(&cfg.setup, "setup", false, "set up the instance by pushing GOROOT and building the Go toolchain") |
| fs.StringVar(&cfg.newGroup, "new-group", "", "also create a new group and add the new instances to it") |
| fs.BoolVar(&cfg.useGolangbuild, "use-golangbuild", true, "disable the installation of build dependencies installed by golangbuild") |
| |
| fs.Parse(args) |
| if listBuilders { |
| if luciDisabled() { |
| for _, bt := range builders() { |
| fmt.Fprintln(os.Stdout, bt.Name) |
| } |
| return nil |
| } |
| swarmingBuilders, err := swarmingBuilders() |
| if err != nil { |
| return err |
| } |
| for _, builder := range swarmingBuilders { |
| fmt.Fprintln(os.Stdout, builder) |
| } |
| return nil |
| } |
| if fs.NArg() != 1 { |
| fs.Usage() |
| } |
| builderType := fs.Arg(0) |
| _, _, err := createInstances(context.Background(), builderType, &cfg) |
| return err |
| } |
| |
| type createConfig struct { |
| printStatus bool |
| count int |
| setup bool |
| newGroup string |
| useGolangbuild bool |
| } |
| |
| func createInstances(ctx context.Context, builderType string, cfg *createConfig) ([]string, *groupData, error) { |
| var groupMu sync.Mutex |
| group := activeGroup |
| var err error |
| if cfg.newGroup != "" { |
| group, err = doCreateGroup(cfg.newGroup) |
| if err != nil { |
| return nil, nil, err |
| } |
| } |
| if group == nil && os.Getenv("GOMOTE_GROUP") != "" { |
| group, err = doCreateGroup(os.Getenv("GOMOTE_GROUP")) |
| if err != nil { |
| return nil, nil, err |
| } |
| } |
| |
| var instancesMu sync.Mutex |
| var instances []string |
| var tmpOutDir string |
| var tmpOutDirOnce sync.Once |
| eg, ctx := errgroup.WithContext(ctx) |
| client := gomoteServerClient(ctx) |
| for i := 0; i < cfg.count; i++ { |
| i := i |
| eg.Go(func() error { |
| start := time.Now() |
| var exp []string |
| if !cfg.useGolangbuild { |
| exp = append(exp, "disable-golang-build") |
| } |
| stream, err := client.CreateInstance(ctx, &protos.CreateInstanceRequest{BuilderType: builderType, ExperimentOption: exp}) |
| if err != nil { |
| return fmt.Errorf("failed to create buildlet: %w", err) |
| } |
| var inst string |
| updateLoop: |
| for { |
| update, err := stream.Recv() |
| switch { |
| case err == io.EOF: |
| break updateLoop |
| case err != nil: |
| return fmt.Errorf("failed to create buildlet (%d): %w", i+1, err) |
| case update.GetStatus() != protos.CreateInstanceResponse_COMPLETE && cfg.printStatus: |
| log.Printf("still creating %s (%d) after %v; %d requests ahead of you\n", builderType, i+1, time.Since(start).Round(time.Second), update.GetWaitersAhead()) |
| case update.GetStatus() == protos.CreateInstanceResponse_COMPLETE: |
| inst = update.GetInstance().GetGomoteId() |
| } |
| } |
| fmt.Println(inst) |
| |
| instancesMu.Lock() |
| instances = append(instances, inst) |
| instancesMu.Unlock() |
| |
| if group != nil { |
| groupMu.Lock() |
| group.Instances = append(group.Instances, inst) |
| groupMu.Unlock() |
| } |
| if !cfg.setup { |
| return nil |
| } |
| |
| // -setup is set, so push GOROOT and run make.bash. |
| |
| tmpOutDirOnce.Do(func() { |
| tmpOutDir, err = os.MkdirTemp("", "gomote") |
| }) |
| if err != nil { |
| return fmt.Errorf("failed to create a temporary directory for setup output: %w", err) |
| } |
| |
| // Push GOROOT. |
| detailedProgress := cfg.count == 1 |
| goroot, err := getGOROOT() |
| if err != nil { |
| return err |
| } |
| if !detailedProgress { |
| log.Printf("Pushing GOROOT %q to %q...\n", goroot, inst) |
| } |
| if err := doPush(ctx, inst, goroot, false, detailedProgress); err != nil { |
| return err |
| } |
| |
| // Run make.bash or make.bat. |
| cmd := "go/src/make.bash" |
| if strings.Contains(builderType, "windows") { |
| cmd = "go/src/make.bat" |
| } |
| |
| // Create a file to write output to so it doesn't get lost. |
| outf, err := os.Create(filepath.Join(tmpOutDir, fmt.Sprintf("%s.stdout", inst))) |
| if err != nil { |
| return err |
| } |
| defer func() { |
| outf.Close() |
| log.Printf("Wrote results from %q to %q.\n", inst, outf.Name()) |
| }() |
| log.Printf("Streaming results from %q to %q...\n", inst, outf.Name()) |
| |
| // If this is the only command running, print to stdout too, for convenience and |
| // backwards compatibility. |
| outputs := []io.Writer{outf} |
| if detailedProgress { |
| outputs = append(outputs, os.Stdout) |
| } else { |
| log.Printf("Running %q on %q...\n", cmd, inst) |
| } |
| return doRun(ctx, inst, cmd, []string{}, runWriters(outputs...)) |
| }) |
| } |
| if err := eg.Wait(); err != nil { |
| return nil, nil, err |
| } |
| if group != nil { |
| if err := storeGroup(group); err != nil { |
| return nil, nil, err |
| } |
| } |
| return instances, group, nil |
| } |