Validate capture variables to obey Azure's rules. (#3537)

This commit is contained in:
Christopher Boumenot 2016-05-18 17:25:57 -07:00 committed by Chris Bednarski
parent 9c9f8cd451
commit c1e7caf53c
22 changed files with 152 additions and 36 deletions

View File

@ -14,23 +14,24 @@ import (
"io/ioutil" "io/ioutil"
"math/big" "math/big"
"net/http" "net/http"
"regexp"
"strings"
"time" "time"
"github.com/Azure/azure-sdk-for-go/arm/compute" "github.com/Azure/azure-sdk-for-go/arm/compute"
"github.com/Azure/go-autorest/autorest/azure"
"github.com/Azure/go-autorest/autorest/to" "github.com/Azure/go-autorest/autorest/to"
"github.com/Azure/go-ntlmssp" "github.com/Azure/go-ntlmssp"
"github.com/mitchellh/packer/builder/azure/common/constants" "github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/packer/builder/azure/pkcs12" "github.com/mitchellh/packer/builder/azure/pkcs12"
"github.com/mitchellh/packer/common" "github.com/mitchellh/packer/common"
"github.com/mitchellh/packer/helper/communicator" "github.com/mitchellh/packer/helper/communicator"
"github.com/mitchellh/packer/helper/config" "github.com/mitchellh/packer/helper/config"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
"github.com/mitchellh/packer/template/interpolate" "github.com/mitchellh/packer/template/interpolate"
"github.com/Azure/go-autorest/autorest/azure"
"golang.org/x/crypto/ssh" "golang.org/x/crypto/ssh"
"strings"
) )
const ( const (
@ -40,6 +41,11 @@ const (
DefaultVMSize = "Standard_A1" DefaultVMSize = "Standard_A1"
) )
var (
reCaptureContainerName = regexp.MustCompile("^[a-z0-9][a-z0-9\\-]{2,62}$")
reCaptureNamePrefix = regexp.MustCompile("^[A-Za-z0-9][A-Za-z0-9_\\-\\.]{0,23}$")
)
type Config struct { type Config struct {
common.PackerConfig `mapstructure:",squash"` common.PackerConfig `mapstructure:",squash"`
@ -399,11 +405,31 @@ func assertRequiredParametersSet(c *Config, errs *packer.MultiError) {
///////////////////////////////////////////// /////////////////////////////////////////////
// Capture // Capture
if c.CaptureContainerName == "" { if c.CaptureContainerName == "" {
errs = packer.MultiErrorAppend(errs, fmt.Errorf("An capture_container_name must be specified")) errs = packer.MultiErrorAppend(errs, fmt.Errorf("A capture_container_name must be specified"))
}
if !reCaptureContainerName.MatchString(c.CaptureContainerName) {
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A capture_container_name must satisfy the regular expression %q.", reCaptureContainerName.String()))
}
if strings.HasSuffix(c.CaptureContainerName, "-") {
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A capture_container_name must not end with a hyphen, e.g. '-'."))
}
if strings.Contains(c.CaptureContainerName, "--") {
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A capture_container_name must not contain consecutive hyphens, e.g. '--'."))
} }
if c.CaptureNamePrefix == "" { if c.CaptureNamePrefix == "" {
errs = packer.MultiErrorAppend(errs, fmt.Errorf("An capture_name_prefix must be specified")) errs = packer.MultiErrorAppend(errs, fmt.Errorf("A capture_name_prefix must be specified"))
}
if !reCaptureNamePrefix.MatchString(c.CaptureNamePrefix) {
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A capture_name_prefix must satisfy the regular expression %q.", reCaptureNamePrefix.String()))
}
if strings.HasSuffix(c.CaptureNamePrefix, "-") || strings.HasSuffix(c.CaptureNamePrefix, ".") {
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A capture_name_prefix must not end with a hyphen or period."))
} }
///////////////////////////////////////////// /////////////////////////////////////////////

View File

@ -4,7 +4,6 @@
package arm package arm
import ( import (
"encoding/json"
"fmt" "fmt"
"strings" "strings"
"testing" "testing"
@ -401,10 +400,107 @@ func TestUseDeviceLoginIsDisabledForWindows(t *testing.T) {
} }
} }
func TestConfigShouldRejectMalformedCaptureNamePrefix(t *testing.T) {
config := map[string]string{
"capture_container_name": "ignore",
"image_offer": "ignore",
"image_publisher": "ignore",
"image_sku": "ignore",
"location": "ignore",
"storage_account": "ignore",
"subscription_id": "ignore",
// Does not matter for this test case, just pick one.
"os_type": constants.Target_Linux,
}
wellFormedCaptureNamePrefix := []string{
"packer",
"AbcdefghijklmnopqrstuvwX",
"hypen-hypen",
"0leading-number",
"v1.core.local",
}
for _, x := range wellFormedCaptureNamePrefix {
config["capture_name_prefix"] = x
_, _, err := newConfig(config, getPackerConfiguration())
if err != nil {
t.Errorf("Expected test to pass, but it failed with the well-formed capture_name_prefix set to %q.", x)
}
}
malformedCaptureNamePrefix := []string{
"-leading-hypen",
"trailing-hypen-",
"trailing-period.",
"_leading-underscore",
"punc-!@#$%^&*()_+-=-punc",
"There-are-too-many-characters-in-the-name-and-the-limit-is-twenty-four",
}
for _, x := range malformedCaptureNamePrefix {
config["capture_name_prefix"] = x
_, _, err := newConfig(config, getPackerConfiguration())
if err == nil {
t.Errorf("Expected test to fail, but it succeeded with the malformed capture_name_prefix set to %q.", x)
}
}
}
func TestConfigShouldRejectMalformedCaptureContainerName(t *testing.T) {
config := map[string]string{
"capture_name_prefix": "ignore",
"image_offer": "ignore",
"image_publisher": "ignore",
"image_sku": "ignore",
"location": "ignore",
"storage_account": "ignore",
"subscription_id": "ignore",
// Does not matter for this test case, just pick one.
"os_type": constants.Target_Linux,
}
wellFormedCaptureContainerName := []string{
"0leading",
"aleading",
"hype-hypen",
"abcdefghijklmnopqrstuvwxyz0123456789-abcdefghijklmnopqrstuvwxyz", // 63 characters
}
for _, x := range wellFormedCaptureContainerName {
config["capture_container_name"] = x
_, _, err := newConfig(config, getPackerConfiguration())
if err != nil {
t.Errorf("Expected test to pass, but it failed with the well-formed capture_container_name set to %q.", x)
}
}
malformedCaptureContainerName := []string{
"No-Capitals",
"double--hypens",
"-leading-hypen",
"trailing-hypen-",
"punc-!@#$%^&*()_+-=-punc",
"there-are-over-63-characters-in-this-string-and-that-is-a-bad-container-name",
}
for _, x := range malformedCaptureContainerName {
config["capture_container_name"] = x
_, _, err := newConfig(config, getPackerConfiguration())
if err == nil {
t.Errorf("Expected test to fail, but it succeeded with the malformed capture_container_name set to %q.", x)
}
}
}
func getArmBuilderConfiguration() map[string]string { func getArmBuilderConfiguration() map[string]string {
m := make(map[string]string) m := make(map[string]string)
for _, v := range requiredConfigValues { for _, v := range requiredConfigValues {
m[v] = fmt.Sprintf("%s00", v) m[v] = fmt.Sprintf("ignored00")
} }
m["communicator"] = "none" m["communicator"] = "none"
@ -413,19 +509,13 @@ func getArmBuilderConfiguration() map[string]string {
} }
func getPackerConfiguration() interface{} { func getPackerConfiguration() interface{} {
var doc = `{ config := map[string]interface{}{
"packer_user_variables": {
"sa": "my_storage_account"
},
"packer_build_name": "azure-arm-vm", "packer_build_name": "azure-arm-vm",
"packer_builder_type": "azure-arm-vm", "packer_builder_type": "azure-arm-vm",
"packer_debug": "false", "packer_debug": "false",
"packer_force": "false", "packer_force": "false",
"packer_template_path": "/home/jenkins/azure-arm-vm/template.json" "packer_template_path": "/home/jenkins/azure-arm-vm/template.json",
}` }
var config interface{}
json.Unmarshal([]byte(doc), &config)
return config return config
} }

View File

@ -4,9 +4,9 @@
package arm package arm
import ( import (
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common" "github.com/mitchellh/packer/builder/azure/common"
"github.com/mitchellh/packer/builder/azure/common/constants" "github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep"
) )
func processInterruptibleResult( func processInterruptibleResult(

View File

@ -7,9 +7,9 @@ import (
"fmt" "fmt"
"github.com/Azure/azure-sdk-for-go/arm/compute" "github.com/Azure/azure-sdk-for-go/arm/compute"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common" "github.com/mitchellh/packer/builder/azure/common"
"github.com/mitchellh/packer/builder/azure/common/constants" "github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
) )

View File

@ -8,8 +8,8 @@ import (
"testing" "testing"
"github.com/Azure/azure-sdk-for-go/arm/compute" "github.com/Azure/azure-sdk-for-go/arm/compute"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepCaptureImageShouldFailIfCaptureFails(t *testing.T) { func TestStepCaptureImageShouldFailIfCaptureFails(t *testing.T) {

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepCreateResourceGroupShouldFailIfCreateFails(t *testing.T) { func TestStepCreateResourceGroupShouldFailIfCreateFails(t *testing.T) {

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepDeleteOSDiskShouldFailIfGetFails(t *testing.T) { func TestStepDeleteOSDiskShouldFailIfGetFails(t *testing.T) {

View File

@ -6,9 +6,9 @@ package arm
import ( import (
"fmt" "fmt"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common" "github.com/mitchellh/packer/builder/azure/common"
"github.com/mitchellh/packer/builder/azure/common/constants" "github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
) )

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepDeployTemplateShouldFailIfDeployFails(t *testing.T) { func TestStepDeployTemplateShouldFailIfDeployFails(t *testing.T) {

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"time" "time"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
) )

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepGetCertificateShouldFailIfGetFails(t *testing.T) { func TestStepGetCertificateShouldFailIfGetFails(t *testing.T) {

View File

@ -6,8 +6,8 @@ package arm
import ( import (
"fmt" "fmt"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
) )

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepGetIPAddressShouldFailIfGetFails(t *testing.T) { func TestStepGetIPAddressShouldFailIfGetFails(t *testing.T) {

View File

@ -6,9 +6,9 @@ package arm
import ( import (
"fmt" "fmt"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common" "github.com/mitchellh/packer/builder/azure/common"
"github.com/mitchellh/packer/builder/azure/common/constants" "github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
) )

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepPowerOffComputeShouldFailIfPowerOffFails(t *testing.T) { func TestStepPowerOffComputeShouldFailIfPowerOffFails(t *testing.T) {

View File

@ -4,8 +4,8 @@
package arm package arm
import ( import (
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
) )

View File

@ -6,8 +6,8 @@ package arm
import ( import (
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepSetCertificateShouldPassIfGetPasses(t *testing.T) { func TestStepSetCertificateShouldPassIfGetPasses(t *testing.T) {

View File

@ -5,9 +5,9 @@ package arm
import ( import (
"fmt" "fmt"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common" "github.com/mitchellh/packer/builder/azure/common"
"github.com/mitchellh/packer/builder/azure/common/constants" "github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep"
"testing" "testing"
) )

View File

@ -6,8 +6,8 @@ package arm
import ( import (
"fmt" "fmt"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
) )

View File

@ -7,8 +7,8 @@ import (
"fmt" "fmt"
"testing" "testing"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
) )
func TestStepValidateTemplateShouldFailIfValidateFails(t *testing.T) { func TestStepValidateTemplateShouldFailIfValidateFails(t *testing.T) {

View File

@ -11,8 +11,8 @@ import (
"github.com/Azure/go-autorest/autorest" "github.com/Azure/go-autorest/autorest"
"github.com/Azure/go-autorest/autorest/azure" "github.com/Azure/go-autorest/autorest/azure"
"github.com/Azure/go-autorest/autorest/to" "github.com/Azure/go-autorest/autorest/to"
"github.com/mitchellh/packer/version"
"github.com/mitchellh/go-homedir" "github.com/mitchellh/go-homedir"
"github.com/mitchellh/packer/version"
) )
var ( var (

View File

@ -5,8 +5,8 @@ package lin
import ( import (
"fmt" "fmt"
"github.com/mitchellh/packer/builder/azure/common/constants"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
"github.com/mitchellh/packer/builder/azure/common/constants"
"golang.org/x/crypto/ssh" "golang.org/x/crypto/ssh"
) )