2019-04-06 19:32:14 +00:00
|
|
|
package exec
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-11-17 07:42:42 +00:00
|
|
|
"fmt"
|
2019-04-06 19:32:14 +00:00
|
|
|
"io"
|
2020-11-17 07:42:42 +00:00
|
|
|
"io/ioutil"
|
2019-04-06 19:32:14 +00:00
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"runtime"
|
|
|
|
"strings"
|
|
|
|
|
2019-09-14 12:21:16 +00:00
|
|
|
"github.com/drone/envsubst"
|
2021-10-12 07:25:13 +00:00
|
|
|
"github.com/rs/zerolog/log"
|
|
|
|
"github.com/urfave/cli"
|
|
|
|
|
2021-09-24 11:18:34 +00:00
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/backend"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/backend/docker"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/frontend"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/frontend/yaml"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/frontend/yaml/compiler"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/frontend/yaml/linter"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/frontend/yaml/matrix"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/interrupt"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/multipart"
|
2019-04-06 19:32:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Command exports the exec command.
|
|
|
|
var Command = cli.Command{
|
|
|
|
Name: "exec",
|
|
|
|
Usage: "execute a local build",
|
2021-10-02 08:59:34 +00:00
|
|
|
ArgsUsage: "[path/to/.woodpecker.yml]",
|
2019-04-06 19:32:14 +00:00
|
|
|
Action: func(c *cli.Context) {
|
|
|
|
if err := exec(c); err != nil {
|
2021-10-12 07:25:13 +00:00
|
|
|
log.Fatal().Err(err).Msg("")
|
2019-04-06 19:32:14 +00:00
|
|
|
}
|
|
|
|
},
|
2021-10-03 13:07:39 +00:00
|
|
|
Flags: flags,
|
2019-04-06 19:32:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func exec(c *cli.Context) error {
|
|
|
|
file := c.Args().First()
|
|
|
|
if file == "" {
|
Clean up config environment variables for server and agent (#218)
The goal here is to make consistent use of configuration environment variables prefixed `WOODPECKER_`. Where several variants existed, this PR aims to remove all but one option, leaving the most explicit.
This PR only changes server and agent code, but not documentation, in order to keep the PR digestible. Once we have consensus that this is correct, I'll change docs accordingly.
User (rather: admin) facing changes in this PR:
- In general, support for all server and agent config environment variables (env vars) starting with `DRONE_` is removed. The according `WOODPECKER_*` variables must be used instead.
- The env var `WOODPECKER_HOST` replaces `DRONE_HOST`, and `DRONE_SERVER_HOST`.
- The env var `WOODPECKER_AGENT_SECRET` is used to configure the shared secret which agents use to authenticate against the server. It replaces `WOODPECKER_SECRET`, `DRONE_SECRET`, `WOODPECKER_PASSWORD`, `DRONE_PASSWORD`, and `DRONE_AGENT_SECRET`.
- The env var `WOODPECKER_DATABASE_DRIVER` replaces `DRONE_DATABASE_DRIVER` and `DATABASE_DRIVER`.
- The env var `WOODPECKER_DATABASE_DATASOURCE` replaces `DRONE_DATABASE_DATASOURCE` and `DATABASE_CONFIG`.
2021-09-28 13:43:44 +00:00
|
|
|
file = ".woodpecker.yml"
|
2019-04-06 19:32:14 +00:00
|
|
|
}
|
|
|
|
|
2020-11-17 07:42:42 +00:00
|
|
|
dat, err := ioutil.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
axes, err := matrix.ParseString(string(dat))
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Parse matrix fail")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(axes) == 0 {
|
|
|
|
axes = append(axes, matrix.Axis{})
|
|
|
|
}
|
|
|
|
for _, axis := range axes {
|
|
|
|
err := execWithAxis(c, axis)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func execWithAxis(c *cli.Context, axis matrix.Axis) error {
|
|
|
|
file := c.Args().First()
|
|
|
|
if file == "" {
|
Clean up config environment variables for server and agent (#218)
The goal here is to make consistent use of configuration environment variables prefixed `WOODPECKER_`. Where several variants existed, this PR aims to remove all but one option, leaving the most explicit.
This PR only changes server and agent code, but not documentation, in order to keep the PR digestible. Once we have consensus that this is correct, I'll change docs accordingly.
User (rather: admin) facing changes in this PR:
- In general, support for all server and agent config environment variables (env vars) starting with `DRONE_` is removed. The according `WOODPECKER_*` variables must be used instead.
- The env var `WOODPECKER_HOST` replaces `DRONE_HOST`, and `DRONE_SERVER_HOST`.
- The env var `WOODPECKER_AGENT_SECRET` is used to configure the shared secret which agents use to authenticate against the server. It replaces `WOODPECKER_SECRET`, `DRONE_SECRET`, `WOODPECKER_PASSWORD`, `DRONE_PASSWORD`, and `DRONE_AGENT_SECRET`.
- The env var `WOODPECKER_DATABASE_DRIVER` replaces `DRONE_DATABASE_DRIVER` and `DATABASE_DRIVER`.
- The env var `WOODPECKER_DATABASE_DATASOURCE` replaces `DRONE_DATABASE_DATASOURCE` and `DATABASE_CONFIG`.
2021-09-28 13:43:44 +00:00
|
|
|
file = ".woodpecker.yml"
|
2020-11-17 07:42:42 +00:00
|
|
|
}
|
|
|
|
|
2021-09-21 03:47:08 +00:00
|
|
|
metadata := metadataFromContext(c, axis)
|
2019-04-06 19:32:14 +00:00
|
|
|
environ := metadata.Environ()
|
2021-09-24 14:29:26 +00:00
|
|
|
var secrets []compiler.Secret
|
2019-04-06 19:32:14 +00:00
|
|
|
for k, v := range metadata.EnvironDrone() {
|
|
|
|
environ[k] = v
|
|
|
|
}
|
|
|
|
for key, val := range metadata.Job.Matrix {
|
|
|
|
environ[key] = val
|
|
|
|
secrets = append(secrets, compiler.Secret{
|
|
|
|
Name: key,
|
|
|
|
Value: val,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-11-18 06:45:57 +00:00
|
|
|
droneEnv := make(map[string]string)
|
2019-04-06 19:32:14 +00:00
|
|
|
for _, env := range c.StringSlice("env") {
|
|
|
|
envs := strings.SplitN(env, "=", 2)
|
2020-11-18 06:45:57 +00:00
|
|
|
droneEnv[envs[0]] = envs[1]
|
2019-04-06 19:32:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
tmpl, err := envsubst.ParseFile(file)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
confstr, err := tmpl.Execute(func(name string) string {
|
|
|
|
return environ[name]
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
conf, err := yaml.ParseString(confstr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// configure volumes for local execution
|
|
|
|
volumes := c.StringSlice("volumes")
|
|
|
|
if c.Bool("local") {
|
|
|
|
var (
|
|
|
|
workspaceBase = conf.Workspace.Base
|
|
|
|
workspacePath = conf.Workspace.Path
|
|
|
|
)
|
|
|
|
if workspaceBase == "" {
|
|
|
|
workspaceBase = c.String("workspace-base")
|
|
|
|
}
|
|
|
|
if workspacePath == "" {
|
|
|
|
workspacePath = c.String("workspace-path")
|
|
|
|
}
|
|
|
|
dir, _ := filepath.Abs(filepath.Dir(file))
|
|
|
|
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
dir = convertPathForWindows(dir)
|
|
|
|
}
|
|
|
|
volumes = append(volumes, c.String("prefix")+"_default:"+workspaceBase)
|
|
|
|
volumes = append(volumes, dir+":"+path.Join(workspaceBase, workspacePath))
|
|
|
|
}
|
|
|
|
|
|
|
|
// lint the yaml file
|
|
|
|
if lerr := linter.New(linter.WithTrusted(true)).Lint(conf); lerr != nil {
|
|
|
|
return lerr
|
|
|
|
}
|
|
|
|
|
|
|
|
// compiles the yaml file
|
|
|
|
compiled := compiler.New(
|
|
|
|
compiler.WithEscalated(
|
|
|
|
c.StringSlice("privileged")...,
|
|
|
|
),
|
|
|
|
compiler.WithVolumes(volumes...),
|
|
|
|
compiler.WithWorkspace(
|
|
|
|
c.String("workspace-base"),
|
|
|
|
c.String("workspace-path"),
|
|
|
|
),
|
|
|
|
compiler.WithNetworks(
|
|
|
|
c.StringSlice("network")...,
|
|
|
|
),
|
|
|
|
compiler.WithPrefix(
|
|
|
|
c.String("prefix"),
|
|
|
|
),
|
|
|
|
compiler.WithProxy(),
|
|
|
|
compiler.WithLocal(
|
|
|
|
c.Bool("local"),
|
|
|
|
),
|
|
|
|
compiler.WithNetrc(
|
|
|
|
c.String("netrc-username"),
|
|
|
|
c.String("netrc-password"),
|
|
|
|
c.String("netrc-machine"),
|
|
|
|
),
|
|
|
|
compiler.WithMetadata(metadata),
|
|
|
|
compiler.WithSecret(secrets...),
|
2020-11-18 06:45:57 +00:00
|
|
|
compiler.WithEnviron(droneEnv),
|
2019-04-06 19:32:14 +00:00
|
|
|
).Compile(conf)
|
|
|
|
engine, err := docker.NewEnv()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), c.Duration("timeout"))
|
|
|
|
defer cancel()
|
|
|
|
ctx = interrupt.WithContext(ctx)
|
|
|
|
|
|
|
|
return pipeline.New(compiled,
|
|
|
|
pipeline.WithContext(ctx),
|
|
|
|
pipeline.WithTracer(pipeline.DefaultTracer),
|
|
|
|
pipeline.WithLogger(defaultLogger),
|
|
|
|
pipeline.WithEngine(engine),
|
|
|
|
).Run()
|
|
|
|
}
|
|
|
|
|
|
|
|
// return the metadata from the cli context.
|
2021-09-21 03:47:08 +00:00
|
|
|
func metadataFromContext(c *cli.Context, axis matrix.Axis) frontend.Metadata {
|
2019-04-06 19:32:14 +00:00
|
|
|
return frontend.Metadata{
|
|
|
|
Repo: frontend.Repo{
|
|
|
|
Name: c.String("repo-name"),
|
|
|
|
Link: c.String("repo-link"),
|
|
|
|
Remote: c.String("repo-remote-url"),
|
|
|
|
Private: c.Bool("repo-private"),
|
|
|
|
},
|
|
|
|
Curr: frontend.Build{
|
|
|
|
Number: c.Int("build-number"),
|
|
|
|
Parent: c.Int("parent-build-number"),
|
|
|
|
Created: c.Int64("build-created"),
|
|
|
|
Started: c.Int64("build-started"),
|
|
|
|
Finished: c.Int64("build-finished"),
|
|
|
|
Status: c.String("build-status"),
|
|
|
|
Event: c.String("build-event"),
|
|
|
|
Link: c.String("build-link"),
|
|
|
|
Target: c.String("build-target"),
|
|
|
|
Commit: frontend.Commit{
|
|
|
|
Sha: c.String("commit-sha"),
|
|
|
|
Ref: c.String("commit-ref"),
|
|
|
|
Refspec: c.String("commit-refspec"),
|
|
|
|
Branch: c.String("commit-branch"),
|
|
|
|
Message: c.String("commit-message"),
|
|
|
|
Author: frontend.Author{
|
|
|
|
Name: c.String("commit-author-name"),
|
|
|
|
Email: c.String("commit-author-email"),
|
|
|
|
Avatar: c.String("commit-author-avatar"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Prev: frontend.Build{
|
|
|
|
Number: c.Int("prev-build-number"),
|
|
|
|
Created: c.Int64("prev-build-created"),
|
|
|
|
Started: c.Int64("prev-build-started"),
|
|
|
|
Finished: c.Int64("prev-build-finished"),
|
|
|
|
Status: c.String("prev-build-status"),
|
|
|
|
Event: c.String("prev-build-event"),
|
|
|
|
Link: c.String("prev-build-link"),
|
|
|
|
Commit: frontend.Commit{
|
|
|
|
Sha: c.String("prev-commit-sha"),
|
|
|
|
Ref: c.String("prev-commit-ref"),
|
|
|
|
Refspec: c.String("prev-commit-refspec"),
|
|
|
|
Branch: c.String("prev-commit-branch"),
|
|
|
|
Message: c.String("prev-commit-message"),
|
|
|
|
Author: frontend.Author{
|
|
|
|
Name: c.String("prev-commit-author-name"),
|
|
|
|
Email: c.String("prev-commit-author-email"),
|
|
|
|
Avatar: c.String("prev-commit-author-avatar"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Job: frontend.Job{
|
|
|
|
Number: c.Int("job-number"),
|
2021-09-21 03:47:08 +00:00
|
|
|
Matrix: axis,
|
2019-04-06 19:32:14 +00:00
|
|
|
},
|
|
|
|
Sys: frontend.System{
|
|
|
|
Name: c.String("system-name"),
|
|
|
|
Link: c.String("system-link"),
|
|
|
|
Arch: c.String("system-arch"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func convertPathForWindows(path string) string {
|
|
|
|
base := filepath.VolumeName(path)
|
|
|
|
if len(base) == 2 {
|
|
|
|
path = strings.TrimPrefix(path, base)
|
|
|
|
base = strings.ToLower(strings.TrimSuffix(base, ":"))
|
|
|
|
return "/" + base + filepath.ToSlash(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
return filepath.ToSlash(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
var defaultLogger = pipeline.LogFunc(func(proc *backend.Step, rc multipart.Reader) error {
|
|
|
|
part, err := rc.NextPart()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
logstream := NewLineWriter(proc.Alias)
|
|
|
|
io.Copy(logstream, part)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|