mirror of https://github.com/pulumi/pulumi.git
270 lines
6.7 KiB
Go
270 lines
6.7 KiB
Go
// Copyright 2016-2023, 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 main
|
|
|
|
import (
|
|
"fmt"
|
|
"path/filepath"
|
|
"runtime"
|
|
"runtime/debug"
|
|
"sync"
|
|
|
|
mapset "github.com/deckarep/golang-set/v2"
|
|
|
|
"github.com/pulumi/pulumi/pkg/v3/display"
|
|
"github.com/pulumi/pulumi/pkg/v3/resource/deploy"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
|
|
"github.com/pulumi/pulumi/sdk/v3/go/common/tokens"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
// L holds the state for the current language test.
|
|
//
|
|
// It provides an interface similar to testing.T,
|
|
// allowing its use with testing libraries like Testify.
|
|
type L struct {
|
|
mu sync.RWMutex // guards the fields below
|
|
|
|
// Whether this test has already failed.
|
|
failed bool
|
|
|
|
// Messages logged to l.Errorf or l.Logf.
|
|
logs []string
|
|
|
|
// Functions marked helpers with L.Helper().
|
|
|
|
// These names are from the runtime.Frame.Function field.
|
|
// They're fully qualified with package names.
|
|
helpers mapset.Set[string]
|
|
}
|
|
|
|
// Helper marks the calling function as a test helper function.
|
|
// When printing file and line information, that function will be skipped.
|
|
func (l *L) Helper() {
|
|
pc, _, _, ok := runtime.Caller(1) // skip this function
|
|
if !ok {
|
|
return // unlikely but not worth panicking over
|
|
}
|
|
|
|
frame, _ := runtime.CallersFrames([]uintptr{pc}).Next()
|
|
if frame.Function == "" {
|
|
return
|
|
}
|
|
|
|
l.mu.Lock()
|
|
defer l.mu.Unlock()
|
|
if l.helpers == nil {
|
|
l.helpers = mapset.NewSet[string]()
|
|
}
|
|
l.helpers.Add(frame.Function)
|
|
}
|
|
|
|
// FailNow marks this test as having failed and halts execution.
|
|
func (l *L) FailNow() {
|
|
l.Fail()
|
|
runtime.Goexit()
|
|
}
|
|
|
|
// Fail marks this test as failed but keeps executing.
|
|
func (l *L) Fail() {
|
|
l.mu.Lock()
|
|
defer l.mu.Unlock()
|
|
|
|
l.failed = true
|
|
}
|
|
|
|
// Failed returns whether this test has failed.
|
|
func (l *L) Failed() bool {
|
|
l.mu.RLock()
|
|
defer l.mu.RUnlock()
|
|
|
|
return l.failed
|
|
}
|
|
|
|
// Errorf records the given error message and marks this test as failed.
|
|
func (l *L) Errorf(format string, args ...interface{}) {
|
|
l.log(1, fmt.Sprintf(format, args...))
|
|
l.Fail()
|
|
}
|
|
|
|
// Logf records the given message in the L's logs.
|
|
func (l *L) Logf(format string, args ...interface{}) {
|
|
l.log(1, fmt.Sprintf(format, args...))
|
|
}
|
|
|
|
// log records the given message in the L's logs.
|
|
//
|
|
// Skip specifies the number of stack frames to skip
|
|
// when recording the caller's location.
|
|
// 0 refers to the immediate caller of log.
|
|
//
|
|
// Typically, when used from an exported method on L,
|
|
// most callers will want to pass skip=1 to skip themselves
|
|
// and record the location of their caller.
|
|
func (l *L) log(skip int, msg string) {
|
|
file, line := "???", 1
|
|
if frame, ok := l.callerFrame(skip + 1); ok {
|
|
file, line = frame.File, frame.Line
|
|
}
|
|
|
|
msg = fmt.Sprintf("%s:%d: %s", filepath.Base(file), line, msg)
|
|
l.mu.Lock()
|
|
l.logs = append(l.logs, msg)
|
|
l.mu.Unlock()
|
|
}
|
|
|
|
// Maximal stack depth to search for the caller's frame.
|
|
const _maxStackDepth = 50
|
|
|
|
// callerFrame searches the call stack for the first frame
|
|
// that isn't a helper function.
|
|
//
|
|
// skip specifies the initial number of frames to skip
|
|
// with 0 referring to the immediate caller of callerFrame.
|
|
func (l *L) callerFrame(skip int) (frame runtime.Frame, ok bool) {
|
|
var pc [_maxStackDepth]uintptr
|
|
n := runtime.Callers(skip+2, pc[:]) // skip runtime.Callers and callerFrame
|
|
if n == 0 {
|
|
return frame, false
|
|
}
|
|
|
|
l.mu.RLock()
|
|
defer l.mu.RUnlock()
|
|
|
|
frames := runtime.CallersFrames(pc[:n])
|
|
for {
|
|
frame, more := frames.Next()
|
|
if !l.helpers.Contains(frame.Function) {
|
|
// Not a helper. Use this frame.
|
|
return frame, true
|
|
}
|
|
if !more {
|
|
break
|
|
}
|
|
}
|
|
return frame, false // no non-helper frames found
|
|
}
|
|
|
|
// WithL runs the given function with a new L,
|
|
// blocking until the function returns.
|
|
//
|
|
// It returns the information recorded by the L.
|
|
func WithL(f func(*L)) LResult {
|
|
// To be able to implement FailNow in the L,
|
|
// we need to run it in a separate goroutine
|
|
// so that we can call runtime.Goexit.
|
|
done := make(chan struct{})
|
|
var l L
|
|
go func() {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
l.failed = true
|
|
l.logs = append(l.logs,
|
|
fmt.Sprintf("panic: %v\n\n%s", r, debug.Stack()))
|
|
}
|
|
|
|
close(done)
|
|
}()
|
|
|
|
f(&l)
|
|
}()
|
|
<-done
|
|
|
|
return LResult{
|
|
Failed: l.failed,
|
|
Messages: l.logs,
|
|
}
|
|
}
|
|
|
|
// LResult is the result of running a language test.
|
|
type LResult struct {
|
|
// Failed is true if the test failed.
|
|
Failed bool
|
|
|
|
// Messages contains the messages logged by the test.
|
|
//
|
|
// This doesn't necessarily mean that the test failed.
|
|
// For example, a test may log debugging information
|
|
// that is only useful when the test fails.
|
|
Messages []string
|
|
}
|
|
|
|
// TestingT is a subset of the testing.T interface.
|
|
// [L] implements this interface.
|
|
type TestingT interface {
|
|
Helper()
|
|
FailNow()
|
|
Fail()
|
|
Failed() bool
|
|
Errorf(string, ...interface{})
|
|
Logf(string, ...interface{})
|
|
}
|
|
|
|
var (
|
|
_ TestingT = (*L)(nil)
|
|
_ require.TestingT = (TestingT)(nil) // ensure testify compatibility
|
|
)
|
|
|
|
func assertStackResource(t TestingT, err error, changes display.ResourceChanges) (ok bool) {
|
|
t.Helper()
|
|
|
|
ok = true
|
|
ok = ok && assert.Nil(t, err, "expected no error, got %v", err)
|
|
ok = ok && assert.NotEmpty(t, changes, "expected at least 1 StepOp")
|
|
ok = ok && assert.NotZero(t, changes[deploy.OpCreate], "expected at least 1 Create")
|
|
return ok
|
|
}
|
|
|
|
func requireStackResource(t TestingT, err error, changes display.ResourceChanges) {
|
|
t.Helper()
|
|
|
|
if !assertStackResource(t, err, changes) {
|
|
t.FailNow()
|
|
}
|
|
}
|
|
|
|
func requireSingleResource(t TestingT, resources []*resource.State, typ tokens.Type) *resource.State {
|
|
t.Helper()
|
|
|
|
var result *resource.State
|
|
for _, res := range resources {
|
|
if res.Type == typ {
|
|
require.Nil(t, result, "expected exactly 1 resource of type %q, got multiple", typ)
|
|
result = res
|
|
}
|
|
}
|
|
|
|
require.NotNil(t, result, "expected exactly 1 resource of type %q, got none", typ)
|
|
return result
|
|
}
|
|
|
|
// assertPropertyMapMember asserts that the given property map has a member with the given key and value.
|
|
func assertPropertyMapMember(
|
|
t TestingT,
|
|
props resource.PropertyMap,
|
|
key string,
|
|
want resource.PropertyValue,
|
|
) (ok bool) {
|
|
t.Helper()
|
|
|
|
got, ok := props[resource.PropertyKey(key)]
|
|
if !assert.True(t, ok, "expected property %q", key) {
|
|
return false
|
|
}
|
|
|
|
return assert.Equal(t, want, got, "expected property %q to be %v", key, want)
|
|
}
|