2019-01-13 04:45:25 +00:00
package cmd
import (
"context"
"os"
2019-02-10 02:39:09 +00:00
"path/filepath"
2019-01-13 04:45:25 +00:00
2020-02-07 06:17:58 +00:00
"github.com/nektos/act/pkg/common"
2019-02-10 02:39:09 +00:00
fswatch "github.com/andreaskoch/go-fswatch"
2020-02-05 00:38:41 +00:00
"github.com/nektos/act/pkg/model"
2020-02-07 06:17:58 +00:00
"github.com/nektos/act/pkg/runner"
2019-02-10 02:39:09 +00:00
gitignore "github.com/sabhiram/go-gitignore"
2019-01-13 04:45:25 +00:00
log "github.com/sirupsen/logrus"
"github.com/spf13/cobra"
)
// Execute is the entry point to running the CLI
func Execute ( ctx context . Context , version string ) {
2020-02-05 00:38:41 +00:00
input := new ( Input )
2019-01-13 04:45:25 +00:00
var rootCmd = & cobra . Command {
2019-01-17 08:15:35 +00:00
Use : "act [event name to run]" ,
Short : "Run Github actions locally by specifying the event name (e.g. `push`) or an action name directly." ,
Args : cobra . MaximumNArgs ( 1 ) ,
2020-02-05 00:38:41 +00:00
RunE : newRunCommand ( ctx , input ) ,
2019-01-17 08:15:35 +00:00
PersistentPreRun : setupLogging ,
Version : version ,
SilenceUsage : true ,
2019-01-13 04:45:25 +00:00
}
2019-02-10 02:39:09 +00:00
rootCmd . Flags ( ) . BoolP ( "watch" , "w" , false , "watch the contents of the local repo and run when files change" )
2020-02-05 00:38:41 +00:00
rootCmd . Flags ( ) . BoolP ( "list" , "l" , false , "list workflows" )
rootCmd . Flags ( ) . StringP ( "job" , "j" , "" , "run job" )
2020-02-18 05:51:49 +00:00
rootCmd . Flags ( ) . StringArrayVarP ( & input . secrets , "secret" , "s" , [ ] string { } , "secret to make available to actions with optional value (e.g. -s mysecret=foo or -s mysecret)" )
2020-02-20 03:16:40 +00:00
rootCmd . Flags ( ) . StringArrayVarP ( & input . platforms , "platform" , "P" , [ ] string { } , "custom image to use per platform (e.g. -P ubuntu-18.04=nektos/act-environments-ubuntu:18.04)" )
2020-02-05 00:38:41 +00:00
rootCmd . Flags ( ) . BoolVarP ( & input . reuseContainers , "reuse" , "r" , false , "reuse action containers to maintain state" )
rootCmd . Flags ( ) . BoolVarP ( & input . forcePull , "pull" , "p" , false , "pull docker image(s) if already present" )
2020-02-18 05:51:49 +00:00
rootCmd . Flags ( ) . StringVarP ( & input . eventPath , "eventpath" , "e" , "" , "path to event JSON file" )
2020-02-05 00:38:41 +00:00
rootCmd . PersistentFlags ( ) . StringVarP ( & input . workflowsPath , "workflows" , "W" , "./.github/workflows/" , "path to workflow files" )
2020-02-07 06:17:58 +00:00
rootCmd . PersistentFlags ( ) . StringVarP ( & input . workdir , "directory" , "C" , "." , "working directory" )
2019-01-17 08:15:35 +00:00
rootCmd . PersistentFlags ( ) . BoolP ( "verbose" , "v" , false , "verbose output" )
2020-02-12 07:55:20 +00:00
rootCmd . PersistentFlags ( ) . BoolVarP ( & input . logOutput , "output" , "o" , false , "log output from steps" )
2020-02-05 00:38:41 +00:00
rootCmd . PersistentFlags ( ) . BoolVarP ( & input . dryrun , "dryrun" , "n" , false , "dryrun mode" )
2019-01-13 04:45:25 +00:00
if err := rootCmd . Execute ( ) ; err != nil {
os . Exit ( 1 )
}
}
2019-01-17 08:15:35 +00:00
func setupLogging ( cmd * cobra . Command , args [ ] string ) {
verbose , _ := cmd . Flags ( ) . GetBool ( "verbose" )
if verbose {
log . SetLevel ( log . DebugLevel )
}
}
2020-02-05 00:38:41 +00:00
func newRunCommand ( ctx context . Context , input * Input ) func ( * cobra . Command , [ ] string ) error {
2019-01-13 04:45:25 +00:00
return func ( cmd * cobra . Command , args [ ] string ) error {
2020-02-05 00:38:41 +00:00
planner , err := model . NewWorkflowPlanner ( input . WorkflowsPath ( ) )
2019-01-17 08:15:35 +00:00
if err != nil {
return err
}
2020-02-05 00:38:41 +00:00
// Determine the event name
var eventName string
if len ( args ) > 0 {
eventName = args [ 0 ]
2020-02-17 18:11:16 +00:00
} else if events := planner . GetEvents ( ) ; len ( events ) > 0 {
2020-02-17 06:04:13 +00:00
// set default event type to first event
2020-02-05 00:38:41 +00:00
// this way user dont have to specify the event.
2020-02-11 00:53:14 +00:00
log . Debugf ( "Using detected workflow event: %s" , events [ 0 ] )
eventName = events [ 0 ]
2019-01-13 04:45:25 +00:00
}
2019-02-10 02:39:09 +00:00
2020-02-05 00:38:41 +00:00
// build the plan for this run
var plan * model . Plan
if jobID , err := cmd . Flags ( ) . GetString ( "job" ) ; err != nil {
return err
} else if jobID != "" {
log . Debugf ( "Planning job: %s" , jobID )
plan = planner . PlanJob ( jobID )
} else {
log . Debugf ( "Planning event: %s" , eventName )
plan = planner . PlanEvent ( eventName )
2019-02-15 16:34:19 +00:00
}
2020-02-05 00:38:41 +00:00
// check if we should just print the graph
if list , err := cmd . Flags ( ) . GetBool ( "list" ) ; err != nil {
return err
} else if list {
return drawGraph ( plan )
}
2019-02-15 16:34:19 +00:00
2020-02-05 00:38:41 +00:00
// run the plan
2020-02-07 06:17:58 +00:00
config := & runner . Config {
EventName : eventName ,
EventPath : input . EventPath ( ) ,
ForcePull : input . forcePull ,
ReuseContainers : input . reuseContainers ,
Workdir : input . Workdir ( ) ,
2020-02-12 07:55:20 +00:00
LogOutput : input . logOutput ,
2020-02-18 05:51:49 +00:00
Secrets : newSecrets ( input . secrets ) ,
2020-02-20 03:16:40 +00:00
Platforms : input . newPlatforms ( ) ,
2020-02-07 06:17:58 +00:00
}
runner , err := runner . New ( config )
if err != nil {
return err
}
ctx = common . WithDryrun ( ctx , input . dryrun )
if watch , err := cmd . Flags ( ) . GetBool ( "watch" ) ; err != nil {
return err
} else if watch {
return watchAndRun ( ctx , runner . NewPlanExecutor ( plan ) )
}
return runner . NewPlanExecutor ( plan ) ( ctx )
2019-02-10 02:39:09 +00:00
}
}
2020-02-07 06:17:58 +00:00
func watchAndRun ( ctx context . Context , fn common . Executor ) error {
2019-02-10 02:39:09 +00:00
recurse := true
checkIntervalInSeconds := 2
dir , err := os . Getwd ( )
if err != nil {
return err
}
var ignore * gitignore . GitIgnore
if _ , err := os . Stat ( filepath . Join ( dir , ".gitignore" ) ) ; ! os . IsNotExist ( err ) {
ignore , _ = gitignore . CompileIgnoreFile ( filepath . Join ( dir , ".gitignore" ) )
} else {
ignore = & gitignore . GitIgnore { }
}
folderWatcher := fswatch . NewFolderWatcher (
dir ,
recurse ,
ignore . MatchesPath ,
checkIntervalInSeconds ,
)
folderWatcher . Start ( )
go func ( ) {
for folderWatcher . IsRunning ( ) {
2020-02-07 06:17:58 +00:00
if err = fn ( ctx ) ; err != nil {
2019-02-13 04:32:54 +00:00
break
}
log . Debugf ( "Watching %s for changes" , dir )
2019-02-10 02:39:09 +00:00
for changes := range folderWatcher . ChangeDetails ( ) {
log . Debugf ( "%s" , changes . String ( ) )
2020-02-07 06:17:58 +00:00
if err = fn ( ctx ) ; err != nil {
2019-02-13 04:32:54 +00:00
break
}
2019-02-10 02:39:09 +00:00
log . Debugf ( "Watching %s for changes" , dir )
}
}
} ( )
<- ctx . Done ( )
folderWatcher . Stop ( )
2019-02-13 04:32:54 +00:00
return err
2019-01-13 04:45:25 +00:00
}