pulumi/pkg/workspace/templates.go

432 lines
12 KiB
Go

// Copyright 2016-2018, Pulumi Corporation. All rights reserved.
package workspace
import (
"archive/tar"
"compress/gzip"
"fmt"
"io"
"io/ioutil"
"os"
"os/user"
"path/filepath"
"runtime"
"strings"
"gopkg.in/yaml.v2"
"github.com/pkg/errors"
"github.com/pulumi/pulumi/pkg/tokens"
"github.com/pulumi/pulumi/pkg/util/contract"
)
const (
defaultProjectName = "project"
defaultProjectDescription = "A Pulumi project."
// This file will be ignored when copying from the template cache to
// a project directory.
// It's not currently used, but the could be used in the future to contain
// metadata for the template, such as the description, for use offline.
pulumiTemplateManifestFile = ".pulumi.template.yaml"
)
// Template represents a project template.
type Template struct {
Name string `json:"name"`
Description string `json:"description"`
}
// templateManifest represents a template's manifest file.
type templateManifest struct {
Description string `yaml:"description"`
}
// LoadLocalTemplate returns a local template.
func LoadLocalTemplate(name string) (Template, error) {
template := Template{Name: name}
templateDir, err := GetTemplateDir(name)
if err != nil {
return Template{}, err
}
info, err := os.Stat(templateDir)
if err != nil {
return Template{}, err
}
if !info.IsDir() {
return Template{}, errors.Errorf("template '%s' in %s is not a directory", name, templateDir)
}
// Read the description from the manifest (if it exists).
manifest, err := readTemplateManifest(filepath.Join(templateDir, pulumiTemplateManifestFile))
if err != nil && !os.IsNotExist(err) {
return Template{}, err
} else if err == nil && manifest.Description != "" {
template.Description = manifest.Description
}
return template, nil
}
// ListLocalTemplates returns a list of local templates.
func ListLocalTemplates() ([]Template, error) {
templateDir, err := GetTemplateDir("")
if err != nil {
return nil, err
}
infos, err := ioutil.ReadDir(templateDir)
if err != nil {
return nil, err
}
var templates []Template
for _, info := range infos {
if info.IsDir() {
template, err := LoadLocalTemplate(info.Name())
if err != nil {
return nil, err
}
templates = append(templates, template)
}
}
return templates, nil
}
// InstallTemplate installs a template tarball into the local cache.
func InstallTemplate(name string, tarball io.ReadCloser) error {
contract.Require(name != "", "name")
contract.Require(tarball != nil, "tarball")
var templateDir string
var err error
// Get the template directory.
if templateDir, err = GetTemplateDir(name); err != nil {
return err
}
// Delete the directory if it exists.
if err = os.RemoveAll(templateDir); err != nil {
return errors.Wrapf(err, "removing existing template directory %s", templateDir)
}
// Ensure it exists since we may have just deleted it.
if err = os.MkdirAll(templateDir, 0700); err != nil {
return errors.Wrapf(err, "creating template directory %s", templateDir)
}
// Extract the tarball to its directory.
if err = extractTarball(tarball, templateDir); err != nil {
return errors.Wrapf(err, "extracting template to %s", templateDir)
}
// On Windows, we need to replace \n with \r\n. We'll just do this as a separate step.
if runtime.GOOS == "windows" {
if err = fixWindowsLineEndings(templateDir); err != nil {
return errors.Wrapf(err, "fixing line endings in %s", templateDir)
}
}
return nil
}
// CopyTemplateFilesDryRun does a dry run of copying a template to a destination directory,
// to ensure it won't overwrite any files.
func (template Template) CopyTemplateFilesDryRun(destDir string) error {
var err error
var sourceDir string
if sourceDir, err = GetTemplateDir(template.Name); err != nil {
return err
}
var existing []string
err = walkFiles(sourceDir, destDir, func(info os.FileInfo, source string, dest string) error {
if destInfo, statErr := os.Stat(dest); statErr == nil && !destInfo.IsDir() {
existing = append(existing, filepath.Base(dest))
}
return nil
})
if err != nil {
return err
}
if len(existing) > 0 {
return newExistingFilesError(existing)
}
return nil
}
// CopyTemplateFiles does the actual copy operation to a destination directory.
func (template Template) CopyTemplateFiles(
destDir string, force bool, projectName string, projectDescription string) error {
sourceDir, err := GetTemplateDir(template.Name)
if err != nil {
return err
}
return walkFiles(sourceDir, destDir, func(info os.FileInfo, source string, dest string) error {
if info.IsDir() {
// Create the destination directory.
return os.Mkdir(dest, 0700)
}
// Read the source file.
b, err := ioutil.ReadFile(source)
if err != nil {
return err
}
// We assume all template files are text files.
transformed := transform(string(b), projectName, projectDescription)
// Write to the destination file.
err = writeAllText(dest, transformed, force)
if err != nil {
// An existing file has shown up in between the dry run and the actual copy operation.
if os.IsExist(err) {
return newExistingFilesError([]string{filepath.Base(dest)})
}
}
return err
})
}
// GetTemplateDir returns the directory in which templates on the current machine are stored.
func GetTemplateDir(name string) (string, error) {
u, err := user.Current()
if u == nil || err != nil {
return "", errors.Wrap(err, "getting user home directory")
}
dir := filepath.Join(u.HomeDir, BookkeepingDir, TemplateDir)
if name != "" {
dir = filepath.Join(dir, name)
}
return dir, nil
}
// IsValidProjectName returns true if the project name is a valid name.
func IsValidProjectName(name string) bool {
return tokens.IsPackageName(name)
}
// ValueOrSanitizedDefaultProjectName returns the value or a sanitized valid project name
// based on defaultNameToSanitize.
func ValueOrSanitizedDefaultProjectName(name string, defaultNameToSanitize string) string {
if name != "" {
return name
}
return getValidProjectName(defaultNameToSanitize)
}
// ValueOrDefaultProjectDescription returns the value or defaultDescription.
func ValueOrDefaultProjectDescription(description string, defaultDescription string) string {
if description != "" {
return description
}
if defaultDescription != "" {
return defaultDescription
}
return defaultProjectDescription
}
// getValidProjectName returns a valid project name based on the passed-in name.
func getValidProjectName(name string) string {
// If the name is valid, return it.
if IsValidProjectName(name) {
return name
}
// Otherwise, try building-up the name, removing any invalid chars.
var result string
for i := 0; i < len(name); i++ {
temp := result + string(name[i])
if IsValidProjectName(temp) {
result = temp
}
}
// If we couldn't come up with a valid project name, fallback to a default.
if result == "" {
result = defaultProjectName
}
return result
}
// extractTarball extracts the tarball to the specified destination directory.
func extractTarball(tarball io.ReadCloser, destDir string) error {
// Unzip and untar the file as we go.
defer contract.IgnoreClose(tarball)
gzr, err := gzip.NewReader(tarball)
if err != nil {
return errors.Wrapf(err, "unzipping")
}
r := tar.NewReader(gzr)
for {
header, err := r.Next()
if err == io.EOF {
break
} else if err != nil {
return errors.Wrapf(err, "untarring")
}
path := filepath.Join(destDir, header.Name)
switch header.Typeflag {
case tar.TypeDir:
// Create any directories as needed.
if _, err := os.Stat(path); err != nil {
if err = os.MkdirAll(path, 0700); err != nil {
return errors.Wrapf(err, "untarring dir %s", path)
}
}
case tar.TypeReg:
// Expand files into the target directory.
dst, err := os.OpenFile(path, os.O_CREATE|os.O_RDWR, os.FileMode(header.Mode))
if err != nil {
return errors.Wrapf(err, "opening file %s for untar", path)
}
defer contract.IgnoreClose(dst)
if _, err = io.Copy(dst, r); err != nil {
return errors.Wrapf(err, "untarring file %s", path)
}
default:
return errors.Errorf("unexpected plugin file type %s (%v)", header.Name, header.Typeflag)
}
}
return nil
}
// walkFiles is a helper that walks the directories/files in a source directory
// and performs an action for each item.
func walkFiles(sourceDir string, destDir string,
actionFn func(info os.FileInfo, source string, dest string) error) error {
contract.Require(sourceDir != "", "sourceDir")
contract.Require(destDir != "", "destDir")
contract.Require(actionFn != nil, "actionFn")
infos, err := ioutil.ReadDir(sourceDir)
if err != nil {
return err
}
for _, info := range infos {
name := info.Name()
source := filepath.Join(sourceDir, name)
dest := filepath.Join(destDir, name)
if info.IsDir() {
if err := actionFn(info, source, dest); err != nil {
return err
}
if err := walkFiles(source, dest, actionFn); err != nil {
return err
}
} else {
// Ignore template manifest file.
if name == pulumiTemplateManifestFile {
continue
}
if err := actionFn(info, source, dest); err != nil {
return err
}
}
}
return nil
}
// readTemplateManifest reads a template manifest file.
func readTemplateManifest(filename string) (templateManifest, error) {
b, err := ioutil.ReadFile(filename)
if err != nil {
return templateManifest{}, err
}
var manifest templateManifest
err = yaml.Unmarshal(b, &manifest)
if err != nil {
return templateManifest{}, err
}
return manifest, nil
}
// newExistingFilesError returns a new error from a list of existing file names
// that would be overwritten.
func newExistingFilesError(existing []string) error {
contract.Assert(len(existing) > 0)
message := "creating this template will make changes to existing files:\n"
for _, file := range existing {
message = message + fmt.Sprintf(" overwrite %s\n", file)
}
message = message + "\nrerun the command and pass --force to accept and create"
return errors.New(message)
}
// transform returns a new string with ${PROJECT} and ${DESCRIPTION} replaced by
// the value of projectName and projectDescription.
func transform(content string, projectName string, projectDescription string) string {
content = strings.Replace(content, "${PROJECT}", projectName, -1)
content = strings.Replace(content, "${DESCRIPTION}", projectDescription, -1)
return content
}
// writeAllText writes all the text to the specified file, with an option to overwrite.
func writeAllText(filename string, text string, overwrite bool) error {
flag := os.O_WRONLY | os.O_CREATE
if overwrite {
flag = flag | os.O_TRUNC
} else {
flag = flag | os.O_EXCL
}
f, err := os.OpenFile(filename, flag, 0600)
if err != nil {
return err
}
defer contract.IgnoreClose(f)
_, err = f.WriteString(text)
return err
}
// fixWindowsLineEndings will go through the sourceDir, read each file, replace \n with \r\n,
// and save the changes.
// It'd be more efficient to do this during tarball extraction, but this is sufficient for now.
func fixWindowsLineEndings(sourceDir string) error {
return walkFiles(sourceDir, sourceDir, func(info os.FileInfo, source string, dest string) error {
// Skip directories.
if info.IsDir() {
return nil
}
// Read the source file.
b, err := ioutil.ReadFile(source)
if err != nil {
return err
}
// We assume all template files are text files.
content := string(b)
content = strings.Replace(content, "\n", "\r\n", -1)
// Write to the destination file.
err = writeAllText(dest, content, true /*overwrite*/)
if err != nil {
// An existing file has shown up in between the dry run and the actual copy operation.
if os.IsExist(err) {
return newExistingFilesError([]string{filepath.Base(dest)})
}
}
return err
})
}