Skip to content

Commit

Permalink
providers: New Interface methods for ephemeral resource types
Browse files Browse the repository at this point in the history
  • Loading branch information
apparentlymart committed Apr 29, 2024
1 parent 23b57bb commit 5c7deeb
Show file tree
Hide file tree
Showing 11 changed files with 519 additions and 1 deletion.
17 changes: 16 additions & 1 deletion internal/builtin/providers/terraform/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -166,7 +166,7 @@ func (p *Provider) ImportResourceState(req providers.ImportResourceStateRequest)
return importDataStore(req)
}

panic("unimplemented - terraform_remote_state has no resources")
panic("unimplemented - terraform.io/builtin/terraform has no managed resource types")
}

func (p *Provider) MoveResourceState(providers.MoveResourceStateRequest) providers.MoveResourceStateResponse {
Expand All @@ -180,6 +180,21 @@ func (p *Provider) ValidateResourceConfig(req providers.ValidateResourceConfigRe
return validateDataStoreResourceConfig(req)
}

// CloseEphemeral implements providers.Interface.
func (p *Provider) CloseEphemeral(providers.CloseEphemeralRequest) providers.CloseEphemeralResponse {
panic("unimplemented - terraform.io/builtin/terraform has no ephemeral resource types")
}

// OpenEphemeral implements providers.Interface.
func (p *Provider) OpenEphemeral(providers.OpenEphemeralRequest) providers.OpenEphemeralResponse {
panic("unimplemented - terraform.io/builtin/terraform has no ephemeral resource types")
}

// RenewEphemeral implements providers.Interface.
func (p *Provider) RenewEphemeral(providers.RenewEphemeralRequest) providers.RenewEphemeralResponse {
panic("unimplemented - terraform.io/builtin/terraform has no ephemeral resource types")
}

// CallFunction would call a function contributed by this provider, but this
// provider has no functions and so this function just panics.
func (p *Provider) CallFunction(req providers.CallFunctionRequest) providers.CallFunctionResponse {
Expand Down
49 changes: 49 additions & 0 deletions internal/plugin/grpc_provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import (
"github.com/hashicorp/terraform/internal/logging"
"github.com/hashicorp/terraform/internal/plugin/convert"
"github.com/hashicorp/terraform/internal/providers"
"github.com/hashicorp/terraform/internal/tfdiags"
proto "github.com/hashicorp/terraform/internal/tfplugin5"
)

Expand Down Expand Up @@ -767,6 +768,54 @@ func (p *GRPCProvider) ReadDataSource(r providers.ReadDataSourceRequest) (resp p
return resp
}

func (p *GRPCProvider) OpenEphemeral(r providers.OpenEphemeralRequest) (resp providers.OpenEphemeralResponse) {
logger.Trace("GRPCProvider: OpenEphemeral")

// There is not yet any support for plugin-based providers to offer
// ephemeral resource types. We should not be able to get here because
// a GRPCProvider should never advertise in its schema that it supports
// ephemeral resource types.
resp.Diagnostics = resp.Diagnostics.Append(tfdiags.AttributeValue(
tfdiags.Error,
"Ephemeral resources not supported",
"This provider does not offer any ephemeral resource types.",
nil,
))
return resp
}

func (p *GRPCProvider) RenewEphemeral(r providers.RenewEphemeralRequest) (resp providers.RenewEphemeralResponse) {
logger.Trace("GRPCProvider: RenewEphemeral")

// There is not yet any support for plugin-based providers to offer
// ephemeral resource types. We should not be able to get here because
// a GRPCProvider should never advertise in its schema that it supports
// ephemeral resource types.
resp.Diagnostics = resp.Diagnostics.Append(tfdiags.AttributeValue(
tfdiags.Error,
"Ephemeral resources not supported",
"This provider does not offer any ephemeral resource types.",
nil,
))
return resp
}

func (p *GRPCProvider) CloseEphemeral(r providers.CloseEphemeralRequest) (resp providers.CloseEphemeralResponse) {
logger.Trace("GRPCProvider: CloseEphemeral")

// There is not yet any support for plugin-based providers to offer
// ephemeral resource types. We should not be able to get here because
// a GRPCProvider should never advertise in its schema that it supports
// ephemeral resource types.
resp.Diagnostics = resp.Diagnostics.Append(tfdiags.AttributeValue(
tfdiags.Error,
"Ephemeral resources not supported",
"This provider does not offer any ephemeral resource types.",
nil,
))
return resp
}

func (p *GRPCProvider) CallFunction(r providers.CallFunctionRequest) (resp providers.CallFunctionResponse) {
logger.Trace("GRPCProvider", "CallFunction", r.FunctionName)

Expand Down
49 changes: 49 additions & 0 deletions internal/plugin6/grpc_provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import (
"github.com/hashicorp/terraform/internal/logging"
"github.com/hashicorp/terraform/internal/plugin6/convert"
"github.com/hashicorp/terraform/internal/providers"
"github.com/hashicorp/terraform/internal/tfdiags"
proto6 "github.com/hashicorp/terraform/internal/tfplugin6"
)

Expand Down Expand Up @@ -756,6 +757,54 @@ func (p *GRPCProvider) ReadDataSource(r providers.ReadDataSourceRequest) (resp p
return resp
}

func (p *GRPCProvider) OpenEphemeral(r providers.OpenEphemeralRequest) (resp providers.OpenEphemeralResponse) {
logger.Trace("GRPCProvider.v6", "OpenEphemeral", r.TypeName)

// There is not yet any support for plugin-based providers to offer
// ephemeral resource types. We should not be able to get here because
// a GRPCProvider should never advertise in its schema that it supports
// ephemeral resource types.
resp.Diagnostics = resp.Diagnostics.Append(tfdiags.AttributeValue(
tfdiags.Error,
"Ephemeral resources not supported",
"This provider does not offer any ephemeral resource types.",
nil,
))
return resp
}

func (p *GRPCProvider) RenewEphemeral(r providers.RenewEphemeralRequest) (resp providers.RenewEphemeralResponse) {
logger.Trace("GRPCProvider.v6", "RenewEphemeral", r.TypeName)

// There is not yet any support for plugin-based providers to offer
// ephemeral resource types. We should not be able to get here because
// a GRPCProvider should never advertise in its schema that it supports
// ephemeral resource types.
resp.Diagnostics = resp.Diagnostics.Append(tfdiags.AttributeValue(
tfdiags.Error,
"Ephemeral resources not supported",
"This provider does not offer any ephemeral resource types.",
nil,
))
return resp
}

func (p *GRPCProvider) CloseEphemeral(r providers.CloseEphemeralRequest) (resp providers.CloseEphemeralResponse) {
logger.Trace("GRPCProvider.v6", "CloseEphemeral", r.TypeName)

// There is not yet any support for plugin-based providers to offer
// ephemeral resource types. We should not be able to get here because
// a GRPCProvider should never advertise in its schema that it supports
// ephemeral resource types.
resp.Diagnostics = resp.Diagnostics.Append(tfdiags.AttributeValue(
tfdiags.Error,
"Ephemeral resources not supported",
"This provider does not offer any ephemeral resource types.",
nil,
))
return resp
}

func (p *GRPCProvider) CallFunction(r providers.CallFunctionRequest) (resp providers.CallFunctionResponse) {
logger.Trace("GRPCProvider.v6", "CallFunction", r.FunctionName)

Expand Down
18 changes: 18 additions & 0 deletions internal/provider-simple-v6/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -171,6 +171,24 @@ func (s simple) ReadDataSource(req providers.ReadDataSourceRequest) (resp provid
return resp
}

func (s simple) OpenEphemeral(providers.OpenEphemeralRequest) providers.OpenEphemeralResponse {
// Our schema doesn't include any ephemeral resource types, so it should be
// impossible to get in here.
panic("OpenEphemeral on provider that didn't declare any ephemeral resource types")
}

func (s simple) RenewEphemeral(providers.RenewEphemeralRequest) providers.RenewEphemeralResponse {
// Our schema doesn't include any ephemeral resource types, so it should be
// impossible to get in here.
panic("RenewEphemeral on provider that didn't declare any ephemeral resource types")
}

func (s simple) CloseEphemeral(providers.CloseEphemeralRequest) providers.CloseEphemeralResponse {
// Our schema doesn't include any ephemeral resource types, so it should be
// impossible to get in here.
panic("CloseEphemeral on provider that didn't declare any ephemeral resource types")
}

func (s simple) CallFunction(req providers.CallFunctionRequest) (resp providers.CallFunctionResponse) {
if req.FunctionName != "noop" {
resp.Err = fmt.Errorf("CallFunction for undefined function %q", req.FunctionName)
Expand Down
18 changes: 18 additions & 0 deletions internal/provider-simple/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,24 @@ func (s simple) ReadDataSource(req providers.ReadDataSourceRequest) (resp provid
return resp
}

func (s simple) OpenEphemeral(providers.OpenEphemeralRequest) providers.OpenEphemeralResponse {
// Our schema doesn't include any ephemeral resource types, so it should be
// impossible to get in here.
panic("OpenEphemeral on provider that didn't declare any ephemeral resource types")
}

func (s simple) RenewEphemeral(providers.RenewEphemeralRequest) providers.RenewEphemeralResponse {
// Our schema doesn't include any ephemeral resource types, so it should be
// impossible to get in here.
panic("RenewEphemeral on provider that didn't declare any ephemeral resource types")
}

func (s simple) CloseEphemeral(providers.CloseEphemeralRequest) providers.CloseEphemeralResponse {
// Our schema doesn't include any ephemeral resource types, so it should be
// impossible to get in here.
panic("CloseEphemeral on provider that didn't declare any ephemeral resource types")
}

func (s simple) CallFunction(req providers.CallFunctionRequest) (resp providers.CallFunctionResponse) {
// Our schema doesn't include any functions, so it should be impossible
// to get in here.
Expand Down
187 changes: 187 additions & 0 deletions internal/providers/ephemeral.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,187 @@
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1

package providers

import (
"time"

"github.com/hashicorp/terraform/internal/tfdiags"
"github.com/zclconf/go-cty/cty"
)

// OpenEphemeralRequest represents the arguments for the OpenEphemeral
// operation on a provider.
type OpenEphemeralRequest struct {
// TypeName is the type of ephemeral resource to open. This should
// only be one of the type names previously reported in the provider's
// schema.
TypeName string

// Config is an object-typed value representing the configuration for
// the ephemeral resource instance that the caller is trying to open.
//
// The object type of this value always conforms to the resource type
// schema's implied type, and uses null values to represent attributes
// that were not explicitly assigned in the configuration block.
// Computed-only attributes are always null in the configuration, because
// they can be set only in the reponse.
Config cty.Value
}

// OpenEphemeralRequest represents the response from an OpenEphemeral
// operation on a provider.
type OpenEphemeralResponse struct {
// Deferred, if present, signals that the provider doesn't have enough
// information to open this ephemeral resource instance.
//
// This implies that any other side-effect-performing object must have
// its planning deferred if its planning operation indirectly depends on
// this ephemeral resource result. For example, if a provider configuration
// refers to an ephemeral resource whose opening is deferred then the
// affected provider configuration must not be instantiated and any
// resource instances that belong to it must have their planning immediately
// deferred.
Deferred *Deferred

// Result is an object-typed value representing the newly-opened session
// with the opened ephemeral object.
//
// The object type of this value always conforms to the resource type
// schema's implied type. Unknown values are forbidden unless the
// Deferred field is set, in which case the Result represents the provider's
// best approximation of the final object using unknown values in any
// location where a final value cannot be predicted.
Result cty.Value

// InternalContext is any internal data needed by the provider to
// perform a subsequent [Interface.CloseEphemeral] request for the same
// object. The provider may choose any encoding format to represent the
// needed data, because Terraform Core treats this field as opaque.
//
// Providers should aim to keep this data relatively compact to minimize
// overhead. Although Terraform Core does not enforce a specific limit
// just for this field, it would be very unusual for the internal context
// to be more than 256 bytes in size, and in most cases it should be
// on the order of only tens of bytes. For example, a lease ID for the
// remote system is a reasonable thing to encode here.
//
// Because ephemeral resource instances never outlive a single Terraform
// Core phase, it's guaranteed that a CloseEphemeral request will be
// received by exactly the same plugin instance that returned this
// value, and so it's valid for this to refer to in-memory state belonging
// to the provider instance.
InternalContext []byte

// Renew, if set, signals that the opened object has an inherent expration
// time and so must be "renewed" if Terraform needs to use it beyond that
// expiration time.
//
// If a provider sets this field then it may receive a subsequent
// [Interface.RenewEphemeral] call, if Terraform expects to need the
// object beyond the expiration time.
Renew *EphemeralRenew

// Diagnostics describes any problems encountered while opening the
// ephemeral resource. If this contains errors then the other response
// fields must be assumed invalid.
Diagnostics tfdiags.Diagnostics
}

// EphemeralRenew describes when and how Terraform Core must request renewal
// of an ephemeral resource instance in order to continue using it.
type EphemeralRenew struct {
// ExpireTime is the deadline before which Terraform must renew the
// ephemeral resource instance. Terraform will make the renew request
// at least one minute before the expiration time.
ExpireTime time.Time

// InternalContext is any internal data needed by the provider to
// perform a subsequent [Interface.RenewEphemeral] request. The provider
// may choose any encoding format to represent the needed data, because
// Terraform Core treats this field as opaque.
//
// Providers should aim to keep this data relatively compact to minimize
// overhead. Although Terraform Core does not enforce a specific limit
// just for this field, it would be very unusual for the internal context
// to be more than 256 bytes in size, and in most cases it should be
// on the order of only tens of bytes. For example, a lease ID for the
// remote system is a reasonable thing to encode here.
//
// Because ephemeral resource instances never outlive a single Terraform
// Core phase, it's guaranteed that a RenewEphemeral request will be
// received by exactly the same plugin instance that previously handled
// the OpenEphemeral or RenewEphemeral request that produced this internal
// context, and so it's valid for this to refer to in-memory state in the
// provider object.
InternalContext []byte
}

// RenewEphemeralRequest represents the arguments for the RenewEphemeral
// operation on a provider.
type RenewEphemeralRequest struct {
// TypeName is the type of ephemeral resource being renewed. This should
// only be one of the type names previously sent in a successful
// [OpenEphemeralRequest].
TypeName string

// InternalContext echoes verbatim the value from the field of the same
// name from the most recent [EphemeralRenew] object, received from either
// an [OpenEphemeralResponse] or a [RenewEphemeralResponse] object.
InternalContext []byte
}

// RenewEphemeralRequest represents the response from a RenewEphemeral
// operation on a provider.
type RenewEphemeralResponse struct {
// RenewAgain, if set, describes a new expiration deadline for the
// object, possibly causing a further call to [Interface.RenewEphemeral]
// if Terraform needs to exceed the updated deadline.
//
// If this is not set then Terraform Core will not make any further
// renewal requests for the remaining life of the object.
RenewAgain *EphemeralRenew

// Diagnostics describes any problems encountered while renewing the
// ephemeral resource instance. If this contains errors then the other
// response fields must be assumed invalid.
//
// Because renewals happen asynchronously from other uses of the
// ephemeral object, it's unspecified whether a renewal error will block
// any specific usage of the object. For example, a request using the
// object might already be in progress when a renewal error occurs,
// in which case that other request might also fail trying to use a
// now-invalid object, or it might by chance succeed in completing its
// operation before the ephemeral object truly expires.
Diagnostics tfdiags.Diagnostics
}

// CloseEphemeralRequest represents the arguments for the CloseEphemeral
// operation on a provider.
type CloseEphemeralRequest struct {
// TypeName is the type of ephemeral resource being closed. This should
// only be one of the type names previously sent in a successful
// [OpenEphemeralRequest].
TypeName string

// InternalContext echoes verbatim the value from the field of the same
// name from the corresponding [OpenEphemeralResponse] object.
InternalContext []byte
}

// CloseEphemeralRequest represents the response from a CloseEphemeral
// operation on a provider.
type CloseEphemeralResponse struct {
// Diagnostics describes any problems encountered while closing the
// ephemeral resource instance. If this contains errors then the other
// response fields must be assumed invalid.
//
// If closing an ephemeral resource instance fails then it's unspecified
// whether a corresponding remote object remains valid or not.
//
// Providers should make a best effort to treat the closure of an
// already-expired ephemeral object as a success in order to exhibit
// idemponent behavior for closing, but some remote systems do not allow
// distinguishing that case from other error conditions.
Diagnostics tfdiags.Diagnostics
}
Loading

0 comments on commit 5c7deeb

Please sign in to comment.