2018-02-01 22:33:06 -06:00
package configs
import (
2018-04-24 12:06:51 -05:00
"fmt"
2019-09-09 17:58:44 -05:00
"github.com/hashicorp/hcl/v2"
"github.com/hashicorp/hcl/v2/gohcl"
"github.com/hashicorp/hcl/v2/hclsyntax"
2018-02-01 22:33:06 -06:00
)
// ModuleCall represents a "module" block in a module or file.
type ModuleCall struct {
2018-02-02 19:22:25 -06:00
Name string
SourceAddr string
SourceAddrRange hcl . Range
2018-02-06 20:05:14 -06:00
SourceSet bool
2018-02-02 19:22:25 -06:00
Config hcl . Body
2018-02-01 22:33:06 -06:00
Version VersionConstraint
Count hcl . Expression
ForEach hcl . Expression
2018-04-24 12:06:51 -05:00
Providers [ ] PassedProviderConfig
2018-02-02 19:22:25 -06:00
DependsOn [ ] hcl . Traversal
2018-02-01 22:33:06 -06:00
DeclRange hcl . Range
}
2018-02-02 19:22:25 -06:00
2018-02-06 20:05:14 -06:00
func decodeModuleBlock ( block * hcl . Block , override bool ) ( * ModuleCall , hcl . Diagnostics ) {
2020-07-02 09:32:57 -05:00
var diags hcl . Diagnostics
// Produce deprecation messages for any pre-0.12-style
// single-interpolation-only expressions.
moreDiags := warnForDeprecatedInterpolationsInBody ( block . Body )
diags = append ( diags , moreDiags ... )
2018-02-02 19:22:25 -06:00
mc := & ModuleCall {
Name : block . Labels [ 0 ] ,
DeclRange : block . DefRange ,
}
2018-02-06 20:05:14 -06:00
schema := moduleBlockSchema
if override {
schema = schemaForOverrides ( schema )
}
2020-07-02 09:32:57 -05:00
content , remain , moreDiags := block . Body . PartialContent ( schema )
diags = append ( diags , moreDiags ... )
2018-02-02 19:22:25 -06:00
mc . Config = remain
if ! hclsyntax . ValidIdentifier ( mc . Name ) {
diags = append ( diags , & hcl . Diagnostic {
Severity : hcl . DiagError ,
Summary : "Invalid module instance name" ,
Detail : badIdentifierDetail ,
Subject : & block . LabelRanges [ 0 ] ,
} )
}
if attr , exists := content . Attributes [ "source" ] ; exists {
valDiags := gohcl . DecodeExpression ( attr . Expr , nil , & mc . SourceAddr )
diags = append ( diags , valDiags ... )
mc . SourceAddrRange = attr . Expr . Range ( )
2018-02-06 20:05:14 -06:00
mc . SourceSet = true
2018-02-02 19:22:25 -06:00
}
if attr , exists := content . Attributes [ "version" ] ; exists {
var versionDiags hcl . Diagnostics
mc . Version , versionDiags = decodeVersionConstraint ( attr )
diags = append ( diags , versionDiags ... )
}
if attr , exists := content . Attributes [ "count" ] ; exists {
mc . Count = attr . Expr
}
if attr , exists := content . Attributes [ "for_each" ] ; exists {
2020-04-07 13:18:08 -05:00
if mc . Count != nil {
diags = append ( diags , & hcl . Diagnostic {
Severity : hcl . DiagError ,
Summary : ` Invalid combination of "count" and "for_each" ` ,
Detail : ` The "count" and "for_each" meta-arguments are mutually-exclusive, only one should be used to be explicit about the number of resources to be created. ` ,
Subject : & attr . NameRange ,
} )
}
2018-02-02 19:22:25 -06:00
mc . ForEach = attr . Expr
}
if attr , exists := content . Attributes [ "depends_on" ] ; exists {
deps , depsDiags := decodeDependsOn ( attr )
diags = append ( diags , depsDiags ... )
mc . DependsOn = append ( mc . DependsOn , deps ... )
}
2018-04-24 12:06:51 -05:00
if attr , exists := content . Attributes [ "providers" ] ; exists {
seen := make ( map [ string ] hcl . Range )
pairs , pDiags := hcl . ExprMap ( attr . Expr )
diags = append ( diags , pDiags ... )
for _ , pair := range pairs {
key , keyDiags := decodeProviderConfigRef ( pair . Key , "providers" )
diags = append ( diags , keyDiags ... )
value , valueDiags := decodeProviderConfigRef ( pair . Value , "providers" )
diags = append ( diags , valueDiags ... )
if keyDiags . HasErrors ( ) || valueDiags . HasErrors ( ) {
continue
}
matchKey := key . String ( )
if prev , exists := seen [ matchKey ] ; exists {
diags = append ( diags , & hcl . Diagnostic {
Severity : hcl . DiagError ,
Summary : "Duplicate provider address" ,
Detail : fmt . Sprintf ( "A provider configuration was already passed to %s at %s. Each child provider configuration can be assigned only once." , matchKey , prev ) ,
Subject : pair . Value . Range ( ) . Ptr ( ) ,
} )
continue
}
rng := hcl . RangeBetween ( pair . Key . Range ( ) , pair . Value . Range ( ) )
seen [ matchKey ] = rng
mc . Providers = append ( mc . Providers , PassedProviderConfig {
InChild : key ,
InParent : value ,
} )
}
}
2018-11-20 13:53:45 -06:00
// Reserved block types (all of them)
for _ , block := range content . Blocks {
diags = append ( diags , & hcl . Diagnostic {
Severity : hcl . DiagError ,
Summary : "Reserved block type name in module block" ,
Detail : fmt . Sprintf ( "The block type name %q is reserved for use by Terraform in a future version." , block . Type ) ,
Subject : & block . TypeRange ,
} )
}
2018-02-02 19:22:25 -06:00
return mc , diags
}
2018-04-24 12:06:51 -05:00
// PassedProviderConfig represents a provider config explicitly passed down to
// a child module, possibly giving it a new local address in the process.
type PassedProviderConfig struct {
InChild * ProviderConfigRef
InParent * ProviderConfigRef
}
2018-02-02 19:22:25 -06:00
var moduleBlockSchema = & hcl . BodySchema {
Attributes : [ ] hcl . AttributeSchema {
{
Name : "source" ,
Required : true ,
} ,
{
Name : "version" ,
} ,
{
Name : "count" ,
} ,
{
Name : "for_each" ,
} ,
{
Name : "depends_on" ,
} ,
2018-04-24 12:06:51 -05:00
{
Name : "providers" ,
} ,
2018-02-02 19:22:25 -06:00
} ,
2018-11-20 13:53:45 -06:00
Blocks : [ ] hcl . BlockHeaderSchema {
// These are all reserved for future use.
{ Type : "lifecycle" } ,
{ Type : "locals" } ,
{ Type : "provider" , LabelNames : [ ] string { "type" } } ,
} ,
2018-02-02 19:22:25 -06:00
}