diff --git a/examples/experimental/fs/opensync.js b/examples/experimental/fs/opensync.js new file mode 100644 index 00000000000..e217fcc5bfe --- /dev/null +++ b/examples/experimental/fs/opensync.js @@ -0,0 +1,17 @@ +import { openSync } from "k6/experimental/fs"; + +export const options = { + vus: 100, + iterations: 1000, +}; + +// As k6 does not support asynchronous code in the init context, yet, we need to +// use a top-level async function to be able to use the `await` keyword. +const file = openSync("bonjour.txt") + +export default async function () { + const fileinfo = await file.stat(); + if (fileinfo.name != "bonjour.txt") { + throw new Error("Unexpected file name"); + } +} diff --git a/js/modules/k6/experimental/fs/module.go b/js/modules/k6/experimental/fs/module.go index af1b22c8986..17a1d492f41 100644 --- a/js/modules/k6/experimental/fs/module.go +++ b/js/modules/k6/experimental/fs/module.go @@ -50,7 +50,8 @@ func (rm *RootModule) NewModuleInstance(vu modules.VU) modules.Instance { func (mi *ModuleInstance) Exports() modules.Exports { return modules.Exports{ Named: map[string]any{ - "open": mi.Open, + "open": mi.Open, + "openSync": mi.OpenSync, }, } } @@ -90,6 +91,50 @@ func (mi *ModuleInstance) Open(path goja.Value) *goja.Promise { return promise } +// OpenSync opens a file and returns a [File] instance. +// +// This method is synchronous and should only be used in the init context. +// +// This method is intended as a temporary workaround until we have proper +// support for asynchronous functions execution in the k6 init context. +// +// TODO @oleiade: remove this method once we have proper support for +// asynchronous functions execution in the k6 init context. +func (mi *ModuleInstance) OpenSync(path goja.Value) (goja.Value, error) { + rt := mi.vu.Runtime() + + // Files can only be opened in the init context. + if mi.vu.State() != nil { + common.Throw( + rt, + newFsError(ForbiddenError, "openSync() failed; reason: opening a file in the VU context is forbidden"), + ) + } + + if common.IsNullish(path) { + common.Throw( + rt, + newFsError(TypeError, "openSync() failed; reason: path cannot be null or undefined"), + ) + } + + // Obtain the underlying path string from the JS value. + pathStr := path.String() + if pathStr == "" { + common.Throw( + rt, + newFsError(TypeError, "open() failed; reason: path cannot be empty"), + ) + } + + file, err := mi.openImpl(pathStr) + if err != nil { + common.Throw(rt, err) + } + + return rt.ToValue(file), nil +} + func (mi *ModuleInstance) openImpl(path string) (*File, error) { initEnv := mi.vu.InitEnv() diff --git a/js/modules/k6/experimental/fs/module_test.go b/js/modules/k6/experimental/fs/module_test.go index b7c8b7eb1bf..c8062baa8e2 100644 --- a/js/modules/k6/experimental/fs/module_test.go +++ b/js/modules/k6/experimental/fs/module_test.go @@ -173,6 +173,159 @@ func TestOpen(t *testing.T) { }) } +func TestOpenSync(t *testing.T) { + t.Parallel() + + t.Run("opening existing file synchronously should succeed", func(t *testing.T) { + t.Parallel() + + tests := []struct { + name string + openPath string + wantPath string + }{ + { + name: "open absolute path", + openPath: fsext.FilePathSeparator + "bonjour.txt", + wantPath: fsext.FilePathSeparator + "bonjour.txt", + }, + { + name: "open relative path", + openPath: filepath.Join(".", fsext.FilePathSeparator, "bonjour.txt"), + wantPath: fsext.FilePathSeparator + "bonjour.txt", + }, + { + name: "open path with ..", + openPath: fsext.FilePathSeparator + "dir" + fsext.FilePathSeparator + ".." + fsext.FilePathSeparator + "bonjour.txt", + wantPath: fsext.FilePathSeparator + "bonjour.txt", + }, + } + + for _, tt := range tests { + tt := tt + + t.Run(tt.name, func(t *testing.T) { + t.Parallel() + + runtime, err := newConfiguredRuntime(t) + require.NoError(t, err) + + fs := newTestFs(t, func(fs afero.Fs) error { + fileErr := afero.WriteFile(fs, tt.wantPath, []byte("Bonjour, le monde"), 0o644) + if fileErr != nil { + return fileErr + } + + return fs.Mkdir(fsext.FilePathSeparator+"dir", 0o644) + }) + runtime.VU.InitEnvField.FileSystems["file"] = fs + runtime.VU.InitEnvField.CWD = &url.URL{Scheme: "file", Path: fsext.FilePathSeparator} + + _, err = runtime.VU.RuntimeField.RunString(fmt.Sprintf(` + try { + const file = fs.openSync(%q) + + if (file.path !== %q) { + throw 'unexpected file path ' + file.path + '; expected %q'; + } + } catch (err) { + throw "unexpected error: " + err + } + + `, tt.openPath, tt.wantPath, tt.wantPath)) + + assert.NoError(t, err) + }) + } + }) + + t.Run("opening file synchronously in VU context should fail", func(t *testing.T) { + t.Parallel() + + runtime, err := newConfiguredRuntime(t) + require.NoError(t, err) + + runtime.MoveToVUContext(&lib.State{ + Tags: lib.NewVUStateTags(metrics.NewRegistry().RootTagSet().With("tag-vu", "mytag")), + }) + + _, err = runtime.VU.RuntimeField.RunString(` + fs.openSync('bonjour.txt') + `) + + assert.Error(t, err) + assert.Contains(t, err.Error(), "ForbiddenError") + }) + + t.Run("calling openSync without providing a path should fail", func(t *testing.T) { + t.Parallel() + + tests := []struct { + name string + openPath string + }{ + { + name: "openSync empty path should fail", + openPath: "", + }, + { + name: "openSync null path should fail", + openPath: "null", + }, + { + name: "openSync undefined path should fail", + openPath: "undefined", + }, + } + + for _, tt := range tests { + tt := tt + + t.Run(tt.name, func(t *testing.T) { + t.Parallel() + + runtime, err := newConfiguredRuntime(t) + require.NoError(t, err) + + _, gotErr := runtime.VU.RuntimeField.RunString(fmt.Sprintf(`fs.openSync(%q)`, tt.openPath)) + + assert.Error(t, gotErr) + }) + } + }) + + t.Run("opening directory synchronously should fail", func(t *testing.T) { + t.Parallel() + + runtime, err := newConfiguredRuntime(t) + require.NoError(t, err) + + testDirPath := fsext.FilePathSeparator + "dir" + fs := newTestFs(t, func(fs afero.Fs) error { + return fs.Mkdir(testDirPath, 0o644) + }) + + runtime.VU.InitEnvField.FileSystems["file"] = fs + + _, err = runtime.VU.RuntimeField.RunString(fmt.Sprintf(`fs.openSync(%q)`, testDirPath)) + + assert.Error(t, err) + assert.Contains(t, err.Error(), "InvalidResourceError") + }) + + t.Run("opening non existing file synchronously should fail", func(t *testing.T) { + t.Parallel() + + runtime, err := newConfiguredRuntime(t) + require.NoError(t, err) + + _, err = runtime.VU.RuntimeField.RunString(`fs.openSync('doesnotexist.txt')`) + + assert.Error(t, err) + assert.Contains(t, err.Error(), "NotFoundError") + }) +} + func TestFile(t *testing.T) { t.Parallel()