Skip to content

Commit

Permalink
Merge pull request #17 from speakeasy-api/improve-api-for-repeat-envi…
Browse files Browse the repository at this point in the history
…ronment-use
  • Loading branch information
TristanSpeakEasy authored Mar 7, 2024
2 parents e9d0bbb + 029bd62 commit 4ec8f0c
Show file tree
Hide file tree
Showing 5 changed files with 67 additions and 76 deletions.
118 changes: 49 additions & 69 deletions engine.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,10 @@ import (
)

var (
// ErrAlreadyRan is returned when the engine has already been ran, and can't be ran again. In order to run the engine again, a new engine must be created.
ErrAlreadyRan = errors.New("engine has already been ran")
// ErrAlreadyInitialized is returned when the engine has already been initialized.
ErrAlreadyInitialized = errors.New("engine has already been initialized")
// ErrNotInitialized is returned when the engine has not been initialized.
ErrNotInitialized = errors.New("engine has not been initialized")
// ErrReserved is returned when a template or js function is reserved and can't be overridden.
ErrReserved = errors.New("function is a reserved function and can't be overridden")
// ErrInvalidArg is returned when an invalid argument is passed to a function.
Expand Down Expand Up @@ -120,11 +122,12 @@ type Engine struct {

templator *template.Templator

ran bool
jsFuncs map[string]func(call CallContext) goja.Value
jsFiles map[string]string

tracer trace.Tracer

vm *vm.VM
}

// New creates a new Engine with the provided options.
Expand Down Expand Up @@ -170,59 +173,55 @@ func New(opts ...Opt) *Engine {
return e
}

// RunScript runs the provided script file, with the provided data, starting the template engine and templating any templates triggered from the script.
func (e *Engine) RunScript(scriptFile string, data any) error {
return e.RunScriptWithContext(context.Background(), scriptFile, data)
}

// RunScriptWithContext runs the provided script file, with the provided data, starting the template engine and templating any templates triggered from the script.
func (e *Engine) RunScriptWithContext(ctx context.Context, scriptFile string, data any) error {
vm, err := e.init(ctx, data)
// Init initializes the engine with global data available to all following methods, and should be called before any other methods are called but only once.
// When using any of the Run or Template methods after init, they will share the global data, but just be careful they will also share any changes made to the environment
// by previous runs.
func (e *Engine) Init(ctx context.Context, data any) error {
v, err := e.init(ctx, data)
if err != nil {
return err
}

e.vm = v

return nil
}

// RunScript runs the provided script file within the environment initialized by Init.
// This is useful for setting up the environment with global variables and functions,
// or running code that is not directly related to templating but might setup the environment for templating.
func (e *Engine) RunScript(scriptFile string) error {
if e.vm == nil {
return ErrNotInitialized
}

script, err := e.readFile(scriptFile)
if err != nil {
return fmt.Errorf("failed to read script file: %w", err)
}

if _, err := vm.Run(scriptFile, string(script)); err != nil {
if _, err := e.vm.Run(scriptFile, string(script)); err != nil {
return err
}

return nil
}

// RunMethod enables calls to global template methods from easytemplate.
func (e *Engine) RunMethod(scriptFile string, data any, fnName string, args ...any) (goja.Value, error) {
return e.RunMethodWithContext(context.Background(), scriptFile, data, fnName, args...)
}

// RunMethodWithContext enables calls to global template methods from easytemplate.
func (e *Engine) RunMethodWithContext(ctx context.Context, scriptFile string, data any, fnName string, args ...any) (goja.Value, error) {
vm, err := e.init(ctx, data)
if err != nil {
return nil, err
}

script, err := e.readFile(scriptFile)
if err != nil {
return nil, fmt.Errorf("failed to read script file: %w", err)
}

if _, err := vm.Run(scriptFile, string(script)); err != nil {
return nil, err
// RunFunction will run the named function if it already exists within the environment, for example if it was defined in a script run by RunScript.
// The provided args will be passed to the function, and the result will be returned.
func (e *Engine) RunFunction(fnName string, args ...any) (goja.Value, error) {
if e.vm == nil {
return nil, ErrNotInitialized
}

fn, ok := goja.AssertFunction(vm.Get(fnName))
fn, ok := goja.AssertFunction(e.vm.Get(fnName))
if !ok {
return nil, fmt.Errorf("%w: %s", ErrFunctionNotFound, fnName)
}

gojaArgs := make([]goja.Value, len(args))
for i, arg := range args {
gojaArgs[i] = vm.ToValue(arg)
gojaArgs[i] = e.vm.ToValue(arg)
}
val, err := fn(goja.Undefined(), gojaArgs...)
if err != nil {
Expand All @@ -232,57 +231,38 @@ func (e *Engine) RunMethodWithContext(ctx context.Context, scriptFile string, da
return val, nil
}

// RunTemplate runs the provided template file, with the provided data, starting the template engine and templating the provided template to a file.
func (e *Engine) RunTemplate(templateFile string, outFile string, data any) error {
return e.RunTemplateWithContext(context.Background(), templateFile, outFile, data)
}

// RunTemplateWithContext runs the provided template file, with the provided data, starting the template engine and templating the provided template to a file.
func (e *Engine) RunTemplateWithContext(ctx context.Context, templateFile string, outFile string, data any) error {
vm, err := e.init(ctx, data)
if err != nil {
return err
// TemplateFile runs the provided template file, with the provided data and writes the result to the provided outFile.
func (e *Engine) TemplateFile(templateFile string, outFile string, data any) error {
if e.vm == nil {
return ErrNotInitialized
}

return e.templator.TemplateFile(vm, templateFile, outFile, data)
}

// RunTemplateString runs the provided template file, with the provided data, starting the template engine and templating the provided template, returning the rendered result.
func (e *Engine) RunTemplateString(templateFile string, data any) (string, error) {
return e.RunTemplateStringWithContext(context.Background(), templateFile, data)
return e.templator.TemplateFile(e.vm, templateFile, outFile, data)
}

// RunTemplateStringWithContext runs the provided template file, with the provided data, starting the template engine and templating the provided template, returning the rendered result.
func (e *Engine) RunTemplateStringWithContext(ctx context.Context, templateFile string, data any) (string, error) {
vm, err := e.init(ctx, data)
if err != nil {
return "", err
// TemplateString runs the provided template file, with the provided data and returns the rendered result.
func (e *Engine) TemplateString(templateFilePath string, data any) (string, error) {
if e.vm == nil {
return "", ErrNotInitialized
}

return e.templator.TemplateString(vm, templateFile, data)
return e.templator.TemplateString(e.vm, templateFilePath, data)
}

// RunTemplateStringInput runs the provided input template string, with the provided data, starting the template engine and templating the provided template, returning the rendered result.
func (e *Engine) RunTemplateStringInput(name, template string, data any) (string, error) {
return e.RunTemplateStringInputWithContext(context.Background(), name, template, data)
}

// RunTemplateStringInputWithContext runs the provided input template string, with the provided data, starting the template engine and templating the provided template, returning the rendered result.
func (e *Engine) RunTemplateStringInputWithContext(ctx context.Context, name, template string, data any) (string, error) {
vm, err := e.init(ctx, data)
if err != nil {
return "", err
// TemplateStringInput runs the provided template string, with the provided data and returns the rendered result.
func (e *Engine) TemplateStringInput(name, template string, data any) (string, error) {
if e.vm == nil {
return "", ErrNotInitialized
}

return e.templator.TemplateStringInput(vm, name, template, data)
return e.templator.TemplateStringInput(e.vm, name, template, data)
}

//nolint:funlen
func (e *Engine) init(ctx context.Context, data any) (*vm.VM, error) {
if e.ran {
return nil, ErrAlreadyRan
if e.vm != nil {
return nil, ErrAlreadyInitialized
}
e.ran = true

v, err := vm.New()
if err != nil {
Expand Down
7 changes: 6 additions & 1 deletion engine_integration_test.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package easytemplate_test

import (
"context"
"fmt"
"os"
"testing"
Expand Down Expand Up @@ -51,10 +52,14 @@ func TestEngine_RunScript_Success(t *testing.T) {
},
}),
)
err = e.RunScript("scripts/test.js", map[string]interface{}{

err = e.Init(context.Background(), map[string]interface{}{
"Test": "global",
})
require.NoError(t, err)

err = e.RunScript("scripts/test.js")
require.NoError(t, err)

assert.Empty(t, expectedFiles, "not all expected files were written")
}
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ module github.com/speakeasy-api/easytemplate
go 1.19

require (
github.com/dop251/goja v0.0.0-20231027120936-b396bb4c349d
github.com/dop251/goja v0.0.0-20240220182346-e401ed450204
github.com/dop251/goja_nodejs v0.0.0-20231122114759-e84d9a924c5c
github.com/evanw/esbuild v0.19.11
github.com/go-sourcemap/sourcemap v2.1.3+incompatible
Expand Down
2 changes: 2 additions & 0 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@ github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cn
github.com/dop251/goja v0.0.0-20211022113120-dc8c55024d06/go.mod h1:R9ET47fwRVRPZnOGvHxxhuZcbrMCuiqOz3Rlrh4KSnk=
github.com/dop251/goja v0.0.0-20231027120936-b396bb4c349d h1:wi6jN5LVt/ljaBG4ue79Ekzb12QfJ52L9Q98tl8SWhw=
github.com/dop251/goja v0.0.0-20231027120936-b396bb4c349d/go.mod h1:QMWlm50DNe14hD7t24KEqZuUdC9sOTy8W6XbCU1mlw4=
github.com/dop251/goja v0.0.0-20240220182346-e401ed450204 h1:O7I1iuzEA7SG+dK8ocOBSlYAA9jBUmCYl/Qa7ey7JAM=
github.com/dop251/goja v0.0.0-20240220182346-e401ed450204/go.mod h1:QMWlm50DNe14hD7t24KEqZuUdC9sOTy8W6XbCU1mlw4=
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-20231122114759-e84d9a924c5c h1:hLoodLRD4KLWIH8eyAQCLcH8EqIrjac7fCkp/fHnvuQ=
Expand Down
14 changes: 9 additions & 5 deletions internal/vm/vm.go
Original file line number Diff line number Diff line change
Expand Up @@ -80,13 +80,17 @@ func (v *VM) Run(name string, src string, opts ...Option) (goja.Value, error) {
return nil, err
}

m, err := sourcemap.Parse("", p.sourceMap)
if err != nil {
return nil, fmt.Errorf("failed to compile source map for script: %w", err)
if len(p.sourceMap) > 0 {
m, err := sourcemap.Parse("", p.sourceMap)
if err != nil {
if !strings.Contains(err.Error(), "mappings are empty") {
return nil, fmt.Errorf("failed to compile source map for script: %w", err)
}
} else {
v.globalSourceMapCache[name] = m
}
}

v.globalSourceMapCache[name] = m

res, err := v.Runtime.RunProgram(p.prog)
if err == nil {
return res, nil
Expand Down

0 comments on commit 4ec8f0c

Please sign in to comment.