pulumi/tests/integration/resource_refs_get_resource/go/main.go

190 lines
5.2 KiB
Go

// Copyright 2016-2022, Pulumi Corporation. All rights reserved.
//go:build !all
// +build !all
package main
import (
"errors"
"fmt"
"reflect"
"time"
"github.com/blang/semver"
"github.com/pulumi/pulumi/sdk/v3/go/pulumi"
"github.com/pulumi/pulumi/sdk/v3/go/pulumi/internals"
)
type Child struct {
pulumi.ResourceState
Message pulumi.StringOutput `pulumi:"message"`
}
func NewChild(ctx *pulumi.Context, name string, message pulumi.StringInput,
opts ...pulumi.ResourceOption,
) (*Child, error) {
component := &Child{}
if err := ctx.RegisterComponentResource("test:index:Child", name, component, opts...); err != nil {
return nil, err
}
if err := ctx.RegisterResourceOutputs(component, pulumi.Map{
"message": message,
}); err != nil {
return nil, err
}
return component, nil
}
type ChildInput interface {
pulumi.Input
}
func (*Child) ElementType() reflect.Type {
return reflect.TypeOf((**Child)(nil)).Elem()
}
type ChildOutput struct{ *pulumi.OutputState }
func (ChildOutput) ElementType() reflect.Type {
return reflect.TypeOf((**Child)(nil)).Elem()
}
type Container struct {
pulumi.ResourceState
Child ChildOutput `pulumi:"child"`
}
type CustomContainer struct {
pulumi.CustomResource
Child ChildOutput `pulumi:"child"`
}
func NewContainer(ctx *pulumi.Context, name string, child ChildInput,
opts ...pulumi.ResourceOption,
) (*Container, error) {
component := &Container{}
if err := ctx.RegisterComponentResource("test:index:Container", name, component, opts...); err != nil {
return nil, err
}
if err := ctx.RegisterResourceOutputs(component, pulumi.Map{
"child": child,
}); err != nil {
return nil, err
}
// Wait to make sure RegisterResourceOutputs has actually finished registering the resource outputs.
//
// RegisterResourceOutputs does most of its work in a a goroutine, as does RegisterComponentResource. This
// means RegisterResourceOutputs is inheritly racy with the resource being read later. This test explicitly
// tests roundtripping a container component resource, which means we need to read the outputs registered
// through RegisterResourceOutputs later, making the test racy. We can work around this by making sure the
// outputs are registered before we return the container. Ideally we should find a way to make this non-racy
// (see the issue linked below)
//
// TODO: make RegisterResourceOutputs not racy [pulumi/pulumi#16896]
containerURNResult, err := internals.UnsafeAwaitOutput(ctx.Context(), component.URN())
if err != nil {
return nil, err
}
containerURN := containerURNResult.Value.(pulumi.URN)
sleep := 20 * time.Millisecond
for i := 0; ; i++ {
roundTrippedContainer := &Container{}
if err := ctx.RegisterComponentResource("test:index:Container", "mycontainer", roundTrippedContainer,
pulumi.URN_(string(containerURN))); err != nil {
return nil, err
}
child, err := internals.UnsafeAwaitOutput(ctx.Context(), roundTrippedContainer.Child)
if err != nil {
return nil, err
}
if child.Value.(*Child) != nil {
break
} else if i > 10 {
return nil, errors.New("outputs were not registered successfully")
}
time.Sleep(sleep)
sleep *= 2
}
return component, nil
}
type module struct {
version semver.Version
}
func (m *module) Version() semver.Version {
return m.version
}
func (m *module) Construct(ctx *pulumi.Context, name, typ, urn string) (r pulumi.Resource, err error) {
switch typ {
case "test:index:Child":
r = &Child{}
default:
return nil, fmt.Errorf("unknown resource type: %s", typ)
}
err = ctx.RegisterResource(typ, name, nil, r, pulumi.URN_(urn))
return
}
func main() {
pulumi.RegisterOutputType(ChildOutput{})
pulumi.RegisterResourceModule(
"test",
"index",
&module{semver.MustParse("1.0.0")},
)
pulumi.Run(func(ctx *pulumi.Context) error {
child, err := NewChild(ctx, "mychild", pulumi.String("hello world!"))
if err != nil {
return err
}
container, err := NewContainer(ctx, "mycontainer", child)
if err != nil {
return err
}
containerURNResult, err := internals.UnsafeAwaitOutput(ctx.Context(), container.URN())
if err != nil {
return err
}
containerURN := containerURNResult.Value.(pulumi.URN)
roundTrippedContainer := &Container{}
if err := ctx.RegisterComponentResource("test:index:Container", "mycontainer", roundTrippedContainer,
pulumi.URN_(string(containerURN))); err != nil {
return err
}
roundTrippedContainerChildResult, err := internals.UnsafeAwaitOutput(ctx.Context(), roundTrippedContainer.Child)
if err != nil {
return err
}
roundTrippedContainerChild := roundTrippedContainerChildResult.Value.(*Child)
pulumi.All(child.URN(), roundTrippedContainerChild.URN(), roundTrippedContainerChild.Message).ApplyT(
func(args []interface{}) (*string, error) {
const expectedMessage = "hello world!"
expectedURN := args[0].(pulumi.URN)
actualURN := args[1].(pulumi.URN)
actualMessage := args[2].(string)
if expectedURN != actualURN {
panic(fmt.Errorf("expected urn %q not equal to actual urn %q", expectedURN, actualMessage))
}
if expectedMessage != actualMessage {
panic(fmt.Errorf(
"expected message %q not equal to actual message %q", expectedMessage, actualMessage))
}
return nil, nil
})
return nil
})
}