pulumi/sdk/nodejs/cmd/pulumi-language-nodejs/main_test.go

488 lines
13 KiB
Go

// Copyright 2016-2018, Pulumi Corporation.
//
// 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 main
import (
"context"
"os"
"os/exec"
"path/filepath"
"strings"
"syscall"
"testing"
"time"
"github.com/blang/semver"
"github.com/pulumi/pulumi/sdk/v3/nodejs/npm"
pulumirpc "github.com/pulumi/pulumi/sdk/v3/proto/go"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
func TestArgumentConstruction(t *testing.T) {
t.Parallel()
info := &pulumirpc.ProgramInfo{
RootDirectory: "/foo/bar",
ProgramDirectory: "/foo/bar",
EntryPoint: ".",
}
t.Run("DryRun-NoArguments", func(t *testing.T) {
t.Parallel()
host := &nodeLanguageHost{}
rr := &pulumirpc.RunRequest{DryRun: true, Info: info}
args := host.constructArguments(rr, "", "", "")
assert.Contains(t, args, "--dry-run")
assert.NotContains(t, args, "true")
})
t.Run("OptionalArgs-PassedIfSpecified", func(t *testing.T) {
t.Parallel()
host := &nodeLanguageHost{}
rr := &pulumirpc.RunRequest{Project: "foo", Info: info}
args := strings.Join(host.constructArguments(rr, "", "", ""), " ")
assert.Contains(t, args, "--project foo")
})
t.Run("OptionalArgs-NotPassedIfNotSpecified", func(t *testing.T) {
t.Parallel()
host := &nodeLanguageHost{}
rr := &pulumirpc.RunRequest{Info: info}
args := strings.Join(host.constructArguments(rr, "", "", ""), " ")
assert.NotContains(t, args, "--stack")
})
t.Run("DotIfProgramNotSpecified", func(t *testing.T) {
t.Parallel()
host := &nodeLanguageHost{}
rr := &pulumirpc.RunRequest{Info: info}
args := strings.Join(host.constructArguments(rr, "", "", ""), " ")
assert.Contains(t, args, ".")
})
t.Run("ProgramIfProgramSpecified", func(t *testing.T) {
t.Parallel()
host := &nodeLanguageHost{}
rr := &pulumirpc.RunRequest{
Program: "foobar",
Info: &pulumirpc.ProgramInfo{
RootDirectory: "/foo/bar",
ProgramDirectory: "/foo/bar",
EntryPoint: "foobar",
},
}
args := strings.Join(host.constructArguments(rr, "", "", ""), " ")
assert.Contains(t, args, "foobar")
})
}
func TestConfig(t *testing.T) {
t.Parallel()
t.Run("Config-Empty", func(t *testing.T) {
t.Parallel()
host := &nodeLanguageHost{}
rr := &pulumirpc.RunRequest{Project: "foo"}
str, err := host.constructConfig(rr)
assert.NoError(t, err)
assert.JSONEq(t, "{}", str)
})
}
func TestCompatibleVersions(t *testing.T) {
t.Parallel()
cases := []struct {
a string
b string
compatible bool
errmsg string
}{
{"0.17.1", "0.16.2", false, "Differing major or minor versions are not supported."},
{"0.17.1", "1.0.0", true, ""},
{"1.0.0", "0.17.1", true, ""},
{"1.13.0", "1.13.0", true, ""},
{"1.1.1", "1.13.0", true, ""},
{"1.13.0", "1.1.1", true, ""},
{"1.1.0", "2.1.0", true, ""},
{"2.1.0", "1.1.0", true, ""},
{"1.1.0", "2.0.0-beta1", true, ""},
{"2.0.0-beta1", "1.1.0", true, ""},
{"2.1.0", "3.1.0", false, "Differing major versions are not supported."},
{"0.16.1", "1.0.0", false, "Differing major or minor versions are not supported."},
}
for _, c := range cases {
compatible, errmsg := compatibleVersions(semver.MustParse(c.a), semver.MustParse(c.b))
assert.Equal(t, c.errmsg, errmsg)
assert.Equal(t, c.compatible, compatible)
}
}
func TestGetRequiredPlugins(t *testing.T) {
t.Parallel()
dir := t.TempDir()
files := []struct {
path string
content string
}{
{
filepath.Join(dir, "node_modules", "@pulumi", "foo", "package.json"),
`{ "name": "@pulumi/foo", "version": "1.2.3", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "node_modules", "@pulumi", "bar", "package.json"),
`{ "name": "@pulumi/bar", "version": "4.5.6", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "node_modules", "@pulumi", "baz", "package.json"),
`{ "name": "@pulumi/baz", "version": "4.5.6", "pulumi": { "resource": false } }`,
},
{
filepath.Join(dir, "node_modules", "malformed", "tests", "malformed_test", "package.json"),
`{`,
},
}
for _, file := range files {
err := os.MkdirAll(filepath.Dir(file.path), 0o755)
require.NoError(t, err)
err = os.WriteFile(file.path, []byte(file.content), 0o600)
require.NoError(t, err)
}
host := &nodeLanguageHost{}
resp, err := host.GetRequiredPlugins(context.Background(), &pulumirpc.GetRequiredPluginsRequest{
Program: dir,
Info: &pulumirpc.ProgramInfo{
RootDirectory: dir,
ProgramDirectory: dir,
EntryPoint: ".",
},
})
require.NoError(t, err)
actual := make(map[string]string)
for _, plugin := range resp.GetPlugins() {
actual[plugin.Name] = plugin.Version
}
assert.Equal(t, map[string]string{
"foo": "v1.2.3",
"bar": "v4.5.6",
}, actual)
}
func TestGetRequiredPluginsSymlinkCycles(t *testing.T) {
t.Parallel()
dir := t.TempDir()
files := []struct {
path string
content string
}{
{
filepath.Join(dir, "node_modules", "@pulumi", "foo", "package.json"),
`{ "name": "@pulumi/foo", "version": "1.2.3", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "node_modules", "@pulumi", "bar", "package.json"),
`{ "name": "@pulumi/bar", "version": "4.5.6", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "node_modules", "@pulumi", "baz", "package.json"),
`{ "name": "@pulumi/baz", "version": "4.5.6", "pulumi": { "resource": false } }`,
},
{
filepath.Join(dir, "node_modules", "malformed", "tests", "malformed_test", "package.json"),
`{`,
},
}
for _, file := range files {
err := os.MkdirAll(filepath.Dir(file.path), 0o755)
require.NoError(t, err)
err = os.WriteFile(file.path, []byte(file.content), 0o600)
require.NoError(t, err)
}
// Add a symlink cycle in
err := os.Symlink(filepath.Join(dir, "node_modules"), filepath.Join(dir, "node_modules", "@node_modules"))
require.NoError(t, err)
host := &nodeLanguageHost{}
resp, err := host.GetRequiredPlugins(context.Background(), &pulumirpc.GetRequiredPluginsRequest{
Program: dir,
Info: &pulumirpc.ProgramInfo{
RootDirectory: dir,
ProgramDirectory: dir,
EntryPoint: ".",
},
})
require.NoError(t, err)
actual := make(map[string]string)
for _, plugin := range resp.GetPlugins() {
actual[plugin.Name] = plugin.Version
}
assert.Equal(t, map[string]string{
"foo": "v1.2.3",
"bar": "v4.5.6",
}, actual)
}
func TestGetRequiredPluginsSymlinkCycles2(t *testing.T) {
t.Parallel()
dir := filepath.Join(t.TempDir(), "testdir")
err := os.Mkdir(dir, 0o755)
require.NoError(t, err)
files := []struct {
path string
content string
}{
{
filepath.Join(dir, "node_modules", "@pulumi", "foo", "package.json"),
`{ "name": "@pulumi/foo", "version": "1.2.3", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "node_modules", "@pulumi", "bar", "package.json"),
`{ "name": "@pulumi/bar", "version": "4.5.6", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "node_modules", "@pulumi", "baz", "package.json"),
`{ "name": "@pulumi/baz", "version": "4.5.6", "pulumi": { "resource": false } }`,
},
{
filepath.Join(dir, "node_modules", "malformed", "tests", "malformed_test", "package.json"),
`{`,
},
}
for _, file := range files {
err := os.MkdirAll(filepath.Dir(file.path), 0o755)
require.NoError(t, err)
err = os.WriteFile(file.path, []byte(file.content), 0o600)
require.NoError(t, err)
}
// Add a symlink cycle in
err = os.Symlink(filepath.Join("..", ".."), filepath.Join(dir, "node_modules", "@node_modules"))
require.NoError(t, err)
host := &nodeLanguageHost{}
resp, err := host.GetRequiredPlugins(context.Background(), &pulumirpc.GetRequiredPluginsRequest{
Program: dir,
Info: &pulumirpc.ProgramInfo{
RootDirectory: dir,
ProgramDirectory: dir,
EntryPoint: ".",
},
})
require.NoError(t, err)
actual := make(map[string]string)
for _, plugin := range resp.GetPlugins() {
actual[plugin.Name] = plugin.Version
}
assert.Equal(t, map[string]string{
"foo": "v1.2.3",
"bar": "v4.5.6",
}, actual)
}
func TestGetRequiredPluginsNestedPolicyPack(t *testing.T) {
t.Parallel()
dir := filepath.Join(t.TempDir(), "testdir")
err := os.Mkdir(dir, 0o755)
require.NoError(t, err)
files := []struct {
path string
content string
}{
{
filepath.Join(dir, "node_modules", "@pulumi", "foo", "package.json"),
`{ "name": "@pulumi/foo", "version": "1.2.3", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "node_modules", "@pulumi", "bar", "package.json"),
`{ "name": "@pulumi/bar", "version": "4.5.6", "pulumi": { "resource": true } }`,
},
{
filepath.Join(dir, "policy", "PulumiPolicy.yaml"),
`name: my-policy`,
},
{
filepath.Join(dir, "policy", "node_modules", "@pulumi", "baz", "package.json"),
`{ "name": "@pulumi/baz", "version": "7.8.9", "pulumi": { "resource": true } }`,
},
}
for _, file := range files {
err := os.MkdirAll(filepath.Dir(file.path), 0o755)
require.NoError(t, err)
err = os.WriteFile(file.path, []byte(file.content), 0o600)
require.NoError(t, err)
}
host := &nodeLanguageHost{}
resp, err := host.GetRequiredPlugins(context.Background(), &pulumirpc.GetRequiredPluginsRequest{
Program: dir,
Info: &pulumirpc.ProgramInfo{
RootDirectory: dir,
ProgramDirectory: dir,
EntryPoint: ".",
},
})
require.NoError(t, err)
actual := make(map[string]string)
for _, plugin := range resp.GetPlugins() {
actual[plugin.Name] = plugin.Version
}
assert.Equal(t, map[string]string{
"foo": "v1.2.3",
"bar": "v4.5.6",
// baz: v7.8.9 is not included because it is in a nested policy pack
}, actual)
}
func TestParseOptions(t *testing.T) {
t.Parallel()
opts, err := parseOptions(nil)
require.NoError(t, err)
require.Equal(t, npm.AutoPackageManager, opts.packagemanager)
_, err = parseOptions(map[string]interface{}{
"typescript": 123,
})
require.ErrorContains(t, err, "typescript option must be a boolean")
_, err = parseOptions(map[string]interface{}{
"packagemanager": "poetry",
})
require.ErrorContains(t, err, "packagemanager option must be one of")
for _, tt := range []struct {
input string
expected npm.PackageManagerType
}{
{"auto", npm.AutoPackageManager},
{"npm", npm.NpmPackageManager},
{"yarn", npm.YarnPackageManager},
{"pnpm", npm.PnpmPackageManager},
} {
opts, err = parseOptions(map[string]interface{}{
"packagemanager": tt.input,
})
require.NoError(t, err)
require.Equal(t, tt.expected, opts.packagemanager)
}
}
// Nodejs sometimes sets stdout/stderr to non-blocking mode. When a nodejs subprocess is directly
// handed the go process's stdout/stderr file descriptors, nodejs's non-blocking configuration goes
// unnoticed by go, and a write from go can result in an error `write /dev/stdout: resource
// temporarily unavailable`. See runWithOutput for more details.
func TestNonblockingStdout(t *testing.T) {
// Regression test for https://github.com/pulumi/pulumi/issues/16503
t.Parallel()
script := `import os, time
os.set_blocking(1, False) # set stdout to non-blocking
time.sleep(3)
`
// Create a named pipe to use as stdout
tmp := os.TempDir()
p := filepath.Join(tmp, "fake-stdout")
err := syscall.Mkfifo(p, 0o644)
defer os.Remove(p)
require.NoError(t, err)
// Open fd without O_NONBLOCK, ensuring that os.NewFile does not return a pollable file.
// When our python script changes the file to non-blocking, Go does not notice and continues to
// expect the file to be blocking, and we can trigger the bug.
fd, err := syscall.Open(p, syscall.O_CREAT|syscall.O_RDWR, 0o644)
require.NoError(t, err)
fakeStdout := os.NewFile(uintptr(fd), p)
defer fakeStdout.Close()
require.NotNil(t, fakeStdout)
cmd := exec.Command("python3", "-c", script)
var done bool
go func() {
time.Sleep(2 * time.Second)
for !done {
s := "....................\n"
n, err := fakeStdout.Write([]byte(s))
require.NoError(t, err)
require.Equal(t, n, len(s))
}
}()
require.NoError(t, runWithOutput(cmd, fakeStdout, os.Stderr))
done = true
}
type slowWriter struct {
nWrites *int
}
func (s slowWriter) Write(b []byte) (int, error) {
time.Sleep(100 * time.Millisecond)
l := len(b)
*s.nWrites += l
return l, nil
}
func TestRunWithOutputDoesNotMissData(t *testing.T) {
// This test ensures that runWithOutput writes all the data from the command and does not miss
// any data that might be buffered when the command exits.
t.Parallel()
// Write `o` to stdout 100 times at 10 ms interval, followed by `x\n`
// Write `e` to stderr 100 times at 10 ms interval, followed by `x\n`
script := `let i = 0;
let interval = setInterval(() => {
process.stdout.write("o");
process.stderr.write("e");
i++;
if (i == 100) {
process.stdout.write("x\n");
process.stderr.write("x\n");
clearInterval(interval);
}
}, 10)
`
cmd := exec.Command("node", "-e", script)
stdout := slowWriter{nWrites: new(int)}
stderr := slowWriter{nWrites: new(int)}
require.NoError(t, runWithOutput(cmd, stdout, stderr))
require.Equal(t, 100+2 /* "x\n" */, *stdout.nWrites)
require.Equal(t, 100+2 /* "x\n" */, *stderr.nWrites)
}