pulumi/sdk/nodejs/npm/npm.go

90 lines
2.4 KiB
Go

package npm
import (
"bytes"
"context"
"errors"
"fmt"
"io"
"os"
"os/exec"
"path/filepath"
"strings"
)
// NPM is the canonical "Node Package Manager".
type npmManager struct {
executable string
}
// Assert that NPM is an instance of PackageManager.
var _ PackageManager = &npmManager{}
func newNPM() (*npmManager, error) {
npmPath, err := exec.LookPath("npm")
if err != nil {
if errors.Is(err, exec.ErrNotFound) {
return nil, errors.New("Could not find `npm` executable.\n" +
"Install npm and make sure it is in your PATH.")
}
return nil, err
}
return &npmManager{
executable: npmPath,
}, nil
}
func (node *npmManager) Name() string {
return "npm"
}
func (node *npmManager) Install(ctx context.Context, dir string, production bool, stdout, stderr io.Writer) error {
command := node.installCmd(ctx, production)
command.Dir = dir
command.Stdout = stdout
command.Stderr = stderr
return command.Run()
}
func (node *npmManager) installCmd(ctx context.Context, production bool) *exec.Cmd {
// We pass `--loglevel=error` to prevent `npm` from printing warnings about missing
// `description`, `repository`, and `license` fields in the package.json file.
args := []string{"install", "--loglevel=error"}
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, node.executable, args...)
}
func (node *npmManager) 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, node.executable, "pack", "--loglevel=error")
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 `npm 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("'npm pack' completed successfully but the package .tgz file was not generated: %w", err)
return nil, newErr
}
return packTarball, nil
}