orchard/internal/command/worker/run.go

185 lines
5.3 KiB
Go

package worker
import (
"errors"
"fmt"
"github.com/cirruslabs/orchard/internal/bootstraptoken"
"github.com/cirruslabs/orchard/internal/netconstants"
"github.com/cirruslabs/orchard/internal/worker"
"github.com/cirruslabs/orchard/pkg/client"
v1 "github.com/cirruslabs/orchard/pkg/resource/v1"
"github.com/spf13/cobra"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
"gopkg.in/natefinch/lumberjack.v2"
"io"
"os"
"strings"
)
var (
ErrRunFailed = errors.New("failed to run worker")
ErrNoBootstrapTokenProvided = errors.New("no bootstrap token was provided")
ErrEmptyBootstrapTokenProvided = errors.New("empty bootstrap token was provided")
)
var name string
var bootstrapTokenRaw string
var bootstrapTokenStdin bool
var logFilePath string
var stringToStringResources map[string]string
var labels map[string]string
var noPKI bool
var defaultCPU uint64
var defaultMemory uint64
var debug bool
func newRunCommand() *cobra.Command {
cmd := &cobra.Command{
Use: "run CONTROLLER_URL",
Short: "Run worker",
RunE: runWorker,
Args: cobra.ExactArgs(1),
}
cmd.PersistentFlags().StringVar(&name, "name", "",
"name of the worker (defaults to the hostname)")
cmd.PersistentFlags().StringVar(&bootstrapTokenRaw, "bootstrap-token", "",
"a bootstrap token retrieved via \"orchard get bootstrap-token <service-account-name-for-workers>\"")
cmd.PersistentFlags().BoolVar(&bootstrapTokenStdin, "bootstrap-token-stdin", false,
"use this flag to provide a bootstrap token via the standard input")
cmd.PersistentFlags().StringVar(&logFilePath, "log-file", "",
"optional path to a file where logs (up to 100 Mb) will be written.")
cmd.PersistentFlags().StringToStringVar(&stringToStringResources, "resources", map[string]string{},
"resources that this worker provides")
cmd.PersistentFlags().StringToStringVar(&labels, "labels", map[string]string{},
"labels that this worker supports")
cmd.PersistentFlags().BoolVar(&noPKI, "no-pki", false,
"do not use the host's root CA set and instead validate the Controller's presented "+
"certificate using a bootstrap token (or manually via fingerprint, "+
"if no bootstrap token is provided)")
cmd.PersistentFlags().Uint64Var(&defaultCPU, "default-cpu", 4, "number of CPUs to use for VMs "+
"that do not explicitly specify a value")
cmd.PersistentFlags().Uint64Var(&defaultMemory, "default-memory", 8*1024, "megabytes of memory "+
"to use for VMs that do not explicitly specify a value")
cmd.PersistentFlags().BoolVar(&debug, "debug", false, "enable debug logging")
return cmd
}
func runWorker(cmd *cobra.Command, args []string) (err error) {
// Parse controller URL
controllerURL, err := netconstants.NormalizeAddress(args[0])
if err != nil {
return err
}
// Parse bootstrap token
bootstrapTokenRaw, err := readBootstrapToken()
if err != nil {
return err
}
if bootstrapTokenRaw == "" {
return ErrEmptyBootstrapTokenProvided
}
bootstrapToken, err := bootstraptoken.NewFromString(bootstrapTokenRaw)
if err != nil {
return err
}
// Convert resources
resources, err := v1.NewResourcesFromStringToString(stringToStringResources)
if err != nil {
return fmt.Errorf("%w: %v", ErrRunFailed, err)
}
clientOpts := []client.Option{
client.WithAddress(controllerURL.String()),
client.WithCredentials(bootstrapToken.ServiceAccountName(), bootstrapToken.ServiceAccountToken()),
}
if trustedCertificate := bootstrapToken.Certificate(); trustedCertificate != nil {
clientOpts = append(clientOpts, client.WithTrustedCertificate(trustedCertificate))
} else if noPKI {
return fmt.Errorf("%w: --no-pki was specified, but not trusted certificate was provided "+
"in the bootstrap token", ErrRunFailed)
}
controllerClient, err := client.New(clientOpts...)
if err != nil {
return err
}
// Initialize the logger
logger, err := createLogger()
if err != nil {
return err
}
defer func() {
if syncErr := logger.Sync(); syncErr != nil && err == nil {
err = syncErr
}
}()
workerInstance, err := worker.New(
controllerClient,
worker.WithName(name),
worker.WithResources(resources),
worker.WithLabels(labels),
worker.WithDefaultCPUAndMemory(defaultCPU, defaultMemory),
worker.WithLogger(logger),
)
if err != nil {
return err
}
defer workerInstance.Close()
return workerInstance.Run(cmd.Context())
}
func readBootstrapToken() (string, error) {
if bootstrapTokenRaw != "" && bootstrapTokenStdin {
return "", fmt.Errorf("--bootstrap-token and --bootstrap-token-stdin are mutually exclusive")
}
if bootstrapTokenRaw != "" {
return bootstrapTokenRaw, nil
}
if bootstrapTokenStdin {
stdinBytes, err := io.ReadAll(os.Stdin)
if err != nil {
return "", fmt.Errorf("failed to read the bootstrap token from the standard input: %w", err)
}
return strings.TrimSuffix(string(stdinBytes), "\n"), nil
}
return "", ErrNoBootstrapTokenProvided
}
func createLogger() (*zap.Logger, error) {
level := zap.InfoLevel
if debug {
level = zap.DebugLevel
}
if logFilePath == "" {
zapConfig := zap.NewProductionConfig()
zapConfig.Level = zap.NewAtomicLevelAt(level)
return zapConfig.Build()
}
logFileWriter := zapcore.AddSync(&lumberjack.Logger{
Filename: logFilePath,
MaxSize: 100, // megabytes
})
core := zapcore.NewCore(
zapcore.NewJSONEncoder(zap.NewProductionEncoderConfig()),
logFileWriter,
level,
)
return zap.New(core), nil
}