pulumi/tests/config_test.go

418 lines
12 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 tests
import (
"encoding/json"
"os"
"path/filepath"
"regexp"
"strings"
"testing"
"github.com/stretchr/testify/assert"
"github.com/pulumi/pulumi/pkg/v3/testing/integration"
ptesting "github.com/pulumi/pulumi/sdk/v3/go/common/testing"
)
const pulumiTestOrg = "moolumi"
func TestConfigCommands(t *testing.T) {
t.Parallel()
t.Run("SanityTest", func(t *testing.T) {
t.Parallel()
e := ptesting.NewEnvironment(t)
defer func() {
if !t.Failed() {
e.DeleteEnvironment()
}
}()
integration.CreateBasicPulumiRepo(e)
e.SetBackend(e.LocalURL())
e.RunCommand("pulumi", "stack", "init", "test")
// check config is empty
stdout, _ := e.RunCommand("pulumi", "config")
assert.Equal(t, "KEY VALUE", strings.Trim(stdout, "\r\n"))
// set a bunch of config
e.RunCommand("pulumi", "config", "set-all",
"--plaintext", "key1=value1",
"--plaintext", "outer.inner=value2",
"--secret", "my_token=my_secret_token",
"--plaintext", "myList[0]=foo")
// check that it all exists
stdout, _ = e.RunCommand("pulumi", "config", "get", "key1")
assert.Equal(t, "value1", strings.Trim(stdout, "\r\n"))
stdout, _ = e.RunCommand("pulumi", "config", "get", "outer.inner")
assert.Equal(t, "value2", strings.Trim(stdout, "\r\n"))
stdout, _ = e.RunCommand("pulumi", "config", "get", "my_token")
assert.Equal(t, "my_secret_token", strings.Trim(stdout, "\r\n"))
stdout, _ = e.RunCommand("pulumi", "config", "get", "myList[0]")
assert.Equal(t, "foo", strings.Trim(stdout, "\r\n"))
// check that the nested config does not exist because we didn't use path
_, stderr := e.RunCommandExpectError("pulumi", "config", "get", "outer")
assert.Equal(t,
"error: configuration key 'outer' not found for stack 'test'",
strings.Trim(stderr, "\r\n"))
_, stderr = e.RunCommandExpectError("pulumi", "config", "get", "myList")
assert.Equal(t,
"error: configuration key 'myList' not found for stack 'test'",
strings.Trim(stderr, "\r\n"))
// set the nested config using --path
e.RunCommand("pulumi", "config", "set-all", "--path",
"--plaintext", "outer.inner=value2",
"--plaintext", "myList[0]=foo")
// check that the nested config now exists
stdout, _ = e.RunCommand("pulumi", "config", "get", "outer")
assert.Equal(t, "{\"inner\":\"value2\"}", strings.Trim(stdout, "\r\n"))
stdout, _ = e.RunCommand("pulumi", "config", "get", "myList")
assert.Equal(t, "[\"foo\"]", strings.Trim(stdout, "\r\n"))
// remove the nested config values
e.RunCommand("pulumi", "config", "rm-all", "--path", "outer.inner", "myList[0]")
// check that it worked
stdout, _ = e.RunCommand("pulumi", "config", "get", "outer")
assert.Equal(t, "{}", strings.Trim(stdout, "\r\n"))
stdout, _ = e.RunCommand("pulumi", "config", "get", "myList")
assert.Equal(t, "[]", strings.Trim(stdout, "\r\n"))
// remove other config values
e.RunCommand("pulumi", "config", "rm-all",
"outer.inner", "myList[0]", "outer", "myList", "key1", "my_token")
// check that you can add keys with '=' in it
e.RunCommand("pulumi", "config", "set-all",
"--plaintext", "\"foo=\"=value2", "--plaintext", "'=some-weird=key='=value3")
// check that they registered correctly
stdout, _ = e.RunCommand("pulumi", "config", "get", "foo=")
assert.Equal(t, "value2", strings.Trim(stdout, "\r\n"))
stdout, _ = e.RunCommand("pulumi", "config", "get", "=some-weird=key=")
assert.Equal(t, "value3", strings.Trim(stdout, "\r\n"))
// remove the config key
e.RunCommand("pulumi", "config", "rm-all", "foo=", "=some-weird=key=")
// check config is empty again
stdout, _ = e.RunCommand("pulumi", "config")
assert.Equal(t, "KEY VALUE", strings.Trim(stdout, "\r\n"))
e.RunCommand("pulumi", "stack", "rm", "--yes")
})
t.Run("YAMLTest", func(t *testing.T) {
t.Parallel()
e := ptesting.NewEnvironment(t)
defer func() {
if !t.Failed() {
e.DeleteEnvironment()
}
}()
integration.CreateBasicPulumiRepo(e)
e.SetBackend(e.LocalURL())
e.RunCommand("pulumi", "stack", "init", "test")
// check config is empty
stdout, _ := e.RunCommand("pulumi", "config")
assert.Equal(t, "KEY VALUE", strings.Trim(stdout, "\r\n"))
// set a config and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"a", "A",
)
expected := `^encryptionsalt: \S*
config:
pulumi-test:a: A
$`
b, err := os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
// set an additional secret config and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"b", "B", "--secret",
)
expected = `^encryptionsalt: \S*
config:
pulumi-test:a: A
pulumi-test:b:
secure: \S*
$`
b, err = os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
// update a config and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"a", "AA",
)
expected = `^encryptionsalt: \S*
config:
pulumi-test:a: AA
pulumi-test:b:
secure: \S*
$`
b, err = os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
// update the secret config and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"b", "BB", "--secret",
)
expected = `^encryptionsalt: \S*
config:
pulumi-test:a: AA
pulumi-test:b:
secure: \S*
$`
b, err = os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
// set a config with path=true and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"--path",
"c", "C",
)
expected = `^encryptionsalt: \S*
config:
pulumi-test:a: AA
pulumi-test:b:
secure: \S*
pulumi-test:c: C
$`
b, err = os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
// set a nested config and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"--path",
"d.a", "D",
)
expected = `^encryptionsalt: \S*
config:
pulumi-test:a: AA
pulumi-test:b:
secure: \S*
pulumi-test:c: C
pulumi-test:d:
a: D
$`
b, err = os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
// set an array config and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"--path",
"e[0]", "E",
)
expected = `^encryptionsalt: \S*
config:
pulumi-test:a: AA
pulumi-test:b:
secure: \S*
pulumi-test:c: C
pulumi-test:d:
a: D
pulumi-test:e:
- E
$`
b, err = os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
// set a nested array config and verify that the generated config file matches the expected values
e.RunCommand("pulumi", "config", "set",
"--path",
"f.g[0]", "F",
)
expected = `^encryptionsalt: \S*
config:
pulumi-test:a: AA
pulumi-test:b:
secure: \S*
pulumi-test:c: C
pulumi-test:d:
a: D
pulumi-test:e:
- E
pulumi-test:f:
g:
- F
$`
b, err = os.ReadFile(filepath.Join(e.CWD, "Pulumi.test.yaml"))
assert.NoError(t, err)
assert.Regexp(t, regexp.MustCompile(expected), string(b))
e.RunCommand("pulumi", "stack", "rm", "--yes")
})
}
func TestBasicConfigGetRetrievedValueFromProject(t *testing.T) {
t.Parallel()
e := ptesting.NewEnvironment(t)
defer func() {
if !t.Failed() {
e.DeleteEnvironment()
}
}()
pulumiProject := `
name: pulumi-test
runtime: go
config:
first-value:
type: string
default: first`
integration.CreatePulumiRepo(e, pulumiProject)
e.SetBackend(e.LocalURL())
e.RunCommand("pulumi", "stack", "init", "test")
stdout, _ := e.RunCommand("pulumi", "config", "get", "first-value")
assert.Equal(t, "first", strings.Trim(stdout, "\r\n"))
}
func TestConfigGetRetrievedValueFromBothStackAndProjectUsingJson(t *testing.T) {
t.Parallel()
e := ptesting.NewEnvironment(t)
defer func() {
if !t.Failed() {
e.DeleteEnvironment()
}
}()
pulumiProject := `
name: pulumi-test
runtime: go
config:
first-value:
type: string
default: first
second-value:
type: string
third-value:
type: array
items:
type: string
default: [third]`
integration.CreatePulumiRepo(e, pulumiProject)
e.SetBackend(e.LocalURL())
e.RunCommand("pulumi", "stack", "init", "test")
e.RunCommand("pulumi", "config", "set", "second-value", "second")
stdout, _ := e.RunCommand("pulumi", "config", "--json")
// check that stdout is an array containing 2 objects
var config map[string]interface{}
jsonError := json.Unmarshal([]byte(stdout), &config)
assert.Nil(t, jsonError)
assert.Equal(t, 3, len(config))
assert.Equal(t, "first", config["pulumi-test:first-value"].(map[string]interface{})["value"])
assert.Equal(t, "second", config["pulumi-test:second-value"].(map[string]interface{})["value"])
thirdValue := config["pulumi-test:third-value"].(map[string]interface{})
assert.Equal(t, "[\"third\"]", thirdValue["value"])
assert.Equal(t, []interface{}{"third"}, thirdValue["objectValue"])
}
func TestConfigCommandsUsingEnvironments(t *testing.T) {
if getTestOrg() != pulumiTestOrg {
t.Skip("Skipping test because the required environment is in the moolumi org.")
}
t.Parallel()
e := ptesting.NewEnvironment(t)
defer func() {
deleteIfNotFailed(e)
}()
integration.CreateBasicPulumiRepo(e)
e.RunCommand("pulumi", "org", "set-default", getTestOrg())
e.RunCommand("pulumi", "stack", "init", "test")
// check config is empty
stdout, _ := e.RunCommand("pulumi", "config")
assert.Equal(t, "KEY VALUE", strings.Trim(stdout, "\r\n"))
// set an esc environment
e.RunCommand("pulumi", "config", "env", "add", "secrets-test-env-DO-NOT-DELETE", "--yes")
// just `pulumi config`
stdout, _ = e.RunCommand("pulumi", "config")
assert.Equal(t, `KEY VALUE
test_secret [secret]`, strings.Trim(stdout, "\r\n"))
// `pulumi config --show-secrets`
stdout, _ = e.RunCommand("pulumi", "config", "--show-secrets")
assert.Equal(t, `KEY VALUE
test_secret this_is_my_secret`, strings.Trim(stdout, "\r\n"))
// `pulumi config --open`
stdout, _ = e.RunCommand("pulumi", "config", "--open")
assert.Equal(t, `KEY VALUE
test_secret [secret]`, strings.Trim(stdout, "\r\n"))
// `pulumi config --show-secrets --open`
stdout, _ = e.RunCommand("pulumi", "config", "--show-secrets", "--open")
assert.Equal(t, `KEY VALUE
test_secret this_is_my_secret`, strings.Trim(stdout, "\r\n"))
// `pulumi config --show-secrets --open=false`
stdout, _ = e.RunCommand("pulumi", "config", "--show-secrets", "--open=false")
assert.Equal(t, `KEY VALUE
test_secret [unknown]`, strings.Trim(stdout, "\r\n"))
// `pulumi config get`
stdout, _ = e.RunCommand("pulumi", "config", "get", "test_secret")
assert.Equal(t, "this_is_my_secret", strings.Trim(stdout, "\r\n"))
// `pulumi config get --open=false`
stdout, _ = e.RunCommand("pulumi", "config", "get", "test_secret", "--open=false")
assert.Equal(t, "[unknown]", strings.Trim(stdout, "\r\n"))
// delete the stack
e.RunCommand("pulumi", "stack", "rm", "-s", "test", "--yes")
}
func getTestOrg() string {
testOrg := pulumiTestOrg
if _, set := os.LookupEnv("PULUMI_TEST_ORG"); set {
testOrg = os.Getenv("PULUMI_TEST_ORG")
}
return testOrg
}