2015-02-04 17:44:23 -06:00
|
|
|
package terraform
|
|
|
|
|
2015-02-08 19:20:46 -06:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/config"
|
|
|
|
)
|
|
|
|
|
2015-02-04 17:44:23 -06:00
|
|
|
// EvalValidateError is the error structure returned if there were
|
|
|
|
// validation errors.
|
|
|
|
type EvalValidateError struct {
|
|
|
|
Warnings []string
|
|
|
|
Errors []error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *EvalValidateError) Error() string {
|
2015-02-10 01:32:28 -06:00
|
|
|
return fmt.Sprintf("Warnings: %s. Errors: %s", e.Warnings, e.Errors)
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|
|
|
|
|
2015-02-08 19:20:46 -06:00
|
|
|
// EvalValidateCount is an EvalNode implementation that validates
|
|
|
|
// the count of a resource.
|
|
|
|
type EvalValidateCount struct {
|
|
|
|
Resource *config.Resource
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: test
|
2015-02-14 00:58:41 -06:00
|
|
|
func (n *EvalValidateCount) Eval(ctx EvalContext) (interface{}, error) {
|
2015-02-08 19:20:46 -06:00
|
|
|
var count int
|
|
|
|
var errs []error
|
|
|
|
var err error
|
|
|
|
if _, err := ctx.Interpolate(n.Resource.RawCount, nil); err != nil {
|
|
|
|
errs = append(errs, fmt.Errorf(
|
|
|
|
"Failed to interpolate count: %s", err))
|
|
|
|
goto RETURN
|
|
|
|
}
|
|
|
|
|
|
|
|
count, err = n.Resource.Count()
|
|
|
|
if err != nil {
|
2015-02-10 14:16:55 -06:00
|
|
|
// If we can't get the count during validation, then
|
|
|
|
// just replace it with the number 1.
|
|
|
|
c := n.Resource.RawCount.Config()
|
|
|
|
c[n.Resource.RawCount.Key] = "1"
|
|
|
|
count = 1
|
2015-02-08 19:20:46 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if count < 0 {
|
|
|
|
errs = append(errs, fmt.Errorf(
|
|
|
|
"Count is less than zero: %d", count))
|
|
|
|
}
|
|
|
|
|
|
|
|
RETURN:
|
2015-10-11 12:45:33 -05:00
|
|
|
if len(errs) != 0 {
|
|
|
|
err = &EvalValidateError{
|
|
|
|
Errors: errs,
|
|
|
|
}
|
2015-02-08 19:20:46 -06:00
|
|
|
}
|
2015-10-11 12:45:33 -05:00
|
|
|
return nil, err
|
2015-02-08 19:20:46 -06:00
|
|
|
}
|
|
|
|
|
2015-02-04 19:23:26 -06:00
|
|
|
// EvalValidateProvider is an EvalNode implementation that validates
|
|
|
|
// the configuration of a resource.
|
|
|
|
type EvalValidateProvider struct {
|
2015-03-25 18:28:52 -05:00
|
|
|
Provider *ResourceProvider
|
|
|
|
Config **ResourceConfig
|
2015-02-04 19:23:26 -06:00
|
|
|
}
|
|
|
|
|
2015-02-14 00:58:41 -06:00
|
|
|
func (n *EvalValidateProvider) Eval(ctx EvalContext) (interface{}, error) {
|
|
|
|
provider := *n.Provider
|
|
|
|
config := *n.Config
|
2015-02-10 01:32:28 -06:00
|
|
|
|
2015-02-04 19:23:26 -06:00
|
|
|
warns, errs := provider.Validate(config)
|
|
|
|
if len(warns) == 0 && len(errs) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, &EvalValidateError{
|
|
|
|
Warnings: warns,
|
|
|
|
Errors: errs,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-09 13:15:54 -06:00
|
|
|
// EvalValidateProvisioner is an EvalNode implementation that validates
|
|
|
|
// the configuration of a resource.
|
|
|
|
type EvalValidateProvisioner struct {
|
2015-02-14 00:58:41 -06:00
|
|
|
Provisioner *ResourceProvisioner
|
|
|
|
Config **ResourceConfig
|
2015-02-09 13:15:54 -06:00
|
|
|
}
|
|
|
|
|
2015-02-14 00:58:41 -06:00
|
|
|
func (n *EvalValidateProvisioner) Eval(ctx EvalContext) (interface{}, error) {
|
|
|
|
provisioner := *n.Provisioner
|
|
|
|
config := *n.Config
|
|
|
|
warns, errs := provisioner.Validate(config)
|
2015-02-09 13:15:54 -06:00
|
|
|
if len(warns) == 0 && len(errs) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, &EvalValidateError{
|
|
|
|
Warnings: warns,
|
|
|
|
Errors: errs,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-04 17:44:23 -06:00
|
|
|
// EvalValidateResource is an EvalNode implementation that validates
|
|
|
|
// the configuration of a resource.
|
|
|
|
type EvalValidateResource struct {
|
2015-02-14 00:58:41 -06:00
|
|
|
Provider *ResourceProvider
|
|
|
|
Config **ResourceConfig
|
2015-02-09 11:50:20 -06:00
|
|
|
ResourceName string
|
2015-02-08 19:58:02 -06:00
|
|
|
ResourceType string
|
2016-05-01 21:01:48 -05:00
|
|
|
ResourceMode config.ResourceMode
|
2016-06-30 18:22:20 -05:00
|
|
|
|
|
|
|
// IgnoreWarnings means that warnings will not be passed through. This allows
|
|
|
|
// "just-in-time" passes of validation to continue execution through warnings.
|
|
|
|
IgnoreWarnings bool
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|
|
|
|
|
2015-02-14 00:58:41 -06:00
|
|
|
func (n *EvalValidateResource) Eval(ctx EvalContext) (interface{}, error) {
|
|
|
|
provider := *n.Provider
|
|
|
|
cfg := *n.Config
|
2016-05-01 21:01:48 -05:00
|
|
|
var warns []string
|
|
|
|
var errs []error
|
|
|
|
// Provider entry point varies depending on resource mode, because
|
|
|
|
// managed resources and data resources are two distinct concepts
|
|
|
|
// in the provider abstraction.
|
|
|
|
switch n.ResourceMode {
|
|
|
|
case config.ManagedResourceMode:
|
|
|
|
warns, errs = provider.ValidateResource(n.ResourceType, cfg)
|
|
|
|
case config.DataResourceMode:
|
|
|
|
warns, errs = provider.ValidateDataSource(n.ResourceType, cfg)
|
|
|
|
}
|
2015-02-09 11:50:20 -06:00
|
|
|
|
2016-06-30 18:22:20 -05:00
|
|
|
// If the resource name doesn't match the name regular
|
|
|
|
// expression, show an error.
|
2015-02-09 11:50:20 -06:00
|
|
|
if !config.NameRegexp.Match([]byte(n.ResourceName)) {
|
2016-02-23 10:32:02 -06:00
|
|
|
errs = append(errs, fmt.Errorf(
|
2015-02-09 11:50:20 -06:00
|
|
|
"%s: resource name can only contain letters, numbers, "+
|
2016-06-30 18:22:20 -05:00
|
|
|
"dashes, and underscores.", n.ResourceName))
|
2015-02-09 11:50:20 -06:00
|
|
|
}
|
2015-02-08 19:58:02 -06:00
|
|
|
|
2016-06-30 18:22:20 -05:00
|
|
|
if (len(warns) == 0 || n.IgnoreWarnings) && len(errs) == 0 {
|
2015-02-08 19:58:02 -06:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, &EvalValidateError{
|
|
|
|
Warnings: warns,
|
|
|
|
Errors: errs,
|
|
|
|
}
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|