Skip to content

Commit

Permalink
tfprotov5+tfprotov6: Require FunctionServer in ProviderServer and Mov…
Browse files Browse the repository at this point in the history
…eResourceState in ResourceServer

Reference: #353
Reference: #363

This removes the temporary handling to smoothly release the new Terraform 1.8 and later RPC handling for provider servers. These changes codify this Go module's desired design that it reflects the protocol via its required implementations.
  • Loading branch information
bflad committed Mar 11, 2024
1 parent a99dc45 commit e68d80d
Show file tree
Hide file tree
Showing 9 changed files with 31 additions and 164 deletions.
7 changes: 7 additions & 0 deletions .changes/unreleased/BREAKING CHANGES-20240311-093208.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
kind: BREAKING CHANGES
body: 'tfprotov5+tfprotov6: `FunctionServer` interface is now required in `ProviderServer`.
Implementations not needing function support can return errors from the `GetFunctions`
and `CallFunction` methods.'
time: 2024-03-11T09:32:08.806864-04:00
custom:
Issue: "9999"
8 changes: 8 additions & 0 deletions .changes/unreleased/BREAKING CHANGES-20240311-093436.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
kind: BREAKING CHANGES
body: 'tfprotov5+tfprotov6: Moved `MoveResourceState` method to `ResourceServer`
and removed temporary `ResourceServerWithMoveResourceState` interface.
Implementations not needing move state support can return errors from the
`MoveResourceState` method.'
time: 2024-03-11T09:34:36.800884-04:00
custom:
Issue: "9999"
8 changes: 8 additions & 0 deletions .changes/unreleased/NOTES-20240311-093736.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
kind: NOTES
body: 'all: To prevent compilation errors, ensure your Go module is updated to at
least [email protected], [email protected],
terraform-plugin-sdk/[email protected], and [email protected] before
upgrading this dependency.'
time: 2024-03-11T09:37:36.503168-04:00
custom:
Issue: "9999"
5 changes: 1 addition & 4 deletions tfprotov5/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,10 +53,7 @@ type ProviderServer interface {
// are a handy interface for defining what a function is to
// terraform-plugin-go, so they are their own interface that is composed
// into ProviderServer.
//
// This will be required in an upcoming release.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
// FunctionServer
FunctionServer
}

// GetMetadataRequest represents a GetMetadata RPC request.
Expand Down
10 changes: 0 additions & 10 deletions tfprotov5/resource.go
Original file line number Diff line number Diff line change
Expand Up @@ -52,16 +52,6 @@ type ResourceServer interface {
// specified by the passed ID and return it as one or more resource
// states for Terraform to assume control of.
ImportResourceState(context.Context, *ImportResourceStateRequest) (*ImportResourceStateResponse, error)
}

