pulumi/pkg/backend/httpstate/stack.go
Chris Smith 792c316e5e
Change backend.ListStacks to return a new StackSummary interface (#1931)
* Have backend.ListStacks return a new StackSummary interface

* Update filestake backend to use new type

* Update httpstate backend to use new type

* Update commands to use new type

* lint

* Address PR feedback

* Lint
2018-09-13 20:54:42 -07:00

184 lines
5.8 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 httpstate
import (
"context"
"fmt"
"time"
"github.com/pkg/errors"
"github.com/pulumi/pulumi/pkg/apitype"
"github.com/pulumi/pulumi/pkg/backend"
"github.com/pulumi/pulumi/pkg/engine"
"github.com/pulumi/pulumi/pkg/operations"
"github.com/pulumi/pulumi/pkg/resource/config"
"github.com/pulumi/pulumi/pkg/resource/deploy"
"github.com/pulumi/pulumi/pkg/tokens"
)
// Stack is a cloud stack. This simply adds some cloud-specific properties atop the standard backend stack interface.
type Stack interface {
backend.Stack
CloudURL() string // the URL to the cloud containing this stack.
OrgName() string // the organization that owns this stack.
ConsoleURL() (string, error) // the URL to view the stack's information on Pulumi.com
Tags() map[apitype.StackTagName]string // the stack's tags.
}
type cloudBackendReference struct {
name tokens.QName
owner string
b *cloudBackend
}
func (c cloudBackendReference) String() string {
curUser, err := c.b.client.GetPulumiAccountName(context.Background())
if err != nil {
curUser = ""
}
if c.owner == curUser {
return string(c.name)
}
return fmt.Sprintf("%s/%s", c.owner, c.name)
}
func (c cloudBackendReference) Name() tokens.QName {
return c.name
}
// nolint: lll
// cloudStack is a cloud stack descriptor.
type cloudStack struct {
ref backend.StackReference // the stack's ref (unique name).
cloudURL string // the URL to the cloud containing this stack.
orgName string // the organization that owns this stack.
config config.Map // the stack's config bag.
snapshot **deploy.Snapshot // a snapshot representing the latest deployment state (allocated on first use)
b *cloudBackend // a pointer to the backend this stack belongs to.
tags map[apitype.StackTagName]string // the stack's tags.
}
func newStack(apistack apitype.Stack, b *cloudBackend) Stack {
// Now assemble all the pieces into a stack structure.
return &cloudStack{
ref: cloudBackendReference{
owner: apistack.OrgName,
name: apistack.StackName,
b: b,
},
cloudURL: b.CloudURL(),
orgName: apistack.OrgName,
config: nil, // TODO[pulumi/pulumi-service#249]: add the config variables.
snapshot: nil, // We explicitly allocate the snapshot on first use, since it is expensive to compute.
tags: apistack.Tags,
b: b,
}
}
func (s *cloudStack) Ref() backend.StackReference { return s.ref }
func (s *cloudStack) Config() config.Map { return s.config }
func (s *cloudStack) Backend() backend.Backend { return s.b }
func (s *cloudStack) CloudURL() string { return s.cloudURL }
func (s *cloudStack) OrgName() string { return s.orgName }
func (s *cloudStack) Tags() map[apitype.StackTagName]string { return s.tags }
func (s *cloudStack) Snapshot(ctx context.Context) (*deploy.Snapshot, error) {
if s.snapshot != nil {
return *s.snapshot, nil
}
snap, err := s.b.getSnapshot(ctx, s.ref)
if err != nil {
return nil, err
}
s.snapshot = &snap
return *s.snapshot, nil
}
func (s *cloudStack) Remove(ctx context.Context, force bool) (bool, error) {
return backend.RemoveStack(ctx, s, force)
}
func (s *cloudStack) Preview(ctx context.Context, op backend.UpdateOperation) (engine.ResourceChanges, error) {
return backend.PreviewStack(ctx, s, op)
}
func (s *cloudStack) Update(ctx context.Context, op backend.UpdateOperation) (engine.ResourceChanges, error) {
return backend.UpdateStack(ctx, s, op)
}
func (s *cloudStack) Refresh(ctx context.Context, op backend.UpdateOperation) (engine.ResourceChanges, error) {
return backend.RefreshStack(ctx, s, op)
}
func (s *cloudStack) Destroy(ctx context.Context, op backend.UpdateOperation) (engine.ResourceChanges, error) {
return backend.DestroyStack(ctx, s, op)
}
func (s *cloudStack) GetLogs(ctx context.Context, query operations.LogQuery) ([]operations.LogEntry, error) {
return backend.GetStackLogs(ctx, s, query)
}
func (s *cloudStack) ExportDeployment(ctx context.Context) (*apitype.UntypedDeployment, error) {
return backend.ExportStackDeployment(ctx, s)
}
func (s *cloudStack) ImportDeployment(ctx context.Context, deployment *apitype.UntypedDeployment) error {
return backend.ImportStackDeployment(ctx, s, deployment)
}
func (s *cloudStack) ConsoleURL() (string, error) {
path, err := s.b.StackConsoleURL(s.ref)
if err != nil {
return "", nil
}
url := s.b.CloudConsoleURL(path)
if url == "" {
return "", errors.New("could not determine clould console URL")
}
return url, nil
}
// cloudStackSummary implements the backend.StackSummary interface, by wrapping
// an apitype.StackSummary struct.
type cloudStackSummary struct {
summary apitype.StackSummary
b *cloudBackend
}
func (css cloudStackSummary) Name() backend.StackReference {
return cloudBackendReference{
owner: css.summary.OrgName,
name: tokens.QName(css.summary.StackName),
b: css.b,
}
}
func (css cloudStackSummary) LastUpdate() *time.Time {
if css.summary.LastUpdate == nil {
return nil
}
t := time.Unix(*css.summary.LastUpdate, 0)
return &t
}
func (css cloudStackSummary) ResourceCount() *int {
return css.summary.ResourceCount
}