mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2024-12-26 10:20:29 +00:00
80d5d233b3
close #1705
172 lines
4.5 KiB
Go
172 lines
4.5 KiB
Go
// Copyright 2022 Woodpecker Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package local
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/alessio/shellescape"
|
|
"golang.org/x/exp/slices"
|
|
|
|
"github.com/woodpecker-ci/woodpecker/pipeline/backend/types"
|
|
"github.com/woodpecker-ci/woodpecker/shared/constant"
|
|
)
|
|
|
|
const (
|
|
workingSubDir = "work"
|
|
homeSubDir = "home"
|
|
)
|
|
|
|
// notAllowedEnvVarOverwrites are all env vars that can not be overwritten by step config
|
|
var notAllowedEnvVarOverwrites = []string{
|
|
"CI_NETRC_MACHINE",
|
|
"CI_NETRC_USERNAME",
|
|
"CI_NETRC_PASSWORD",
|
|
"CI_SCRIPT",
|
|
"HOME",
|
|
"SHELL",
|
|
}
|
|
|
|
type local struct {
|
|
// TODO: make cmd a cmd list to iterate over, the hard part is to have a common ReadCloser
|
|
cmd *exec.Cmd
|
|
output io.ReadCloser
|
|
workflowBaseDir string
|
|
}
|
|
|
|
// New returns a new local Engine.
|
|
func New() types.Engine {
|
|
return &local{}
|
|
}
|
|
|
|
func (e *local) Name() string {
|
|
return "local"
|
|
}
|
|
|
|
func (e *local) IsAvailable(context.Context) bool {
|
|
return true
|
|
}
|
|
|
|
func (e *local) Load(context.Context) error {
|
|
dir, err := os.MkdirTemp("", "woodpecker-local-*")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
e.workflowBaseDir = dir
|
|
|
|
if err := os.Mkdir(filepath.Join(e.workflowBaseDir, workingSubDir), 0o700); err != nil {
|
|
return err
|
|
}
|
|
return os.Mkdir(filepath.Join(e.workflowBaseDir, homeSubDir), 0o700)
|
|
}
|
|
|
|
// Setup the pipeline environment.
|
|
func (e *local) Setup(_ context.Context, _ *types.Config) error {
|
|
return nil
|
|
}
|
|
|
|
// Exec the pipeline step.
|
|
func (e *local) Exec(ctx context.Context, step *types.Step) error {
|
|
// Get environment variables
|
|
env := os.Environ()
|
|
for a, b := range step.Environment {
|
|
// append allowed env vars to command env
|
|
if !slices.Contains(notAllowedEnvVarOverwrites, a) {
|
|
env = append(env, a+"="+b)
|
|
}
|
|
}
|
|
|
|
// Set HOME
|
|
env = append(env, "HOME="+filepath.Join(e.workflowBaseDir, homeSubDir))
|
|
|
|
var command []string
|
|
if step.Image == constant.DefaultCloneImage {
|
|
// Default clone step
|
|
// TODO: use tmp HOME and insert netrc and delete it after clone
|
|
// TODO: download plugin-git binary if not exist
|
|
env = append(env, "CI_WORKSPACE="+filepath.Join(e.workflowBaseDir, workingSubDir, step.Environment["CI_REPO"]))
|
|
command = append(command, "plugin-git")
|
|
} else {
|
|
// Use "image name" as run command
|
|
command = append(command, step.Image)
|
|
command = append(command, "-c")
|
|
|
|
// TODO: use commands directly
|
|
script := ""
|
|
for _, cmd := range step.Commands {
|
|
script += fmt.Sprintf("echo + %s\n%s\n\n", shellescape.Quote(cmd), cmd)
|
|
}
|
|
script = strings.TrimSpace(script)
|
|
|
|
// Deleting the initial lines removes netrc support but adds compatibility for more shells like fish
|
|
command = append(command, script)
|
|
}
|
|
|
|
// Prepare command
|
|
e.cmd = exec.CommandContext(ctx, command[0], command[1:]...)
|
|
e.cmd.Env = env
|
|
|
|
// Prepare working directory
|
|
if step.Image == constant.DefaultCloneImage {
|
|
e.cmd.Dir = filepath.Join(e.workflowBaseDir, workingSubDir, step.Environment["CI_REPO_OWNER"])
|
|
} else {
|
|
e.cmd.Dir = filepath.Join(e.workflowBaseDir, workingSubDir, step.Environment["CI_REPO"])
|
|
}
|
|
err := os.MkdirAll(e.cmd.Dir, 0o700)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Get output and redirect Stderr to Stdout
|
|
e.output, _ = e.cmd.StdoutPipe()
|
|
e.cmd.Stderr = e.cmd.Stdout
|
|
|
|
return e.cmd.Start()
|
|
}
|
|
|
|
// Wait for the pipeline step to complete and returns
|
|
// the completion results.
|
|
func (e *local) Wait(context.Context, *types.Step) (*types.State, error) {
|
|
err := e.cmd.Wait()
|
|
ExitCode := 0
|
|
|
|
var execExitError *exec.ExitError
|
|
if errors.As(err, &execExitError) {
|
|
ExitCode = execExitError.ExitCode()
|
|
// Non-zero exit code is a pipeline failure, but not an agent error.
|
|
err = nil
|
|
}
|
|
|
|
return &types.State{
|
|
Exited: true,
|
|
ExitCode: ExitCode,
|
|
}, err
|
|
}
|
|
|
|
// Tail the pipeline step logs.
|
|
func (e *local) Tail(context.Context, *types.Step) (io.ReadCloser, error) {
|
|
return e.output, nil
|
|
}
|
|
|
|
// Destroy the pipeline environment.
|
|
func (e *local) Destroy(context.Context, *types.Config) error {
|
|
return os.RemoveAll(e.workflowBaseDir)
|
|
}
|