mirror of https://github.com/pulumi/pulumi.git
339 lines
7.6 KiB
Go
339 lines
7.6 KiB
Go
// Copyright 2016-2022, 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 config
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
yaml "gopkg.in/yaml.v2"
|
|
)
|
|
|
|
func TestMarshallNormalValueYAML(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
v := NewValue("value")
|
|
|
|
b, err := yaml.Marshal(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, []byte("value\n"), b)
|
|
|
|
newV, err := roundtripValueYAML(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, v, newV)
|
|
}
|
|
|
|
func TestMarshallSecureValueYAML(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
v := NewSecureValue("value")
|
|
|
|
b, err := yaml.Marshal(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, []byte("secure: value\n"), b)
|
|
|
|
newV, err := roundtripValueYAML(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, v, newV)
|
|
}
|
|
|
|
func TestMarshallNormalValueJSON(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
v := NewValue("value")
|
|
|
|
b, err := json.Marshal(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, []byte("\"value\""), b)
|
|
|
|
newV, err := roundtripValueJSON(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, v, newV)
|
|
}
|
|
|
|
func TestMarshallSecureValueJSON(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
v := NewSecureValue("value")
|
|
|
|
b, err := json.Marshal(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, []byte("{\"secure\":\"value\"}"), b)
|
|
|
|
newV, err := roundtripValueJSON(v)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, v, newV)
|
|
}
|
|
|
|
func TestHasSecureValue(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
Value interface{}
|
|
Expected bool
|
|
}{
|
|
{
|
|
Value: []interface{}{"a", "b", "c"},
|
|
Expected: false,
|
|
},
|
|
{
|
|
Value: map[string]interface{}{
|
|
"foo": "bar",
|
|
"hi": map[string]interface{}{"secure": "securevalue", "but": "not"},
|
|
},
|
|
Expected: false,
|
|
},
|
|
{
|
|
Value: []interface{}{"a", "b", map[string]interface{}{"secure": "securevalue"}},
|
|
Expected: true,
|
|
},
|
|
{
|
|
Value: map[string]interface{}{
|
|
"foo": "bar",
|
|
"hi": map[string]interface{}{"secure": "securevalue"},
|
|
},
|
|
Expected: true,
|
|
},
|
|
{
|
|
Value: map[string]interface{}{
|
|
"foo": "bar",
|
|
"array": []interface{}{"a", "b", map[string]interface{}{"secure": "securevalue"}},
|
|
},
|
|
Expected: true,
|
|
},
|
|
{
|
|
Value: map[string]interface{}{
|
|
"foo": "bar",
|
|
"map": map[string]interface{}{
|
|
"nest": "blah",
|
|
"hi": map[string]interface{}{"secure": "securevalue"},
|
|
},
|
|
},
|
|
Expected: true,
|
|
},
|
|
}
|
|
|
|
//nolint:paralleltest // false positive because range var isn't used directly in t.Run(name) arg
|
|
for _, test := range tests {
|
|
test := test
|
|
t.Run(fmt.Sprintf("%v", test.Value), func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
jsonBytes, err := json.Marshal(test.Value)
|
|
assert.NoError(t, err)
|
|
|
|
var val object
|
|
err = json.Unmarshal(jsonBytes, &val)
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.Expected, val.Secure())
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestDecryptingValue(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
Value Value
|
|
Expected string
|
|
}{
|
|
{
|
|
Value: NewValue("value"),
|
|
Expected: "value",
|
|
},
|
|
{
|
|
Value: NewValue(`{"foo":"bar"}`),
|
|
Expected: `{"foo":"bar"}`,
|
|
},
|
|
{
|
|
Value: NewValue(`["a","b"]`),
|
|
Expected: `["a","b"]`,
|
|
},
|
|
{
|
|
Value: NewObjectValue(`{"foo":"bar"}`),
|
|
Expected: `{"foo":"bar"}`,
|
|
},
|
|
{
|
|
Value: NewObjectValue(`["a","b"]`),
|
|
Expected: `["a","b"]`,
|
|
},
|
|
{
|
|
Value: NewSecureValue("securevalue"),
|
|
Expected: "[secret]",
|
|
},
|
|
{
|
|
Value: NewSecureObjectValue(`{"foo":{"secure":"securevalue"}}`),
|
|
Expected: `{"foo":"[secret]"}`,
|
|
},
|
|
{
|
|
Value: NewSecureObjectValue(`["a",{"secure":"securevalue"}]`),
|
|
Expected: `["a","[secret]"]`,
|
|
},
|
|
}
|
|
|
|
decrypter := NewBlindingDecrypter()
|
|
|
|
//nolint:paralleltest // false positive because range var isn't used directly in t.Run(name) arg
|
|
for _, test := range tests {
|
|
test := test
|
|
t.Run(fmt.Sprintf("%v", test.Value), func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
actual, err := test.Value.Value(decrypter)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, test.Expected, actual)
|
|
|
|
// Ensure the same value is returned when the NopDecrypter is used.
|
|
actualNop, err := test.Value.Value(NopDecrypter)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, test.Value.value, actualNop)
|
|
})
|
|
}
|
|
}
|
|
|
|
type passThroughDecrypter struct{}
|
|
|
|
func (d passThroughDecrypter) DecryptValue(
|
|
ctx context.Context, ciphertext string,
|
|
) (string, error) {
|
|
return ciphertext, nil
|
|
}
|
|
|
|
func (d passThroughDecrypter) BulkDecrypt(
|
|
ctx context.Context, ciphertexts []string,
|
|
) (map[string]string, error) {
|
|
return DefaultBulkDecrypt(ctx, d, ciphertexts)
|
|
}
|
|
|
|
func TestSecureValues(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
Value Value
|
|
Expected []string
|
|
}{
|
|
{
|
|
Value: NewValue("value"),
|
|
Expected: nil,
|
|
},
|
|
{
|
|
Value: NewObjectValue(`{"foo":"bar"}`),
|
|
Expected: nil,
|
|
},
|
|
{
|
|
Value: NewObjectValue(`["a","b"]`),
|
|
Expected: nil,
|
|
},
|
|
{
|
|
Value: NewSecureValue("securevalue"),
|
|
Expected: []string{"securevalue"},
|
|
},
|
|
{
|
|
Value: NewSecureObjectValue(`{"foo":{"secure":"securevalue"}}`),
|
|
Expected: []string{"securevalue"},
|
|
},
|
|
{
|
|
Value: NewSecureObjectValue(`["a",{"secure":"securevalue"}]`),
|
|
Expected: []string{"securevalue"},
|
|
},
|
|
{
|
|
Value: NewSecureObjectValue(`["a",{"secure":"alpha"},{"test":{"secure":"beta"}}]`),
|
|
Expected: []string{"alpha", "beta"},
|
|
},
|
|
}
|
|
|
|
decrypter := passThroughDecrypter{}
|
|
|
|
//nolint:paralleltest // false positive because range var isn't used directly in t.Run(name) arg
|
|
for _, test := range tests {
|
|
test := test
|
|
t.Run(fmt.Sprintf("%v", test.Value), func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
actual, err := test.Value.SecureValues(decrypter)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, test.Expected, actual)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestCopyValue(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
Val Value
|
|
Expected Value
|
|
}{
|
|
{
|
|
Val: NewValue("value"),
|
|
Expected: NewValue("value"),
|
|
},
|
|
{
|
|
Val: NewObjectValue(`{"foo":"bar"}`),
|
|
Expected: NewObjectValue(`{"foo":"bar"}`),
|
|
},
|
|
{
|
|
Val: NewSecureObjectValue(`{"foo":{"secure":"stackAsecurevalue"}}`),
|
|
Expected: NewSecureObjectValue(`{"foo":{"secure":"stackBsecurevalue"}}`),
|
|
},
|
|
{
|
|
Val: NewSecureValue("stackAsecurevalue"),
|
|
Expected: NewSecureValue("stackBsecurevalue"),
|
|
},
|
|
{
|
|
Val: NewSecureObjectValue(`["a",{"secure":"stackAalpha"},{"test":{"secure":"stackAbeta"}}]`),
|
|
Expected: NewSecureObjectValue(`["a",{"secure":"stackBalpha"},{"test":{"secure":"stackBbeta"}}]`),
|
|
},
|
|
}
|
|
|
|
//nolint:paralleltest // false positive because range var isn't used directly in t.Run(name) arg
|
|
for _, test := range tests {
|
|
test := test
|
|
t.Run(fmt.Sprintf("%v", test), func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
newConfig, err := test.Val.Copy(newPrefixCrypter("stackA"), newPrefixCrypter("stackB"))
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.Expected, newConfig)
|
|
})
|
|
}
|
|
}
|
|
|
|
func roundtripValueYAML(v Value) (Value, error) {
|
|
return roundtripValue(v, yaml.Marshal, yaml.Unmarshal)
|
|
}
|
|
|
|
func roundtripValueJSON(v Value) (Value, error) {
|
|
return roundtripValue(v, json.Marshal, json.Unmarshal)
|
|
}
|
|
|
|
func roundtripValue(v Value, marshal func(v interface{}) ([]byte, error),
|
|
unmarshal func([]byte, interface{}) error,
|
|
) (Value, error) {
|
|
b, err := marshal(v)
|
|
if err != nil {
|
|
return Value{}, err
|
|
}
|
|
|
|
var newV Value
|
|
err = unmarshal(b, &newV)
|
|
return newV, err
|
|
}
|