From a844d5e28dad3fbd65be215c01ecb76a9b91d9da Mon Sep 17 00:00:00 2001 From: Paul Annesley Date: Mon, 14 Nov 2022 22:36:34 +1030 Subject: [PATCH 1/4] Embedded JavaScript pipeline PoC --- clicommand/pipeline_eval.go | 217 ++++++++++++++++++++++ go.mod | 4 + go.sum | 22 +++ main.go | 1 + resources/node_modules/buildkite/hello.js | 1 + resources/resources.go | 8 + test/fixtures/pipelines/buildkite.js | 26 +++ 7 files changed, 279 insertions(+) create mode 100644 clicommand/pipeline_eval.go create mode 100644 resources/node_modules/buildkite/hello.js create mode 100644 resources/resources.go create mode 100644 test/fixtures/pipelines/buildkite.js diff --git a/clicommand/pipeline_eval.go b/clicommand/pipeline_eval.go new file mode 100644 index 0000000000..f73e10489e --- /dev/null +++ b/clicommand/pipeline_eval.go @@ -0,0 +1,217 @@ +package clicommand + +import ( + "errors" + "fmt" + "io" + "io/fs" + "os" + "path" + "path/filepath" + "strings" + + "github.com/buildkite/agent/v3/cliconfig" + "github.com/buildkite/agent/v3/resources" + "github.com/buildkite/agent/v3/stdin" + "github.com/buildkite/yaml" + "github.com/dop251/goja" + "github.com/dop251/goja_nodejs/console" + "github.com/dop251/goja_nodejs/process" + "github.com/dop251/goja_nodejs/require" + "github.com/urfave/cli" +) + +const evalDescription = `Usage: + buildkite-agent pipeline eval [options] + +Description: + Something something JavaScript? + +Example: + $ buildkite-agent pipeline eval buildkite.js + + Evaluates buildkite.js as JavaScript and perhaps uploads the stdout as JSON/YAML pipeline? +` + +type PipelineEvalConfig struct { + FilePath string `cli:"arg:0" label:"upload paths"` + + // Global flags + Debug bool `cli:"debug"` + LogLevel string `cli:"log-level"` + NoColor bool `cli:"no-color"` + Experiments []string `cli:"experiment" normalize:"list"` + Profile string `cli:"profile"` +} + +var PipelineEvalCommand = cli.Command{ + Name: "eval", + Usage: "Evaluates a JavaScript pipeline", + Description: evalDescription, + Flags: []cli.Flag{ + // Global flags + NoColorFlag, + DebugFlag, + LogLevelFlag, + ExperimentsFlag, + ProfileFlag, + }, + Action: func(c *cli.Context) error { + // The configuration will be loaded into this struct + cfg := PipelineEvalConfig{} + + loader := cliconfig.Loader{CLI: c, Config: &cfg} + warnings, err := loader.Load() + if err != nil { + fmt.Printf("%s", err) + os.Exit(1) + } + + l := CreateLogger(&cfg) + + // Now that we have a logger, log out the warnings that loading config generated + for _, warning := range warnings { + l.Warn("%s", warning) + } + + // Setup any global configuration options + done := HandleGlobalFlags(l, cfg) + defer done() + + // Find the pipeline file either from STDIN or the first + // argument + var input []byte + var filename string + + if cfg.FilePath != "" { + l.Info("Reading pipeline config from \"%s\"", cfg.FilePath) + + filename = filepath.Base(cfg.FilePath) + input, err = os.ReadFile(cfg.FilePath) + if err != nil { + l.Fatal("Failed to read file: %s", err) + } + } else if stdin.IsReadable() { + l.Info("Reading pipeline config from STDIN") + + // Actually read the file from STDIN + input, err = io.ReadAll(os.Stdin) + if err != nil { + l.Fatal("Failed to read from STDIN: %s", err) + } + } else { + l.Info("Searching for pipeline config...") + + paths := []string{ + "buildkite.js", + filepath.FromSlash(".buildkite/buildkite.js"), + filepath.FromSlash("buildkite/buildkite.js"), + } + + // Collect all the files that exist + exists := []string{} + for _, path := range paths { + if _, err := os.Stat(path); err == nil { + exists = append(exists, path) + } + } + + // If more than 1 of the config files exist, throw an + // error. There can only be one!! + if len(exists) > 1 { + l.Fatal("Found multiple configuration files: %s. Please only have 1 configuration file present.", strings.Join(exists, ", ")) + } else if len(exists) == 0 { + l.Fatal("Could not find a default pipeline configuration file. See `buildkite-agent pipeline upload --help` for more information.") + } + + found := exists[0] + + l.Info("Found config file \"%s\"", found) + + // Read the default file + filename = path.Base(found) + input, err = os.ReadFile(found) + if err != nil { + l.Fatal("Failed to read file \"%s\" (%s)", found, err) + } + } + + if err := evalJS(filename, input, c.App.Writer); err != nil { + panic(err) + } + return nil + }, +} + +func evalJS(filename string, input []byte, output io.Writer) error { + runtime := goja.New() + runtime.SetFieldNameMapper(goja.TagFieldNameMapper("json", true)) + + // Add support for require() CommonJS modules. + // require("buildkite/*") is handled by embedded resources/node_modules/buildkite/* filesystem. + // Other paths are loaded from the host filesystem. + registry := require.NewRegistry( + require.WithLoader(func(name string) ([]byte, error) { + if !strings.HasPrefix(name, "node_modules/buildkite/") { + return require.DefaultSourceLoader(name) + } + res := resources.FS + data, err := res.ReadFile(name) + if errors.Is(err, fs.ErrNotExist) { + return nil, require.ModuleFileDoesNotExistError + } else if err != nil { + return nil, err + } + return data, nil + }), + ) + registry.Enable(runtime) + + // Add basic utilities + console.Enable(runtime) // console.log() + process.Enable(runtime) // process.env() + + // provide plugin() as a native module (implemented in Go) + registry.RegisterNativeModule("buildkite/plugin", func(runtime *goja.Runtime, module *goja.Object) { + module.Set("exports", func(call goja.FunctionCall) goja.Value { + name := call.Argument(0) + ref := call.Argument(1) + config := call.Argument(2) + plugin := runtime.NewObject() + plugin.Set(name.String()+"#"+ref.String(), config) + return plugin + }) + }) + + // provide assignable module.exports for Pipeline result + rootModule := runtime.NewObject() + rootModule.Set("exports", runtime.NewObject()) + err := runtime.Set("module", rootModule) + if err != nil { + return err + } + + if filename == "" { + filename = "(stdin)" + } + + v, err := runtime.RunScript(filename, string(input)) + if err != nil { + panic(err) + } + + y, err := yaml.Marshal(v.Export()) + if err != nil { + return err + } + + n, err := output.Write(y) + if err != nil { + return nil + } + if n != len(y) { + return errors.New("short write") + } + + return nil +} diff --git a/go.mod b/go.mod index 437cf40aab..901cf0e87a 100644 --- a/go.mod +++ b/go.mod @@ -58,9 +58,13 @@ require ( github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/dgraph-io/ristretto v0.1.0 // indirect + github.com/dlclark/regexp2 v1.7.0 // indirect + github.com/dop251/goja v0.0.0-20221106173738-3b8a68ca89b4 // indirect + github.com/dop251/goja_nodejs v0.0.0-20221009164102-3aa5028e57f6 // indirect github.com/dustin/go-humanize v1.0.0 // indirect github.com/go-logr/logr v1.2.3 // indirect github.com/go-logr/stdr v1.2.2 // indirect + github.com/go-sourcemap/sourcemap v2.1.3+incompatible // indirect github.com/golang/glog v1.0.0 // indirect github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e // indirect github.com/golang/protobuf v1.5.2 // indirect diff --git a/go.sum b/go.sum index 0a4e626ba4..9b54411203 100644 --- a/go.sum +++ b/go.sum @@ -105,6 +105,7 @@ github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWH github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d h1:U+s90UTSYgptZMwQh2aRr3LuazLJIa+Pg3Kc1ylSYVY= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= +github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/creack/pty v1.1.18 h1:n56/Zwd5o6whRC5PMGretI4IdRLlmBXYNjScPaBgsbY= github.com/creack/pty v1.1.18/go.mod h1:MOBLtS5ELjhRRrroQr9kyvTxUAFNvYEK993ew/Vr4O4= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= @@ -116,6 +117,17 @@ github.com/dgraph-io/ristretto v0.1.0 h1:Jv3CGQHp9OjuMBSne1485aDpUkTKEcUqF+jm/Lu github.com/dgraph-io/ristretto v0.1.0/go.mod h1:fux0lOrBhrVCJd3lcTHsIJhq1T2rokOu6v9Vcb3Q9ug= github.com/dgryski/go-farm v0.0.0-20190423205320-6a90982ecee2 h1:tdlZCpZ/P9DhczCTSixgIKmwPv6+wP5DGjqLYw5SUiA= github.com/dgryski/go-farm v0.0.0-20190423205320-6a90982ecee2/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= +github.com/dlclark/regexp2 v1.4.1-0.20201116162257-a2a8dda75c91/go.mod h1:2pZnwuY/m+8K6iRw6wQdMtk+rH5tNGR1i55kozfMjCc= +github.com/dlclark/regexp2 v1.7.0 h1:7lJfhqlPssTb1WQx4yvTHN0uElPEv52sbaECrAQxjAo= +github.com/dlclark/regexp2 v1.7.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= +github.com/dop251/goja v0.0.0-20211022113120-dc8c55024d06/go.mod h1:R9ET47fwRVRPZnOGvHxxhuZcbrMCuiqOz3Rlrh4KSnk= +github.com/dop251/goja v0.0.0-20220815083517-0c74f9139fd6/go.mod h1:yRkwfj0CBpOGre+TwBsqPV0IH0Pk73e4PXJOeNDboGs= +github.com/dop251/goja v0.0.0-20221106173738-3b8a68ca89b4 h1:arM6Tq1Ba+a9FWuq3S6Qgrfd5MD0slQdMnCKI2VclFg= +github.com/dop251/goja v0.0.0-20221106173738-3b8a68ca89b4/go.mod h1:yRkwfj0CBpOGre+TwBsqPV0IH0Pk73e4PXJOeNDboGs= +github.com/dop251/goja_nodejs v0.0.0-20210225215109-d91c329300e7/go.mod h1:hn7BA7c8pLvoGndExHudxTDKZ84Pyvv+90pbBjbTz0Y= +github.com/dop251/goja_nodejs v0.0.0-20211022123610-8dd9abb0616d/go.mod h1:DngW8aVqWbuLRMHItjPUyqdj+HWPvnQe8V8y1nDpIbM= +github.com/dop251/goja_nodejs v0.0.0-20221009164102-3aa5028e57f6 h1:p3QZwRRfCN7Qr3GNBTMKBkLFjEm3DHR4MaJABvsiqgk= +github.com/dop251/goja_nodejs v0.0.0-20221009164102-3aa5028e57f6/go.mod h1:+CJy9V5cGycP5qwp6RM5jLg+TFEMyGtD7A9xUbU/BOQ= github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo= github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= github.com/dvyukov/go-fuzz v0.0.0-20210103155950-6a8e9d1f2415/go.mod h1:11Gm+ccJnvAhCNLlf5+cS9KjtbaD5I5zaZpFMsTHWTw= @@ -140,6 +152,8 @@ github.com/go-logr/logr v1.2.3 h1:2DntVwHkVopvECVRSlL5PSo9eG+cAkDCuckLubN+rq0= github.com/go-logr/logr v1.2.3/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= +github.com/go-sourcemap/sourcemap v2.1.3+incompatible h1:W1iEw64niKVGogNgBN3ePyLFfuisuzeidWPMPWmECqU= +github.com/go-sourcemap/sourcemap v2.1.3+incompatible/go.mod h1:F8jJfvm2KbVjc5NqelyYJmf/v5J0dwNLS2mL4sNA1Jg= github.com/gofrs/flock v0.8.1 h1:+gYjHKf32LDeiEEFhQaotPbLuUXjY5ZqxKgXy7n59aw= github.com/gofrs/flock v0.8.1/go.mod h1:F1TvTiK9OcQqauNUHlbJvyl9Qa1QvF/gOUDKA14jxHU= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= @@ -251,9 +265,13 @@ github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/X github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= +github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= +github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= github.com/mattn/go-zglob v0.0.0-20180803001819-2ea3427bfa53 h1:tGfIHhDghvEnneeRhODvGYOt305TPwingKt6p90F4MU= @@ -285,6 +303,7 @@ github.com/rjeczalik/interfaces v0.1.1 h1:xhFQNGtz3T3CQgtJJwWn+i3Ekl1WeObh7wtTtC github.com/rjeczalik/interfaces v0.1.1/go.mod h1:TNwD+kCGmXYrXksRDD5ikspp08m/Aosbr67zVLMjnOY= github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= github.com/russross/blackfriday/v2 v2.0.1 h1:lPqVAte+HuHNfhJ/0LC98ESWRz8afy9tM/0RK8m9o+Q= github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/sasha-s/go-deadlock v0.0.0-20180226215254-237a9547c8a5 h1:T7hUw7pBSINuHQyWwMdfIWZZH5M3ju4yXIbuV/Upp+4= @@ -811,11 +830,14 @@ gopkg.in/DataDog/dd-trace-go.v1 v1.43.1/go.mod h1:YL9g+nlUY7ByCffD5pDytAqy99GNby gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= diff --git a/main.go b/main.go index 595e93be47..8d1080d95b 100644 --- a/main.go +++ b/main.go @@ -95,6 +95,7 @@ func main() { Usage: "Make changes to the pipeline of the currently running build", Subcommands: []cli.Command{ clicommand.PipelineUploadCommand, + clicommand.PipelineEvalCommand, }, }, { diff --git a/resources/node_modules/buildkite/hello.js b/resources/node_modules/buildkite/hello.js new file mode 100644 index 0000000000..a67fa04b25 --- /dev/null +++ b/resources/node_modules/buildkite/hello.js @@ -0,0 +1 @@ +console.log("hello embedded FS!"); diff --git a/resources/resources.go b/resources/resources.go new file mode 100644 index 0000000000..153ee10d14 --- /dev/null +++ b/resources/resources.go @@ -0,0 +1,8 @@ +package resources + +import "embed" + +// FS is an embedded filesystem. +// +//go:embed node_modules +var FS embed.FS diff --git a/test/fixtures/pipelines/buildkite.js b/test/fixtures/pipelines/buildkite.js new file mode 100644 index 0000000000..9ffc5dde9e --- /dev/null +++ b/test/fixtures/pipelines/buildkite.js @@ -0,0 +1,26 @@ +plugin = require("buildkite/plugin"); +require("buildkite/hello"); + +const dockerCompose = plugin("docker-compose", "v3.0.0", { + config: ".buildkite/docker-compose.yml", + run: "agent", +}); + +pipeline = { + env: { + DRY_RUN: !!process.env.DRY_RUN, + }, + agents: { + queue: "agent-runners-linux-amd64", + }, + steps: [ + { + name: ":go: go fmt", + key: "test-go-fmt", + command: ".buildkite/steps/test-go-fmt.sh", + plugins: [dockerCompose], + }, + ], +}; + +module.exports = pipeline; From a17963d864ec8b621a66bdcde1f44dbf8581b9ec Mon Sep 17 00:00:00 2001 From: Paul Annesley Date: Tue, 15 Nov 2022 17:41:17 +1030 Subject: [PATCH 2/4] package js extracted from PipelineEvalCommand package resources (embedded filesystem) is also folded into package js. --- clicommand/pipeline_eval.go | 94 ++----------- js/js.go | 126 ++++++++++++++++++ .../node_modules/buildkite/hello.js | 0 resources/resources.go | 8 -- 4 files changed, 139 insertions(+), 89 deletions(-) create mode 100644 js/js.go rename {resources => js}/node_modules/buildkite/hello.js (100%) delete mode 100644 resources/resources.go diff --git a/clicommand/pipeline_eval.go b/clicommand/pipeline_eval.go index f73e10489e..ec75393996 100644 --- a/clicommand/pipeline_eval.go +++ b/clicommand/pipeline_eval.go @@ -4,20 +4,14 @@ import ( "errors" "fmt" "io" - "io/fs" "os" "path" "path/filepath" "strings" "github.com/buildkite/agent/v3/cliconfig" - "github.com/buildkite/agent/v3/resources" + "github.com/buildkite/agent/v3/js" "github.com/buildkite/agent/v3/stdin" - "github.com/buildkite/yaml" - "github.com/dop251/goja" - "github.com/dop251/goja_nodejs/console" - "github.com/dop251/goja_nodejs/process" - "github.com/dop251/goja_nodejs/require" "github.com/urfave/cli" ) @@ -95,6 +89,7 @@ var PipelineEvalCommand = cli.Command{ l.Info("Reading pipeline config from STDIN") // Actually read the file from STDIN + filename = "(stdin)" input, err = io.ReadAll(os.Stdin) if err != nil { l.Fatal("Failed to read from STDIN: %s", err) @@ -136,82 +131,19 @@ var PipelineEvalCommand = cli.Command{ } } - if err := evalJS(filename, input, c.App.Writer); err != nil { + pipelineYAML, err := js.EvalJS(filename, input) + if err != nil { panic(err) } - return nil - }, -} -func evalJS(filename string, input []byte, output io.Writer) error { - runtime := goja.New() - runtime.SetFieldNameMapper(goja.TagFieldNameMapper("json", true)) - - // Add support for require() CommonJS modules. - // require("buildkite/*") is handled by embedded resources/node_modules/buildkite/* filesystem. - // Other paths are loaded from the host filesystem. - registry := require.NewRegistry( - require.WithLoader(func(name string) ([]byte, error) { - if !strings.HasPrefix(name, "node_modules/buildkite/") { - return require.DefaultSourceLoader(name) - } - res := resources.FS - data, err := res.ReadFile(name) - if errors.Is(err, fs.ErrNotExist) { - return nil, require.ModuleFileDoesNotExistError - } else if err != nil { - return nil, err - } - return data, nil - }), - ) - registry.Enable(runtime) - - // Add basic utilities - console.Enable(runtime) // console.log() - process.Enable(runtime) // process.env() - - // provide plugin() as a native module (implemented in Go) - registry.RegisterNativeModule("buildkite/plugin", func(runtime *goja.Runtime, module *goja.Object) { - module.Set("exports", func(call goja.FunctionCall) goja.Value { - name := call.Argument(0) - ref := call.Argument(1) - config := call.Argument(2) - plugin := runtime.NewObject() - plugin.Set(name.String()+"#"+ref.String(), config) - return plugin - }) - }) - - // provide assignable module.exports for Pipeline result - rootModule := runtime.NewObject() - rootModule.Set("exports", runtime.NewObject()) - err := runtime.Set("module", rootModule) - if err != nil { - return err - } - - if filename == "" { - filename = "(stdin)" - } - - v, err := runtime.RunScript(filename, string(input)) - if err != nil { - panic(err) - } - - y, err := yaml.Marshal(v.Export()) - if err != nil { - return err - } - - n, err := output.Write(y) - if err != nil { - return nil - } - if n != len(y) { - return errors.New("short write") - } + n, err := c.App.Writer.Write(pipelineYAML) + if err != nil { + return nil + } + if n != len(pipelineYAML) { + return errors.New("short write") + } - return nil + return nil + }, } diff --git a/js/js.go b/js/js.go new file mode 100644 index 0000000000..8e76f3356f --- /dev/null +++ b/js/js.go @@ -0,0 +1,126 @@ +package js + +import ( + "embed" + "errors" + "fmt" + "io/fs" + "strings" + + "github.com/buildkite/yaml" + "github.com/dop251/goja" + "github.com/dop251/goja_nodejs/console" + "github.com/dop251/goja_nodejs/process" + "github.com/dop251/goja_nodejs/require" +) + +const ( + // nameModule is the name of the top-level object injected into the VM + nameModule = "module" + + // nameExports is the key/name of the object expected to be assigned within + // the top-level module, e.g. `module.exports = { hello: "world" }` + nameExports = "exports" +) + +// EvalJS takes JavaScript code (loaded from file or stdin etc) and returns +// a YAML serialization of the exported value (e.g. a YAML Pipeline). +// The name arg is the name of the file/stream/source of the JavaScript code, +// used for stack/error messages. +func EvalJS(name string, input []byte) ([]byte, error) { + runtime, rootModule, err := newJavaScriptRuntime() + if err != nil { + return nil, err + } + + // Run the script; we don't need to capture the return value of the script, + // we'll access module.exports instead. + returnValue, err := runtime.RunScript(name, string(input)) + if err != nil { + return nil, err + } + + // Get the module.exports valus assigned by the script + value := rootModule.Get(nameExports) + if value == nil { + // if module.exports wasn't assigned, try the return value of the script + value = returnValue + + if value == nil { + return nil, errors.New("Script neither assigned module.exports nor returned a value") + } + } + result := value.Export() + + // Rather than returning the interface{} from goja.Value.Export(), we'll + // serialize it to YAML here. + pipelineYAML, err := yaml.Marshal(result) + if err != nil { + return nil, fmt.Errorf("Serializing JavaScript result to YAML: %w", err) + } + + return pipelineYAML, nil +} + +func newJavaScriptRuntime() (*goja.Runtime, *goja.Object, error) { + runtime := goja.New() + + // Add support for require() CommonJS modules. + // require("buildkite/*") is handled by embedded resources/node_modules/buildkite/* filesystem. + // Other paths are loaded from the host filesystem. + registry := require.NewRegistry( + require.WithLoader(requireSourceLoader), + ) + + // Add basic utilities + registry.Enable(runtime) // require(); must be enabled before console, process + console.Enable(runtime) // console.log() + process.Enable(runtime) // process.env + + // provide plugin() as a native module (implemented in Go) + // This is implemented natively as a proof-of-concept; there's no good reason + // for this to be implemented in Go rather than an embedded .js file. + registry.RegisterNativeModule("buildkite/plugin", pluginNativeModule) + + // provide assignable module.exports for Pipeline result + rootModule := runtime.NewObject() + err := runtime.Set(nameModule, rootModule) + if err != nil { + return nil, nil, err + } + + return runtime, rootModule, nil +} + +// FS is an embedded filesystem. +// +//go:embed node_modules +var embeddedFS embed.FS + +// requireSourceLoader is a require.SourceLoader which loads +// require("buildkite/*") from a filesystem embedded in the compiled binary, +// and delegates other paths to require.DefaultSourceLoader to be loaded from +// the host filesystem. +func requireSourceLoader(name string) ([]byte, error) { + if !strings.HasPrefix(name, "node_modules/buildkite/") { + return require.DefaultSourceLoader(name) + } + data, err := embeddedFS.ReadFile(name) + if errors.Is(err, fs.ErrNotExist) { + return nil, require.ModuleFileDoesNotExistError + } else if err != nil { + return nil, err + } + return data, nil +} + +func pluginNativeModule(runtime *goja.Runtime, module *goja.Object) { + module.Set("exports", func(call goja.FunctionCall) goja.Value { + name := call.Argument(0) + ref := call.Argument(1) + config := call.Argument(2) + plugin := runtime.NewObject() + plugin.Set(name.String()+"#"+ref.String(), config) + return plugin + }) +} diff --git a/resources/node_modules/buildkite/hello.js b/js/node_modules/buildkite/hello.js similarity index 100% rename from resources/node_modules/buildkite/hello.js rename to js/node_modules/buildkite/hello.js diff --git a/resources/resources.go b/resources/resources.go deleted file mode 100644 index 153ee10d14..0000000000 --- a/resources/resources.go +++ /dev/null @@ -1,8 +0,0 @@ -package resources - -import "embed" - -// FS is an embedded filesystem. -// -//go:embed node_modules -var FS embed.FS From 49f09c633e4cf2a1af1f6ba5b7cd69d040707594 Mon Sep 17 00:00:00 2001 From: Paul Annesley Date: Tue, 15 Nov 2022 20:38:04 +1030 Subject: [PATCH 3/4] js: debug logger, JS require() load order tweak Now require("buildkite/*") will attempt to load from host filesystem if the requested file doesn't exist in the embedded filesystem. --- clicommand/pipeline_eval.go | 2 +- js/js.go | 44 +++++++++++++++++++++++++------------ 2 files changed, 31 insertions(+), 15 deletions(-) diff --git a/clicommand/pipeline_eval.go b/clicommand/pipeline_eval.go index ec75393996..e50b05dad4 100644 --- a/clicommand/pipeline_eval.go +++ b/clicommand/pipeline_eval.go @@ -131,7 +131,7 @@ var PipelineEvalCommand = cli.Command{ } } - pipelineYAML, err := js.EvalJS(filename, input) + pipelineYAML, err := js.EvalJS(filename, input, l) if err != nil { panic(err) } diff --git a/js/js.go b/js/js.go index 8e76f3356f..0b60b95764 100644 --- a/js/js.go +++ b/js/js.go @@ -7,6 +7,7 @@ import ( "io/fs" "strings" + "github.com/buildkite/agent/v3/logger" "github.com/buildkite/yaml" "github.com/dop251/goja" "github.com/dop251/goja_nodejs/console" @@ -27,8 +28,8 @@ const ( // a YAML serialization of the exported value (e.g. a YAML Pipeline). // The name arg is the name of the file/stream/source of the JavaScript code, // used for stack/error messages. -func EvalJS(name string, input []byte) ([]byte, error) { - runtime, rootModule, err := newJavaScriptRuntime() +func EvalJS(name string, input []byte, log logger.Logger) ([]byte, error) { + runtime, rootModule, err := newJavaScriptRuntime(log) if err != nil { return nil, err } @@ -62,14 +63,14 @@ func EvalJS(name string, input []byte) ([]byte, error) { return pipelineYAML, nil } -func newJavaScriptRuntime() (*goja.Runtime, *goja.Object, error) { +func newJavaScriptRuntime(log logger.Logger) (*goja.Runtime, *goja.Object, error) { runtime := goja.New() // Add support for require() CommonJS modules. // require("buildkite/*") is handled by embedded resources/node_modules/buildkite/* filesystem. // Other paths are loaded from the host filesystem. registry := require.NewRegistry( - require.WithLoader(requireSourceLoader), + require.WithLoader(requireSourceLoader(log)), ) // Add basic utilities @@ -101,17 +102,32 @@ var embeddedFS embed.FS // require("buildkite/*") from a filesystem embedded in the compiled binary, // and delegates other paths to require.DefaultSourceLoader to be loaded from // the host filesystem. -func requireSourceLoader(name string) ([]byte, error) { - if !strings.HasPrefix(name, "node_modules/buildkite/") { - return require.DefaultSourceLoader(name) - } - data, err := embeddedFS.ReadFile(name) - if errors.Is(err, fs.ErrNotExist) { - return nil, require.ModuleFileDoesNotExistError - } else if err != nil { - return nil, err +func requireSourceLoader(log logger.Logger) require.SourceLoader { + return func(name string) ([]byte, error) { + // attempt to load require("buildkite/*") from embedded FS, + // but continue to default filesystem loader when not found. + if strings.HasPrefix(name, "node_modules/buildkite/") { + data, err := embeddedFS.ReadFile(name) + if errors.Is(err, fs.ErrNotExist) { + log.Debug("js require() embedded: %q %v", name, require.ModuleFileDoesNotExistError) + // continue to default loader + } else if err != nil { + log.Debug("js require() embedded: %q %v", name, err) + return nil, err + } else { + log.Debug("js require() embedded: %q loaded %d bytes", name, len(data)) + return data, nil + } + } + + data, err := require.DefaultSourceLoader(name) + if err != nil { + log.Debug("js require() default: %q %v", name, err) + return data, err + } + log.Debug("js require() default: %q loaded %d bytes", name, len(data)) + return data, err } - return data, nil } func pluginNativeModule(runtime *goja.Runtime, module *goja.Object) { From e5b88fed8f3bce57f0282585d471367b653ecc56 Mon Sep 17 00:00:00 2001 From: Paul Annesley Date: Tue, 15 Nov 2022 22:55:33 +1030 Subject: [PATCH 4/4] js: better error/debug reporting, especially for require() --- clicommand/pipeline_eval.go | 2 +- js/js.go | 81 +++++++++++++++++++++++++++++++------ 2 files changed, 70 insertions(+), 13 deletions(-) diff --git a/clicommand/pipeline_eval.go b/clicommand/pipeline_eval.go index e50b05dad4..f2eafaa8d7 100644 --- a/clicommand/pipeline_eval.go +++ b/clicommand/pipeline_eval.go @@ -133,7 +133,7 @@ var PipelineEvalCommand = cli.Command{ pipelineYAML, err := js.EvalJS(filename, input, l) if err != nil { - panic(err) + return fmt.Errorf("JavaScript evaluation: %w", err) } n, err := c.App.Writer.Write(pipelineYAML) diff --git a/js/js.go b/js/js.go index 0b60b95764..831b1c47f2 100644 --- a/js/js.go +++ b/js/js.go @@ -34,14 +34,21 @@ func EvalJS(name string, input []byte, log logger.Logger) ([]byte, error) { return nil, err } - // Run the script; we don't need to capture the return value of the script, - // we'll access module.exports instead. + // Run the script; capture the return value as a fallback in case the + // preferred module.exports wasn't assigned. returnValue, err := runtime.RunScript(name, string(input)) if err != nil { + if exception, ok := err.(*goja.Exception); ok { + if exception.Value().String() == "GoError: Invalid module" { + log.Info("Use --debug to trace require() load attempts") + } + // log the exception and multi-line stack trace + log.Error("%s", exception.String()) + } return nil, err } - // Get the module.exports valus assigned by the script + // Get the module.exports value assigned by the script value := rootModule.Get(nameExports) if value == nil { // if module.exports wasn't assigned, try the return value of the script @@ -63,6 +70,8 @@ func EvalJS(name string, input []byte, log logger.Logger) ([]byte, error) { return pipelineYAML, nil } +// newJavaScriptRuntime builds and configures a goja.Runtime, with various +// modules loaded, and custom require() source loading. func newJavaScriptRuntime(log logger.Logger) (*goja.Runtime, *goja.Object, error) { runtime := goja.New() @@ -74,9 +83,9 @@ func newJavaScriptRuntime(log logger.Logger) (*goja.Runtime, *goja.Object, error ) // Add basic utilities - registry.Enable(runtime) // require(); must be enabled before console, process - console.Enable(runtime) // console.log() - process.Enable(runtime) // process.env + enableRequireModule(runtime, registry, log) // require(); must be enabled before console, process + console.Enable(runtime) // console.log() + process.Enable(runtime) // process.env // provide plugin() as a native module (implemented in Go) // This is implemented natively as a proof-of-concept; there's no good reason @@ -93,7 +102,8 @@ func newJavaScriptRuntime(log logger.Logger) (*goja.Runtime, *goja.Object, error return runtime, rootModule, nil } -// FS is an embedded filesystem. +// embeddedFS embeds node_modules from the source tree into the compiled binary +// as a virtual filesystem, which requireSourceLoader accesses. // //go:embed node_modules var embeddedFS embed.FS @@ -109,27 +119,31 @@ func requireSourceLoader(log logger.Logger) require.SourceLoader { if strings.HasPrefix(name, "node_modules/buildkite/") { data, err := embeddedFS.ReadFile(name) if errors.Is(err, fs.ErrNotExist) { - log.Debug("js require() embedded: %q %v", name, require.ModuleFileDoesNotExistError) + log.Debug(" loader=embedded %q %v", name, require.ModuleFileDoesNotExistError) // continue to default loader } else if err != nil { - log.Debug("js require() embedded: %q %v", name, err) + log.Debug(" loader=embedded %q %v", name, err) return nil, err } else { - log.Debug("js require() embedded: %q loaded %d bytes", name, len(data)) + log.Debug(" loader=embedded %q loaded %d bytes", name, len(data)) return data, nil } } data, err := require.DefaultSourceLoader(name) if err != nil { - log.Debug("js require() default: %q %v", name, err) + log.Debug(" loader=default %q %v", name, err) return data, err } - log.Debug("js require() default: %q loaded %d bytes", name, len(data)) + log.Debug(" loader=default %q loaded %d bytes", name, len(data)) return data, err } } +// pluginNativeModule implements a basic `plugin(name, ver, config)` JS function, +// as a proof of concept of native modules. It should really be implemented as +// an embedded JavaScript file, but this demonstrates how to implement native +// functions that interact with Go code in more complex ways. func pluginNativeModule(runtime *goja.Runtime, module *goja.Object) { module.Set("exports", func(call goja.FunctionCall) goja.Value { name := call.Argument(0) @@ -140,3 +154,46 @@ func pluginNativeModule(runtime *goja.Runtime, module *goja.Object) { return plugin }) } + +// enableRequireModule adds goja_nodejs's require() function to the runtime, +// wrapped in some custom debug logging and error reporting. +func enableRequireModule(runtime *goja.Runtime, registry *require.Registry, log logger.Logger) { + // enable goja_nodejs's require() + registry.Enable(runtime) + + // get a reference to goja_nodejs's require() + orig, ok := goja.AssertFunction(runtime.Get("require")) + if !ok { + panic("expected `require` to be a function") + } + + // a stack of names being recursively loaded + var stack []string + + // wrap require() to log/track the name being required + runtime.Set("require", func(call goja.FunctionCall) goja.Value { + name := call.Argument(0) + + // track this name on our stack + stack = append(stack, name.String()) + defer func() { stack = stack[:len(stack)-1] }() + + log.Debug("require(%q) [%s]", name, strings.Join(stack, " → ")) + + // call the original goja_nodejs require() + res, err := orig(goja.Undefined(), name) + if err != nil { + if exception, ok := err.(*goja.Exception); ok { + if exception.Value().String() == "GoError: Invalid module" { + // report the head of the require() name stack + log.Error("require(%q)", stack[len(stack)-1]) + } + } + // propagate the error to goja.Runtime + panic(err) + } + + log.Debug(" require(%q) finished", name) + return res + }) +}