2019-01-13 04:45:25 +00:00
package cmd
import (
2020-02-25 06:11:33 +00:00
"bufio"
2019-01-13 04:45:25 +00:00
"context"
"os"
2019-02-10 02:39:09 +00:00
"path/filepath"
2020-03-02 16:11:46 +00:00
"regexp"
2020-02-25 06:11:33 +00:00
"strings"
2019-01-13 04:45:25 +00:00
2020-02-07 06:17:58 +00:00
"github.com/nektos/act/pkg/common"
2021-01-19 14:30:17 +00:00
"github.com/AlecAivazis/survey/v2"
2021-01-12 06:39:43 +00:00
"github.com/andreaskoch/go-fswatch"
2020-03-06 20:30:24 +00:00
"github.com/joho/godotenv"
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"
2021-01-19 14:30:17 +00:00
"github.com/nektos/act/pkg/model"
"github.com/nektos/act/pkg/runner"
2019-01-13 04:45:25 +00:00
)
// 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 {
2021-01-18 19:42:55 +00:00
Use : "act [event name to run]\nIf no event name passed, will default to \"on: push\"" ,
2019-01-17 08:15:35 +00:00
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" )
2020-10-12 17:26:22 +00:00
rootCmd . Flags ( ) . BoolP ( "graph" , "g" , false , "draw workflows" )
2020-02-05 00:38:41 +00:00
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)" )
2021-01-15 05:26:01 +00:00
rootCmd . Flags ( ) . StringArrayVarP ( & input . envs , "env" , "" , [ ] string { } , "env to make available to actions with optional value (e.g. --e myenv=foo or -s myenv)" )
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" )
2020-02-25 01:48:21 +00:00
rootCmd . Flags ( ) . BoolVarP ( & input . bindWorkdir , "bind" , "b" , false , "bind working directory to container, rather than copy" )
2020-02-05 00:38:41 +00:00
rootCmd . Flags ( ) . BoolVarP ( & input . forcePull , "pull" , "p" , false , "pull docker image(s) if already present" )
2021-01-18 19:42:55 +00:00
rootCmd . Flags ( ) . BoolVarP ( & input . autodetectEvent , "detect-event" , "" , false , "Use first event type from workflow as event that triggered the workflow" )
2020-02-18 05:51:49 +00:00
rootCmd . Flags ( ) . StringVarP ( & input . eventPath , "eventpath" , "e" , "" , "path to event JSON file" )
2020-09-02 14:56:44 +00:00
rootCmd . Flags ( ) . StringVar ( & input . defaultBranch , "defaultbranch" , "" , "the name of the main branch" )
2020-08-01 20:21:49 +00:00
rootCmd . Flags ( ) . BoolVar ( & input . privileged , "privileged" , false , "use privileged mode" )
2020-05-12 07:14:56 +00:00
rootCmd . PersistentFlags ( ) . StringVarP ( & input . actor , "actor" , "a" , "nektos/act" , "user that triggered the event" )
2020-05-27 03:29:50 +00:00
rootCmd . PersistentFlags ( ) . StringVarP ( & input . workflowsPath , "workflows" , "W" , "./.github/workflows/" , "path to workflow file(s)" )
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-20 16:57:18 +00:00
rootCmd . PersistentFlags ( ) . BoolVarP ( & input . noOutput , "quiet" , "q" , false , "disable logging of output from steps" )
2020-02-05 00:38:41 +00:00
rootCmd . PersistentFlags ( ) . BoolVarP ( & input . dryrun , "dryrun" , "n" , false , "dryrun mode" )
2021-01-19 14:31:46 +00:00
rootCmd . PersistentFlags ( ) . StringVarP ( & input . secretfile , "secret-file" , "" , ".secrets" , "file with list of secrets to read from (e.g. --secret-file .secrets)" )
2021-01-12 06:28:45 +00:00
rootCmd . PersistentFlags ( ) . BoolVarP ( & input . insecureSecrets , "insecure-secrets" , "" , false , "NOT RECOMMENDED! Doesn't hide secrets while printing logs." )
2020-04-17 17:04:40 +00:00
rootCmd . PersistentFlags ( ) . StringVarP ( & input . envfile , "env-file" , "" , ".env" , "environment file to read and use as env in the containers" )
2020-02-25 06:11:33 +00:00
rootCmd . SetArgs ( args ( ) )
2020-03-06 20:30:24 +00:00
2019-01-13 04:45:25 +00:00
if err := rootCmd . Execute ( ) ; err != nil {
os . Exit ( 1 )
}
}
2020-02-25 06:11:33 +00:00
func args ( ) [ ] string {
args := make ( [ ] string , 0 )
2021-01-19 14:30:17 +00:00
actrc := [ ] string {
filepath . Join ( os . Getenv ( "HOME" ) , ".actrc" ) ,
filepath . Join ( "." , ".actrc" ) ,
}
for _ , f := range actrc {
args = append ( args , readArgsFile ( f ) ... )
2020-02-25 06:11:33 +00:00
}
2021-01-19 14:30:17 +00:00
if len ( args ) == 0 {
var answer string
confirmation := & survey . Select {
Message : "Please choose the default image you want to use with act:\n\n - Large size image: +20GB Docker image, includes almost all tools used on GitHub Actions\n - Medium size image: ~500MB, includes only necessary tools to bootstrap actions and aims to be compatible with all actions\n - Micro size image: <200MB, contains only NodeJS required to bootstrap actions, doesn't work with all actions\n\nDefault image and other options can be changed manually in ~/.actrc" ,
Help : "If you want to know why act asks you that, please go to https://github.com/nektos/act/issues/107" ,
Default : "Medium" ,
Options : [ ] string { "Large" , "Medium" , "Micro" } ,
}
err := survey . AskOne ( confirmation , & answer )
if err != nil {
log . Error ( err )
os . Exit ( 1 )
}
_ , err = os . Stat ( actrc [ 0 ] )
if os . IsNotExist ( err ) {
var option string
switch answer {
case "Large" :
option = "-P ubuntu-latest=nektos/act-environments-ubuntu:18.04\n-P ubuntu-18.04=nektos/act-environments-ubuntu:18.04"
case "Medium" :
option = "-P ubuntu-latest=catthehacker/ubuntu:act-latest\n-P ubuntu-20.04=catthehacker/ubuntu:act-20.04\n-P ubuntu-18.04=catthehacker/ubuntu:act-18.04\nubuntu-16.04=catthehacker/ubuntu:act-16.04"
case "Micro" :
option = "-P ubuntu-latest=node:12.6-buster-slim\n-P ubuntu-12.04=node:12.6-buster-slim\n-P ubuntu-18.04=node:12.6-buster-slim\n-P ubuntu-16.04=node:12.6-stretch-slim"
}
f , err := os . Create ( actrc [ 0 ] )
if err != nil {
log . Fatal ( err )
}
_ , err = f . WriteString ( option )
if err != nil {
log . Fatal ( err )
_ = f . Close ( )
}
err = f . Close ( )
if err != nil {
log . Fatal ( err )
}
args = append ( args , readArgsFile ( actrc [ 0 ] ) ... )
} else {
log . Error ( "File ~/.actrc already exists" )
}
}
2020-02-25 06:11:33 +00:00
args = append ( args , os . Args [ 1 : ] ... )
2021-01-19 14:30:17 +00:00
2020-02-25 06:11:33 +00:00
return args
}
func readArgsFile ( file string ) [ ] string {
args := make ( [ ] string , 0 )
f , err := os . Open ( file )
if err != nil {
return args
}
2021-01-12 06:39:43 +00:00
defer func ( ) {
err := f . Close ( )
if err != nil {
log . Errorf ( "Failed to close args file: %v" , err )
}
} ( )
2020-02-25 06:11:33 +00:00
scanner := bufio . NewScanner ( f )
for scanner . Scan ( ) {
arg := scanner . Text ( )
if strings . HasPrefix ( arg , "-" ) {
2020-03-02 16:11:46 +00:00
args = append ( args , regexp . MustCompile ( ` \s ` ) . Split ( arg , 2 ) ... )
2020-02-25 06:11:33 +00:00
}
}
return args
}
2021-01-12 06:39:43 +00:00
func setupLogging ( cmd * cobra . Command , _ [ ] string ) {
2019-01-17 08:15:35 +00:00
verbose , _ := cmd . Flags ( ) . GetBool ( "verbose" )
if verbose {
log . SetLevel ( log . DebugLevel )
}
}
2020-04-17 17:04:40 +00:00
func readEnvs ( path string , envs map [ string ] string ) bool {
if _ , err := os . Stat ( path ) ; err == nil {
env , err := godotenv . Read ( path )
if err != nil {
log . Fatalf ( "Error loading from %s: %v" , path , err )
}
for k , v := range env {
envs [ k ] = v
}
return true
}
return false
}
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-04-17 17:04:40 +00:00
log . Debugf ( "Loading environment from %s" , input . Envfile ( ) )
envs := make ( map [ string ] string )
2021-01-15 05:26:01 +00:00
if input . envs != nil {
for _ , envVar := range input . envs {
e := strings . SplitN ( envVar , ` = ` , 2 )
if len ( e ) == 2 {
envs [ e [ 0 ] ] = e [ 1 ]
} else {
envs [ e [ 0 ] ] = ""
}
}
}
2020-04-17 17:04:40 +00:00
_ = readEnvs ( input . Envfile ( ) , envs )
log . Debugf ( "Loading secrets from %s" , input . Secretfile ( ) )
secrets := newSecrets ( input . secrets )
_ = readEnvs ( input . Secretfile ( ) , secrets )
2020-03-06 20:30:24 +00:00
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
2021-01-18 19:42:55 +00:00
events := planner . GetEvents ( )
if input . autodetectEvent && len ( events ) > 0 {
// 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 ]
2021-01-18 19:42:55 +00:00
} else {
if len ( args ) > 0 {
eventName = args [ 0 ]
} else if plan := planner . PlanEvent ( "push" ) ; plan != nil {
eventName = "push"
}
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-10-12 17:26:22 +00:00
// check if we should just list the workflows
2020-02-05 00:38:41 +00:00
if list , err := cmd . Flags ( ) . GetBool ( "list" ) ; err != nil {
return err
2020-10-12 17:26:22 +00:00
} else if list {
return printList ( plan )
}
// check if we should just print the graph
if list , err := cmd . Flags ( ) . GetBool ( "graph" ) ; err != nil {
return err
2020-02-05 00:38:41 +00:00
} else if list {
return drawGraph ( plan )
}
2019-02-15 16:34:19 +00:00
2020-09-02 14:56:44 +00:00
// check to see if the main branch was defined
defaultbranch , err := cmd . Flags ( ) . GetString ( "defaultbranch" )
if err != nil {
return err
}
2020-02-05 00:38:41 +00:00
// run the plan
2020-02-07 06:17:58 +00:00
config := & runner . Config {
2020-05-12 07:14:56 +00:00
Actor : input . actor ,
2020-02-07 06:17:58 +00:00
EventName : eventName ,
EventPath : input . EventPath ( ) ,
2020-09-02 14:56:44 +00:00
DefaultBranch : defaultbranch ,
2020-02-07 06:17:58 +00:00
ForcePull : input . forcePull ,
ReuseContainers : input . reuseContainers ,
Workdir : input . Workdir ( ) ,
2020-02-25 01:48:21 +00:00
BindWorkdir : input . bindWorkdir ,
2020-02-20 16:57:18 +00:00
LogOutput : ! input . noOutput ,
2020-04-17 17:04:40 +00:00
Env : envs ,
Secrets : secrets ,
2021-01-12 06:28:45 +00:00
InsecureSecrets : input . insecureSecrets ,
2020-02-20 03:16:40 +00:00
Platforms : input . newPlatforms ( ) ,
2020-08-01 20:21:49 +00:00
Privileged : input . privileged ,
2020-02-07 06:17:58 +00:00
}
2021-01-12 06:39:43 +00:00
r , err := runner . New ( config )
2020-02-07 06:17:58 +00:00
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 {
2021-01-12 06:39:43 +00:00
return watchAndRun ( ctx , r . NewPlanExecutor ( plan ) )
2020-02-07 06:17:58 +00:00
}
2021-01-12 06:39:43 +00:00
return r . 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
}