blob: 38b292376a03d17408c089c07524cc7bcafe71f6 [file] [log] [blame]
// Copyright 2021 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package rbcrun
import (
"fmt"
"os"
"path/filepath"
"runtime"
"strings"
"testing"
"go.starlark.net/resolve"
"go.starlark.net/starlark"
"go.starlark.net/starlarktest"
)
// In order to use "assert.star" from go/starlark.net/starlarktest in the tests,
// provide:
// * load function that handles "assert.star"
// * starlarktest.DataFile function that finds its location
func init() {
starlarktestSetup()
}
func starlarktestSetup() {
resolve.AllowLambda = true
starlarktest.DataFile = func(pkgdir, filename string) string {
// The caller expects this function to return the path to the
// data file. The implementation assumes that the source file
// containing the caller and the data file are in the same
// directory. It's ugly. Not sure what's the better way.
// TODO(asmundak): handle Bazel case
_, starlarktestSrcFile, _, _ := runtime.Caller(1)
if filepath.Base(starlarktestSrcFile) != "starlarktest.go" {
panic(fmt.Errorf("this function should be called from starlarktest.go, got %s",
starlarktestSrcFile))
}
return filepath.Join(filepath.Dir(starlarktestSrcFile), filename)
}
}
// Common setup for the tests: create thread, change to the test directory
func testSetup(t *testing.T) *starlark.Thread {
thread := &starlark.Thread{
Load: func(thread *starlark.Thread, module string) (starlark.StringDict, error) {
if module == "assert.star" {
return starlarktest.LoadAssertModule()
}
return nil, fmt.Errorf("load not implemented")
}}
starlarktest.SetReporter(thread, t)
if err := os.Chdir(dataDir()); err != nil {
t.Fatal(err)
}
return thread
}
func dataDir() string {
_, thisSrcFile, _, _ := runtime.Caller(0)
return filepath.Join(filepath.Dir(thisSrcFile), "testdata")
}
func exerciseStarlarkTestFile(t *testing.T, starFile string) {
// In order to use "assert.star" from go/starlark.net/starlarktest in the tests, provide:
// * load function that handles "assert.star"
// * starlarktest.DataFile function that finds its location
if err := os.Chdir(dataDir()); err != nil {
t.Fatal(err)
}
thread := &starlark.Thread{
Load: func(thread *starlark.Thread, module string) (starlark.StringDict, error) {
if module == "assert.star" {
return starlarktest.LoadAssertModule()
}
return nil, fmt.Errorf("load not implemented")
}}
starlarktest.SetReporter(thread, t)
_, thisSrcFile, _, _ := runtime.Caller(0)
filename := filepath.Join(filepath.Dir(thisSrcFile), starFile)
thread.SetLocal(executionModeKey, ExecutionModeRbc)
thread.SetLocal(shellKey, "/bin/sh")
if _, err := starlark.ExecFile(thread, filename, nil, rbcBuiltins); err != nil {
if err, ok := err.(*starlark.EvalError); ok {
t.Fatal(err.Backtrace())
}
t.Fatal(err)
}
}
func TestFileOps(t *testing.T) {
// TODO(asmundak): convert this to use exerciseStarlarkTestFile
thread := testSetup(t)
if _, err := starlark.ExecFile(thread, "file_ops.star", nil, rbcBuiltins); err != nil {
if err, ok := err.(*starlark.EvalError); ok {
t.Fatal(err.Backtrace())
}
t.Fatal(err)
}
}
func TestLoad(t *testing.T) {
// TODO(asmundak): convert this to use exerciseStarlarkTestFile
thread := testSetup(t)
thread.Load = func(thread *starlark.Thread, module string) (starlark.StringDict, error) {
if module == "assert.star" {
return starlarktest.LoadAssertModule()
} else {
return loader(thread, module)
}
}
dir := dataDir()
if err := os.Chdir(filepath.Dir(dir)); err != nil {
t.Fatal(err)
}
thread.SetLocal(allowExternalEntrypointKey, false)
thread.SetLocal(callingFileKey, "testdata/load.star")
thread.SetLocal(executionModeKey, ExecutionModeRbc)
if _, err := starlark.ExecFile(thread, "testdata/load.star", nil, rbcBuiltins); err != nil {
if err, ok := err.(*starlark.EvalError); ok {
t.Fatal(err.Backtrace())
}
t.Fatal(err)
}
}
func TestBzlLoadsScl(t *testing.T) {
moduleCache = make(map[string]*modentry)
dir := dataDir()
if err := os.Chdir(filepath.Dir(dir)); err != nil {
t.Fatal(err)
}
vars, _, err := Run("testdata/bzl_loads_scl.bzl", nil, ExecutionModeRbc, false)
if err != nil {
t.Fatal(err)
}
if val, ok := vars["foo"]; !ok {
t.Fatalf("Failed to load foo variable")
} else if val.(starlark.String) != "bar" {
t.Fatalf("Expected \"bar\", got %q", val)
}
}
func TestNonEntrypointBzlLoadsScl(t *testing.T) {
moduleCache = make(map[string]*modentry)
dir := dataDir()
if err := os.Chdir(filepath.Dir(dir)); err != nil {
t.Fatal(err)
}
vars, _, err := Run("testdata/bzl_loads_scl_2.bzl", nil, ExecutionModeRbc, false)
if err != nil {
t.Fatal(err)
}
if val, ok := vars["foo"]; !ok {
t.Fatalf("Failed to load foo variable")
} else if val.(starlark.String) != "bar" {
t.Fatalf("Expected \"bar\", got %q", val)
}
}
func TestSclLoadsBzl(t *testing.T) {
moduleCache = make(map[string]*modentry)
dir := dataDir()
if err := os.Chdir(filepath.Dir(dir)); err != nil {
t.Fatal(err)
}
_, _, err := Run("testdata/scl_incorrectly_loads_bzl.scl", nil, ExecutionModeScl, false)
if err == nil {
t.Fatal("Expected failure")
}
if !strings.Contains(err.Error(), ".scl files can only load other .scl files") {
t.Fatalf("Expected error to contain \".scl files can only load other .scl files\": %q", err.Error())
}
}
func TestCantLoadSymlink(t *testing.T) {
moduleCache = make(map[string]*modentry)
dir := dataDir()
if err := os.Chdir(filepath.Dir(dir)); err != nil {
t.Fatal(err)
}
_, _, err := Run("testdata/test_scl_symlink.scl", nil, ExecutionModeScl, false)
if err == nil {
t.Fatal("Expected failure")
}
if !strings.Contains(err.Error(), "symlinks to starlark files are not allowed") {
t.Fatalf("Expected error to contain \"symlinks to starlark files are not allowed\": %q", err.Error())
}
}
func TestShell(t *testing.T) {
exerciseStarlarkTestFile(t, "testdata/shell.star")
}