mirror of https://github.com/pulumi/pulumi.git
505 lines
14 KiB
Go
505 lines
14 KiB
Go
// Copyright 2024, 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 toolchain
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"runtime"
|
|
"sort"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestValidateVenv(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, opts := range []PythonOptions{
|
|
{
|
|
Toolchain: Pip,
|
|
Virtualenv: "venv",
|
|
},
|
|
{
|
|
Toolchain: Poetry,
|
|
},
|
|
} {
|
|
opts := opts
|
|
t.Run("Doesnt-exist-"+Name(opts.Toolchain), func(t *testing.T) {
|
|
t.Parallel()
|
|
opts := copyOptions(opts)
|
|
|
|
opts.Root = t.TempDir()
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
|
|
err = tc.ValidateVenv(context.Background())
|
|
require.Error(t, err)
|
|
})
|
|
t.Run("Exists-"+Name(opts.Toolchain), func(t *testing.T) {
|
|
t.Parallel()
|
|
opts := copyOptions(opts)
|
|
opts.Root = t.TempDir()
|
|
createVenv(t, opts)
|
|
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
err = tc.InstallDependencies(context.Background(), opts.Root, false, true, os.Stdout, os.Stderr)
|
|
require.NoError(t, err)
|
|
err = tc.ValidateVenv(context.Background())
|
|
require.NoError(t, err)
|
|
})
|
|
}
|
|
}
|
|
|
|
//nolint:paralleltest // modifies environment variables
|
|
func TestCommand(t *testing.T) {
|
|
// Poetry with `in-project = true` uses `.venv` as the default virtualenv directory.
|
|
// Use the same for pip to keep the tests consistent.
|
|
venvDir := ".venv"
|
|
|
|
for _, opts := range []PythonOptions{
|
|
{
|
|
Toolchain: Pip,
|
|
Virtualenv: venvDir,
|
|
},
|
|
{
|
|
Toolchain: Poetry,
|
|
},
|
|
} {
|
|
opts := opts
|
|
t.Run("empty/"+Name(opts.Toolchain), func(t *testing.T) {
|
|
opts := copyOptions(opts)
|
|
opts.Root = t.TempDir()
|
|
createVenv(t, opts)
|
|
|
|
t.Setenv("MY_ENV_VAR", "HELLO")
|
|
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
|
|
cmd, err := tc.Command(context.Background())
|
|
require.NoError(t, err)
|
|
|
|
var venvBin string
|
|
if runtime.GOOS == windows {
|
|
venvBin = filepath.Join(opts.Root, venvDir, "Scripts")
|
|
require.Equal(t, filepath.Join("python.exe"), cmd.Path)
|
|
} else {
|
|
venvBin = filepath.Join(opts.Root, venvDir, "bin")
|
|
cmdPath := normalizePath(t, cmd.Path)
|
|
require.Equal(t, filepath.Join(normalizePath(t, venvBin), "python"), cmdPath)
|
|
}
|
|
|
|
foundPath := false
|
|
foundMyEnvVar := false
|
|
for _, env := range cmd.Env {
|
|
if strings.HasPrefix(env, "PATH=") {
|
|
require.Contains(t, env, venvBin, "venv binary directory should in PATH")
|
|
foundPath = true
|
|
}
|
|
if strings.HasPrefix(env, "MY_ENV_VAR=") {
|
|
require.Equal(t, "MY_ENV_VAR=HELLO", env, "Env variables should be passed through")
|
|
foundMyEnvVar = true
|
|
}
|
|
}
|
|
require.True(t, foundPath, "PATH was not found in cmd.Env")
|
|
require.True(t, foundMyEnvVar, "MY_ENV_VAR was not found in cmd.Env")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestListPackages(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, test := range []struct {
|
|
opts PythonOptions
|
|
expectedPackages []string
|
|
}{
|
|
{
|
|
opts: PythonOptions{
|
|
Toolchain: Pip,
|
|
Virtualenv: "venv",
|
|
},
|
|
// The virtualenv created by the pip toolchain always has pip
|
|
// installed. Additionally it always installs setuptools and wheel
|
|
// into the virtualenv.
|
|
expectedPackages: []string{"pip", "setuptools", "wheel"},
|
|
},
|
|
{
|
|
opts: PythonOptions{
|
|
Toolchain: Poetry,
|
|
},
|
|
// Virtual environments created by Poetry always include pip.
|
|
expectedPackages: []string{"pip"},
|
|
},
|
|
{
|
|
opts: PythonOptions{
|
|
Toolchain: Uv,
|
|
},
|
|
// Virtual environments created by uv are empty.
|
|
expectedPackages: []string{},
|
|
},
|
|
} {
|
|
test := test
|
|
|
|
t.Run("empty/"+Name(test.opts.Toolchain), func(t *testing.T) {
|
|
t.Parallel()
|
|
opts := copyOptions(test.opts)
|
|
opts.Root = t.TempDir()
|
|
createVenv(t, opts)
|
|
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
|
|
packages, err := tc.ListPackages(context.Background(), false)
|
|
require.NoError(t, err)
|
|
require.Len(t, packages, len(test.expectedPackages))
|
|
for i, pkg := range test.expectedPackages {
|
|
require.Equal(t, pkg, packages[i].Name)
|
|
}
|
|
})
|
|
|
|
t.Run("non-empty/"+Name(test.opts.Toolchain), func(t *testing.T) {
|
|
t.Parallel()
|
|
opts := copyOptions(test.opts)
|
|
opts.Root = t.TempDir()
|
|
createVenv(t, opts, "pulumi-random")
|
|
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
|
|
packages, err := tc.ListPackages(context.Background(), false)
|
|
require.NoError(t, err)
|
|
sort.Slice(packages, func(i, j int) bool {
|
|
return packages[i].Name < packages[j].Name
|
|
})
|
|
// pulumi_random depends on pulumi, which has pip as a dependency.
|
|
// We are looking for packages that are not dependencies of other
|
|
// packages, so we have to exclude pip here.
|
|
var expectedPackages []string
|
|
for _, pkg := range append([]string{"pulumi_random"}, test.expectedPackages...) {
|
|
if pkg != "pip" || opts.Toolchain == Poetry {
|
|
expectedPackages = append(expectedPackages, pkg)
|
|
}
|
|
}
|
|
sort.Strings(expectedPackages)
|
|
|
|
require.Len(t, packages, len(expectedPackages))
|
|
for i, pkg := range expectedPackages {
|
|
require.Equal(t, pkg, packages[i].Name)
|
|
}
|
|
})
|
|
|
|
t.Run("non-empty-with-pip/"+Name(test.opts.Toolchain), func(t *testing.T) {
|
|
t.Parallel()
|
|
opts := copyOptions(test.opts)
|
|
opts.Root = t.TempDir()
|
|
createVenv(t, opts, "pulumi-random", "pip")
|
|
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
|
|
packages, err := tc.ListPackages(context.Background(), false)
|
|
require.NoError(t, err)
|
|
sort.Slice(packages, func(i, j int) bool {
|
|
return packages[i].Name < packages[j].Name
|
|
})
|
|
|
|
// pulumi_random depends on pulumi, which has pip as a dependency.
|
|
// We are looking for packages that are not dependencies of other
|
|
// packages, so we have to exclude pip here.
|
|
var expectedPackages []string
|
|
for _, pkg := range append([]string{"pulumi_random"}, test.expectedPackages...) {
|
|
if pkg != "pip" || opts.Toolchain == Poetry {
|
|
expectedPackages = append(expectedPackages, pkg)
|
|
}
|
|
}
|
|
expectedPackages = unique(expectedPackages)
|
|
sort.Strings(expectedPackages)
|
|
|
|
require.Len(t, packages, len(expectedPackages))
|
|
for i, pkg := range expectedPackages {
|
|
require.Equal(t, pkg, packages[i].Name)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestAbout(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, opts := range []PythonOptions{
|
|
{
|
|
Toolchain: Pip,
|
|
Virtualenv: "venv",
|
|
},
|
|
{
|
|
Toolchain: Poetry,
|
|
},
|
|
} {
|
|
opts := opts
|
|
t.Run(Name(opts.Toolchain), func(t *testing.T) {
|
|
t.Parallel()
|
|
opts := copyOptions(opts)
|
|
opts.Root = t.TempDir()
|
|
createVenv(t, opts)
|
|
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
info, err := tc.About(context.Background())
|
|
require.NoError(t, err)
|
|
require.Regexp(t, "[0-9]+\\.[0-9]+\\.[0-9]+", info.Version)
|
|
require.Regexp(t, "python$", info.Executable)
|
|
})
|
|
}
|
|
}
|
|
|
|
//nolint:paralleltest // mutates environment variables
|
|
func TestPyenv(t *testing.T) {
|
|
if runtime.GOOS == windows {
|
|
t.Skip("pyenv is not supported on Windows")
|
|
}
|
|
tmpDir := t.TempDir()
|
|
|
|
// Test without pyenv, a .python-version file
|
|
use, _, _, err := usePyenv(tmpDir)
|
|
require.NoError(t, err)
|
|
require.False(t, use)
|
|
|
|
// Add a fake pyenv binary to $tmp/bin and set $PATH to $tmp/bin
|
|
require.NoError(t, os.MkdirAll(filepath.Join(tmpDir, "bin"), 0o755))
|
|
//nolint:gosec // we want this file to be executable
|
|
require.NoError(t, os.WriteFile(filepath.Join(tmpDir, "bin", "pyenv"), []byte("#!/bin/sh\nexit 0;\n"), 0o700))
|
|
t.Setenv("PATH", filepath.Join(tmpDir, "bin"))
|
|
|
|
// Test witbout .python-version file
|
|
use, _, _, err = usePyenv(tmpDir)
|
|
require.NoError(t, err)
|
|
require.False(t, use)
|
|
|
|
// Create a .python-version file
|
|
require.NoError(t, os.WriteFile(filepath.Join(tmpDir, ".python-version"), []byte("3.9.0"), 0o600))
|
|
|
|
use, pyenvPath, versionFile, err := usePyenv(tmpDir)
|
|
t.Log("X", use, pyenvPath, versionFile, err)
|
|
require.NoError(t, err)
|
|
require.True(t, use)
|
|
require.Equal(t, filepath.Join(tmpDir, ".python-version"), versionFile)
|
|
require.Equal(t, filepath.Join(tmpDir, "bin", "pyenv"), pyenvPath)
|
|
}
|
|
|
|
//nolint:paralleltest // mutates environment variables
|
|
func TestPyenvInstall(t *testing.T) {
|
|
if runtime.GOOS == windows {
|
|
t.Skip("pyenv is not supported on Windows")
|
|
}
|
|
tmpDir := t.TempDir()
|
|
|
|
t.Log("tmpDir", tmpDir)
|
|
|
|
// Add a fake pyenv binary to $tmp/bin and set $PATH to $tmp/bin.
|
|
// The binary will write its arguments to a file, that we read back later to verify that it was called with the
|
|
// correct arguments.
|
|
require.NoError(t, os.MkdirAll(filepath.Join(tmpDir, "bin"), 0o755))
|
|
outPath := filepath.Join(tmpDir, "out.txt")
|
|
script := fmt.Sprintf("#!/bin/sh\necho $@ > %s\n", outPath)
|
|
//nolint:gosec // we want this file to be executable
|
|
require.NoError(t, os.WriteFile(filepath.Join(tmpDir, "bin", "pyenv"), []byte(script), 0o700))
|
|
t.Setenv("PATH", filepath.Join(tmpDir, "bin"))
|
|
|
|
// Create a .python-version file
|
|
require.NoError(t, os.WriteFile(filepath.Join(tmpDir, ".python-version"), []byte("3.9.0"), 0o600))
|
|
|
|
stdout := &bytes.Buffer{}
|
|
stderr := &bytes.Buffer{}
|
|
err := installPython(context.Background(), tmpDir, false, stdout, stderr)
|
|
require.NoError(t, err)
|
|
|
|
b, err := os.ReadFile(outPath)
|
|
require.NoError(t, err)
|
|
require.Equal(t, "install --skip-existing", strings.TrimSpace(string(b)))
|
|
}
|
|
|
|
func createVenv(t *testing.T, opts PythonOptions, packages ...string) {
|
|
t.Helper()
|
|
|
|
if opts.Toolchain == Pip {
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
err = tc.InstallDependencies(context.Background(), opts.Root, false, /*useLanguageVersionTools*/
|
|
true /*showOutput */, os.Stdout, os.Stderr)
|
|
require.NoError(t, err)
|
|
|
|
for _, pkg := range packages {
|
|
cmd, err := tc.Command(context.Background(), "-m", "pip", "install", pkg)
|
|
require.NoError(t, err)
|
|
require.NoError(t, cmd.Run())
|
|
}
|
|
} else if opts.Toolchain == Poetry {
|
|
writePyprojectForPoetry(t, opts.Root)
|
|
// Write poetry.toml file to enable in-project virtualenvs. This ensures we delete the
|
|
// virtualenv with the tmp directory after the test is done.
|
|
writePoetryToml(t, opts.Root)
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
err = tc.InstallDependencies(context.Background(), opts.Root, false, /*useLanguageVersionTools*/
|
|
true /*showOutput */, os.Stdout, os.Stderr)
|
|
require.NoError(t, err)
|
|
|
|
for _, pkg := range packages {
|
|
cmd := exec.Command("poetry", "add", pkg)
|
|
cmd.Dir = opts.Root
|
|
out, err := cmd.CombinedOutput()
|
|
require.NoError(t, err, string(out))
|
|
}
|
|
} else if opts.Toolchain == Uv {
|
|
writePyprojectForUv(t, opts.Root)
|
|
tc, err := ResolveToolchain(opts)
|
|
require.NoError(t, err)
|
|
err = tc.InstallDependencies(context.Background(), opts.Root, false, /*useLanguageVersionTools*/
|
|
true /*showOutput */, os.Stdout, os.Stderr)
|
|
require.NoError(t, err)
|
|
|
|
for _, pkg := range packages {
|
|
cmd := exec.Command("uv", "add", pkg)
|
|
cmd.Dir = opts.Root
|
|
out, err := cmd.CombinedOutput()
|
|
require.NoError(t, err, string(out))
|
|
}
|
|
}
|
|
}
|
|
|
|
func writePyprojectForUv(t *testing.T, root string) {
|
|
t.Helper()
|
|
|
|
f, err := os.OpenFile(filepath.Join(root, "pyproject.toml"), os.O_CREATE|os.O_WRONLY, 0o600)
|
|
require.NoError(t, err)
|
|
fmt.Fprint(f, `
|
|
[project]
|
|
name = "list-packages-test"
|
|
version = "0.0.1"
|
|
requires-python = ">=3.8"
|
|
dependencies = []
|
|
`)
|
|
err = f.Close()
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func writePyprojectForPoetry(t *testing.T, root string) {
|
|
t.Helper()
|
|
|
|
f, err := os.OpenFile(filepath.Join(root, "pyproject.toml"), os.O_CREATE|os.O_WRONLY, 0o600)
|
|
require.NoError(t, err)
|
|
fmt.Fprint(f, `
|
|
[build-system]
|
|
requires = ["poetry-core"]
|
|
build-backend = "poetry.core.masonry.api"
|
|
|
|
[tool.poetry]
|
|
name = "test_pulumi_venv"
|
|
version = "0.1.0"
|
|
description = ""
|
|
authors = []
|
|
readme = "README.md"
|
|
package-mode = false
|
|
packages = [{include = "test_pulumi_venv"}]
|
|
|
|
[tool.poetry.dependencies]
|
|
python = "^3.8"
|
|
`)
|
|
err = f.Close()
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func writePoetryToml(t *testing.T, path string) {
|
|
t.Helper()
|
|
|
|
f, err := os.OpenFile(filepath.Join(path, "poetry.toml"), os.O_CREATE|os.O_WRONLY, 0o600)
|
|
require.NoError(t, err)
|
|
fmt.Fprint(f, `[virtualenvs]
|
|
in-project = true
|
|
[virtualenvs.options]
|
|
no-setuptools = true
|
|
`)
|
|
err = f.Close()
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func copyOptions(opts PythonOptions) PythonOptions {
|
|
return PythonOptions{
|
|
Root: opts.Root,
|
|
ProgramDir: opts.ProgramDir,
|
|
Virtualenv: opts.Virtualenv,
|
|
Typechecker: opts.Typechecker,
|
|
Toolchain: opts.Toolchain,
|
|
}
|
|
}
|
|
|
|
func unique(s []string) []string {
|
|
u := make([]string, 0, len(s))
|
|
m := make(map[string]bool)
|
|
for _, val := range s {
|
|
if _, ok := m[val]; !ok {
|
|
m[val] = true
|
|
u = append(u, val)
|
|
}
|
|
}
|
|
return u
|
|
}
|
|
|
|
// normalizePath resolves symlinks within the directory part of the given path.
|
|
// This helps avoid test issues on macOS where for example /var -> /private/var.
|
|
// Importantly, we do not evaluate the symlink of the whole path, as that would
|
|
// resolve the python binary to the system python, since within in a virtualenv
|
|
// bin/python points to the system python.
|
|
func normalizePath(t *testing.T, path string) string {
|
|
t.Helper()
|
|
dir, bin := filepath.Split(path)
|
|
normDir, err := filepath.EvalSymlinks(dir)
|
|
require.NoError(t, err)
|
|
return filepath.Join(normDir, bin)
|
|
}
|
|
|
|
type ProcessState struct{}
|
|
|
|
func (p *ProcessState) Pid() int {
|
|
return 123
|
|
}
|
|
|
|
func (p *ProcessState) String() string {
|
|
return "exit status 139 "
|
|
}
|
|
|
|
func TestErrorWithStderr(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
err := errors.New("error")
|
|
require.Equal(t, "the error message: error", errorWithStderr(err, "the error message").Error())
|
|
|
|
exitErr := &exec.ExitError{ProcessState: &os.ProcessState{}, Stderr: []byte("command said something")}
|
|
require.Equal(t, "the error message: exit status 0: command said something",
|
|
errorWithStderr(exitErr, "the error message").Error())
|
|
|
|
exitErrNoStderr := &exec.ExitError{ProcessState: &os.ProcessState{}}
|
|
require.Equal(t, "the error message: exit status 0", errorWithStderr(exitErrNoStderr, "the error message").Error())
|
|
}
|