pulumi/sdk/go/pulumi/resource.go

891 lines
28 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 pulumi
import (
"context"
"fmt"
"reflect"
"sort"
"sync"
"github.com/pulumi/pulumi/sdk/v3/go/common/resource"
"github.com/pulumi/pulumi/sdk/v3/go/common/slice"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
"github.com/pulumi/pulumi/sdk/v3/go/internal"
)
type (
// ID is a unique identifier assigned by a resource provider to a resource.
ID string
// URN is an automatically generated logical URN, used to stably identify resources.
URN string
)
var (
resourceStateType = reflect.TypeOf(ResourceState{})
customResourceStateType = reflect.TypeOf(CustomResourceState{})
providerResourceStateType = reflect.TypeOf(ProviderResourceState{})
)
// This type alias is a hack to embed the internal.ResourceState type
// into pulumi.ResourceState without exporting the field to the public API.
//
//nolint:unused
type internalResourceState = internal.ResourceState
// ResourceState is the base
type ResourceState struct {
// internalResourceState marks this ResourceState as a resource
// recognized by the internal package.
internalResourceState
m sync.RWMutex
urn URNOutput `pulumi:"urn"`
rawOutputs Output
children resourceSet
providers map[string]ProviderResource
provider ProviderResource
version string
pluginDownloadURL string
aliases []URNOutput
name string
transformations []ResourceTransformation
keepDep bool
}
var _ Resource = (*ResourceState)(nil)
func (s *ResourceState) URN() URNOutput {
return s.urn
}
func (s *ResourceState) GetProvider(token string) ProviderResource {
return s.providers[getPackage(token)]
}
// This is an internal method and future versions of the sdk may not support this API.
//
// InternalGetRawOutputs obtains the full PropertyMap returned during resource registration,
// allowing a caller of RegisterResource to obtain directly information about the outputs and their
// known and secret attributes.
func InternalGetRawOutputs(res *ResourceState) Output {
return res.rawOutputs
}
func (s *ResourceState) getChildren() []Resource {
s.m.RLock()
defer s.m.RUnlock()
var children []Resource
if len(s.children) != 0 {
children = slice.Prealloc[Resource](len(s.children))
for r := range s.children {
children = append(children, r)
}
}
return children
}
func (s *ResourceState) addChild(r Resource) {
s.m.Lock()
defer s.m.Unlock()
if s.children == nil {
s.children = resourceSet{}
}
s.children.add(r)
}
func (s *ResourceState) getProviders() map[string]ProviderResource {
return s.providers
}
func (s *ResourceState) getProvider() ProviderResource {
return s.provider
}
func (s *ResourceState) getVersion() string {
return s.version
}
func (s *ResourceState) getPluginDownloadURL() string {
return s.pluginDownloadURL
}
func (s *ResourceState) getAliases() []URNOutput {
return s.aliases
}
func (s *ResourceState) getName() string {
return s.name
}
func (s *ResourceState) getTransformations() []ResourceTransformation {
return s.transformations
}
func (s *ResourceState) addTransformation(t ResourceTransformation) {
s.transformations = append(s.transformations, t)
}
func (s *ResourceState) setKeepDependency() {
s.keepDep = true
}
func (s *ResourceState) keepDependency() bool {
return s.keepDep
}
func (ctx *Context) newDependencyResource(urn URN) Resource {
var res ResourceState
res.urn.OutputState = ctx.newOutputState(res.urn.ElementType(), &res)
internal.ResolveOutput(res.urn, urn, true, false, resourcesToInternal(nil))
res.keepDep = true
return &res
}
type CustomResourceState struct {
ResourceState
id IDOutput `pulumi:"id"`
}
func (s *CustomResourceState) ID() IDOutput {
return s.id
}
func (*CustomResourceState) isCustomResource() {}
func (ctx *Context) newDependencyCustomResource(urn URN, id ID) CustomResource {
var res CustomResourceState
res.urn.OutputState = ctx.newOutputState(res.urn.ElementType(), &res)
internal.ResolveOutput(res.urn, urn, true, false, resourcesToInternal(nil))
res.id.OutputState = ctx.newOutputState(res.id.ElementType(), &res)
internal.ResolveOutput(res.id, id, id != "", false, resourcesToInternal(nil))
return &res
}
type ProviderResourceState struct {
CustomResourceState
pkg string
}
func (s *ProviderResourceState) getPackage() string {
return s.pkg
}
func (ctx *Context) newDependencyProviderResource(urn URN, id ID) ProviderResource {
var res ProviderResourceState
res.urn.OutputState = ctx.newOutputState(res.urn.ElementType(), &res)
res.id.OutputState = ctx.newOutputState(res.id.ElementType(), &res)
internal.ResolveOutput(res.urn, urn, true, false, resourcesToInternal(nil))
internal.ResolveOutput(res.id, id, id != "", false, resourcesToInternal(nil))
res.pkg = string(resource.URN(urn).Type().Name())
return &res
}
// Resource represents a cloud resource managed by Pulumi.
type Resource interface {
internal.Resource
// URN is this resource's stable logical URN used to distinctly address it before, during, and after deployments.
URN() URNOutput
// getChildren returns the resource's children.
getChildren() []Resource
// addChild adds a child to the resource.
addChild(r Resource)
// getProviders returns the provider map for this resource.
getProviders() map[string]ProviderResource
// getProvider returns the provider for the resource.
getProvider() ProviderResource
// getVersion returns the version for the resource.
getVersion() string
// getPluginDownloadURL returns the provider plugin download url
getPluginDownloadURL() string
// getAliases returns the list of aliases for this resource
getAliases() []URNOutput
// getName returns the name of the resource
getName() string
// getTransformations returns the transformations for the resource.
getTransformations() []ResourceTransformation
// addTransformation adds a single transformation to the resource.
addTransformation(t ResourceTransformation)
// setKeepDependency marks this resource as a resource that should be kept as a dependency.
// This is done for remote component resources, dependency resources, and rehydrated component resources.
setKeepDependency()
// keepDependency returns true if the resource should be kept as a dependency, which is the case for
// remote component resources, dependency resources, and rehydrated component resources.
keepDependency() bool
}
var _ internal.Resource = (Resource)(nil)
// CustomResource is a cloud resource whose create, read, update, and delete (CRUD) operations are managed by performing
// external operations on some physical entity. The engine understands how to diff and perform partial updates of them,
// and these CRUD operations are implemented in a dynamically loaded plugin for the defining package.
type CustomResource interface {
Resource
// ID is the provider-assigned unique identifier for this managed resource. It is set during deployments,
// but might be missing ("") during planning phases.
ID() IDOutput
isCustomResource()
}
// ComponentResource is a resource that aggregates one or more other child resources into a higher level abstraction.
// The component resource itself is a resource, but does not require custom CRUD operations for provisioning.
type ComponentResource interface {
Resource
}
// ProviderResource is a resource that represents a configured instance of a particular package's provider plugin.
// These resources are supply the implementations of their package's CRUD operations. A specific provider instance can
// be used for a given resource by passing it in ResourceOpt.Provider.
type ProviderResource interface {
CustomResource
getPackage() string
}
// CustomTimeouts specifies timeouts for resource provisioning operations.
// Use it with the [Timeouts] option when creating new resources
// to override default timeouts.
//
// Each timeout is specified as a duration string such as,
// "5ms" (5 milliseconds), "40s" (40 seconds),
// and "1m30s" (1 minute, 30 seconds).
//
// The following units are accepted.
//
// - ns: nanoseconds
// - us: microseconds
// - µs: microseconds
// - ms: milliseconds
// - s: seconds
// - m: minutes
// - h: hours
type CustomTimeouts struct {
Create string
Update string
Delete string
}
// ResourceOptions is a snapshot of one or more [ResourceOption]s.
//
// You cannot pass a ResourceOptions struct to a resource constructor.
// Instead, use individual [ResourceOption] values to configure a resource.
// The ResourceOptions struct only provides a read-only preview
// of the collective effect of the options.
//
// See https://www.pulumi.com/docs/intro/concepts/resources/options/
// for more details on individual options.
type ResourceOptions struct {
// AdditionalSecretOutputs lists output properties
// that must be encrypted as secrets.
AdditionalSecretOutputs []string
// Aliases lists aliases for this resource
// that are used to find and use existing resources.
Aliases []Alias
// CustomTimeouts, if set, overrides the default timeouts
// for resource CRUD operations.
CustomTimeouts *CustomTimeouts
// DeleteBeforeReplace specifies that resources being replaced
// should be deleted before creating the replacement
// instead of Pulumi's default behavior of creating the replacement
// before performing deletion.
DeleteBeforeReplace bool
// DependsOn lists additional explicit dependencies for the resource
// in addition to those tracked automatically by Pulumi.
DependsOn []Resource
// DependsOnInputs holds explicit dependencies for the resource
// that may not be fully known yet.
DependsOnInputs []ResourceArrayInput
// IgnoreChanges lists properties changes to which should be ignored.
IgnoreChanges []string
// Import specifies that the provider for this resource
// should import its state from a cloud resource with the given ID.
Import IDInput
// Parent is the parent resource for the resource being created,
// or nil if this resource does not have a parent.
Parent Resource
// Protect prevents this resource from being deleted.
Protect bool
// Provider is the provider resource to use for this resource's CRUD operations.
// It's nil if the default provider should be used.
Provider ProviderResource
// Providers is a bag of providers available
// to instantiate resources of various types.
// These are used for a type when a provider for that type
// was not explicitly supplied.
Providers []ProviderResource
// ReplaceOnChanges lists properties that, when modified,
// force a replacement of the resource.
// The list may include '*' to indicate that all properties trigger
// replacements.
ReplaceOnChanges []string
// Transformations is a list of functions that transform
// the resource's properties during construction.
Transformations []ResourceTransformation
// URN is the URN of a previously-registered resource of this type.
URN string
// Version changes the version of the provider plugin that should be used
// when operating on this resource.
// This will be blank if the version was automatically inferred.
Version string
// PluginDownloadURL specifies the URL from which the provider plugin
// should be downloaded.
// This will be blank if the URL was inferred automatically.
PluginDownloadURL string
// RetainOnDelete specifies that the resource should not be deleted
// in the cloud provider, even if it's deleted from Pulumi.
RetainOnDelete bool
// DeletedWith holds a container resource that, if deleted,
// also deletes this resource.
DeletedWith Resource
}
// NewResourceOptions builds a preview of the effect of the provided options.
//
// Use this to get a read-only snapshot of a list of options
// inside mocks and component resources.
func NewResourceOptions(opts ...ResourceOption) (*ResourceOptions, error) {
// The error return is currently unused,
// but it's foreseeable that we'll need it
// if we begin doing option validation at option merge time.
return resourceOptionsSnapshot(merge(opts...)), nil
}
// resourceOptions is the internal representation of the effect of
// [ResourceOption]s.
type resourceOptions struct {
AdditionalSecretOutputs []string
Aliases []Alias
CustomTimeouts *CustomTimeouts
DeleteBeforeReplace bool
DependsOn []dependencySet
IgnoreChanges []string
Import IDInput
Parent Resource
Protect bool
Provider ProviderResource
Providers map[string]ProviderResource
ReplaceOnChanges []string
Transformations []ResourceTransformation
URN string
Version string
PluginDownloadURL string
RetainOnDelete bool
DeletedWith Resource
}
func resourceOptionsSnapshot(ro *resourceOptions) *ResourceOptions {
var (
dependsOn []Resource
dependsOnInputs []ResourceArrayInput
)
for _, d := range ro.DependsOn {
switch d := d.(type) {
case resourceDependencySet:
dependsOn = append(dependsOn, []Resource(d)...)
case *resourceArrayInputDependencySet:
dependsOnInputs = append(dependsOnInputs, d.input)
default:
// Unreachable.
// We control all implementations of dependencySet.
contract.Failf("Unknown dependencySet %T", d)
}
}
sort.Slice(dependsOn, func(i, j int) bool {
return dependsOn[i].getName() < dependsOn[j].getName()
})
var providers []ProviderResource
if len(ro.Providers) > 0 {
providers = slice.Prealloc[ProviderResource](len(ro.Providers))
for _, p := range ro.Providers {
providers = append(providers, p)
}
sort.Slice(providers, func(i, j int) bool {
return providers[i].getPackage() < providers[j].getPackage()
})
}
return &ResourceOptions{
AdditionalSecretOutputs: ro.AdditionalSecretOutputs,
Aliases: ro.Aliases,
CustomTimeouts: ro.CustomTimeouts,
DeleteBeforeReplace: ro.DeleteBeforeReplace,
DependsOn: dependsOn,
DependsOnInputs: dependsOnInputs,
IgnoreChanges: ro.IgnoreChanges,
Import: ro.Import,
Parent: ro.Parent,
Protect: ro.Protect,
Provider: ro.Provider,
Providers: providers,
ReplaceOnChanges: ro.ReplaceOnChanges,
Transformations: ro.Transformations,
URN: ro.URN,
Version: ro.Version,
PluginDownloadURL: ro.PluginDownloadURL,
RetainOnDelete: ro.RetainOnDelete,
DeletedWith: ro.DeletedWith,
}
}
// InvokeOptions is a snapshot of one or more [InvokeOption]s.
//
// You cannot pass an InvokeOptions struct to a provider function.
// Instead, use individual [InvokeOption] values to configure a call.
// The InvokeOptions struct only provides a read-only preview
// of the collective effect of the options.
type InvokeOptions struct {
// Parent is the parent resource for this operation.
// It may be used to determine the provider to use.
Parent Resource
// Provider specifies the provider to use for this operation.
// This is nil if the default provider should be used.
Provider ProviderResource
// Version is the version of the provider plugin that should be used.
// This will be blank if the version was automatically inferred.
Version string
// PluginDownloadURL is the URL from which the provider plugin
// should be downloaded.
// This will be blank if the URL was inferred automatically.
PluginDownloadURL string
}
// NOTE:
// InvokeOptions is part of the public API.
// If you introduce a new option,
// consider what its "snapshot" should look like,
// not the internal representation.
// For example, a list of resources should be []Resource,
// even if it's stored as a map[string]Resource.
// If the snapshot representation diverges from the internal,
// mirror this struct into a private invokeOptions struct.
// See resourceOptions for an example.
// NewInvokeOptions builds a preview of the effect of the provided options.
//
// Use this to get a read-only snapshot of the collective effect
// of a list of [InvokeOption]s.
func NewInvokeOptions(opts ...InvokeOption) (*InvokeOptions, error) {
var options InvokeOptions
for _, o := range opts {
if o != nil {
o.applyInvokeOption(&options)
}
}
// The error return is currently unused,
// but it's foreseeable that we'll need it
// if we begin doing option validation at option merge time.
return &options, nil
}
type ResourceOption interface {
applyResourceOption(*resourceOptions)
}
type InvokeOption interface {
applyInvokeOption(*InvokeOptions)
}
type ResourceOrInvokeOption interface {
ResourceOption
InvokeOption
}
type resourceOption func(*resourceOptions)
func (o resourceOption) applyResourceOption(opts *resourceOptions) {
o(opts)
}
type invokeOption func(*InvokeOptions)
func (o invokeOption) applyInvokeOption(opts *InvokeOptions) {
o(opts)
}
type resourceOrInvokeOption func(ro *resourceOptions, io *InvokeOptions)
func (o resourceOrInvokeOption) applyResourceOption(opts *resourceOptions) {
o(opts, nil)
}
func (o resourceOrInvokeOption) applyInvokeOption(opts *InvokeOptions) {
o(nil, opts)
}
// merging is handled by each functional options call
// properties that are arrays/maps are always appended/merged together
// last value wins for non-array/map values and for conflicting map values (bool, struct, etc)
func merge(opts ...ResourceOption) *resourceOptions {
options := &resourceOptions{}
for _, o := range opts {
if o != nil {
o.applyResourceOption(options)
}
}
return options
}
// AdditionalSecretOutputs specifies a list of output properties to mark as secret.
func AdditionalSecretOutputs(o []string) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.AdditionalSecretOutputs = append(ro.AdditionalSecretOutputs, o...)
})
}
// Aliases applies a list of identifiers to find and use existing resources.
func Aliases(o []Alias) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.Aliases = append(ro.Aliases, o...)
})
}
// DeleteBeforeReplace, when set to true, ensures that this resource is deleted prior to replacement.
func DeleteBeforeReplace(o bool) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.DeleteBeforeReplace = o
})
}
// Composite is a resource option that contains other resource options.
func Composite(opts ...ResourceOption) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
for _, o := range opts {
o.applyResourceOption(ro)
}
})
}
// CompositeInvoke is an invoke option that contains other invoke options.
func CompositeInvoke(opts ...InvokeOption) InvokeOption {
return invokeOption(func(ro *InvokeOptions) {
for _, o := range opts {
o.applyInvokeOption(ro)
}
})
}
// dependencySet unifies types that can provide dependencies for a
// resource.
type dependencySet interface {
// Adds URNs for addURNs from this set
// into the given urnSet.
// Optionally pass the last Resource arg to short-circuit component
// children cycles.
addURNs(context.Context, urnSet, Resource) error
}
// DependsOn is an optional array of explicit dependencies on other resources.
func DependsOn(o []Resource) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.DependsOn = append(ro.DependsOn, resourceDependencySet(o))
})
}
// resourceDependencySet is a dependencySet comprised of references to
// resources.
type resourceDependencySet []Resource
var _ dependencySet = (resourceDependencySet)(nil)
func (rs resourceDependencySet) addURNs(ctx context.Context, urns urnSet, from Resource) error {
for _, r := range rs {
if err := addDependency(ctx, urns, r, from); err != nil {
return err
}
}
return nil
}
// Declares explicit dependencies on other resources. Similar to
// `DependsOn`, but also admits resource inputs and outputs:
//
// var r Resource
// var ri ResourceInput
// var ro ResourceOutput
// allDeps := NewResourceArrayOutput(NewResourceOutput(r), ri.ToResourceOutput(), ro)
// DependsOnInputs(allDeps)
func DependsOnInputs(o ResourceArrayInput) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.DependsOn = append(ro.DependsOn, &resourceArrayInputDependencySet{o})
})
}
// resourceArrayInputDependencySet is a dependencySet built from
// collections of resources that are not yet known.
type resourceArrayInputDependencySet struct{ input ResourceArrayInput }
var _ dependencySet = (*resourceArrayInputDependencySet)(nil)
func (ra *resourceArrayInputDependencySet) addURNs(ctx context.Context, urns urnSet, from Resource) error {
out := ra.input.ToResourceArrayOutput()
value, known, _ /* secret */, _ /* deps */, err := internal.AwaitOutput(ctx, out)
if err != nil || !known {
return err
}
resources, ok := value.([]Resource)
if !ok {
return fmt.Errorf("ResourceArrayInput resolved to a value of unexpected type %v, expected []Resource",
reflect.TypeOf(value))
}
// For some reason, deps returned above are incorrect; instead:
toplevelDeps := getOutputDeps(out)
for _, r := range append(resources, toplevelDeps...) {
if err := addDependency(ctx, urns, r, from); err != nil {
return err
}
}
return nil
}
// Ignore changes to any of the specified properties.
func IgnoreChanges(o []string) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.IgnoreChanges = append(ro.IgnoreChanges, o...)
})
}
// Import, when provided with a resource ID, indicates that this resource's provider should import its state from
// the cloud resource with the given ID. The inputs to the resource's constructor must align with the resource's
// current state. Once a resource has been imported, the import property must be removed from the resource's
// options.
func Import(o IDInput) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.Import = o
})
}
// Parent sets the parent resource to which this resource or invoke belongs.
func Parent(r Resource) ResourceOrInvokeOption {
return resourceOrInvokeOption(func(ro *resourceOptions, io *InvokeOptions) {
switch {
case ro != nil:
ro.Parent = r
case io != nil:
io.Parent = r
}
})
}
// Protect, when set to true, ensures that this resource cannot be deleted (without first setting it to false).
func Protect(o bool) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.Protect = o
})
}
////////////////////////////////////////////////////////////////////////
// NOTE(Provider and Providers)
//
// For Provider vs Providers, there's a bit of complexity.
//
// # Background
//
// First, here's the desired behavior across languages,
// as standardized in #8796:
//
// - Providers is a bag of providers available to a resource.
// These are used by the resource and its children.
// - Provider passed to a resource indicates that that specific resource
// MUST use this provider over all else,
// and it should add it to the bag of providers for use by its children.
//
// One discrepancy from the above is that originally, for the Provider option,
// mismatch between the provider package and resource type was an error.
// However, this appears to have been relaxed over time.
// In such a case, the provider is added to the bag of providers
// and used by the children of the resource.
// This is necessary, for example, for resources to inherit a provider
// passed to a component resource via the Provider option.
//
// # Go-specific difference
//
// In other languages, the Provider and Providers options can be provided at
// most once per resource.
// In Go, because we use functional options, we allow multiple calls to
// the Provider and Providers options.
// This has allowed for the following to be equivalen in Go:
//
// NewFoo(..., Provider(p1), Provider(p2), Provider(p3))
// NewFoo(..., Providers(p1, p2, p3))
//
// To support this while still having the Provider option take precedence,
// we need to do the following:
//
// 1. All providers (whether passed with Provider or Providers)
// are merged into a single map.
// Last provider for a given package wins.
// 2. For the Provider option, we additionally track the last
// passed value in a separate field.
// If this provider handles the current resource,
// it takes precedence over the map.
////////////////////////////////////////////////////////////////////////
// Provider sets the provider resource to use for a resource's CRUD operations or an invoke's call.
func Provider(r ProviderResource) ResourceOrInvokeOption {
return resourceOrInvokeOption(func(ro *resourceOptions, io *InvokeOptions) {
switch {
case ro != nil:
ro.Provider = r
Providers(r).applyResourceOption(ro)
case io != nil:
io.Provider = r
}
})
}
// ProviderMap is an optional map of package to provider resource for a component resource.
func ProviderMap(o map[string]ProviderResource) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
if o != nil {
if ro.Providers == nil {
ro.Providers = make(map[string]ProviderResource)
}
for k, v := range o {
ro.Providers[k] = v
}
}
})
}
// Providers is an optional list of providers to use for a resource's children.
func Providers(o ...ProviderResource) ResourceOption {
m := map[string]ProviderResource{}
for _, p := range o {
m[p.getPackage()] = p
}
return ProviderMap(m)
}
// ReplaceOnChanges will force a replacement when any of these property paths are set. If this list includes `"*"`,
// changes to any properties will force a replacement. Initialization errors from previous deployments will
// require replacement instead of update only if `"*"` is passed.
func ReplaceOnChanges(o []string) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.ReplaceOnChanges = append(ro.ReplaceOnChanges, o...)
})
}
// Timeouts is an optional configuration block used for CRUD operations
func Timeouts(o *CustomTimeouts) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.CustomTimeouts = o
})
}
// Transformations is an optional list of transformations to be applied to the resource.
func Transformations(o []ResourceTransformation) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.Transformations = append(ro.Transformations, o...)
})
}
// URN_ is an optional URN of a previously-registered resource of this type to read from the engine.
//
//nolint:revive
func URN_(o string) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.URN = o
})
}
// Version is an optional version, corresponding to the version of the provider plugin that should be used when
// operating on this resource. This version overrides the version information inferred from the current package and
// should rarely be used.
func Version(o string) ResourceOrInvokeOption {
return resourceOrInvokeOption(func(ro *resourceOptions, io *InvokeOptions) {
switch {
case ro != nil:
ro.Version = o
case io != nil:
io.Version = o
}
})
}
// PluginDownloadURL is an optional url, corresponding to the download url of the provider plugin
// that should be used when operating on this resource. This url overrides the url information
// inferred from the current package and should rarely be used.
func PluginDownloadURL(o string) ResourceOrInvokeOption {
return resourceOrInvokeOption(func(ro *resourceOptions, io *InvokeOptions) {
switch {
case ro != nil:
ro.PluginDownloadURL = o
case io != nil:
io.PluginDownloadURL = o
}
})
}
// If set to True, the providers Delete method will not be called for this resource.
func RetainOnDelete(b bool) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.RetainOnDelete = b
})
}
// If set, the providers Delete method will not be called for this resource
// if specified resource is being deleted as well.
func DeletedWith(r Resource) ResourceOption {
return resourceOption(func(ro *resourceOptions) {
ro.DeletedWith = r
})
}