9f90cba993
Reviewed-on: https://gitea.com/gitea/act_runner/pulls/65 Reviewed-by: Jason Song <i@wolfogre.com> Co-authored-by: Benjamin Loison <benjamin.loison@orange.fr> Co-committed-by: Benjamin Loison <benjamin.loison@orange.fr>
310 lines
7.7 KiB
Go
310 lines
7.7 KiB
Go
// Copyright 2022 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"bufio"
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"os/signal"
|
|
"runtime"
|
|
"strings"
|
|
"time"
|
|
|
|
pingv1 "code.gitea.io/actions-proto-go/ping/v1"
|
|
"gitea.com/gitea/act_runner/client"
|
|
"gitea.com/gitea/act_runner/config"
|
|
"gitea.com/gitea/act_runner/register"
|
|
|
|
"github.com/bufbuild/connect-go"
|
|
"github.com/joho/godotenv"
|
|
"github.com/mattn/go-isatty"
|
|
log "github.com/sirupsen/logrus"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
// runRegister registers a runner to the server
|
|
func runRegister(ctx context.Context, regArgs *registerArgs, envFile string) func(*cobra.Command, []string) error {
|
|
return func(cmd *cobra.Command, args []string) error {
|
|
log.SetReportCaller(false)
|
|
isTerm := isatty.IsTerminal(os.Stdout.Fd())
|
|
log.SetFormatter(&log.TextFormatter{
|
|
DisableColors: !isTerm,
|
|
DisableTimestamp: true,
|
|
})
|
|
log.SetLevel(log.DebugLevel)
|
|
|
|
log.Infof("Registering runner, arch=%s, os=%s, version=%s.",
|
|
runtime.GOARCH, runtime.GOOS, version)
|
|
|
|
// runner always needs root permission
|
|
if os.Getuid() != 0 {
|
|
// TODO: use a better way to check root permission
|
|
log.Warnf("Runner in user-mode.")
|
|
}
|
|
|
|
if regArgs.NoInteractive {
|
|
if err := registerNoInteractive(envFile, regArgs); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
go func() {
|
|
if err := registerInteractive(envFile); err != nil {
|
|
// log.Errorln(err)
|
|
os.Exit(2)
|
|
return
|
|
}
|
|
os.Exit(0)
|
|
}()
|
|
|
|
c := make(chan os.Signal, 1)
|
|
signal.Notify(c, os.Interrupt)
|
|
<-c
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// registerArgs represents the arguments for register command
|
|
type registerArgs struct {
|
|
NoInteractive bool
|
|
InstanceAddr string
|
|
Insecure bool
|
|
Token string
|
|
RunnerName string
|
|
Labels string
|
|
}
|
|
|
|
type registerStage int8
|
|
|
|
const (
|
|
StageUnknown registerStage = -1
|
|
StageOverwriteLocalConfig registerStage = iota + 1
|
|
StageInputInstance
|
|
StageInputToken
|
|
StageInputRunnerName
|
|
StageInputCustomLabels
|
|
StageWaitingForRegistration
|
|
StageExit
|
|
)
|
|
|
|
var defaultLabels = []string{
|
|
"ubuntu-latest:docker://node:16-bullseye",
|
|
"ubuntu-22.04:docker://node:16-bullseye", // There's no node:16-bookworm yet
|
|
"ubuntu-20.04:docker://node:16-bullseye",
|
|
"ubuntu-18.04:docker://node:16-buster",
|
|
}
|
|
|
|
type registerInputs struct {
|
|
InstanceAddr string
|
|
Insecure bool
|
|
Token string
|
|
RunnerName string
|
|
CustomLabels []string
|
|
}
|
|
|
|
func (r *registerInputs) validate() error {
|
|
if r.InstanceAddr == "" {
|
|
return fmt.Errorf("instance address is empty")
|
|
}
|
|
if r.Token == "" {
|
|
return fmt.Errorf("token is empty")
|
|
}
|
|
if len(r.CustomLabels) > 0 {
|
|
return validateLabels(r.CustomLabels)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func validateLabels(labels []string) error {
|
|
for _, label := range labels {
|
|
values := strings.SplitN(label, ":", 2)
|
|
if len(values) > 2 {
|
|
return fmt.Errorf("Invalid label: %s", label)
|
|
}
|
|
// len(values) == 1, label for non docker execution environment
|
|
// TODO: validate value format, like docker://node:16-buster
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *registerInputs) assignToNext(stage registerStage, value string) registerStage {
|
|
// must set instance address and token.
|
|
// if empty, keep current stage.
|
|
if stage == StageInputInstance || stage == StageInputToken {
|
|
if value == "" {
|
|
return stage
|
|
}
|
|
}
|
|
|
|
// set hostname for runner name if empty
|
|
if stage == StageInputRunnerName && value == "" {
|
|
value, _ = os.Hostname()
|
|
}
|
|
|
|
switch stage {
|
|
case StageOverwriteLocalConfig:
|
|
if value == "Y" || value == "y" {
|
|
return StageInputInstance
|
|
}
|
|
return StageExit
|
|
case StageInputInstance:
|
|
r.InstanceAddr = value
|
|
return StageInputToken
|
|
case StageInputToken:
|
|
r.Token = value
|
|
return StageInputRunnerName
|
|
case StageInputRunnerName:
|
|
r.RunnerName = value
|
|
return StageInputCustomLabels
|
|
case StageInputCustomLabels:
|
|
r.CustomLabels = defaultLabels
|
|
if value != "" {
|
|
r.CustomLabels = strings.Split(value, ",")
|
|
}
|
|
|
|
if validateLabels(r.CustomLabels) != nil {
|
|
log.Infoln("Invalid labels, please input again, leave blank to use the default labels (for example, ubuntu-20.04:docker://node:16-bullseye,ubuntu-18.04:docker://node:16-buster)")
|
|
return StageInputCustomLabels
|
|
}
|
|
return StageWaitingForRegistration
|
|
}
|
|
return StageUnknown
|
|
}
|
|
|
|
func registerInteractive(envFile string) error {
|
|
var (
|
|
reader = bufio.NewReader(os.Stdin)
|
|
stage = StageInputInstance
|
|
inputs = new(registerInputs)
|
|
)
|
|
|
|
// check if overwrite local config
|
|
_ = godotenv.Load(envFile)
|
|
cfg, _ := config.FromEnviron()
|
|
if f, err := os.Stat(cfg.Runner.File); err == nil && !f.IsDir() {
|
|
stage = StageOverwriteLocalConfig
|
|
}
|
|
|
|
for {
|
|
printStageHelp(stage)
|
|
|
|
cmdString, err := reader.ReadString('\n')
|
|
if err != nil {
|
|
return err
|
|
}
|
|
stage = inputs.assignToNext(stage, strings.TrimSpace(cmdString))
|
|
|
|
if stage == StageWaitingForRegistration {
|
|
log.Infof("Registering runner, name=%s, instance=%s, labels=%v.", inputs.RunnerName, inputs.InstanceAddr, inputs.CustomLabels)
|
|
if err := doRegister(&cfg, inputs); err != nil {
|
|
log.Errorf("Failed to register runner: %v", err)
|
|
} else {
|
|
log.Infof("Runner registered successfully.")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
if stage == StageExit {
|
|
return nil
|
|
}
|
|
|
|
if stage <= StageUnknown {
|
|
log.Errorf("Invalid input, please re-run act command.")
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
func printStageHelp(stage registerStage) {
|
|
switch stage {
|
|
case StageOverwriteLocalConfig:
|
|
log.Infoln("Runner is already registered, overwrite local config? [y/N]")
|
|
case StageInputInstance:
|
|
log.Infoln("Enter the Gitea instance URL (for example, https://gitea.com/):")
|
|
case StageInputToken:
|
|
log.Infoln("Enter the runner token:")
|
|
case StageInputRunnerName:
|
|
hostname, _ := os.Hostname()
|
|
log.Infof("Enter the runner name (if set empty, use hostname: %s):\n", hostname)
|
|
case StageInputCustomLabels:
|
|
log.Infoln("Enter the runner labels, leave blank to use the default labels (comma-separated, for example, self-hosted,ubuntu-20.04:docker://node:16-bullseye,ubuntu-18.04:docker://node:16-buster):")
|
|
case StageWaitingForRegistration:
|
|
log.Infoln("Waiting for registration...")
|
|
}
|
|
}
|
|
|
|
func registerNoInteractive(envFile string, regArgs *registerArgs) error {
|
|
_ = godotenv.Load(envFile)
|
|
cfg, _ := config.FromEnviron()
|
|
inputs := ®isterInputs{
|
|
InstanceAddr: regArgs.InstanceAddr,
|
|
Insecure: regArgs.Insecure,
|
|
Token: regArgs.Token,
|
|
RunnerName: regArgs.RunnerName,
|
|
CustomLabels: defaultLabels,
|
|
}
|
|
regArgs.Labels = strings.TrimSpace(regArgs.Labels)
|
|
if regArgs.Labels != "" {
|
|
inputs.CustomLabels = strings.Split(regArgs.Labels, ",")
|
|
}
|
|
if inputs.RunnerName == "" {
|
|
inputs.RunnerName, _ = os.Hostname()
|
|
log.Infof("Runner name is empty, use hostname '%s'.", inputs.RunnerName)
|
|
}
|
|
if err := inputs.validate(); err != nil {
|
|
log.WithError(err).Errorf("Invalid input, please re-run act command.")
|
|
return nil
|
|
}
|
|
if err := doRegister(&cfg, inputs); err != nil {
|
|
log.Errorf("Failed to register runner: %v", err)
|
|
return nil
|
|
}
|
|
log.Infof("Runner registered successfully.")
|
|
return nil
|
|
}
|
|
|
|
func doRegister(cfg *config.Config, inputs *registerInputs) error {
|
|
ctx := context.Background()
|
|
|
|
// initial http client
|
|
cli := client.New(
|
|
inputs.InstanceAddr,
|
|
inputs.Insecure,
|
|
"",
|
|
"",
|
|
version,
|
|
)
|
|
|
|
for {
|
|
_, err := cli.Ping(ctx, connect.NewRequest(&pingv1.PingRequest{
|
|
Data: inputs.RunnerName,
|
|
}))
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil
|
|
default:
|
|
}
|
|
if ctx.Err() != nil {
|
|
break
|
|
}
|
|
if err != nil {
|
|
log.WithError(err).
|
|
Errorln("Cannot ping the Gitea instance server")
|
|
// TODO: if ping failed, retry or exit
|
|
time.Sleep(time.Second)
|
|
} else {
|
|
log.Debugln("Successfully pinged the Gitea instance server")
|
|
break
|
|
}
|
|
}
|
|
|
|
cfg.Runner.Name = inputs.RunnerName
|
|
cfg.Runner.Token = inputs.Token
|
|
cfg.Runner.Labels = inputs.CustomLabels
|
|
_, err := register.New(cli).Register(ctx, cfg.Runner)
|
|
return err
|
|
}
|