pulumi/pkg/cmd/pulumi/login.go

167 lines
5.7 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 main
import (
"fmt"
"os"
"path/filepath"
"strings"
"github.com/pkg/errors"
"github.com/spf13/cobra"
"github.com/pulumi/pulumi/pkg/v3/backend"
"github.com/pulumi/pulumi/pkg/v3/backend/display"
"github.com/pulumi/pulumi/pkg/v3/backend/filestate"
"github.com/pulumi/pulumi/pkg/v3/backend/httpstate"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/cmdutil"
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
)
func newLoginCmd() *cobra.Command {
var cloudURL string
var localMode bool
cmd := &cobra.Command{
Use: "login [<url>]",
Short: "Log in to the Pulumi service",
Long: "Log in to the Pulumi service.\n" +
"\n" +
"The service manages your stack's state reliably. Simply run\n" +
"\n" +
" $ pulumi login\n" +
"\n" +
"and this command will prompt you for an access token, including a way to launch your web browser to\n" +
"easily obtain one. You can script by using `PULUMI_ACCESS_TOKEN` environment variable.\n" +
"\n" +
"By default, this will log in to the managed Pulumi service backend.\n" +
"If you prefer to log in to a self-hosted Pulumi service backend, specify a URL. For example, run\n" +
"\n" +
" $ pulumi login https://api.pulumi.acmecorp.com\n" +
"\n" +
"to log in to a self-hosted Pulumi service running at the api.pulumi.acmecorp.com domain.\n" +
"\n" +
"For `https://` URLs, the CLI will speak REST to a service that manages state and concurrency control.\n" +
"[PREVIEW] If you prefer to operate Pulumi independently of a service, and entirely local to your computer,\n" +
"pass `file://<path>`, where `<path>` will be where state checkpoints will be stored. For instance,\n" +
"\n" +
" $ pulumi login file://~\n" +
"\n" +
"will store your state information on your computer underneath `~/.pulumi`. It is then up to you to\n" +
"manage this state, including backing it up, using it in a team environment, and so on.\n" +
"\n" +
"As a shortcut, you may pass --local to use your home directory (this is an alias for `file://~`):\n" +
"\n" +
" $ pulumi login --local\n" +
"\n" +
"[PREVIEW] Additionally, you may leverage supported object storage backends from one of the cloud providers " +
"to manage the state independent of the service. For instance,\n" +
"\n" +
"AWS S3:\n" +
"\n" +
" $ pulumi login s3://my-pulumi-state-bucket\n" +
"\n" +
"GCP GCS:\n" +
"\n" +
" $ pulumi login gs://my-pulumi-state-bucket\n" +
"\n" +
"Azure Blob:\n" +
"\n" +
" $ pulumi login azblob://my-pulumi-state-bucket\n",
Args: cmdutil.MaximumNArgs(1),
Run: cmdutil.RunFunc(func(cmd *cobra.Command, args []string) error {
displayOptions := display.Options{
Color: cmdutil.GetGlobalColorization(),
}
// If a <cloud> was specified as an argument, use it.
if len(args) > 0 {
if cloudURL != "" {
return errors.New("only one of --cloud-url or argument URL may be specified, not both")
}
cloudURL = args[0]
}
// For local mode, store state by default in the user's home directory.
if localMode {
if cloudURL != "" {
return errors.New("a URL may not be specified when --local mode is enabled")
}
cloudURL = filestate.FilePathPrefix + "~"
}
// If we're on Windows, and this is a local login path, then allow the user to provide
// backslashes as path separators. We will normalize them here to forward slashes as that's
// what the gocloud blob system requires.
if strings.HasPrefix(cloudURL, filestate.FilePathPrefix) && os.PathSeparator != '/' {
cloudURL = filepath.ToSlash(cloudURL)
}
if cloudURL == "" {
var err error
cloudURL, err = workspace.GetCurrentCloudURL()
if err != nil {
return errors.Wrap(err, "could not determine current cloud")
}
} else {
// Ensure we have the correct cloudurl type before logging in
if err := validateCloudBackendType(cloudURL); err != nil {
return err
}
}
var be backend.Backend
var err error
if filestate.IsFileStateBackendURL(cloudURL) {
be, err = filestate.Login(cmdutil.Diag(), cloudURL)
} else {
be, err = httpstate.Login(commandContext(), cmdutil.Diag(), cloudURL, displayOptions)
}
if err != nil {
return errors.Wrapf(err, "problem logging in")
}
if currentUser, err := be.CurrentUser(); err == nil {
fmt.Printf("Logged in to %s as %s (%s)\n", be.Name(), currentUser, be.URL())
} else {
fmt.Printf("Logged in to %s (%s)\n", be.Name(), be.URL())
}
return nil
}),
}
cmd.PersistentFlags().StringVarP(&cloudURL, "cloud-url", "c", "", "A cloud URL to log in to")
cmd.PersistentFlags().BoolVarP(&localMode, "local", "l", false, "Use Pulumi in local-only mode")
return cmd
}
func validateCloudBackendType(typ string) error {
kind := strings.SplitN(typ, ":", 2)[0]
supportedKinds := []string{"azblob", "gs", "s3", "file", "https", "http"}
for _, supportedKind := range supportedKinds {
if kind == supportedKind {
return nil
}
}
return errors.Errorf(
"unknown backend cloudUrl format '%s' (supported Url formats are: "+
"azblob://, gs://, s3://, file://, https:// and http://)",
kind,
)
}