Terraform is trusted for managing many facets of infrastructure across many organizations. Part of that trust is due to consistent versioning guidelines and setting expectations for various levels of upgrades. Ensuring backwards compatibility for all patch and minor releases, potentially in concert with any upcoming major changes, is recommended and supported by the Terraform development framework. This allows operators to iteratively update their Terraform configurations rather than require massive refactoring.
This guide is designed to walk through various scenarios where existing Terraform functionality requires future removal, while maintaining backwards compatibility. Further information about the versioning terminology (e.g. MAJOR
.MINOR
.PATCH
) in this guide can be found in the versioning guidelines documentation.
NOTE: Removals should only ever occur in MAJOR
version upgrades.
NOTE: This documentation references usage of the DeprecationMessage
field, please see the schema documentation for more detailed guidance on how to structure warning messages and when those warnings will be raised to practitioners.
The recommended process for removing an attribute from a data source or resource in a provider is as follows:
DeprecationMessage
in the attribute schema definition. Set this field to a practitioner actionable message such as "Remove this attribute's configuration as it's no longer in use and the attribute will be removed in the next major version of the provider."
MINOR
version with the deprecation.MAJOR
version, remove all code associated with the attribute including the schema definition.MAJOR
version.When renaming an attribute from one name to another, it is important to keep backwards compatibility with both existing Terraform configurations and the Terraform state while operators migrate. To accomplish this, there will be some duplicated logic to support both attributes until the next MAJOR
release. Once both attributes are appropriately handled, the process for deprecating and removing the old attribute is the same as noted in the Provider Attribute Removal section.
The procedure for renaming an attribute depends on what type of attribute it is:
Renaming a Required AttributeNOTE: If the schema definition does not contain Optional
or Required
, see the Renaming a Computed Attribute section instead. If the schema definition contains Optional
instead of Required
, see the Renaming an Optional Attribute section.
Required attributes are also referred to as required "arguments" throughout the Terraform documentation.
In general, the procedure here does two things:
terraform plan
output format:existing_attribute: "" => "value"
new_attribute: "value" => ""
The recommended process is as follows:
Required: true
with Optional: true
in the existing attribute schema definition.Required
with Optional
in the existing attribute documentation.DeprecationMessage
to the schema definition of the existing (now the "old") attribute, noting to use the new attribute in the message.**Deprecated**
to the documentation of the existing (now the "old") attribute, noting to use the new attribute.{type}validator.ExactlyOneOf
to the schema definition of the new attribute, with a path expression matching the old attribute. This will ensure at least one of the attributes is configured, but present an error to the operator if both are configured at the same time. For example, an attribute of type string would use the stringvalidator.ExactlyOneOf
validator.Create
and Update
functions of the data source or resource to handle both attributes. Generally, this involves using {type}.IsNull()
.Required
, and remove the {type}validator.ExactlyOneOf
validator.Given this sample resource:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"existing_attribute": schema.StringAttribute{
Required: true,
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
ExistingAttribute types.String `tfsdk:"existing_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add attribute to provider create API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
// ... other logic ...
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add attribute to provider update API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
In order to support renaming existing_attribute
to new_attribute
, this sample can be written as the following to support both attributes simultaneously until the existing_attribute
is removed:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
"github.com/hashicorp/terraform-plugin-framework/path"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/schema/validator"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"existing_attribute": schema.StringAttribute{
Optional: true,
DeprecationMessage: "use new_attribute instead",
},
"new_attribute": schema.StringAttribute{
Optional: true,
Validators: []validator.String{
stringvalidator.ExactlyOneOf(path.Expressions{
path.MatchRoot("existing_attribute"),
}...),
},
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
ExistingAttribute types.String `tfsdk:"existing_attribute"`
NewAttribute types.String `tfsdk:"new_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
if !data.NewAttribute.IsNull() {
// add NewAttribute to provider create API call
} else {
// add ExistingAttribute to provider create API call
}
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
// ... other logic ...
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
if !data.NewAttribute.IsNull() {
// add NewAttribute to provider create API call
} else {
// add ExistingAttribute to provider create API call
}
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
When the existing_attribute
is ready for removal, then this can be written as:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"new_attribute": schema.StringAttribute{
Required: true,
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
NewAttribute types.String `tfsdk:"new_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add NewAttribute to provider create API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
// ... other logic ...
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add NewAttribute to provider create API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
Renaming an Optional Attribute
NOTE: If the schema definition does not contain Optional
or Required
, see the Renaming a Computed Attribute section instead. If the schema definition contains Required
instead of Optional
, see the Renaming a Required Attribute section.
Optional attributes are also referred to as optional "arguments" throughout the Terraform documentation.
In general, the procedure here allows the operator to migrate the configuration to the new attribute at the same time requiring that any other references only work with the new attribute. This is to prevent a situation with Terraform showing a difference when the existing attribute is configured, but the new attribute is saved into the Terraform state. For example, in terraform plan
output format:
existing_attribute: "" => "value"
new_attribute: "value" => ""
The recommended process is as follows:
DeprecationMessage
to the schema definition of the existing (now the "old") attribute, noting to use the new attribute in the message.**Deprecated**
to the documentation of the existing (now the "old") attribute, noting to use the new attribute.{type}validator.ExactlyOneOf
to the schema definition of the new attribute, with a path expression matching the old attribute. This will ensure at least one of the attributes is configured, but present an error to the operator if both are configured at the same time. For example, an attribute of type string would use the stringvalidator.ExactlyOneOf
validator.Create
and Update
functions of the data source or resource to handle both attributes. Generally, this involves using {type}.IsNull()
.{type}validator.ExactlyOneOf
validator.Given this sample resource:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"existing_attribute": schema.StringAttribute{
Optional: true,
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
ExistingAttribute types.String `tfsdk:"existing_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add attribute to provider create API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
// ... other logic ...
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add attribute to provider update API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
In order to support renaming existing_attribute
to new_attribute
, this sample can be written as the following to support both attributes simultaneously until the existing_attribute
is removed:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
"github.com/hashicorp/terraform-plugin-framework/path"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/schema/validator"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"existing_attribute": schema.StringAttribute{
Optional: true,
DeprecationMessage: "use new_attribute instead",
},
"new_attribute": schema.StringAttribute{
Optional: true,
Validators: []validator.String{
stringvalidator.ExactlyOneOf(path.Expressions{
path.MatchRoot("existing_attribute"),
}...),
},
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
ExistingAttribute types.String `tfsdk:"existing_attribute"`
NewAttribute types.String `tfsdk:"new_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
if !data.NewAttribute.IsNull() {
// add NewAttribute to provider create API call
} else {
// add ExistingAttribute to provider create API call
}
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
// ... other logic ...
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
if !data.NewAttribute.IsNull() {
// add NewAttribute to provider create API call
} else {
// add ExistingAttribute to provider create API call
}
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
When the existing_attribute
is ready for removal, then this can be written as:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"new_attribute": schema.StringAttribute{
Optional: true,
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
NewAttribute types.String `tfsdk:"new_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add NewAttribute to provider create API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
// ... other logic ...
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Plan.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// add NewAttribute to provider create API call
// ... other logic ...
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
Renaming a Computed Attribute
NOTE: If the schema definition contains Optional
see the Renaming an Optional Attribute section instead. If the schema definition contains Required
see the Renaming a Required Attribute section instead.
The recommended process is as follows:
DeprecationMessage
to the schema definition of the existing (now the "old") attribute, noting to use the new attribute in the message.**Deprecated**
to the documentation of the existing (now the "old") attribute, noting to use the new attribute.Create
, Update
, and Read
functions of the resource (Read
only for data source).Given this sample resource:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"existing_attribute": schema.StringAttribute{
Computed: true,
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
ExistingAttribute types.String `tfsdk:"existing_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.ExistingAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.State.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.ExistingAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.ExistingAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
In order to support renaming existing_attribute
to new_attribute
, this sample can be written as the following to support both attributes simultaneously until the existing_attribute
is removed:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"existing_attribute": schema.StringAttribute{
Computed: true,
DeprecationMessage: "use new_attribute instead",
},
"new_attribute": schema.StringAttribute{
Computed: true,
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
ExistingAttribute types.String `tfsdk:"existing_attribute"`
NewAttribute types.String `tfsdk:"new_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.ExistingAttribute = // set to computed value
data.NewAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.State.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.ExistingAttribute = // set to computed value
data.NewAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.ExistingAttribute = // set to computed value
data.NewAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
When the existing_attribute
is ready for removal, then this can be written as:
package provider
import (
"context"
"github.com/hashicorp/terraform-plugin-framework/resource"
"github.com/hashicorp/terraform-plugin-framework/resource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
)
var _ resource.Resource = (*exampleWidgetResource)(nil)
type exampleWidgetResource struct{}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Metadata(_ context.Context, req resource.MetadataRequest, resp *resource.MetadataResponse) {
resp.TypeName = req.ProviderTypeName + "_widget"
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
"new_attribute": schema.StringAttribute{
Computed: true,
},
},
}
}
type exampleWidgetResourceData struct {
// ... other attributes ...
NewAttribute types.String `tfsdk:"new_attribute"`
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.NewAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.State.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.NewAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
var data exampleWidgetResourceData
resp.Diagnostics.Append(req.Config.Get(ctx, &data)...)
if resp.Diagnostics.HasError() {
return
}
// ... other logic ...
data.NewAttribute = // set to computed value
resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
// ... other logic ...
}
The recommended process for removing a data source or resource from a provider is as follows:
DeprecationMessage
in the data source or resource schema definition. After an operator upgrades to this version, they will be shown a warning with the message provided when using the deprecated data source or resource, but the Terraform run will still complete.MINOR
version with the deprecation.MAJOR
version, remove all code associated with the deprecated data source or resource except for the schema and replace the Create
, Read
, Update
, and Delete
functions to always return an error diagnostic. Remove the documentation sidebar link and update the resource or data source documentation page to include information about the removal and any potential migration information. After an operator upgrades to this version, they will be shown an error about the missing data source or resource.MAJOR
version.MAJOR
version, remove all code associated with the removed data source or resource. Remove the resource or data source documentation page.MAJOR
version.Given this sample provider and resource:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
NewWidgetResource,
}
}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
}
}
// ... resource implementation ...
In order to deprecate example_widget
, this sample can be written as:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
NewWidgetResource,
}
}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
DeprecationMessage: "use example_thing resource instead",
}
}
// ... resource implementation ...
To soft remove example_widget
with a friendly error message, this sample can be written as:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
NewWidgetResource,
}
}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
DeprecationMessage: "use example_thing resource instead",
}
}
func (e *exampleWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_widget resource deprecated", "use example_thing resource instead"),
)
}
func (e *exampleWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_widget resource deprecated", "use example_thing resource instead"),
)
}
func (e *exampleWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_widget resource deprecated", "use example_thing resource instead"),
)
}
func (e *exampleWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_widget resource deprecated", "use example_thing resource instead"),
)
}
To remove example_widget
:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
}
}
When renaming a resource from one name to another, it is important to keep backwards compatibility with both existing Terraform configurations and the Terraform state while operators migrate. To accomplish this, there will be some duplicated logic to support both resources until the next MAJOR
release. Once both resources are appropriately handled, the process for deprecating and removing the old resource is the same as noted in the Provider Data Source or Resource Removal section.
The recommended process is as follows:
DeprecationMessage
to the schema definition of the existing (now the "old") resource, noting to use the new resource in the message.~> This resource is deprecated and will be removed in the next major version
to the documentation of the existing (now the "old") resource, noting to use the new resource.Resources
functionGiven this sample provider and resource:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
NewExistingWidgetResource,
}
}
func NewExistingWidgetResource() resource.Resource {
return &exampleExistingWidgetResource{}
}
func (e *exampleExistingWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
}
}
// ... resource implementation ...
In order to support renaming example_existing_widget
to example_new_widget
, this sample can be written as the following to support both resources simultaneously until the example_existing_widget
resource is removed:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
NewExistingWidgetResource,
NewWidgetResource,
}
}
func NewExistingWidgetResource() resource.Resource {
return &exampleExistingWidgetResource{}
}
func (e *exampleExistingWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
DeprecationMessage: "use example_new_widget resource instead",
}
}
// ... resource implementation ...
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
}
}
// ... resource implementation ...
To soft remove example_existing_widget
with a friendly error message:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
NewExistingWidgetResource,
NewWidgetResource,
}
}
func NewExistingWidgetResource() resource.Resource {
return &exampleExistingWidgetResource{}
}
func (e *exampleExistingWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
DeprecationMessage: "use example_new_widget resource instead",
}
}
func (e *exampleExistingWidgetResource) Create(ctx context.Context, req resource.CreateRequest, resp *resource.CreateResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_existing_widget resource deprecated", "use example_new_widget resource instead"),
)
}
func (e *exampleExistingWidgetResource) Read(ctx context.Context, req resource.ReadRequest, resp *resource.ReadResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_existing_widget resource deprecated", "use example_new_widget resource instead"),
)
}
func (e *exampleExistingWidgetResource) Update(ctx context.Context, req resource.UpdateRequest, resp *resource.UpdateResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_existing_widget resource deprecated", "use example_new_widget resource instead"),
)
}
func (e *exampleExistingWidgetResource) Delete(ctx context.Context, req resource.DeleteRequest, resp *resource.DeleteResponse) {
resp.Diagnostics.Append(
diag.NewErrorDiagnostic("example_existing_widget resource deprecated", "use example_new_widget resource instead"),
)
}
// ... resource implementation ...
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
}
}
// ... resource implementation ...
To remove example_existing_widget
:
// ... provider implementation ...
func (p *exampleProvider) Resources(ctx context.Context) []func() resource.Resource {
return []func() resource.Resource{
//... other resources ...
NewWidgetResource,
}
}
func NewWidgetResource() resource.Resource {
return &exampleWidgetResource{}
}
func (e *exampleWidgetResource) Schema(ctx context.Context, req resource.SchemaRequest, resp *resource.SchemaResponse) {
resp.Schema = schema.Schema{
// ... other configuration ...
Attributes: map[string]schema.Attribute{
// ... other attributes ...
},
}
}
// ... resource implementation ...
RetroSearch is an open source project built by @garambo | Open a GitHub Issue
Search and Browse the WWW like it's 1997 | Search results from DuckDuckGo
HTML:
3.2
| Encoding:
UTF-8
| Version:
0.7.4