Skip to content

Commit adf67ff

Browse files
support deprecating variables
1 parent b4a634c commit adf67ff

File tree

5 files changed

+186
-0
lines changed

5 files changed

+186
-0
lines changed

internal/configs/named_values.go

+14
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,10 @@ type Variable struct {
4848
Nullable bool
4949
NullableSet bool
5050

51+
Deprecated string
52+
DeprecatedSet bool
53+
DeprecatedRange hcl.Range
54+
5155
DeclRange hcl.Range
5256
}
5357

@@ -186,6 +190,13 @@ func decodeVariableBlock(block *hcl.Block, override bool) (*Variable, hcl.Diagno
186190
v.Default = val
187191
}
188192

193+
if attr, exists := content.Attributes["deprecated"]; exists {
194+
valDiags := gohcl.DecodeExpression(attr.Expr, nil, &v.Deprecated)
195+
diags = append(diags, valDiags...)
196+
v.DeprecatedSet = true
197+
v.DeprecatedRange = attr.Range
198+
}
199+
189200
for _, block := range content.Blocks {
190201
switch block.Type {
191202

@@ -499,6 +510,9 @@ var variableBlockSchema = &hcl.BodySchema{
499510
{
500511
Name: "nullable",
501512
},
513+
{
514+
Name: "deprecated",
515+
},
502516
},
503517
Blocks: []hcl.BlockHeaderSchema{
504518
{

internal/configs/named_values_test.go

+27
Original file line numberDiff line numberDiff line change
@@ -48,3 +48,30 @@ func TestVariableInvalidDefault(t *testing.T) {
4848
}
4949
}
5050
}
51+
52+
func TestVariableDeprecation(t *testing.T) {
53+
src := `
54+
variable "foo" {
55+
type = string
56+
deprecated = "This variable is deprecated, use bar instead"
57+
}
58+
`
59+
60+
hclF, diags := hclsyntax.ParseConfig([]byte(src), "test.tf", hcl.InitialPos)
61+
if diags.HasErrors() {
62+
t.Fatal(diags.Error())
63+
}
64+
65+
b, diags := parseConfigFile(hclF.Body, nil, false, false)
66+
if diags.HasErrors() {
67+
t.Fatalf("unexpected error: %q", diags)
68+
}
69+
70+
if !b.Variables[0].DeprecatedSet {
71+
t.Fatalf("expected variable to be deprecated")
72+
}
73+
74+
if b.Variables[0].Deprecated != "This variable is deprecated, use bar instead" {
75+
t.Fatalf("expected variable to have deprecation message")
76+
}
77+
}

internal/terraform/context_plan2_test.go

+127
Original file line numberDiff line numberDiff line change
@@ -6068,3 +6068,130 @@ data "test_data_source" "foo" {
60686068
_, diags := ctx.Plan(m, states.NewState(), SimplePlanOpts(plans.NormalMode, testInputValuesUnset(m.Module.Variables)))
60696069
assertNoErrors(t, diags)
60706070
}
6071+
6072+
func TestContext2Plan_deprecated_variable(t *testing.T) {
6073+
m := testModuleInline(t, map[string]string{
6074+
"mod/main.tf": `
6075+
variable "old-and-used" {
6076+
type = string
6077+
deprecated = "module variable deprecation"
6078+
default = "optional"
6079+
}
6080+
6081+
variable "old-and-unused" {
6082+
type = string
6083+
deprecated = "module variable deprecation"
6084+
default = "optional"
6085+
}
6086+
6087+
variable "new" {
6088+
type = string
6089+
default = "optional"
6090+
}
6091+
6092+
output "use-everything" {
6093+
value = {
6094+
used = var.old-and-used
6095+
unused = var.old-and-unused
6096+
new = var.new
6097+
}
6098+
}
6099+
`,
6100+
"main.tf": `
6101+
variable "root-old-and-used" {
6102+
type = string
6103+
deprecated = "root variable deprecation"
6104+
default = "optional"
6105+
}
6106+
6107+
variable "root-old-and-unused" {
6108+
type = string
6109+
deprecated = "root variable deprecation"
6110+
default = "optional"
6111+
}
6112+
6113+
variable "new" {
6114+
type = string
6115+
default = "new"
6116+
}
6117+
6118+
module "old-mod" {
6119+
source = "./mod"
6120+
old-and-used = "old"
6121+
}
6122+
6123+
module "new-mod" {
6124+
source = "./mod"
6125+
new = "new"
6126+
}
6127+
6128+
output "use-everything" {
6129+
value = {
6130+
used = var.root-old-and-used
6131+
unused = var.root-old-and-unused
6132+
new = var.new
6133+
}
6134+
}
6135+
`,
6136+
})
6137+
6138+
p := new(testing_provider.MockProvider)
6139+
p.GetProviderSchemaResponse = getProviderSchemaResponseFromProviderSchema(&providerSchema{
6140+
ResourceTypes: map[string]*configschema.Block{
6141+
"test_resource": {
6142+
Attributes: map[string]*configschema.Attribute{
6143+
"attr": {
6144+
Type: cty.String,
6145+
Computed: true,
6146+
},
6147+
},
6148+
},
6149+
},
6150+
})
6151+
6152+
ctx := testContext2(t, &ContextOpts{
6153+
Providers: map[addrs.Provider]providers.Factory{
6154+
addrs.NewDefaultProvider("test"): testProviderFuncFixed(p),
6155+
},
6156+
})
6157+
6158+
vars := InputValues{
6159+
"root-old-and-used": {
6160+
Value: cty.StringVal("root-old-and-used"),
6161+
},
6162+
"root-old-and-unused": {
6163+
Value: cty.NullVal(cty.String),
6164+
},
6165+
"new": {
6166+
Value: cty.StringVal("new"),
6167+
},
6168+
}
6169+
6170+
_, diags := ctx.Plan(m, states.NewState(), SimplePlanOpts(plans.NormalMode, vars))
6171+
6172+
var expectedDiags tfdiags.Diagnostics
6173+
expectedDiags = expectedDiags.Append(
6174+
&hcl.Diagnostic{
6175+
Severity: hcl.DiagWarning,
6176+
Summary: "Usage of deprecated variable",
6177+
Detail: "root variable deprecation",
6178+
Subject: &hcl.Range{
6179+
Filename: filepath.Join(m.Module.SourceDir, "main.tf"),
6180+
Start: hcl.Pos{Line: 4, Column: 2, Byte: 48},
6181+
End: hcl.Pos{Line: 4, Column: 42, Byte: 88},
6182+
},
6183+
},
6184+
&hcl.Diagnostic{
6185+
Severity: hcl.DiagWarning,
6186+
Summary: "Usage of deprecated variable",
6187+
Detail: "module variable deprecation",
6188+
Subject: &hcl.Range{
6189+
Filename: filepath.Join(m.Module.SourceDir, "main.tf"),
6190+
Start: hcl.Pos{Line: 21, Column: 20, Byte: 346},
6191+
End: hcl.Pos{Line: 21, Column: 25, Byte: 351},
6192+
},
6193+
},
6194+
)
6195+
6196+
assertDiagnosticsMatch(t, diags, expectedDiags)
6197+
}

internal/terraform/node_module_variable.go

+9
Original file line numberDiff line numberDiff line change
@@ -309,6 +309,15 @@ func (n *nodeModuleVariable) evalModuleVariable(ctx EvalContext, validateOnly bo
309309
finalVal, moreDiags := prepareFinalInputVariableValue(n.Addr, rawVal, n.Config)
310310
diags = diags.Append(moreDiags)
311311

312+
if n.Config.DeprecatedSet && givenVal.IsWhollyKnown() && !givenVal.IsNull() && validateOnly {
313+
diags = diags.Append(&hcl.Diagnostic{
314+
Severity: hcl.DiagWarning,
315+
Summary: "Usage of deprecated variable",
316+
Detail: n.Config.Deprecated,
317+
Subject: n.Expr.Range().Ptr(),
318+
})
319+
}
320+
312321
return finalVal, diags.ErrWithWarnings()
313322
}
314323

internal/terraform/node_root_variable.go

+9
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,15 @@ func (n *NodeRootVariable) Execute(ctx EvalContext, op walkOperation) tfdiags.Di
9292
}
9393
}
9494

95+
if n.Config.DeprecatedSet && givenVal.Value.IsWhollyKnown() && !givenVal.Value.IsNull() && op == walkPlan {
96+
diags = diags.Append(&hcl.Diagnostic{
97+
Severity: hcl.DiagWarning,
98+
Summary: "Usage of deprecated variable",
99+
Detail: n.Config.Deprecated,
100+
Subject: &n.Config.DeprecatedRange,
101+
})
102+
}
103+
95104
if n.Planning {
96105
if checkState := ctx.Checks(); checkState.ConfigHasChecks(n.Addr.InModule(addrs.RootModule)) {
97106
ctx.Checks().ReportCheckableObjects(

0 commit comments

Comments
 (0)