// ResourceServerWithMoveResourceState is a temporary interface for servers
// to implement MoveResourceState RPC handling.
//
// Deprecated: The MoveResourceState method will be moved into the
// ResourceServer interface and this interface will be removed in a future
// version.
type ResourceServerWithMoveResourceState interface {
ResourceServer

// MoveResourceState is called when Terraform is asked to change a resource
// type for an existing resource. The provider must accept the change as
Expand Down
71 changes: 3 additions & 68 deletions tfprotov5/tf5server/server.go
Original file line number Diff line number Diff line change
Expand Up @@ -900,37 +900,11 @@ func (s *server) MoveResourceState(ctx context.Context, protoReq *tfplugin5.Move
logging.ProtocolTrace(ctx, "Received request")
defer logging.ProtocolTrace(ctx, "Served request")

// Remove this check and error in preference of
// s.downstream.MoveResourceState below once ResourceServer interface
// implements the MoveResourceState method.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/363
// nolint:staticcheck
resourceServerWMRS, ok := s.downstream.(tfprotov5.ResourceServerWithMoveResourceState)

if !ok {
logging.ProtocolError(ctx, "ProviderServer does not implement ResourceServerWithMoveResourceState")

protoResp := &tfplugin5.MoveResourceState_Response{
Diagnostics: []*tfplugin5.Diagnostic{
{
Severity: tfplugin5.Diagnostic_ERROR,
Summary: "Provider Move Resource State Not Implemented",
Detail: "A MoveResourceState call was received by the provider, however the provider does not implement the call. " +
"Either upgrade the provider to a version that implements move resource state support or this is a bug in Terraform that should be reported to the Terraform maintainers.",
},
},
}

return protoResp, nil
}

req := fromproto.MoveResourceStateRequest(protoReq)

ctx = tf5serverlogging.DownstreamRequest(ctx)

// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/363
// resp, err := s.downstream.MoveResourceState(ctx, req)
resp, err := resourceServerWMRS.MoveResourceState(ctx, req)
resp, err := s.downstream.MoveResourceState(ctx, req)

if err != nil {
logging.ProtocolError(ctx, "Error from downstream", map[string]interface{}{logging.KeyError: err})
Expand All @@ -954,26 +928,6 @@ func (s *server) CallFunction(ctx context.Context, protoReq *tfplugin5.CallFunct
logging.ProtocolTrace(ctx, "Received request")
defer logging.ProtocolTrace(ctx, "Served request")

// Remove this check and error in preference of s.downstream.CallFunction
// below once ProviderServer interface requires FunctionServer.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
functionServer, ok := s.downstream.(tfprotov5.FunctionServer)

if !ok {
logging.ProtocolError(ctx, "ProviderServer does not implement FunctionServer")

text := "Provider Functions Not Implemented: A provider-defined function call was received by the provider, however the provider does not implement functions. " +
"Either upgrade the provider to a version that implements provider-defined functions or this is a bug in Terraform that should be reported to the Terraform maintainers."

protoResp := &tfplugin5.CallFunction_Response{
Error: &tfplugin5.FunctionError{
Text: text,
},
}

return protoResp, nil
}

req := fromproto.CallFunctionRequest(protoReq)

for position, argument := range req.Arguments {
Expand All @@ -982,9 +936,7 @@ func (s *server) CallFunction(ctx context.Context, protoReq *tfplugin5.CallFunct

ctx = tf5serverlogging.DownstreamRequest(ctx)

// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
// resp, err := s.downstream.CallFunction(ctx, req)
resp, err := functionServer.CallFunction(ctx, req)
resp, err := s.downstream.CallFunction(ctx, req)

if err != nil {
logging.ProtocolError(ctx, "Error from downstream", map[string]any{logging.KeyError: err})
Expand All @@ -1007,28 +959,11 @@ func (s *server) GetFunctions(ctx context.Context, protoReq *tfplugin5.GetFuncti
logging.ProtocolTrace(ctx, "Received request")
defer logging.ProtocolTrace(ctx, "Served request")

// Remove this check and response in preference of s.downstream.GetFunctions
// below once ProviderServer interface requires FunctionServer.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
functionServer, ok := s.downstream.(tfprotov5.FunctionServer)

if !ok {
logging.ProtocolWarn(ctx, "ProviderServer does not implement FunctionServer")

protoResp := &tfplugin5.GetFunctions_Response{
Functions: map[string]*tfplugin5.Function{},
}

return protoResp, nil
}

req := fromproto.GetFunctionsRequest(protoReq)

ctx = tf5serverlogging.DownstreamRequest(ctx)

// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
// resp, err := s.downstream.GetFunctions(ctx, req)
resp, err := functionServer.GetFunctions(ctx, req)
resp, err := s.downstream.GetFunctions(ctx, req)

if err != nil {
logging.ProtocolError(ctx, "Error from downstream", map[string]any{logging.KeyError: err})
Expand Down
5 changes: 1 addition & 4 deletions tfprotov6/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,10 +53,7 @@ type ProviderServer interface {
// are a handy interface for defining what a function is to
// terraform-plugin-go, so they are their own interface that is composed
// into ProviderServer.
//
// This will be required in an upcoming release.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
// FunctionServer
FunctionServer
}

// GetMetadataRequest represents a GetMetadata RPC request.
Expand Down
10 changes: 0 additions & 10 deletions tfprotov6/resource.go
Original file line number Diff line number Diff line change
Expand Up @@ -52,16 +52,6 @@ type ResourceServer interface {
// specified by the passed ID and return it as one or more resource
// states for Terraform to assume control of.
ImportResourceState(context.Context, *ImportResourceStateRequest) (*ImportResourceStateResponse, error)
}

// ResourceServerWithMoveResourceState is a temporary interface for servers
// to implement MoveResourceState RPC handling.
//
// Deprecated: The MoveResourceState method will be moved into the
// ResourceServer interface and this interface will be removed in a future
// version.
type ResourceServerWithMoveResourceState interface {
ResourceServer

// MoveResourceState is called when Terraform is asked to change a resource
// type for an existing resource. The provider must accept the change as
Expand Down
71 changes: 3 additions & 68 deletions tfprotov6/tf6server/server.go
Original file line number Diff line number Diff line change
Expand Up @@ -900,37 +900,11 @@ func (s *server) MoveResourceState(ctx context.Context, protoReq *tfplugin6.Move
logging.ProtocolTrace(ctx, "Received request")
defer logging.ProtocolTrace(ctx, "Served request")

// Remove this check and error in preference of
// s.downstream.MoveResourceState below once ResourceServer interface
// implements the MoveResourceState method.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/363
// nolint:staticcheck
resourceServerWMRS, ok := s.downstream.(tfprotov6.ResourceServerWithMoveResourceState)

if !ok {
logging.ProtocolError(ctx, "ProviderServer does not implement ResourceServerWithMoveResourceState")

protoResp := &tfplugin6.MoveResourceState_Response{
Diagnostics: []*tfplugin6.Diagnostic{
{
Severity: tfplugin6.Diagnostic_ERROR,
Summary: "Provider Move Resource State Not Implemented",
Detail: "A MoveResourceState call was received by the provider, however the provider does not implement the call. " +
"Either upgrade the provider to a version that implements move resource state support or this is a bug in Terraform that should be reported to the Terraform maintainers.",
},
},
}

return protoResp, nil
}

req := fromproto.MoveResourceStateRequest(protoReq)

ctx = tf6serverlogging.DownstreamRequest(ctx)

// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/363
// resp, err := s.downstream.MoveResourceState(ctx, req)
resp, err := resourceServerWMRS.MoveResourceState(ctx, req)
resp, err := s.downstream.MoveResourceState(ctx, req)

if err != nil {
logging.ProtocolError(ctx, "Error from downstream", map[string]interface{}{logging.KeyError: err})
Expand All @@ -954,26 +928,6 @@ func (s *server) CallFunction(ctx context.Context, protoReq *tfplugin6.CallFunct
logging.ProtocolTrace(ctx, "Received request")
defer logging.ProtocolTrace(ctx, "Served request")

// Remove this check and error in preference of s.downstream.CallFunction
// below once ProviderServer interface requires FunctionServer.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
functionServer, ok := s.downstream.(tfprotov6.FunctionServer)

if !ok {
logging.ProtocolError(ctx, "ProviderServer does not implement FunctionServer")

text := "Provider Functions Not Implemented: A provider-defined function call was received by the provider, however the provider does not implement functions. " +
"Either upgrade the provider to a version that implements provider-defined functions or this is a bug in Terraform that should be reported to the Terraform maintainers."

protoResp := &tfplugin6.CallFunction_Response{
Error: &tfplugin6.FunctionError{
Text: text,
},
}

return protoResp, nil
}

req := fromproto.CallFunctionRequest(protoReq)

for position, argument := range req.Arguments {
Expand All @@ -982,9 +936,7 @@ func (s *server) CallFunction(ctx context.Context, protoReq *tfplugin6.CallFunct

ctx = tf6serverlogging.DownstreamRequest(ctx)

// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
// resp, err := s.downstream.CallFunction(ctx, req)
resp, err := functionServer.CallFunction(ctx, req)
resp, err := s.downstream.CallFunction(ctx, req)

if err != nil {
logging.ProtocolError(ctx, "Error from downstream", map[string]any{logging.KeyError: err})
Expand All @@ -1007,28 +959,11 @@ func (s *server) GetFunctions(ctx context.Context, protoReq *tfplugin6.GetFuncti
logging.ProtocolTrace(ctx, "Received request")
defer logging.ProtocolTrace(ctx, "Served request")

// Remove this check and response in preference of s.downstream.GetFunctions
// below once ProviderServer interface requires FunctionServer.
// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
functionServer, ok := s.downstream.(tfprotov6.FunctionServer)

if !ok {
logging.ProtocolWarn(ctx, "ProviderServer does not implement FunctionServer")

protoResp := &tfplugin6.GetFunctions_Response{
Functions: map[string]*tfplugin6.Function{},
}

return protoResp, nil
}

req := fromproto.GetFunctionsRequest(protoReq)

ctx = tf6serverlogging.DownstreamRequest(ctx)

// Reference: https://github.com/hashicorp/terraform-plugin-go/issues/353
// resp, err := s.downstream.GetFunctions(ctx, req)
resp, err := functionServer.GetFunctions(ctx, req)
resp, err := s.downstream.GetFunctions(ctx, req)

if err != nil {
logging.ProtocolError(ctx, "Error from downstream", map[string]any{logging.KeyError: err})
Expand Down

0 comments on commit e68d80d

Please sign in to comment.