pulumi/sdk/go/common/workspace/creds.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.CreateGlobalFilter(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
}