pulumi/tests/config_test.go

349 lines
9.9 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"
)
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"])
}