mirror of https://github.com/pulumi/pulumi.git
1352 lines
42 KiB
Go
1352 lines
42 KiB
Go
package filestate
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"regexp"
|
|
"runtime"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
user "github.com/tweekmonster/luser"
|
|
"gocloud.dev/blob/fileblob"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/backend"
|
|
"github.com/pulumi/pulumi/pkg/v3/operations"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/stack"
|
|
"github.com/pulumi/pulumi/pkg/v3/secrets/b64"
|
|
"github.com/pulumi/pulumi/pkg/v3/secrets/passphrase"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/apitype"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/diag"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/diag/colors"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/encoding"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/env"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource/config"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/testing/diagtest"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/testing/iotest"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/workspace"
|
|
)
|
|
|
|
func TestMassageBlobPath(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
testMassagePath := func(t *testing.T, s string, want string) {
|
|
massaged, err := massageBlobPath(s)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, want, massaged,
|
|
"massageBlobPath(%s) didn't return expected result.\nWant: %q\nGot: %q", s, want, massaged)
|
|
}
|
|
|
|
// URLs not prefixed with "file://" are kept as-is. Also why we add FilePathPrefix as a prefix for other tests.
|
|
t.Run("NonFilePrefixed", func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
testMassagePath(t, "asdf-123", "asdf-123")
|
|
})
|
|
|
|
// The home directory is converted into the user's actual home directory.
|
|
// Which requires even more tweaks to work on Windows.
|
|
t.Run("PrefixedWithTilde", func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
usr, err := user.Current()
|
|
if err != nil {
|
|
t.Fatalf("Unable to get current user: %v", err)
|
|
}
|
|
|
|
homeDir := usr.HomeDir
|
|
|
|
// When running on Windows, the "home directory" takes on a different meaning.
|
|
if runtime.GOOS == "windows" {
|
|
t.Logf("Running on %v", runtime.GOOS)
|
|
|
|
t.Run("NormalizeDirSeparator", func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
testMassagePath(t, FilePathPrefix+`C:\Users\steve\`, FilePathPrefix+"/C:/Users/steve")
|
|
})
|
|
|
|
newHomeDir := "/" + filepath.ToSlash(homeDir)
|
|
t.Logf("Changed homeDir to expect from %q to %q", homeDir, newHomeDir)
|
|
homeDir = newHomeDir
|
|
}
|
|
|
|
testMassagePath(t, FilePathPrefix+"~", FilePathPrefix+homeDir)
|
|
testMassagePath(t, FilePathPrefix+"~/alpha/beta", FilePathPrefix+homeDir+"/alpha/beta")
|
|
})
|
|
|
|
t.Run("MakeAbsolute", func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Run the expected result through filepath.Abs, since on Windows we expect "C:\1\2".
|
|
expected := "/1/2"
|
|
abs, err := filepath.Abs(expected)
|
|
assert.NoError(t, err)
|
|
|
|
expected = filepath.ToSlash(abs)
|
|
if expected[0] != '/' {
|
|
expected = "/" + expected // A leading slash is added on Windows.
|
|
}
|
|
|
|
testMassagePath(t, FilePathPrefix+"/1/2/3/../4/..", FilePathPrefix+expected)
|
|
})
|
|
}
|
|
|
|
func TestGetLogsForTargetWithNoSnapshot(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
target := &deploy.Target{
|
|
Name: tokens.MustParseStackName("test"),
|
|
Config: config.Map{},
|
|
Decrypter: config.NopDecrypter,
|
|
Snapshot: nil,
|
|
}
|
|
query := operations.LogQuery{}
|
|
res, err := GetLogsForTarget(target, query)
|
|
assert.NoError(t, err)
|
|
assert.Nil(t, res)
|
|
}
|
|
|
|
func makeUntypedDeployment(name tokens.QName, phrase, state string) (*apitype.UntypedDeployment, error) {
|
|
return makeUntypedDeploymentTimestamp(name, phrase, state, nil, nil)
|
|
}
|
|
|
|
func makeUntypedDeploymentTimestamp(
|
|
name tokens.QName,
|
|
phrase, state string,
|
|
created, modified *time.Time,
|
|
) (*apitype.UntypedDeployment, error) {
|
|
sm, err := passphrase.GetPassphraseSecretsManager(phrase, state)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
resources := []*resource.State{
|
|
{
|
|
URN: resource.NewURN("a", "proj", "d:e:f", "a:b:c", name),
|
|
Type: "a:b:c",
|
|
Inputs: resource.PropertyMap{
|
|
resource.PropertyKey("secret"): resource.MakeSecret(resource.NewStringProperty("s3cr3t")),
|
|
},
|
|
Created: created,
|
|
Modified: modified,
|
|
},
|
|
}
|
|
|
|
snap := deploy.NewSnapshot(deploy.Manifest{}, sm, resources, nil)
|
|
|
|
sdep, err := stack.SerializeDeployment(snap, snap.SecretsManager, false /* showSecrsts */)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
data, err := json.Marshal(sdep)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &apitype.UntypedDeployment{
|
|
Version: 3,
|
|
Deployment: json.RawMessage(data),
|
|
}, nil
|
|
}
|
|
|
|
//nolint:paralleltest // mutates environment variables
|
|
func TestListStacksWithMultiplePassphrases(t *testing.T) {
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Create stack "a" and import a checkpoint with a secret
|
|
aStackRef, err := b.ParseStackReference("organization/project/a")
|
|
assert.NoError(t, err)
|
|
aStack, err := b.CreateStack(ctx, aStackRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, aStack)
|
|
defer func() {
|
|
t.Setenv("PULUMI_CONFIG_PASSPHRASE", "abc123")
|
|
_, err := b.RemoveStack(ctx, aStack, true)
|
|
assert.NoError(t, err)
|
|
}()
|
|
deployment, err := makeUntypedDeployment("a", "abc123",
|
|
"v1:4iF78gb0nF0=:v1:Co6IbTWYs/UdrjgY:FSrAWOFZnj9ealCUDdJL7LrUKXX9BA==")
|
|
assert.NoError(t, err)
|
|
t.Setenv("PULUMI_CONFIG_PASSPHRASE", "abc123")
|
|
err = b.ImportDeployment(ctx, aStack, deployment)
|
|
assert.NoError(t, err)
|
|
|
|
// Create stack "b" and import a checkpoint with a secret
|
|
bStackRef, err := b.ParseStackReference("organization/project/b")
|
|
assert.NoError(t, err)
|
|
bStack, err := b.CreateStack(ctx, bStackRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, bStack)
|
|
defer func() {
|
|
t.Setenv("PULUMI_CONFIG_PASSPHRASE", "123abc")
|
|
_, err := b.RemoveStack(ctx, bStack, true)
|
|
assert.NoError(t, err)
|
|
}()
|
|
deployment, err = makeUntypedDeployment("b", "123abc",
|
|
"v1:C7H2a7/Ietk=:v1:yfAd1zOi6iY9DRIB:dumdsr+H89VpHIQWdB01XEFqYaYjAg==")
|
|
assert.NoError(t, err)
|
|
t.Setenv("PULUMI_CONFIG_PASSPHRASE", "123abc")
|
|
err = b.ImportDeployment(ctx, bStack, deployment)
|
|
assert.NoError(t, err)
|
|
|
|
// Remove the config passphrase so that we can no longer deserialize the checkpoints
|
|
err = os.Unsetenv("PULUMI_CONFIG_PASSPHRASE")
|
|
assert.NoError(t, err)
|
|
|
|
// Ensure that we can list the stacks we created even without a passphrase
|
|
stacks, outContToken, err := b.ListStacks(ctx, backend.ListStacksFilter{}, nil /* inContToken */)
|
|
assert.NoError(t, err)
|
|
assert.Nil(t, outContToken)
|
|
assert.Len(t, stacks, 2)
|
|
for _, stack := range stacks {
|
|
assert.NotNil(t, stack.ResourceCount())
|
|
assert.Equal(t, 1, *stack.ResourceCount())
|
|
}
|
|
}
|
|
|
|
func TestDrillError(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Get a non-existent stack and expect a nil error because it won't be found.
|
|
stackRef, err := b.ParseStackReference("organization/project/dev")
|
|
if err != nil {
|
|
t.Fatalf("unexpected error %v when parsing stack reference", err)
|
|
}
|
|
_, err = b.GetStack(ctx, stackRef)
|
|
assert.NoError(t, err)
|
|
}
|
|
|
|
func TestCancel(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Check that trying to cancel a stack that isn't created yet doesn't error
|
|
aStackRef, err := b.ParseStackReference("organization/project/a")
|
|
assert.NoError(t, err)
|
|
err = b.CancelCurrentUpdate(ctx, aStackRef)
|
|
assert.NoError(t, err)
|
|
|
|
// Check that trying to cancel a stack that isn't locked doesn't error
|
|
aStack, err := b.CreateStack(ctx, aStackRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, aStack)
|
|
err = b.CancelCurrentUpdate(ctx, aStackRef)
|
|
assert.NoError(t, err)
|
|
|
|
// Locking and lock checks are only part of the internal interface
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
|
|
// Lock the stack and check CancelCurrentUpdate deletes the lock file
|
|
err = lb.Lock(ctx, aStackRef)
|
|
assert.NoError(t, err)
|
|
// check the lock file exists
|
|
lockExists, err := lb.bucket.Exists(ctx, lb.lockPath(aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.True(t, lockExists)
|
|
// Call CancelCurrentUpdate
|
|
err = lb.CancelCurrentUpdate(ctx, aStackRef)
|
|
assert.NoError(t, err)
|
|
// Now check the lock file no longer exists
|
|
lockExists, err = lb.bucket.Exists(ctx, lb.lockPath(aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.False(t, lockExists)
|
|
|
|
// Make another filestate backend which will have a different lockId
|
|
ob, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
otherBackend, ok := ob.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
|
|
// Lock the stack with this new backend, then check that checkForLocks on the first backend now errors
|
|
err = otherBackend.Lock(ctx, aStackRef)
|
|
assert.NoError(t, err)
|
|
err = lb.checkForLock(ctx, aStackRef)
|
|
assert.Error(t, err)
|
|
// Now call CancelCurrentUpdate and check that checkForLocks no longer errors
|
|
err = lb.CancelCurrentUpdate(ctx, aStackRef)
|
|
assert.NoError(t, err)
|
|
err = lb.checkForLock(ctx, aStackRef)
|
|
assert.NoError(t, err)
|
|
}
|
|
|
|
func TestRemoveMakesBackups(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Grab the bucket interface to test with
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
|
|
// Check that creating a new stack doesn't make a backup file
|
|
aStackRef, err := lb.parseStackReference("organization/project/a")
|
|
assert.NoError(t, err)
|
|
aStack, err := b.CreateStack(ctx, aStackRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, aStack)
|
|
|
|
// Check the stack file now exists, but the backup file doesn't
|
|
stackFileExists, err := lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
backupFileExists, err := lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef)+".bak")
|
|
assert.NoError(t, err)
|
|
assert.False(t, backupFileExists)
|
|
|
|
// Now remove the stack
|
|
removed, err := b.RemoveStack(ctx, aStack, false)
|
|
assert.NoError(t, err)
|
|
assert.False(t, removed)
|
|
|
|
// Check the stack file is now gone, but the backup file exists
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.False(t, stackFileExists)
|
|
backupFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef)+".bak")
|
|
assert.NoError(t, err)
|
|
assert.True(t, backupFileExists)
|
|
}
|
|
|
|
func TestRenameWorks(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Grab the bucket interface to test with
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
|
|
// Create a new stack
|
|
aStackRef, err := lb.parseStackReference("organization/project/a")
|
|
assert.NoError(t, err)
|
|
aStack, err := b.CreateStack(ctx, aStackRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, aStack)
|
|
|
|
// Check the stack file now exists
|
|
stackFileExists, err := lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
|
|
// Fake up some history
|
|
err = lb.addToHistory(ctx, aStackRef, backend.UpdateInfo{Kind: apitype.DestroyUpdate})
|
|
assert.NoError(t, err)
|
|
// And pollute the history folder
|
|
err = lb.bucket.WriteAll(ctx, path.Join(aStackRef.HistoryDir(), "randomfile.txt"), []byte{0, 13}, nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Rename the stack
|
|
bStackRefI, err := b.RenameStack(ctx, aStack, "organization/project/b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "organization/project/b", bStackRefI.String())
|
|
bStackRef := bStackRefI.(*localBackendReference)
|
|
|
|
// Check the new stack file now exists and the old one is gone
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, bStackRef))
|
|
assert.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.False(t, stackFileExists)
|
|
|
|
// Rename again
|
|
bStack, err := b.GetStack(ctx, bStackRef)
|
|
assert.NoError(t, err)
|
|
cStackRefI, err := b.RenameStack(ctx, bStack, "organization/project/c")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "organization/project/c", cStackRefI.String())
|
|
cStackRef := cStackRefI.(*localBackendReference)
|
|
|
|
// Check the new stack file now exists and the old one is gone
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, cStackRef))
|
|
assert.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, bStackRef))
|
|
assert.NoError(t, err)
|
|
assert.False(t, stackFileExists)
|
|
|
|
// Check we can still get the history
|
|
history, err := b.GetHistory(ctx, cStackRef, 10, 0)
|
|
assert.NoError(t, err)
|
|
assert.Len(t, history, 1)
|
|
assert.Equal(t, apitype.DestroyUpdate, history[0].Kind)
|
|
}
|
|
|
|
func TestRenameProjectWorks(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Grab the bucket interface to test with
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
|
|
// Create a new stack
|
|
aStackRef, err := lb.parseStackReference("organization/project/a")
|
|
assert.NoError(t, err)
|
|
aStack, err := b.CreateStack(ctx, aStackRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, aStack)
|
|
|
|
// Check the stack file now exists
|
|
stackFileExists, err := lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
|
|
// Fake up some history
|
|
err = lb.addToHistory(ctx, aStackRef, backend.UpdateInfo{Kind: apitype.DestroyUpdate})
|
|
assert.NoError(t, err)
|
|
// And pollute the history folder
|
|
err = lb.bucket.WriteAll(ctx, path.Join(aStackRef.HistoryDir(), "randomfile.txt"), []byte{0, 13}, nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Rename the project and stack
|
|
bStackRefI, err := b.RenameStack(ctx, aStack, "organization/newProject/b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "organization/newProject/b", bStackRefI.String())
|
|
bStackRef := bStackRefI.(*localBackendReference)
|
|
|
|
// Check the new stack file now exists and the old one is gone
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, bStackRef))
|
|
assert.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef))
|
|
assert.NoError(t, err)
|
|
assert.False(t, stackFileExists)
|
|
|
|
// Check we can still get the history
|
|
history, err := b.GetHistory(ctx, bStackRef, 10, 0)
|
|
assert.NoError(t, err)
|
|
assert.Len(t, history, 1)
|
|
assert.Equal(t, apitype.DestroyUpdate, history[0].Kind)
|
|
}
|
|
|
|
func TestLoginToNonExistingFolderFails(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
fakeDir := "file://" + filepath.ToSlash(os.TempDir()) + "/non-existing"
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), fakeDir, nil)
|
|
assert.Error(t, err)
|
|
assert.Nil(t, b)
|
|
}
|
|
|
|
// TestParseEmptyStackFails demonstrates that ParseStackReference returns
|
|
// an error when the stack name is the empty string.TestParseEmptyStackFails
|
|
func TestParseEmptyStackFails(t *testing.T) {
|
|
t.Parallel()
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
_, err = b.ParseStackReference("")
|
|
assert.Error(t, err)
|
|
}
|
|
|
|
// Regression test for https://github.com/pulumi/pulumi/issues/10439
|
|
func TestHtmlEscaping(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sm := b64.NewBase64SecretsManager()
|
|
resources := []*resource.State{
|
|
{
|
|
URN: resource.NewURN("a", "proj", "d:e:f", "a:b:c", "name"),
|
|
Type: "a:b:c",
|
|
Inputs: resource.PropertyMap{
|
|
resource.PropertyKey("html"): resource.NewStringProperty("<html@tags>"),
|
|
},
|
|
},
|
|
}
|
|
|
|
snap := deploy.NewSnapshot(deploy.Manifest{}, sm, resources, nil)
|
|
|
|
sdep, err := stack.SerializeDeployment(snap, snap.SecretsManager, false /* showSecrsts */)
|
|
assert.NoError(t, err)
|
|
|
|
data, err := encoding.JSON.Marshal(sdep)
|
|
assert.NoError(t, err)
|
|
|
|
// Ensure data has the string contents "<html@tags>"", not "\u003chtml\u0026tags\u003e"
|
|
// ImportDeployment below should not modify the data
|
|
assert.Contains(t, string(data), "<html@tags>")
|
|
|
|
udep := &apitype.UntypedDeployment{
|
|
Version: 3,
|
|
Deployment: json.RawMessage(data),
|
|
}
|
|
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Create stack "a" and import a checkpoint with a secret
|
|
aStackRef, err := b.ParseStackReference("organization/project/a")
|
|
assert.NoError(t, err)
|
|
aStack, err := b.CreateStack(ctx, aStackRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, aStack)
|
|
err = b.ImportDeployment(ctx, aStack, udep)
|
|
assert.NoError(t, err)
|
|
|
|
// Ensure the file has the string contents "<html@tags>"", not "\u003chtml\u0026tags\u003e"
|
|
|
|
// Grab the bucket interface to read the file with
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
|
|
chkpath := lb.stackPath(ctx, aStackRef.(*localBackendReference))
|
|
bytes, err := lb.bucket.ReadAll(context.Background(), chkpath)
|
|
assert.NoError(t, err)
|
|
state := string(bytes)
|
|
assert.Contains(t, state, "<html@tags>")
|
|
}
|
|
|
|
func TestLocalBackendRejectsStackInitOptions(t *testing.T) {
|
|
t.Parallel()
|
|
// Here, we provide options that illegally specify a team on a
|
|
// backend that does not support teams. We expect this to create
|
|
// an error later when we call CreateStack.
|
|
illegalOptions := &backend.CreateStackOptions{Teams: []string{"red-team"}}
|
|
|
|
// • Create a mock local backend
|
|
tmpDir := t.TempDir()
|
|
dirURI := fmt.Sprintf("file://%s", filepath.ToSlash(tmpDir))
|
|
local, err := New(context.Background(), diagtest.LogSink(t), dirURI, nil)
|
|
assert.NoError(t, err)
|
|
ctx := context.Background()
|
|
|
|
// • Simulate `pulumi stack init`, passing non-nil init options
|
|
fakeStackRef, err := local.ParseStackReference("organization/b/foobar")
|
|
assert.NoError(t, err)
|
|
_, err = local.CreateStack(ctx, fakeStackRef, "", illegalOptions)
|
|
assert.ErrorIs(t, err, backend.ErrTeamsNotSupported)
|
|
}
|
|
|
|
func TestLegacyFolderStructure(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Make a dummy stack file in the legacy location
|
|
tmpDir := t.TempDir()
|
|
err := os.MkdirAll(path.Join(tmpDir, ".pulumi", "stacks"), os.ModePerm)
|
|
require.NoError(t, err)
|
|
err = os.WriteFile(path.Join(tmpDir, ".pulumi", "stacks", "a.json"), []byte("{}"), os.ModePerm)
|
|
require.NoError(t, err)
|
|
|
|
// Login to a temp dir filestate backend
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
require.NoError(t, err)
|
|
// Check the backend says it's NOT in project mode
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
assert.IsType(t, &legacyReferenceStore{}, lb.store)
|
|
|
|
// Check that list stack shows that stack
|
|
stacks, token, err := b.ListStacks(ctx, backend.ListStacksFilter{}, nil /* inContToken */)
|
|
assert.NoError(t, err)
|
|
assert.Nil(t, token)
|
|
assert.Len(t, stacks, 1)
|
|
assert.Equal(t, "a", stacks[0].Name().String())
|
|
|
|
// Create a new non-project stack
|
|
bRef, err := b.ParseStackReference("b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "b", bRef.String())
|
|
bStack, err := b.CreateStack(ctx, bRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "b", bStack.Ref().String())
|
|
assert.FileExists(t, path.Join(tmpDir, ".pulumi", "stacks", "b.json"))
|
|
}
|
|
|
|
func TestListStacksFilter(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Login to a temp dir filestate backend
|
|
ctx := context.Background()
|
|
tmpDir := t.TempDir()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
require.NoError(t, err)
|
|
|
|
// Create two different project stack
|
|
aRef, err := b.ParseStackReference("organization/proj1/a")
|
|
assert.NoError(t, err)
|
|
_, err = b.CreateStack(ctx, aRef, "", nil)
|
|
assert.NoError(t, err)
|
|
|
|
bRef, err := b.ParseStackReference("organization/proj2/b")
|
|
assert.NoError(t, err)
|
|
_, err = b.CreateStack(ctx, bRef, "", nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Check that list stack with a filter only shows one stack
|
|
filter := "proj1"
|
|
stacks, token, err := b.ListStacks(ctx, backend.ListStacksFilter{
|
|
Project: &filter,
|
|
}, nil /* inContToken */)
|
|
assert.NoError(t, err)
|
|
assert.Nil(t, token)
|
|
assert.Len(t, stacks, 1)
|
|
assert.Equal(t, "organization/proj1/a", stacks[0].Name().String())
|
|
}
|
|
|
|
func TestOptIntoLegacyFolderStructure(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
s := make(env.MapStore)
|
|
s[env.SelfManagedStateLegacyLayout.Var().Name()] = "true"
|
|
b, err := newLocalBackend(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil,
|
|
&localBackendOptions{Env: env.NewEnv(s)},
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
// Verify that a new stack is created in the legacy location.
|
|
foo, err := b.ParseStackReference("foo")
|
|
require.NoError(t, err)
|
|
|
|
_, err = b.CreateStack(ctx, foo, "", nil)
|
|
require.NoError(t, err)
|
|
assert.FileExists(t, filepath.Join(tmpDir, ".pulumi", "stacks", "foo.json"))
|
|
}
|
|
|
|
// Verifies that the StackReference.String method
|
|
// takes the current project name into account,
|
|
// even if the current project name changes
|
|
// after the stack reference is created.
|
|
func TestStackReferenceString_currentProjectChange(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
dir := t.TempDir()
|
|
ctx := context.Background()
|
|
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(dir), nil)
|
|
require.NoError(t, err)
|
|
|
|
foo, err := b.ParseStackReference("organization/proj1/foo")
|
|
require.NoError(t, err)
|
|
|
|
bar, err := b.ParseStackReference("organization/proj2/bar")
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, "organization/proj1/foo", foo.String())
|
|
assert.Equal(t, "organization/proj2/bar", bar.String())
|
|
|
|
// Change the current project name
|
|
b.SetCurrentProject(&workspace.Project{Name: "proj1"})
|
|
|
|
assert.Equal(t, "foo", foo.String())
|
|
assert.Equal(t, "organization/proj2/bar", bar.String())
|
|
}
|
|
|
|
// Verifies that there's no data race in calling StackReference.String
|
|
// and localBackend.SetCurrentProject concurrently.
|
|
func TestStackReferenceString_currentProjectChange_race(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const N = 1000
|
|
|
|
dir := t.TempDir()
|
|
ctx := context.Background()
|
|
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(dir), nil)
|
|
require.NoError(t, err)
|
|
|
|
projects := make([]*workspace.Project, N)
|
|
refs := make([]backend.StackReference, N)
|
|
for i := 0; i < N; i++ {
|
|
name := fmt.Sprintf("proj%d", i)
|
|
projects[i] = &workspace.Project{Name: tokens.PackageName(name)}
|
|
refs[i], err = b.ParseStackReference(fmt.Sprintf("organization/%v/foo", name))
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
// To exercise this data race, we'll have two goroutines.
|
|
// One goroutine will call StackReference.String repeatedly
|
|
// on all the stack references,
|
|
// and the other goroutine will call localBackend.SetCurrentProject
|
|
// with all the projects.
|
|
|
|
var wg sync.WaitGroup
|
|
ready := make(chan struct{}) // both goroutines wait on this
|
|
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
<-ready
|
|
for i := 0; i < N; i++ {
|
|
_ = refs[i].String()
|
|
}
|
|
}()
|
|
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
<-ready
|
|
for i := 0; i < N; i++ {
|
|
b.SetCurrentProject(projects[i])
|
|
}
|
|
}()
|
|
|
|
close(ready) // start racing
|
|
wg.Wait()
|
|
}
|
|
|
|
func TestProjectFolderStructure(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Login to a temp dir filestate backend
|
|
|
|
// Make a dummy file in the legacy location which isn't a stack file, we should still automatically turn
|
|
// this into project mode.
|
|
tmpDir := t.TempDir()
|
|
err := os.MkdirAll(path.Join(tmpDir, ".pulumi", "plugins"), os.ModePerm)
|
|
require.NoError(t, err)
|
|
err = os.MkdirAll(path.Join(tmpDir, ".pulumi", "stacks"), os.ModePerm)
|
|
require.NoError(t, err)
|
|
err = os.WriteFile(path.Join(tmpDir, ".pulumi", "stacks", "a.txt"), []byte("{}"), os.ModePerm)
|
|
require.NoError(t, err)
|
|
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), nil)
|
|
assert.NoError(t, err)
|
|
|
|
// Check the backend says it's in project mode
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
assert.IsType(t, &projectReferenceStore{}, lb.store)
|
|
|
|
// Make a dummy stack file in the new project location
|
|
err = os.MkdirAll(path.Join(tmpDir, ".pulumi", "stacks", "testproj"), os.ModePerm)
|
|
assert.NoError(t, err)
|
|
err = os.WriteFile(path.Join(tmpDir, ".pulumi", "stacks", "testproj", "a.json"), []byte("{}"), os.ModePerm)
|
|
assert.NoError(t, err)
|
|
|
|
// Check that testproj is reported as existing
|
|
exists, err := b.DoesProjectExist(ctx, "", "testproj")
|
|
assert.NoError(t, err)
|
|
assert.True(t, exists)
|
|
|
|
// Check that list stack shows that stack
|
|
stacks, token, err := b.ListStacks(ctx, backend.ListStacksFilter{}, nil /* inContToken */)
|
|
assert.NoError(t, err)
|
|
assert.Nil(t, token)
|
|
assert.Len(t, stacks, 1)
|
|
assert.Equal(t, "organization/testproj/a", stacks[0].Name().String())
|
|
|
|
// Create a new project stack
|
|
bRef, err := b.ParseStackReference("organization/testproj/b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "organization/testproj/b", bRef.String())
|
|
bStack, err := b.CreateStack(ctx, bRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "organization/testproj/b", bStack.Ref().String())
|
|
assert.FileExists(t, path.Join(tmpDir, ".pulumi", "stacks", "testproj", "b.json"))
|
|
}
|
|
|
|
func chdir(t *testing.T, dir string) {
|
|
cwd, err := os.Getwd()
|
|
require.NoError(t, err)
|
|
require.NoError(t, os.Chdir(dir)) // Set directory
|
|
t.Cleanup(func() {
|
|
require.NoError(t, os.Chdir(cwd)) // Restore directory
|
|
restoredDir, err := os.Getwd()
|
|
require.NoError(t, err)
|
|
require.Equal(t, cwd, restoredDir)
|
|
})
|
|
}
|
|
|
|
//nolint:paralleltest // mutates cwd
|
|
func TestProjectNameMustMatch(t *testing.T) {
|
|
// Create a new project
|
|
projectDir := t.TempDir()
|
|
pyaml := filepath.Join(projectDir, "Pulumi.yaml")
|
|
err := os.WriteFile(pyaml, []byte("name: my-project\nruntime: test"), 0o600)
|
|
require.NoError(t, err)
|
|
proj, err := workspace.LoadProject(pyaml)
|
|
require.NoError(t, err)
|
|
|
|
chdir(t, projectDir)
|
|
|
|
// Login to a temp dir filestate backend
|
|
tmpDir := t.TempDir()
|
|
ctx := context.Background()
|
|
b, err := New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(tmpDir), proj)
|
|
require.NoError(t, err)
|
|
|
|
// Create a new implicit-project stack
|
|
aRef, err := b.ParseStackReference("a")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "a", aRef.String())
|
|
aStack, err := b.CreateStack(ctx, aRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "a", aStack.Ref().String())
|
|
assert.FileExists(t, path.Join(tmpDir, ".pulumi", "stacks", "my-project", "a.json"))
|
|
|
|
// Create a new project stack with the wrong project name
|
|
bRef, err := b.ParseStackReference("organization/not-my-project/b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "organization/not-my-project/b", bRef.String())
|
|
bStack, err := b.CreateStack(ctx, bRef, "", nil)
|
|
assert.Error(t, err)
|
|
assert.Nil(t, bStack)
|
|
|
|
// Create a new project stack with the right project name
|
|
cRef, err := b.ParseStackReference("organization/my-project/c")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "c", cRef.String())
|
|
cStack, err := b.CreateStack(ctx, cRef, "", nil)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "c", cStack.Ref().String())
|
|
assert.FileExists(t, path.Join(tmpDir, ".pulumi", "stacks", "my-project", "c.json"))
|
|
}
|
|
|
|
func TestNew_legacyFileWarning(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Verifies the names of files printed in warnings
|
|
// when legacy files are found while running in project mode.
|
|
|
|
tests := []struct {
|
|
desc string
|
|
files map[string]string
|
|
env env.MapStore
|
|
wantOut string
|
|
}{
|
|
{
|
|
desc: "no legacy stacks",
|
|
files: map[string]string{
|
|
// Should ignore non-stack files.
|
|
".pulumi/foo/extraneous_file": "",
|
|
},
|
|
},
|
|
{
|
|
desc: "legacy stacks",
|
|
files: map[string]string{
|
|
".pulumi/stacks/a.json": "{}",
|
|
".pulumi/stacks/b.json": "{}",
|
|
".pulumi/stacks/c.json.bak": "{}", // should ignore backup files
|
|
},
|
|
wantOut: "warning: Found legacy stack files in state store:\n" +
|
|
" - a\n" +
|
|
" - b\n" +
|
|
"Please run 'pulumi state upgrade' to migrate them to the new format.\n" +
|
|
"Set PULUMI_SELF_MANAGED_STATE_NO_LEGACY_WARNING=1 to disable this warning.\n",
|
|
},
|
|
{
|
|
desc: "warning opt-out",
|
|
files: map[string]string{
|
|
".pulumi/stacks/a.json": "{}",
|
|
".pulumi/stacks/b.json": "{}",
|
|
},
|
|
env: map[string]string{
|
|
"PULUMI_SELF_MANAGED_STATE_NO_LEGACY_WARNING": "true",
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
tt := tt
|
|
t.Run(tt.desc, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
stateDir := t.TempDir()
|
|
bucket, err := fileblob.OpenBucket(stateDir, nil)
|
|
require.NoError(t, err)
|
|
|
|
ctx := context.Background()
|
|
require.NoError(t,
|
|
bucket.WriteAll(ctx, ".pulumi/meta.yaml", []byte("version: 1"), nil),
|
|
"write meta.yaml")
|
|
|
|
for path, contents := range tt.files {
|
|
require.NoError(t, bucket.WriteAll(ctx, path, []byte(contents), nil),
|
|
"write %q", path)
|
|
}
|
|
|
|
var buff bytes.Buffer
|
|
sink := diag.DefaultSink(io.Discard, &buff, diag.FormatOptions{Color: colors.Never})
|
|
|
|
_, err = newLocalBackend(ctx, sink, "file://"+filepath.ToSlash(stateDir), nil,
|
|
&localBackendOptions{Env: env.NewEnv(tt.env)})
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, tt.wantOut, buff.String())
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestLegacyUpgrade(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Make a dummy stack file in the legacy location
|
|
tmpDir := t.TempDir()
|
|
err := os.MkdirAll(path.Join(tmpDir, ".pulumi", "stacks"), os.ModePerm)
|
|
require.NoError(t, err)
|
|
err = os.WriteFile(path.Join(tmpDir, ".pulumi", "stacks", "a.json"), []byte(`{
|
|
"latest": {
|
|
"resources": [
|
|
{
|
|
"type": "package:module:resource",
|
|
"urn": "urn:pulumi:stack::project::package:module:resource::name"
|
|
}
|
|
]
|
|
}
|
|
}`), os.ModePerm)
|
|
require.NoError(t, err)
|
|
|
|
var output bytes.Buffer
|
|
sink := diag.DefaultSink(&output, &output, diag.FormatOptions{Color: colors.Never})
|
|
|
|
// Login to a temp dir filestate backend
|
|
ctx := context.Background()
|
|
b, err := New(ctx, sink, "file://"+filepath.ToSlash(tmpDir), nil)
|
|
require.NoError(t, err)
|
|
// Check the backend says it's NOT in project mode
|
|
lb, ok := b.(*localBackend)
|
|
assert.True(t, ok)
|
|
assert.NotNil(t, lb)
|
|
assert.IsType(t, &legacyReferenceStore{}, lb.store)
|
|
|
|
err = lb.Upgrade(ctx, nil /* opts */)
|
|
require.NoError(t, err)
|
|
assert.IsType(t, &projectReferenceStore{}, lb.store)
|
|
|
|
assert.Contains(t, output.String(), "Upgraded 1 stack(s) to project mode")
|
|
|
|
// Check that a has been moved
|
|
aStackRef, err := lb.parseStackReference("organization/project/a")
|
|
require.NoError(t, err)
|
|
stackFileExists, err := lb.bucket.Exists(ctx, lb.stackPath(ctx, aStackRef))
|
|
require.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
|
|
// Write b.json and upgrade again
|
|
err = os.WriteFile(path.Join(tmpDir, ".pulumi", "stacks", "b.json"), []byte(`{
|
|
"latest": {
|
|
"resources": [
|
|
{
|
|
"type": "package:module:resource",
|
|
"urn": "urn:pulumi:stack::other-project::package:module:resource::name"
|
|
}
|
|
]
|
|
}
|
|
}`), os.ModePerm)
|
|
require.NoError(t, err)
|
|
|
|
err = lb.Upgrade(ctx, nil /* opts */)
|
|
require.NoError(t, err)
|
|
|
|
// Check that b has been moved
|
|
bStackRef, err := lb.parseStackReference("organization/other-project/b")
|
|
require.NoError(t, err)
|
|
stackFileExists, err = lb.bucket.Exists(ctx, lb.stackPath(ctx, bStackRef))
|
|
require.NoError(t, err)
|
|
assert.True(t, stackFileExists)
|
|
}
|
|
|
|
func TestLegacyUpgrade_partial(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Verifies that we can upgrade a subset of stacks.
|
|
|
|
stateDir := t.TempDir()
|
|
bucket, err := fileblob.OpenBucket(stateDir, nil)
|
|
require.NoError(t, err)
|
|
|
|
ctx := context.Background()
|
|
require.NoError(t,
|
|
bucket.WriteAll(ctx, ".pulumi/stacks/foo.json", []byte(`{
|
|
"latest": {
|
|
"resources": [
|
|
{
|
|
"type": "package:module:resource",
|
|
"urn": "urn:pulumi:stack::project::package:module:resource::name"
|
|
}
|
|
]
|
|
}
|
|
}`), nil))
|
|
require.NoError(t,
|
|
// no resources, can't guess project name
|
|
bucket.WriteAll(ctx, ".pulumi/stacks/bar.json",
|
|
[]byte(`{"latest": {"resources": []}}`), nil))
|
|
|
|
var buff bytes.Buffer
|
|
sink := diag.DefaultSink(io.Discard, &buff, diag.FormatOptions{Color: colors.Never})
|
|
b, err := New(ctx, sink, "file://"+filepath.ToSlash(stateDir), nil)
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, b.Upgrade(ctx, nil /* opts */))
|
|
assert.Contains(t, buff.String(), `Skipping stack "bar": no project name found`)
|
|
|
|
exists, err := bucket.Exists(ctx, ".pulumi/stacks/project/foo.json")
|
|
require.NoError(t, err)
|
|
assert.True(t, exists, "foo was not migrated")
|
|
|
|
ref, err := b.ParseStackReference("organization/project/foo")
|
|
require.NoError(t, err)
|
|
assert.Equal(t, tokens.QName("organization/project/foo"), ref.FullyQualifiedName())
|
|
}
|
|
|
|
// When a stack project could not be determined,
|
|
// we should fill it in with ProjectsForDetachedStacks.
|
|
func TestLegacyUpgrade_ProjectsForDetachedStacks(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
stateDir := t.TempDir()
|
|
bucket, err := fileblob.OpenBucket(stateDir, nil)
|
|
require.NoError(t, err)
|
|
|
|
// Write a few empty stacks.
|
|
// These stacks have no resources, so we can't guess the project name.
|
|
ctx := context.Background()
|
|
for _, stack := range []string{"foo", "bar", "baz"} {
|
|
statePath := path.Join(".pulumi", "stacks", stack+".json")
|
|
require.NoError(t,
|
|
bucket.WriteAll(ctx, statePath,
|
|
[]byte(`{"latest": {"resources": []}}`), nil),
|
|
"write stack %s", stack)
|
|
}
|
|
|
|
var stderr bytes.Buffer
|
|
sink := diag.DefaultSink(io.Discard, &stderr, diag.FormatOptions{Color: colors.Never})
|
|
b, err := New(ctx, sink, "file://"+filepath.ToSlash(stateDir), nil)
|
|
require.NoError(t, err, "initialize backend")
|
|
|
|
// For the first two stacks, we'll return project names to upgrade them.
|
|
// For the third stack, we will not set a project name, and it should be skipped.
|
|
err = b.Upgrade(ctx, &UpgradeOptions{
|
|
ProjectsForDetachedStacks: func(stacks []tokens.StackName) (projects []tokens.Name, err error) {
|
|
assert.ElementsMatch(t, []tokens.StackName{
|
|
tokens.MustParseStackName("foo"),
|
|
tokens.MustParseStackName("bar"),
|
|
tokens.MustParseStackName("baz"),
|
|
}, stacks)
|
|
|
|
projects = make([]tokens.Name, len(stacks))
|
|
for idx, stack := range stacks {
|
|
switch stack.String() {
|
|
case "foo":
|
|
projects[idx] = "proj1"
|
|
case "bar":
|
|
projects[idx] = "proj2"
|
|
case "baz":
|
|
// Leave baz detached.
|
|
}
|
|
}
|
|
return projects, nil
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
for _, stack := range []string{"foo", "bar"} {
|
|
assert.NotContains(t, stderr.String(), fmt.Sprintf("Skipping stack %q", stack))
|
|
}
|
|
assert.Contains(t, stderr.String(), fmt.Sprintf("Skipping stack %q", "baz"))
|
|
|
|
wantFiles := []string{
|
|
".pulumi/stacks/proj1/foo.json",
|
|
".pulumi/stacks/proj2/bar.json",
|
|
".pulumi/stacks/baz.json",
|
|
}
|
|
for _, file := range wantFiles {
|
|
exists, err := bucket.Exists(ctx, file)
|
|
require.NoError(t, err, "exists(%q)", file)
|
|
assert.True(t, exists, "file %q must exist", file)
|
|
}
|
|
}
|
|
|
|
// When a stack project could not be determined
|
|
// and ProjectsForDetachedStacks returns an error,
|
|
// the upgrade should fail.
|
|
func TestLegacyUpgrade_ProjectsForDetachedStacks_error(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
stateDir := t.TempDir()
|
|
bucket, err := fileblob.OpenBucket(stateDir, nil)
|
|
require.NoError(t, err)
|
|
|
|
ctx := context.Background()
|
|
|
|
// We have one stack with a guessable project name, and one without.
|
|
// If ProjectsForDetachedStacks returns an error, the upgrade should
|
|
// fail for both because the user likely cancelled the upgrade.
|
|
require.NoError(t,
|
|
bucket.WriteAll(ctx, ".pulumi/stacks/foo.json", []byte(`{
|
|
"latest": {
|
|
"resources": [
|
|
{
|
|
"type": "package:module:resource",
|
|
"urn": "urn:pulumi:stack::project::package:module:resource::name"
|
|
}
|
|
]
|
|
}
|
|
}`), nil))
|
|
require.NoError(t,
|
|
bucket.WriteAll(ctx, ".pulumi/stacks/bar.json",
|
|
[]byte(`{"latest": {"resources": []}}`), nil))
|
|
|
|
sink := diag.DefaultSink(io.Discard, iotest.LogWriter(t), diag.FormatOptions{Color: colors.Never})
|
|
b, err := New(ctx, sink, "file://"+filepath.ToSlash(stateDir), nil)
|
|
require.NoError(t, err)
|
|
|
|
giveErr := errors.New("canceled operation")
|
|
err = b.Upgrade(ctx, &UpgradeOptions{
|
|
ProjectsForDetachedStacks: func(stacks []tokens.StackName) (projects []tokens.Name, err error) {
|
|
assert.Equal(t, []tokens.StackName{
|
|
tokens.MustParseStackName("bar"),
|
|
}, stacks)
|
|
return nil, giveErr
|
|
},
|
|
})
|
|
require.Error(t, err)
|
|
assert.ErrorIs(t, err, giveErr)
|
|
|
|
wantFiles := []string{
|
|
".pulumi/stacks/foo.json",
|
|
".pulumi/stacks/bar.json",
|
|
}
|
|
for _, file := range wantFiles {
|
|
exists, err := bucket.Exists(ctx, file)
|
|
require.NoError(t, err, "exists(%q)", file)
|
|
assert.True(t, exists, "file %q must exist", file)
|
|
}
|
|
}
|
|
|
|
// If an upgrade failed because we couldn't write the meta.yaml,
|
|
// the stacks should be left in legacy mode.
|
|
func TestLegacyUpgrade_writeMetaError(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
stateDir := t.TempDir()
|
|
bucket, err := fileblob.OpenBucket(stateDir, nil)
|
|
require.NoError(t, err)
|
|
|
|
ctx := context.Background()
|
|
require.NoError(t,
|
|
bucket.WriteAll(ctx, ".pulumi/stacks/foo.json", []byte(`{
|
|
"latest": {
|
|
"resources": [
|
|
{
|
|
"type": "package:module:resource",
|
|
"urn": "urn:pulumi:stack::project::package:module:resource::name"
|
|
}
|
|
]
|
|
}
|
|
}`), nil))
|
|
|
|
// To prevent a write to meta.yaml, we'll create a directory with that name.
|
|
// The system will reject creating a file with the same name.
|
|
require.NoError(t, os.MkdirAll(filepath.Join(stateDir, ".pulumi", "meta.yaml"), 0o755))
|
|
|
|
var buff bytes.Buffer
|
|
sink := diag.DefaultSink(io.Discard, &buff, diag.FormatOptions{Color: colors.Never})
|
|
b, err := New(ctx, sink, "file://"+filepath.ToSlash(stateDir), nil)
|
|
require.NoError(t, err)
|
|
|
|
require.Error(t, b.Upgrade(ctx, nil /* opts */))
|
|
|
|
stderr := buff.String()
|
|
assert.Contains(t, stderr, "error: Could not write new state metadata file")
|
|
assert.Contains(t, stderr, "Please verify that the storage is writable")
|
|
|
|
assert.FileExists(t, filepath.Join(stateDir, ".pulumi", "stacks", "foo.json"),
|
|
"foo.json should not have been upgraded")
|
|
}
|
|
|
|
func TestNew_unsupportedStoreVersion(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Verifies that we fail to initialize a backend if the store version is
|
|
// newer than the CLI version.
|
|
|
|
stateDir := t.TempDir()
|
|
bucket, err := fileblob.OpenBucket(stateDir, nil)
|
|
require.NoError(t, err)
|
|
|
|
// Set up a meta.yaml "from the future".
|
|
ctx := context.Background()
|
|
require.NoError(t,
|
|
bucket.WriteAll(ctx, ".pulumi/meta.yaml", []byte("version: 999999999"), nil))
|
|
|
|
_, err = New(ctx, diagtest.LogSink(t), "file://"+filepath.ToSlash(stateDir), nil)
|
|
assert.ErrorContains(t, err, "state store unsupported")
|
|
assert.ErrorContains(t, err, "'meta.yaml' version (999999999) is not supported")
|
|
}
|
|
|
|
// TestSerializeTimestampRFC3339 captures our expectations that Created and Modified will be serialized to
|
|
// RFC3339.
|
|
func TestSerializeTimestampRFC3339(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
created := time.Now().UTC()
|
|
modified := created.Add(time.Hour)
|
|
|
|
deployment, err := makeUntypedDeploymentTimestamp("b", "123abc",
|
|
"v1:C7H2a7/Ietk=:v1:yfAd1zOi6iY9DRIB:dumdsr+H89VpHIQWdB01XEFqYaYjAg==", &created, &modified)
|
|
assert.NoError(t, err)
|
|
|
|
createdStr := created.Format(time.RFC3339Nano)
|
|
modifiedStr := modified.Format(time.RFC3339Nano)
|
|
assert.Contains(t, string(deployment.Deployment), createdStr)
|
|
assert.Contains(t, string(deployment.Deployment), modifiedStr)
|
|
}
|
|
|
|
func TestUpgrade_manyFailures(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const (
|
|
numStacks = 100
|
|
badStackBody = `{"latest": {"resources": []}}`
|
|
)
|
|
|
|
tmpDir := t.TempDir()
|
|
|
|
bucket, err := fileblob.OpenBucket(tmpDir, nil)
|
|
require.NoError(t, err)
|
|
ctx := context.Background()
|
|
for i := 0; i < numStacks; i++ {
|
|
stackPath := path.Join(".pulumi", "stacks", fmt.Sprintf("stack-%d.json", i))
|
|
require.NoError(t, bucket.WriteAll(ctx, stackPath, []byte(badStackBody), nil))
|
|
}
|
|
|
|
var output bytes.Buffer
|
|
sink := diag.DefaultSink(io.Discard, &output, diag.FormatOptions{Color: colors.Never})
|
|
|
|
// Login to a temp dir filestate backend
|
|
b, err := New(ctx, sink, "file://"+filepath.ToSlash(tmpDir), nil)
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, b.Upgrade(ctx, nil /* opts */))
|
|
out := output.String()
|
|
for i := 0; i < numStacks; i++ {
|
|
assert.Contains(t, out, fmt.Sprintf(`Skipping stack "stack-%d"`, i))
|
|
}
|
|
}
|
|
|
|
func TestCreateStack_gzip(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
stateDir := t.TempDir()
|
|
ctx := context.Background()
|
|
|
|
s := make(env.MapStore)
|
|
s[env.SelfManagedGzip.Var().Name()] = "true"
|
|
|
|
b, err := newLocalBackend(
|
|
ctx,
|
|
diagtest.LogSink(t), "file://"+filepath.ToSlash(stateDir),
|
|
&workspace.Project{Name: "testproj"},
|
|
&localBackendOptions{Env: env.NewEnv(s)},
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
fooRef, err := b.ParseStackReference("foo")
|
|
require.NoError(t, err)
|
|
|
|
_, err = b.CreateStack(ctx, fooRef, "", nil)
|
|
require.NoError(t, err)
|
|
|
|
// With PULUMI_SELF_MANAGED_STATE_GZIP enabled,
|
|
// we'll store state into gzipped files.
|
|
assert.FileExists(t, filepath.Join(stateDir, ".pulumi", "stacks", "testproj", "foo.json.gz"))
|
|
}
|
|
|
|
func TestCreateStack_retainCheckpoints(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
stateDir := t.TempDir()
|
|
ctx := context.Background()
|
|
|
|
s := make(env.MapStore)
|
|
s[env.SelfManagedRetainCheckpoints.Var().Name()] = "true"
|
|
|
|
b, err := newLocalBackend(
|
|
ctx,
|
|
diagtest.LogSink(t), "file://"+filepath.ToSlash(stateDir),
|
|
&workspace.Project{Name: "testproj"},
|
|
&localBackendOptions{Env: env.NewEnv(s)},
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
fooRef, err := b.ParseStackReference("foo")
|
|
require.NoError(t, err)
|
|
|
|
_, err = b.CreateStack(ctx, fooRef, "", nil)
|
|
require.NoError(t, err)
|
|
|
|
// With PULUMI_RETAIN_CHECKPOINTS enabled,
|
|
// we'll make copies of files under $orig.$timestamp.
|
|
// Since we can't predict the timestamp,
|
|
// we'll just check that there's at least one file
|
|
// with a timestamp extension.
|
|
got, err := filepath.Glob(
|
|
filepath.Join(stateDir, ".pulumi", "stacks", "testproj", "foo.json.*"))
|
|
require.NoError(t, err)
|
|
|
|
checkpointExtRe := regexp.MustCompile(`^\.[0-9]+$`)
|
|
var found bool
|
|
for _, f := range got {
|
|
if checkpointExtRe.MatchString(filepath.Ext(f)) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
assert.True(t, found,
|
|
"file with a timestamp extension not found in %v", got)
|
|
}
|