package cli import ( "context" "fmt" "os" "os/signal" "time" "golang.org/x/xerrors" "cdr.dev/slog" "cdr.dev/slog/sloggers/sloghuman" agpl "github.com/coder/coder/cli" "github.com/coder/coder/cli/clibase" "github.com/coder/coder/cli/cliui" "github.com/coder/coder/coderd/database" "github.com/coder/coder/codersdk" "github.com/coder/coder/provisioner/terraform" "github.com/coder/coder/provisionerd" provisionerdproto "github.com/coder/coder/provisionerd/proto" "github.com/coder/coder/provisionersdk" "github.com/coder/coder/provisionersdk/proto" ) func (r *RootCmd) provisionerDaemons() *clibase.Cmd { cmd := &clibase.Cmd{ Use: "provisionerd", Short: "Manage provisioner daemons", Handler: func(inv *clibase.Invocation) error { return inv.Command.HelpHandler(inv) }, Children: []*clibase.Cmd{ r.provisionerDaemonStart(), }, } return cmd } func (r *RootCmd) provisionerDaemonStart() *clibase.Cmd { var ( cacheDir string rawTags []string pollInterval time.Duration pollJitter time.Duration ) client := new(codersdk.Client) cmd := &clibase.Cmd{ Use: "start", Short: "Run a provisioner daemon", Middleware: clibase.Chain( r.InitClient(client), ), Handler: func(inv *clibase.Invocation) error { ctx, cancel := context.WithCancel(inv.Context()) defer cancel() notifyCtx, notifyStop := signal.NotifyContext(ctx, agpl.InterruptSignals...) defer notifyStop() org, err := agpl.CurrentOrganization(inv, client) if err != nil { return xerrors.Errorf("get current organization: %w", err) } tags, err := agpl.ParseProvisionerTags(rawTags) if err != nil { return err } err = os.MkdirAll(cacheDir, 0o700) if err != nil { return xerrors.Errorf("mkdir %q: %w", cacheDir, err) } terraformClient, terraformServer := provisionersdk.MemTransportPipe() go func() { <-ctx.Done() _ = terraformClient.Close() _ = terraformServer.Close() }() logger := slog.Make(sloghuman.Sink(inv.Stderr)) errCh := make(chan error, 1) go func() { defer cancel() err := terraform.Serve(ctx, &terraform.ServeOptions{ ServeOptions: &provisionersdk.ServeOptions{ Listener: terraformServer, }, CachePath: cacheDir, Logger: logger.Named("terraform"), }) if err != nil && !xerrors.Is(err, context.Canceled) { select { case errCh <- err: default: } } }() tempDir, err := os.MkdirTemp("", "provisionerd") if err != nil { return err } logger.Info(ctx, "starting provisioner daemon", slog.F("tags", tags)) provisioners := provisionerd.Provisioners{ string(database.ProvisionerTypeTerraform): proto.NewDRPCProvisionerClient(terraformClient), } srv := provisionerd.New(func(ctx context.Context) (provisionerdproto.DRPCProvisionerDaemonClient, error) { return client.ServeProvisionerDaemon(ctx, org.ID, []codersdk.ProvisionerType{ codersdk.ProvisionerTypeTerraform, }, tags) }, &provisionerd.Options{ Logger: logger, JobPollInterval: pollInterval, JobPollJitter: pollJitter, UpdateInterval: 500 * time.Millisecond, Provisioners: provisioners, WorkDirectory: tempDir, }) var exitErr error select { case <-notifyCtx.Done(): exitErr = notifyCtx.Err() _, _ = fmt.Fprintln(inv.Stdout, cliui.DefaultStyles.Bold.Render( "Interrupt caught, gracefully exiting. Use ctrl+\\ to force quit", )) case exitErr = <-errCh: } if exitErr != nil && !xerrors.Is(exitErr, context.Canceled) { cliui.Errorf(inv.Stderr, "Unexpected error, shutting down server: %s\n", exitErr) } shutdown, shutdownCancel := context.WithTimeout(ctx, time.Minute) defer shutdownCancel() err = srv.Shutdown(shutdown) if err != nil { return xerrors.Errorf("shutdown: %w", err) } cancel() if xerrors.Is(exitErr, context.Canceled) { return nil } return exitErr }, } cmd.Options = clibase.OptionSet{ { Flag: "cache-dir", FlagShorthand: "c", Env: "CODER_CACHE_DIRECTORY", Description: "Directory to store cached data.", Default: codersdk.DefaultCacheDir(), Value: clibase.StringOf(&cacheDir), }, { Flag: "tag", FlagShorthand: "t", Env: "CODER_PROVISIONERD_TAGS", Description: "Tags to filter provisioner jobs by.", Value: clibase.StringArrayOf(&rawTags), }, { Flag: "poll-interval", Env: "CODER_PROVISIONERD_POLL_INTERVAL", Default: time.Second.String(), Description: "How often to poll for provisioner jobs.", Value: clibase.DurationOf(&pollInterval), }, { Flag: "poll-jitter", Env: "CODER_PROVISIONERD_POLL_JITTER", Description: "How much to jitter the poll interval by.", Default: (100 * time.Millisecond).String(), Value: clibase.DurationOf(&pollJitter), }, } return cmd }