mirror of https://github.com/pulumi/pulumi.git
418 lines
12 KiB
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
|
|
}
|