pulumi/sdk/nodejs/npm/pnpm.go

104 lines
3.0 KiB
Go

// Copyright 2016-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 npm
import (
"bytes"
"context"
"fmt"
"io"
"os"
"os/exec"
"path/filepath"
"strings"
)
// pnpm is an alternative package manager for Node.js.
type pnpmManager struct {
executable string
}
// Assert that pnpm is an instance of PackageManager.
var _ PackageManager = &pnpmManager{}
func newPnpm() (*pnpmManager, error) {
pnpmPath, err := exec.LookPath("pnpm")
instance := &pnpmManager{
executable: pnpmPath,
}
return instance, err
}
func (pnpm *pnpmManager) Name() string {
return "pnpm"
}
func (pnpm *pnpmManager) Install(ctx context.Context, dir string, production bool, stdout, stderr io.Writer) error {
command := pnpm.installCmd(ctx, production)
command.Dir = dir
command.Stdout = stdout
command.Stderr = stderr
return command.Run()
}
func (pnpm *pnpmManager) installCmd(ctx context.Context, production bool) *exec.Cmd {
args := []string{"install", "--use-stderr"}
if production {
args = append(args, "--production")
}
//nolint:gosec // False positive on tained command execution. We aren't accepting input from the user here.
return exec.CommandContext(ctx, pnpm.executable, args...)
}
func (pnpm *pnpmManager) Pack(ctx context.Context, dir string, stderr io.Writer) ([]byte, error) {
//nolint:gosec // False positive on tained command execution. We aren't accepting input from the user here.
command := exec.CommandContext(ctx, pnpm.executable, "pack", "--use-stderr")
command.Dir = dir
// We have to read the name of the file from stdout.
var stdout bytes.Buffer
command.Stdout = &stdout
command.Stderr = stderr
err := command.Run()
if err != nil {
return nil, err
}
// Next, we try to read the name of the file from stdout.
// packfile is the name of the file containing the tarball,
// as produced by `pnpm pack`.
packFilename := strings.TrimSpace(stdout.String())
packfile := filepath.Join(dir, packFilename)
defer os.Remove(packfile)
packTarball, err := os.ReadFile(packfile)
if err != nil {
newErr := fmt.Errorf("'pnpm pack' completed successfully but the package .tgz file was not generated: %w", err)
return nil, newErr
}
return packTarball, nil
}
// checkPnpmLock checks if there's a file named pnpm-lock.yaml in pwd.
// This function is used to indicate whether to prefer pnpm over
// other package managers.
func checkPnpmLock(pwd string) bool {
pnpmFile := filepath.Join(pwd, "pnpm-lock.yaml")
_, err := os.Stat(pnpmFile)
return err == nil
}