mirror of https://github.com/pulumi/pulumi.git
380 lines
11 KiB
Go
380 lines
11 KiB
Go
// Copyright 2016-2021, 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 workspace
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/rogpeppe/go-internal/lockedfile"
|
|
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/util/logging"
|
|
)
|
|
|
|
// PulumiCredentialsPathEnvVar is a path to the folder where credentials are stored.
|
|
// We use this in testing so that tests which log in and out do not impact the local developer's
|
|
// credentials or tests interacting with one another
|
|
//
|
|
//nolint:gosec
|
|
const PulumiCredentialsPathEnvVar = "PULUMI_CREDENTIALS_PATH"
|
|
|
|
// PulumiBackendURLEnvVar is an environment variable which can be used to set the backend that will be
|
|
// used instead of the currently logged in backend or the current projects backend.
|
|
const PulumiBackendURLEnvVar = "PULUMI_BACKEND_URL"
|
|
|
|
// GetAccount returns an account underneath a given key.
|
|
//
|
|
// Note that the account may not be fully populated: it may only have a valid AccessToken. In that case, it is up to
|
|
// the caller to fill in the username and last validation time.
|
|
func GetAccount(key string) (Account, error) {
|
|
creds, err := GetStoredCredentials()
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return Account{}, err
|
|
}
|
|
|
|
// Try the account
|
|
if account, ok := creds.Accounts[key]; ok {
|
|
return account, nil
|
|
}
|
|
token, ok := creds.AccessTokens[key]
|
|
if !ok {
|
|
return Account{}, nil
|
|
}
|
|
return Account{AccessToken: token}, nil
|
|
}
|
|
|
|
// DeleteAccount deletes an account underneath the given key.
|
|
func DeleteAccount(key string) error {
|
|
creds, err := GetStoredCredentials()
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
if creds.AccessTokens != nil {
|
|
delete(creds.AccessTokens, key)
|
|
}
|
|
if creds.Accounts != nil {
|
|
delete(creds.Accounts, key)
|
|
}
|
|
if creds.Current == key {
|
|
creds.Current = ""
|
|
}
|
|
return StoreCredentials(creds)
|
|
}
|
|
|
|
func DeleteAllAccounts() error {
|
|
credsFile, err := getCredsFilePath()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err = os.Remove(credsFile); err != nil && !os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// StoreAccount saves the given account underneath the given key.
|
|
func StoreAccount(key string, account Account, current bool) error {
|
|
creds, err := GetStoredCredentials()
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
if creds.AccessTokens == nil {
|
|
creds.AccessTokens = make(map[string]string)
|
|
}
|
|
if creds.Accounts == nil {
|
|
creds.Accounts = make(map[string]Account)
|
|
}
|
|
creds.AccessTokens[key], creds.Accounts[key] = account.AccessToken, account
|
|
if current {
|
|
creds.Current = key
|
|
}
|
|
return StoreCredentials(creds)
|
|
}
|
|
|
|
// Account holds the information associated with a Pulumi account.
|
|
type Account struct {
|
|
// The access token for this account.
|
|
AccessToken string `json:"accessToken,omitempty"`
|
|
// The username for this account.
|
|
Username string `json:"username,omitempty"`
|
|
// The organizations for this account.
|
|
Organizations []string `json:"organizations,omitempty"`
|
|
// The last time this token was validated.
|
|
LastValidatedAt time.Time `json:"lastValidatedAt,omitempty"`
|
|
// Allow insecure server connections when using SSL.
|
|
Insecure bool `json:"insecure,omitempty"`
|
|
// Information about the token used to authenticate.
|
|
TokenInformation *TokenInformation `json:"tokenInformation,omitempty"`
|
|
}
|
|
|
|
// Information about the token that was used to authenticate the current user. One (or none) of Team or Organization
|
|
// will be set, but not both.
|
|
type TokenInformation struct {
|
|
Name string `json:"name"` // The name of the token.
|
|
Organization string `json:"organization,omitempty"` // If this was an organization token, the organization it was for.
|
|
Team string `json:"team,omitempty"` // If this was a team token, the team it was for.
|
|
}
|
|
|
|
// Credentials hold the information necessary for authenticating Pulumi Cloud API requests. It contains
|
|
// a map from the cloud API URL to the associated access token.
|
|
type Credentials struct {
|
|
Current string `json:"current,omitempty"` // the currently selected key.
|
|
AccessTokens map[string]string `json:"accessTokens,omitempty"` // a map of arbitrary key strings to tokens.
|
|
Accounts map[string]Account `json:"accounts,omitempty"` // a map of arbitrary keys to account info.
|
|
}
|
|
|
|
// getCredsFilePath returns the path to the Pulumi credentials file on disk, regardless of
|
|
// whether it exists or not.
|
|
func getCredsFilePath() (string, error) {
|
|
// Allow the folder we use to store credentials to be overridden by tests
|
|
pulumiFolder := os.Getenv(PulumiCredentialsPathEnvVar)
|
|
if pulumiFolder == "" {
|
|
folder, err := GetPulumiHomeDir()
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to get the home path: %w", err)
|
|
}
|
|
pulumiFolder = folder
|
|
}
|
|
|
|
err := os.MkdirAll(pulumiFolder, 0o700)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to create '%s': %w", pulumiFolder, err)
|
|
}
|
|
|
|
return filepath.Join(pulumiFolder, "credentials.json"), nil
|
|
}
|
|
|
|
// GetCurrentCloudURL returns the URL of the cloud we are currently connected to. This may be empty if we
|
|
// have not logged in. Note if PULUMI_BACKEND_URL is set, the corresponding value is returned
|
|
// instead irrespective of the backend for current project or stored credentials.
|
|
func GetCurrentCloudURL(project *Project) (string, error) {
|
|
// Allow PULUMI_BACKEND_URL to override the current cloud URL selection
|
|
if backend := os.Getenv(PulumiBackendURLEnvVar); backend != "" {
|
|
return backend, nil
|
|
}
|
|
|
|
var url string
|
|
if project != nil {
|
|
if project.Backend != nil {
|
|
url = project.Backend.URL
|
|
}
|
|
}
|
|
|
|
if url == "" {
|
|
creds, err := GetStoredCredentials()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
url = creds.Current
|
|
}
|
|
|
|
return url, nil
|
|
}
|
|
|
|
// GetCloudInsecure returns if this cloud url is saved as one that should use insecure transport.
|
|
func GetCloudInsecure(cloudURL string) bool {
|
|
insecure := false
|
|
creds, err := GetStoredCredentials()
|
|
// If this errors just assume insecure == false
|
|
if err == nil {
|
|
if account, has := creds.Accounts[cloudURL]; has {
|
|
insecure = account.Insecure
|
|
}
|
|
}
|
|
return insecure
|
|
}
|
|
|
|
// GetStoredCredentials returns any credentials stored on the local machine.
|
|
func GetStoredCredentials() (Credentials, error) {
|
|
credsFile, err := getCredsFilePath()
|
|
if err != nil {
|
|
return Credentials{}, err
|
|
}
|
|
|
|
c, err := lockedfile.Read(credsFile)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return Credentials{}, nil
|
|
}
|
|
return Credentials{}, fmt.Errorf("reading '%s': %w", credsFile, err)
|
|
}
|
|
|
|
var creds Credentials
|
|
if err = json.Unmarshal(c, &creds); err != nil {
|
|
return Credentials{}, fmt.Errorf("failed to read Pulumi credentials file. Please re-run "+
|
|
"`pulumi login` to reset your credentials file: %w", err)
|
|
}
|
|
|
|
secrets := slice.Prealloc[string](len(creds.AccessTokens))
|
|
for _, v := range creds.AccessTokens {
|
|
secrets = append(secrets, v)
|
|
}
|
|
|
|
logging.AddGlobalFilter(logging.CreateFilter(secrets, "[credential]"))
|
|
|
|
return creds, nil
|
|
}
|
|
|
|
// StoreCredentials updates the stored credentials on the machine, replacing the existing set. If the credentials
|
|
// are empty, the auth file will be deleted rather than just serializing an empty map.
|
|
func StoreCredentials(creds Credentials) error {
|
|
credsFile, err := getCredsFilePath()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(creds.AccessTokens) == 0 {
|
|
err = os.Remove(credsFile)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
raw, err := json.MarshalIndent(creds, "", " ")
|
|
if err != nil {
|
|
return fmt.Errorf("marshalling credentials object: %w", err)
|
|
}
|
|
|
|
return lockedfile.Write(credsFile, bytes.NewReader(raw), 0o600)
|
|
}
|
|
|
|
type BackendConfig struct {
|
|
DefaultOrg string `json:"defaultOrg,omitempty"` // The default org for this backend config.
|
|
}
|
|
|
|
type PulumiConfig struct {
|
|
BackendConfig map[string]BackendConfig `json:"backends,omitempty"` // a map of arbitrary backends configs.
|
|
}
|
|
|
|
func getConfigFilePath() (string, error) {
|
|
// Allow the folder we use to store config in to be overridden by tests
|
|
pulumiFolder := os.Getenv(PulumiCredentialsPathEnvVar)
|
|
if pulumiFolder == "" {
|
|
folder, err := GetPulumiHomeDir()
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to get the home path: %w", err)
|
|
}
|
|
pulumiFolder = folder
|
|
}
|
|
|
|
err := os.MkdirAll(pulumiFolder, 0o700)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to create '%s': %w", pulumiFolder, err)
|
|
}
|
|
|
|
return filepath.Join(pulumiFolder, "config.json"), nil
|
|
}
|
|
|
|
func GetPulumiConfig() (PulumiConfig, error) {
|
|
configFile, err := getConfigFilePath()
|
|
if err != nil {
|
|
return PulumiConfig{}, err
|
|
}
|
|
|
|
c, err := os.ReadFile(configFile)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return PulumiConfig{}, nil
|
|
}
|
|
return PulumiConfig{}, fmt.Errorf("reading '%s': %w", configFile, err)
|
|
}
|
|
|
|
var config PulumiConfig
|
|
if err = json.Unmarshal(c, &config); err != nil {
|
|
return PulumiConfig{}, fmt.Errorf("failed to read Pulumi config file: %w", err)
|
|
}
|
|
|
|
return config, nil
|
|
}
|
|
|
|
func StorePulumiConfig(config PulumiConfig) error {
|
|
configFile, err := getConfigFilePath()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
raw, err := json.MarshalIndent(config, "", " ")
|
|
if err != nil {
|
|
return fmt.Errorf("marshalling config object: %w", err)
|
|
}
|
|
|
|
// Use a temporary file and atomic os.Rename to ensure the file contents are
|
|
// updated atomically to ensure concurrent `pulumi` CLI operations are safe.
|
|
tempConfigFile, err := os.CreateTemp(filepath.Dir(configFile), "config-*.json")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = tempConfigFile.Write(raw)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = tempConfigFile.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = os.Rename(tempConfigFile.Name(), configFile)
|
|
if err != nil {
|
|
contract.IgnoreError(os.Remove(tempConfigFile.Name()))
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func SetBackendConfigDefaultOrg(backendURL, defaultOrg string) error {
|
|
config, err := GetPulumiConfig()
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
|
|
if config.BackendConfig == nil {
|
|
config.BackendConfig = make(map[string]BackendConfig)
|
|
}
|
|
|
|
config.BackendConfig[backendURL] = BackendConfig{
|
|
DefaultOrg: defaultOrg,
|
|
}
|
|
|
|
return StorePulumiConfig(config)
|
|
}
|
|
|
|
func GetBackendConfigDefaultOrg(project *Project) (string, error) {
|
|
config, err := GetPulumiConfig()
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return "", err
|
|
}
|
|
|
|
backendURL, err := GetCurrentCloudURL(project)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if beConfig, ok := config.BackendConfig[backendURL]; ok {
|
|
if beConfig.DefaultOrg != "" {
|
|
return beConfig.DefaultOrg, nil
|
|
}
|
|
}
|
|
|
|
return "", nil
|
|
}
|