77 lines
1.8 KiB
Go
77 lines
1.8 KiB
Go
|
package actions
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"io/ioutil"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
|
||
|
"github.com/hashicorp/hcl"
|
||
|
"github.com/hashicorp/hcl/hcl/ast"
|
||
|
log "github.com/sirupsen/logrus"
|
||
|
)
|
||
|
|
||
|
// ParseWorkflows will read in the set of actions from the workflow file
|
||
|
func ParseWorkflows(workingDir string, workflowPath string) (Workflows, error) {
|
||
|
workingDir, err := filepath.Abs(workingDir)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
log.Debugf("Setting working dir to %s", workingDir)
|
||
|
|
||
|
if !filepath.IsAbs(workflowPath) {
|
||
|
workflowPath = filepath.Join(workingDir, workflowPath)
|
||
|
}
|
||
|
log.Debugf("Loading workflow config from %s", workflowPath)
|
||
|
workflowReader, err := os.Open(workflowPath)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
buf := new(bytes.Buffer)
|
||
|
buf.ReadFrom(workflowReader)
|
||
|
|
||
|
workflows := new(workflowsFile)
|
||
|
workflows.WorkingDir = workingDir
|
||
|
workflows.WorkflowPath = workflowPath
|
||
|
|
||
|
astFile, err := hcl.ParseBytes(buf.Bytes())
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
rootNode := ast.Walk(astFile.Node, cleanWorkflowsAST)
|
||
|
err = hcl.DecodeObject(workflows, rootNode)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
workflows.TempDir, err = ioutil.TempDir("/tmp", "act-")
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
// TODO: add validation logic
|
||
|
// - check for circular dependencies
|
||
|
// - check for valid local path refs
|
||
|
// - check for valid dependencies
|
||
|
|
||
|
return workflows, nil
|
||
|
}
|
||
|
|
||
|
func cleanWorkflowsAST(node ast.Node) (ast.Node, bool) {
|
||
|
if objectItem, ok := node.(*ast.ObjectItem); ok {
|
||
|
key := objectItem.Keys[0].Token.Value()
|
||
|
|
||
|
// handle condition where value is a string but should be a list
|
||
|
switch key {
|
||
|
case "resolves", "needs", "args":
|
||
|
if literalType, ok := objectItem.Val.(*ast.LiteralType); ok {
|
||
|
listType := new(ast.ListType)
|
||
|
listType.Add(literalType)
|
||
|
objectItem.Val = listType
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return node, true
|
||
|
}
|