mirror of https://github.com/pulumi/pulumi.git
260 lines
9.6 KiB
Go
260 lines
9.6 KiB
Go
// Copyright 2016-2018, 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 backend
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/display"
|
|
"github.com/pulumi/pulumi/pkg/v3/operations"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy"
|
|
"github.com/pulumi/pulumi/pkg/v3/secrets"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/apitype"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/config"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/gitutil"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/result"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
|
|
)
|
|
|
|
// Stack is used to manage stacks of resources against a pluggable backend.
|
|
type Stack interface {
|
|
// this stack's identity.
|
|
Ref() StackReference
|
|
// the latest deployment snapshot.
|
|
Snapshot(ctx context.Context, secretsProvider secrets.Provider) (*deploy.Snapshot, error)
|
|
// the backend this stack belongs to.
|
|
Backend() Backend
|
|
// the stack's existing tags.
|
|
Tags() map[apitype.StackTagName]string
|
|
// Preview changes to this stack.
|
|
Preview(ctx context.Context, op UpdateOperation) (*deploy.Plan, display.ResourceChanges, result.Result)
|
|
// Update this stack.
|
|
Update(ctx context.Context, op UpdateOperation) (display.ResourceChanges, result.Result)
|
|
// Import resources into this stack.
|
|
Import(ctx context.Context, op UpdateOperation, imports []deploy.Import) (display.ResourceChanges, result.Result)
|
|
// Refresh this stack's state from the cloud provider.
|
|
Refresh(ctx context.Context, op UpdateOperation) (display.ResourceChanges, result.Result)
|
|
// Destroy this stack's resources.
|
|
Destroy(ctx context.Context, op UpdateOperation) (display.ResourceChanges, result.Result)
|
|
// Watch this stack.
|
|
Watch(ctx context.Context, op UpdateOperation, paths []string) result.Result
|
|
|
|
// remove this stack.
|
|
Remove(ctx context.Context, force bool) (bool, error)
|
|
// rename this stack.
|
|
Rename(ctx context.Context, newName tokens.QName) (StackReference, error)
|
|
// list log entries for this stack.
|
|
GetLogs(ctx context.Context, secretsProvider secrets.Provider,
|
|
cfg StackConfiguration, query operations.LogQuery) ([]operations.LogEntry, error)
|
|
// export this stack's deployment.
|
|
ExportDeployment(ctx context.Context) (*apitype.UntypedDeployment, error)
|
|
// import the given deployment into this stack.
|
|
ImportDeployment(ctx context.Context, deployment *apitype.UntypedDeployment) error
|
|
|
|
// Return the default secrets manager to use for this stack.
|
|
DefaultSecretManager(info *workspace.ProjectStack) (secrets.Manager, error)
|
|
}
|
|
|
|
// RemoveStack returns the stack, or returns an error if it cannot.
|
|
func RemoveStack(ctx context.Context, s Stack, force bool) (bool, error) {
|
|
return s.Backend().RemoveStack(ctx, s, force)
|
|
}
|
|
|
|
// RenameStack renames the stack, or returns an error if it cannot.
|
|
func RenameStack(ctx context.Context, s Stack, newName tokens.QName) (StackReference, error) {
|
|
return s.Backend().RenameStack(ctx, s, newName)
|
|
}
|
|
|
|
// PreviewStack previews changes to this stack.
|
|
func PreviewStack(
|
|
ctx context.Context,
|
|
s Stack,
|
|
op UpdateOperation,
|
|
) (*deploy.Plan, display.ResourceChanges, result.Result) {
|
|
return s.Backend().Preview(ctx, s, op)
|
|
}
|
|
|
|
// UpdateStack updates the target stack with the current workspace's contents (config and code).
|
|
func UpdateStack(ctx context.Context, s Stack, op UpdateOperation) (display.ResourceChanges, result.Result) {
|
|
return s.Backend().Update(ctx, s, op)
|
|
}
|
|
|
|
// ImportStack updates the target stack with the current workspace's contents (config and code).
|
|
func ImportStack(ctx context.Context, s Stack, op UpdateOperation,
|
|
imports []deploy.Import,
|
|
) (display.ResourceChanges, result.Result) {
|
|
return s.Backend().Import(ctx, s, op, imports)
|
|
}
|
|
|
|
// RefreshStack refresh's the stack's state from the cloud provider.
|
|
func RefreshStack(ctx context.Context, s Stack, op UpdateOperation) (display.ResourceChanges, result.Result) {
|
|
return s.Backend().Refresh(ctx, s, op)
|
|
}
|
|
|
|
// DestroyStack destroys all of this stack's resources.
|
|
func DestroyStack(ctx context.Context, s Stack, op UpdateOperation) (display.ResourceChanges, result.Result) {
|
|
return s.Backend().Destroy(ctx, s, op)
|
|
}
|
|
|
|
// WatchStack watches the projects working directory for changes and automatically updates the
|
|
// active stack.
|
|
func WatchStack(ctx context.Context, s Stack, op UpdateOperation, paths []string) result.Result {
|
|
return s.Backend().Watch(ctx, s, op, paths)
|
|
}
|
|
|
|
// GetLatestConfiguration returns the configuration for the most recent deployment of the stack.
|
|
func GetLatestConfiguration(ctx context.Context, s Stack) (config.Map, error) {
|
|
return s.Backend().GetLatestConfiguration(ctx, s)
|
|
}
|
|
|
|
// GetStackLogs fetches a list of log entries for the current stack in the current backend.
|
|
func GetStackLogs(ctx context.Context, secretsProvider secrets.Provider, s Stack, cfg StackConfiguration,
|
|
query operations.LogQuery,
|
|
) ([]operations.LogEntry, error) {
|
|
return s.Backend().GetLogs(ctx, secretsProvider, s, cfg, query)
|
|
}
|
|
|
|
// ExportStackDeployment exports the given stack's deployment as an opaque JSON message.
|
|
func ExportStackDeployment(
|
|
ctx context.Context,
|
|
s Stack,
|
|
) (*apitype.UntypedDeployment, error) {
|
|
return s.Backend().ExportDeployment(ctx, s)
|
|
}
|
|
|
|
// ImportStackDeployment imports the given deployment into the indicated stack.
|
|
func ImportStackDeployment(ctx context.Context, s Stack, deployment *apitype.UntypedDeployment) error {
|
|
return s.Backend().ImportDeployment(ctx, s, deployment)
|
|
}
|
|
|
|
// UpdateStackTags updates the stacks's tags, replacing all existing tags.
|
|
func UpdateStackTags(ctx context.Context, s Stack, tags map[apitype.StackTagName]string) error {
|
|
return s.Backend().UpdateStackTags(ctx, s, tags)
|
|
}
|
|
|
|
// GetMergedStackTags returns the stack's existing tags merged with fresh tags from the environment
|
|
// and Pulumi.yaml file.
|
|
func GetMergedStackTags(ctx context.Context, s Stack,
|
|
root string, project *workspace.Project, cfg config.Map,
|
|
) (map[apitype.StackTagName]string, error) {
|
|
// Get the stack's existing tags.
|
|
tags := s.Tags()
|
|
if tags == nil {
|
|
tags = make(map[apitype.StackTagName]string)
|
|
}
|
|
|
|
// Get latest environment tags for the current stack.
|
|
envTags, err := GetEnvironmentTagsForCurrentStack(root, project, cfg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Add each new environment tag to the existing tags, overwriting existing tags with the
|
|
// latest values.
|
|
for k, v := range envTags {
|
|
tags[k] = v
|
|
}
|
|
|
|
return tags, nil
|
|
}
|
|
|
|
// GetEnvironmentTagsForCurrentStack returns the set of tags for the "current" stack, based on the environment
|
|
// and Pulumi.yaml file.
|
|
func GetEnvironmentTagsForCurrentStack(root string,
|
|
project *workspace.Project, cfg config.Map,
|
|
) (map[apitype.StackTagName]string, error) {
|
|
tags := make(map[apitype.StackTagName]string)
|
|
|
|
// Tags based on Pulumi.yaml.
|
|
if project != nil {
|
|
tags[apitype.ProjectNameTag] = project.Name.String()
|
|
tags[apitype.ProjectRuntimeTag] = project.Runtime.Name()
|
|
if project.Description != nil {
|
|
tags[apitype.ProjectDescriptionTag] = *project.Description
|
|
}
|
|
}
|
|
|
|
// Grab any `pulumi:tag` config values and use those to update the stack's tags.
|
|
configTags, has, err := cfg.Get(config.MustMakeKey("pulumi", "tags"), false)
|
|
contract.AssertNoErrorf(err, "Config.Get(\"pulumi:tags\") failed unexpectedly")
|
|
if has {
|
|
configTagInterface, err := configTags.ToObject()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("pulumi:tags must be an object of strings")
|
|
}
|
|
configTagObject, ok := configTagInterface.(map[string]interface{})
|
|
if !ok {
|
|
return nil, fmt.Errorf("pulumi:tags must be an object of strings")
|
|
}
|
|
|
|
for name, value := range configTagObject {
|
|
stringValue, ok := value.(string)
|
|
if !ok {
|
|
return nil, fmt.Errorf("pulumi:tags[%s] must be a string", name)
|
|
}
|
|
|
|
tags[name] = stringValue
|
|
}
|
|
}
|
|
|
|
// Add the git metadata to the tags, ignoring any errors that come from it.
|
|
if root != "" {
|
|
ignoredErr := addGitMetadataToStackTags(tags, root)
|
|
contract.IgnoreError(ignoredErr)
|
|
}
|
|
|
|
return tags, nil
|
|
}
|
|
|
|
// addGitMetadataToStackTags fetches the git repository from the directory, and attempts to detect
|
|
// and add any relevant git metadata as stack tags.
|
|
func addGitMetadataToStackTags(tags map[apitype.StackTagName]string, projPath string) error {
|
|
repo, err := gitutil.GetGitRepository(projPath)
|
|
if repo == nil {
|
|
return fmt.Errorf("no git repository found from %v", projPath)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
remoteURL, err := gitutil.GetGitRemoteURL(repo, "origin")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if remoteURL == "" {
|
|
return nil
|
|
}
|
|
|
|
if vcsInfo, err := gitutil.TryGetVCSInfo(remoteURL); err == nil {
|
|
tags[apitype.VCSOwnerNameTag] = vcsInfo.Owner
|
|
tags[apitype.VCSRepositoryNameTag] = vcsInfo.Repo
|
|
tags[apitype.VCSRepositoryKindTag] = vcsInfo.Kind
|
|
} else {
|
|
return fmt.Errorf("detecting VCS info for stack tags for remote %v: %w", remoteURL, err)
|
|
}
|
|
// Set the old stack tags keys as GitHub so that the UI will continue to work,
|
|
// regardless of whether the remote URL is a GitHub URL or not.
|
|
// TODO remove these when the UI no longer needs them.
|
|
if tags[apitype.VCSOwnerNameTag] != "" {
|
|
tags[apitype.GitHubOwnerNameTag] = tags[apitype.VCSOwnerNameTag]
|
|
tags[apitype.GitHubRepositoryNameTag] = tags[apitype.VCSRepositoryNameTag]
|
|
}
|
|
|
|
return nil
|
|
}
|