mirror of https://github.com/pulumi/pulumi.git
250 lines
7.8 KiB
Go
250 lines
7.8 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 (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/fsutil"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/logging"
|
|
)
|
|
|
|
type typeChecker int
|
|
|
|
const (
|
|
// TypeCheckerNone is the default typeChecker
|
|
TypeCheckerNone typeChecker = iota
|
|
// TypeCheckerMypy is the mypy typeChecker
|
|
TypeCheckerMypy
|
|
// TypeCheckerPyright is the pyright typeChecker
|
|
TypeCheckerPyright
|
|
)
|
|
|
|
type toolchain int
|
|
|
|
const (
|
|
Pip toolchain = iota
|
|
Poetry
|
|
Uv
|
|
)
|
|
|
|
type PythonOptions struct {
|
|
// The root directory of the project.
|
|
Root string
|
|
// The program directory of the project.
|
|
ProgramDir string
|
|
// Virtual environment to use, relative to `Root`.
|
|
Virtualenv string
|
|
// Use a typechecker to type check.
|
|
Typechecker typeChecker
|
|
// The package manager to use for managing dependencies.
|
|
Toolchain toolchain
|
|
}
|
|
|
|
type PythonPackage struct {
|
|
Name string `json:"name"`
|
|
Version string `json:"version"`
|
|
Location string `json:"location"`
|
|
}
|
|
|
|
type Info struct {
|
|
Executable string
|
|
Version string
|
|
}
|
|
|
|
type Toolchain interface {
|
|
// InstallDependencies installs the dependencies of the project found in `cwd`.
|
|
InstallDependencies(ctx context.Context, cwd string, useLanguageVersionTools,
|
|
showOutput bool, infoWriter, errorWriter io.Writer) error
|
|
// EnsureVenv validates virtual environment of the toolchain and creates it if it doesn't exist.
|
|
EnsureVenv(ctx context.Context, cwd string, useLanguageVersionTools,
|
|
showOutput bool, infoWriter, errorWriter io.Writer) error
|
|
// ValidateVenv checks if the virtual environment of the toolchain is valid.
|
|
ValidateVenv(ctx context.Context) error
|
|
// ListPackages returns a list of Python packages installed in the toolchain.
|
|
ListPackages(ctx context.Context, transitive bool) ([]PythonPackage, error)
|
|
// Command returns an *exec.Cmd for running `python` using the configured toolchain.
|
|
Command(ctx context.Context, args ...string) (*exec.Cmd, error)
|
|
// ModuleCommand returns an *exec.Cmd for running an installed python module using the configured toolchain.
|
|
// https://docs.python.org/3/using/cmdline.html#cmdoption-m
|
|
ModuleCommand(ctx context.Context, module string, args ...string) (*exec.Cmd, error)
|
|
// About returns information about the python executable of the toolchain.
|
|
About(ctx context.Context) (Info, error)
|
|
}
|
|
|
|
func Name(tc toolchain) string {
|
|
switch tc {
|
|
case Pip:
|
|
return "Pip"
|
|
case Poetry:
|
|
return "Poetry"
|
|
case Uv:
|
|
return "Uv"
|
|
default:
|
|
return "Unknown"
|
|
}
|
|
}
|
|
|
|
func ResolveToolchain(options PythonOptions) (Toolchain, error) {
|
|
if options.Toolchain == Poetry {
|
|
dir := options.ProgramDir
|
|
if dir == "" {
|
|
dir = options.Root
|
|
}
|
|
return newPoetry(dir)
|
|
} else if options.Toolchain == Uv {
|
|
return newUv(options.Root, options.Virtualenv)
|
|
}
|
|
return newPip(options.Root, options.Virtualenv)
|
|
}
|
|
|
|
// ActivateVirtualEnv takes an array of environment variables (same format as os.Environ()) and path to
|
|
// a virtual environment directory, and returns a new "activated" array with the virtual environment's
|
|
// "bin" dir ("Scripts" on Windows) prepended to the `PATH` environment variable, the `VIRTUAL_ENV`
|
|
// variable set to the path, and the `PYTHONHOME` variable removed.
|
|
func ActivateVirtualEnv(environ []string, virtualEnvDir string) []string {
|
|
virtualEnvBin := filepath.Join(virtualEnvDir, virtualEnvBinDirName())
|
|
var hasPath bool
|
|
var result []string
|
|
for _, env := range environ {
|
|
split := strings.SplitN(env, "=", 2)
|
|
contract.Assertf(len(split) == 2, "unexpected environment variable: %q", env)
|
|
key, value := split[0], split[1]
|
|
|
|
// Case-insensitive compare, as Windows will normally be "Path", not "PATH".
|
|
if strings.EqualFold(key, "PATH") {
|
|
hasPath = true
|
|
// Prepend the virtual environment bin directory to PATH so any calls to run
|
|
// python or pip will use the binaries in the virtual environment.
|
|
path := fmt.Sprintf("%s=%s%s%s", key, virtualEnvBin, string(os.PathListSeparator), value)
|
|
result = append(result, path)
|
|
} else if strings.EqualFold(key, "PYTHONHOME") {
|
|
// Skip PYTHONHOME to "unset" this value.
|
|
} else if strings.EqualFold(key, "VIRTUAL_ENV") {
|
|
// Skip VIRTUAL_ENV, we always set this to `virtualEnvDir`
|
|
} else {
|
|
result = append(result, env)
|
|
}
|
|
}
|
|
if !hasPath {
|
|
path := "PATH=" + virtualEnvBin
|
|
result = append(result, path)
|
|
}
|
|
virtualEnv := "VIRTUAL_ENV=" + virtualEnvDir
|
|
result = append(result, virtualEnv)
|
|
return result
|
|
}
|
|
|
|
func virtualEnvBinDirName() string {
|
|
if runtime.GOOS == windows {
|
|
return "Scripts"
|
|
}
|
|
return "bin"
|
|
}
|
|
|
|
// Determines if we should use pyenv. To use pyenv we need:
|
|
// - pyenv installed
|
|
// - .python-version file in the current directory or any of its parents
|
|
func usePyenv(cwd string) (bool, string, string, error) {
|
|
versionFile, err := fsutil.Searchup(cwd, ".python-version")
|
|
if err != nil {
|
|
if !errors.Is(err, fsutil.ErrNotFound) {
|
|
return false, "", "", fmt.Errorf("error while looking for .python-version %s", err)
|
|
}
|
|
// No .python-version file found
|
|
return false, "", "", nil
|
|
}
|
|
logging.V(9).Infof("Python toolchain: found .python-version %s", versionFile)
|
|
pyenvPath, err := exec.LookPath("pyenv")
|
|
if err != nil {
|
|
if !errors.Is(err, exec.ErrNotFound) {
|
|
return false, "", "", fmt.Errorf("error while looking for pyenv %+v", err)
|
|
}
|
|
// No pyenv installed
|
|
logging.V(9).Infof("Python toolchain: found .python-version file at %s, but could not find pyenv executable",
|
|
versionFile)
|
|
return false, "", "", nil
|
|
}
|
|
return true, pyenvPath, versionFile, nil
|
|
}
|
|
|
|
func installPython(ctx context.Context, cwd string, showOutput bool, infoWriter, errorWriter io.Writer) error {
|
|
use, pyenv, versionFile, err := usePyenv(cwd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !use {
|
|
return nil
|
|
}
|
|
|
|
if showOutput {
|
|
_, err := infoWriter.Write([]byte(fmt.Sprintf("Installing python version from .python-version file at %s\n",
|
|
versionFile)))
|
|
if err != nil {
|
|
return fmt.Errorf("error while writing to infoWriter %s", err)
|
|
}
|
|
}
|
|
cmd := exec.CommandContext(ctx, pyenv, "install", "--skip-existing")
|
|
cmd.Dir = cwd
|
|
if showOutput {
|
|
cmd.Stdout = infoWriter
|
|
cmd.Stderr = errorWriter
|
|
}
|
|
err = cmd.Run()
|
|
if err != nil {
|
|
var exitErr *exec.ExitError
|
|
if errors.As(err, &exitErr) {
|
|
return fmt.Errorf("error while running pyenv install: %s", string(exitErr.Stderr))
|
|
}
|
|
return fmt.Errorf("error while running pyenv install: %s", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func searchup(currentDir, fileToFind string) (string, error) {
|
|
if _, err := os.Stat(filepath.Join(currentDir, fileToFind)); err == nil {
|
|
return currentDir, nil
|
|
} else if !errors.Is(err, os.ErrNotExist) {
|
|
return "", err
|
|
}
|
|
parentDir := filepath.Dir(currentDir)
|
|
if currentDir == parentDir {
|
|
// Reached the root directory, file not found
|
|
return "", os.ErrNotExist
|
|
}
|
|
return searchup(parentDir, fileToFind)
|
|
}
|
|
|
|
// errorWithStderr returns an error that includes the stderr output if the error is an ExitError.
|
|
func errorWithStderr(err error, message string) error {
|
|
var exitErr *exec.ExitError
|
|
if errors.As(err, &exitErr) {
|
|
stderr := strings.TrimSpace(string(exitErr.Stderr))
|
|
if len(stderr) > 0 {
|
|
return fmt.Errorf("%s: %w: %s", message, exitErr, exitErr.Stderr)
|
|
}
|
|
}
|
|
return fmt.Errorf("%s: %w", message, err)
|
|
}
|