mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2025-02-02 04:32:23 +00:00
Add multi-pipeline support to exec & lint (#568)
close #443 * add support to exec * auto detect multi-pipeline for lint and exec * de-duplicate code
This commit is contained in:
parent
442561fea2
commit
ad509fd86f
3 changed files with 111 additions and 33 deletions
65
cli/common/pipeline.go
Normal file
65
cli/common/pipeline.go
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
package common
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/urfave/cli/v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
func DetectPipelineConfig() (multiplies bool, config string, _ error) {
|
||||||
|
config = ".woodpecker"
|
||||||
|
if fi, err := os.Stat(config); err == nil && fi.IsDir() {
|
||||||
|
return true, config, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
config = ".woodpecker.yml"
|
||||||
|
if fi, err := os.Stat(config); err == nil && !fi.IsDir() {
|
||||||
|
return true, config, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
config = ".drone.yml"
|
||||||
|
fi, err := os.Stat(config)
|
||||||
|
if err == nil && !fi.IsDir() {
|
||||||
|
return false, config, nil
|
||||||
|
}
|
||||||
|
return false, "", fmt.Errorf("could not detect pipeline config")
|
||||||
|
}
|
||||||
|
|
||||||
|
func RunPipelineFunc(c *cli.Context, fileFunc, dirFunc func(*cli.Context, string) error) error {
|
||||||
|
if c.Args().Len() == 0 {
|
||||||
|
isDir, path, err := DetectPipelineConfig()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if isDir {
|
||||||
|
return dirFunc(c, path)
|
||||||
|
}
|
||||||
|
return fileFunc(c, path)
|
||||||
|
}
|
||||||
|
|
||||||
|
multiArgs := c.Args().Len() > 1
|
||||||
|
for _, arg := range c.Args().Slice() {
|
||||||
|
fi, err := os.Stat(arg)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if multiArgs {
|
||||||
|
fmt.Println("#", fi.Name())
|
||||||
|
}
|
||||||
|
if fi.IsDir() {
|
||||||
|
if err := dirFunc(c, arg); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := fileFunc(c, arg); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if multiArgs {
|
||||||
|
fmt.Println("")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -5,6 +5,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
|
@ -31,16 +32,46 @@ var Command = &cli.Command{
|
||||||
Name: "exec",
|
Name: "exec",
|
||||||
Usage: "execute a local build",
|
Usage: "execute a local build",
|
||||||
ArgsUsage: "[path/to/.woodpecker.yml]",
|
ArgsUsage: "[path/to/.woodpecker.yml]",
|
||||||
Action: exec,
|
Action: run,
|
||||||
Flags: append(common.GlobalFlags, flags...),
|
Flags: append(common.GlobalFlags, flags...),
|
||||||
}
|
}
|
||||||
|
|
||||||
func exec(c *cli.Context) error {
|
func run(c *cli.Context) error {
|
||||||
file := c.Args().First()
|
return common.RunPipelineFunc(c, execFile, execDir)
|
||||||
if file == "" {
|
}
|
||||||
file = ".woodpecker.yml"
|
|
||||||
}
|
|
||||||
|
|
||||||
|
func execDir(c *cli.Context, dir string) error {
|
||||||
|
// TODO: respect pipeline dependency
|
||||||
|
repoPath, _ := filepath.Abs(filepath.Dir(dir))
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
repoPath = convertPathForWindows(repoPath)
|
||||||
|
}
|
||||||
|
return filepath.Walk(dir, func(path string, info os.FileInfo, e error) error {
|
||||||
|
if e != nil {
|
||||||
|
return e
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if it is a regular file (not dir)
|
||||||
|
if info.Mode().IsRegular() && strings.HasSuffix(info.Name(), ".yml") {
|
||||||
|
fmt.Println("#", info.Name())
|
||||||
|
_ = runExec(c, path, repoPath) // TODO: should we drop errors or store them and report back?
|
||||||
|
fmt.Println("")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func execFile(c *cli.Context, file string) error {
|
||||||
|
repoPath, _ := filepath.Abs(filepath.Dir(file))
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
repoPath = convertPathForWindows(repoPath)
|
||||||
|
}
|
||||||
|
return runExec(c, file, repoPath)
|
||||||
|
}
|
||||||
|
|
||||||
|
func runExec(c *cli.Context, file, repoPath string) error {
|
||||||
dat, err := ioutil.ReadFile(file)
|
dat, err := ioutil.ReadFile(file)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -55,7 +86,7 @@ func exec(c *cli.Context) error {
|
||||||
axes = append(axes, matrix.Axis{})
|
axes = append(axes, matrix.Axis{})
|
||||||
}
|
}
|
||||||
for _, axis := range axes {
|
for _, axis := range axes {
|
||||||
err := execWithAxis(c, axis)
|
err := execWithAxis(c, file, repoPath, axis)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -63,12 +94,7 @@ func exec(c *cli.Context) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func execWithAxis(c *cli.Context, axis matrix.Axis) error {
|
func execWithAxis(c *cli.Context, file, repoPath string, axis matrix.Axis) error {
|
||||||
file := c.Args().First()
|
|
||||||
if file == "" {
|
|
||||||
file = ".woodpecker.yml"
|
|
||||||
}
|
|
||||||
|
|
||||||
metadata := metadataFromContext(c, axis)
|
metadata := metadataFromContext(c, axis)
|
||||||
environ := metadata.Environ()
|
environ := metadata.Environ()
|
||||||
var secrets []compiler.Secret
|
var secrets []compiler.Secret
|
||||||
|
@ -115,13 +141,9 @@ func execWithAxis(c *cli.Context, axis matrix.Axis) error {
|
||||||
if workspacePath == "" {
|
if workspacePath == "" {
|
||||||
workspacePath = c.String("workspace-path")
|
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, c.String("prefix")+"_default:"+workspaceBase)
|
||||||
volumes = append(volumes, dir+":"+path.Join(workspaceBase, workspacePath))
|
volumes = append(volumes, repoPath+":"+path.Join(workspaceBase, workspacePath))
|
||||||
}
|
}
|
||||||
|
|
||||||
// lint the yaml file
|
// lint the yaml file
|
||||||
|
|
|
@ -22,20 +22,11 @@ var Command = &cli.Command{
|
||||||
}
|
}
|
||||||
|
|
||||||
func lint(c *cli.Context) error {
|
func lint(c *cli.Context) error {
|
||||||
file := c.Args().First()
|
return common.RunPipelineFunc(c, lintFile, lintDir)
|
||||||
if file == "" {
|
}
|
||||||
file = ".woodpecker.yml"
|
|
||||||
}
|
|
||||||
|
|
||||||
fi, err := os.Stat(file)
|
func lintDir(c *cli.Context, dir string) error {
|
||||||
if err != nil {
|
return filepath.Walk(dir, func(path string, info os.FileInfo, e error) error {
|
||||||
return err
|
|
||||||
}
|
|
||||||
if !fi.IsDir() {
|
|
||||||
return lintFile(file)
|
|
||||||
}
|
|
||||||
|
|
||||||
return filepath.Walk(file, func(path string, info os.FileInfo, e error) error {
|
|
||||||
if e != nil {
|
if e != nil {
|
||||||
return e
|
return e
|
||||||
}
|
}
|
||||||
|
@ -43,7 +34,7 @@ func lint(c *cli.Context) error {
|
||||||
// check if it is a regular file (not dir)
|
// check if it is a regular file (not dir)
|
||||||
if info.Mode().IsRegular() && strings.HasSuffix(info.Name(), ".yml") {
|
if info.Mode().IsRegular() && strings.HasSuffix(info.Name(), ".yml") {
|
||||||
fmt.Println("#", info.Name())
|
fmt.Println("#", info.Name())
|
||||||
_ = lintFile(path) // TODO: should we drop errors or store them and report back?
|
_ = lintFile(c, path) // TODO: should we drop errors or store them and report back?
|
||||||
fmt.Println("")
|
fmt.Println("")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -52,7 +43,7 @@ func lint(c *cli.Context) error {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func lintFile(file string) error {
|
func lintFile(_ *cli.Context, file string) error {
|
||||||
configErrors, err := schema.Lint(file)
|
configErrors, err := schema.Lint(file)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println("❌ Config is invalid")
|
fmt.Println("❌ Config is invalid")
|
||||||
|
|
Loading…
Reference in a new issue