2016-03-04 05:14:55 -05:00
|
|
|
package arm
|
|
|
|
|
|
|
|
import (
|
2018-01-22 18:32:33 -05:00
|
|
|
"context"
|
2017-12-12 04:12:53 -05:00
|
|
|
"errors"
|
2016-03-04 05:14:55 -05:00
|
|
|
"fmt"
|
2017-11-06 00:16:58 -05:00
|
|
|
"net/url"
|
|
|
|
"strings"
|
2016-03-04 05:14:55 -05:00
|
|
|
|
2017-04-04 16:39:01 -04:00
|
|
|
"github.com/hashicorp/packer/builder/azure/common/constants"
|
2018-01-19 19:18:44 -05:00
|
|
|
"github.com/hashicorp/packer/helper/multistep"
|
2017-04-04 16:39:01 -04:00
|
|
|
"github.com/hashicorp/packer/packer"
|
2016-03-04 05:14:55 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
type StepDeployTemplate struct {
|
2017-11-06 00:16:58 -05:00
|
|
|
client *AzureClient
|
2018-04-06 04:12:58 -04:00
|
|
|
deploy func(ctx context.Context, resourceGroupName string, deploymentName string) error
|
|
|
|
delete func(ctx context.Context, client *AzureClient, resourceType string, resourceName string, resourceGroupName string) error
|
|
|
|
disk func(ctx context.Context, resourceGroupName string, computeName string) (string, string, error)
|
|
|
|
deleteDisk func(ctx context.Context, imageType string, imageName string, resourceGroupName string) error
|
2017-11-06 00:16:58 -05:00
|
|
|
say func(message string)
|
|
|
|
error func(e error)
|
|
|
|
config *Config
|
|
|
|
factory templateFactoryFunc
|
2017-11-30 03:11:17 -05:00
|
|
|
name string
|
2016-03-04 05:14:55 -05:00
|
|
|
}
|
|
|
|
|
2017-11-30 03:11:17 -05:00
|
|
|
func NewStepDeployTemplate(client *AzureClient, ui packer.Ui, config *Config, deploymentName string, factory templateFactoryFunc) *StepDeployTemplate {
|
2016-03-04 05:14:55 -05:00
|
|
|
var step = &StepDeployTemplate{
|
2016-05-21 02:01:16 -04:00
|
|
|
client: client,
|
|
|
|
say: func(message string) { ui.Say(message) },
|
|
|
|
error: func(e error) { ui.Error(e.Error()) },
|
|
|
|
config: config,
|
|
|
|
factory: factory,
|
2017-11-30 03:11:17 -05:00
|
|
|
name: deploymentName,
|
2016-03-04 05:14:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
step.deploy = step.deployTemplate
|
2018-04-06 04:12:58 -04:00
|
|
|
step.delete = deleteResource
|
2017-11-06 00:16:58 -05:00
|
|
|
step.disk = step.getImageDetails
|
|
|
|
step.deleteDisk = step.deleteImage
|
2016-03-04 05:14:55 -05:00
|
|
|
return step
|
|
|
|
}
|
|
|
|
|
2020-08-10 12:57:06 -04:00
|
|
|
func (s *StepDeployTemplate) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction {
|
|
|
|
s.say("Deploying deployment template ...")
|
|
|
|
|
|
|
|
var resourceGroupName = state.Get(constants.ArmResourceGroupName).(string)
|
|
|
|
s.say(fmt.Sprintf(" -> ResourceGroupName : '%s'", resourceGroupName))
|
|
|
|
s.say(fmt.Sprintf(" -> DeploymentName : '%s'", s.name))
|
|
|
|
|
|
|
|
return processStepResult(
|
|
|
|
s.deploy(ctx, resourceGroupName, s.name),
|
|
|
|
s.error, state)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *StepDeployTemplate) Cleanup(state multistep.StateBag) {
|
2020-08-12 11:30:15 -04:00
|
|
|
defer func() {
|
|
|
|
err := s.deleteTemplate(context.Background(), state)
|
|
|
|
if err != nil {
|
|
|
|
s.say(s.client.LastError.Error())
|
|
|
|
}
|
|
|
|
}()
|
2020-08-10 12:57:06 -04:00
|
|
|
|
|
|
|
//Only clean up if this was an existing resource group and the resource group
|
|
|
|
//is marked as created
|
|
|
|
existingResourceGroup := state.Get(constants.ArmIsExistingResourceGroup).(bool)
|
|
|
|
resourceGroupCreated := state.Get(constants.ArmIsResourceGroupCreated).(bool)
|
|
|
|
if !existingResourceGroup || !resourceGroupCreated {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ui := state.Get("ui").(packer.Ui)
|
|
|
|
ui.Say("\nThe resource group was not created by Packer, deleting individual resources ...")
|
|
|
|
|
2020-08-12 11:03:51 -04:00
|
|
|
deploymentName := s.name
|
2020-08-10 12:57:06 -04:00
|
|
|
resourceGroupName := state.Get(constants.ArmResourceGroupName).(string)
|
2020-08-12 11:03:51 -04:00
|
|
|
|
|
|
|
// Get image disk details before deleting the image; otherwise we won't be able to
|
|
|
|
// delete the disk as the image request will return a 404
|
2020-08-10 12:57:06 -04:00
|
|
|
computeName := state.Get(constants.ArmComputeName).(string)
|
|
|
|
imageType, imageName, err := s.disk(context.TODO(), resourceGroupName, computeName)
|
2020-08-12 11:03:51 -04:00
|
|
|
|
|
|
|
if err != nil && !strings.Contains(err.Error(), "ResourceNotFound") {
|
|
|
|
ui.Error(fmt.Sprintf("Could not retrieve OS Image details: %s", err))
|
2020-08-10 12:57:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say(" -> Deployment Resources within: " + deploymentName)
|
|
|
|
if deploymentName != "" {
|
|
|
|
maxResources := int32(50)
|
|
|
|
deploymentOperations, err := s.client.DeploymentOperationsClient.ListComplete(context.TODO(), resourceGroupName, deploymentName, &maxResources)
|
|
|
|
if err != nil {
|
|
|
|
ui.Error(fmt.Sprintf("Error deleting resources. Please delete them manually.\n\n"+
|
|
|
|
"Name: %s\n"+
|
|
|
|
"Error: %s", resourceGroupName, err))
|
|
|
|
}
|
|
|
|
|
|
|
|
for deploymentOperations.NotDone() {
|
|
|
|
deploymentOperation := deploymentOperations.Value()
|
|
|
|
// Sometimes an empty operation is added to the list by Azure
|
|
|
|
if deploymentOperation.Properties.TargetResource == nil {
|
2020-08-11 16:40:51 -04:00
|
|
|
if err := deploymentOperations.Next(); err != nil {
|
2020-08-12 11:03:51 -04:00
|
|
|
ui.Error(fmt.Sprintf("Error moving to to next deployment operation ...\n\n"+
|
2020-08-11 16:40:51 -04:00
|
|
|
"Name: %s\n"+
|
|
|
|
"Error: %s", resourceGroupName, err))
|
|
|
|
break
|
|
|
|
}
|
2020-08-10 12:57:06 -04:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say(fmt.Sprintf(" -> %s : '%s'",
|
|
|
|
*deploymentOperation.Properties.TargetResource.ResourceType,
|
|
|
|
*deploymentOperation.Properties.TargetResource.ResourceName))
|
|
|
|
|
|
|
|
err = s.delete(context.TODO(), s.client,
|
|
|
|
*deploymentOperation.Properties.TargetResource.ResourceType,
|
|
|
|
*deploymentOperation.Properties.TargetResource.ResourceName,
|
|
|
|
resourceGroupName)
|
|
|
|
if err != nil {
|
|
|
|
ui.Error(fmt.Sprintf("Error deleting resource. Please delete manually.\n\n"+
|
|
|
|
"Name: %s\n"+
|
|
|
|
"Error: %s", *deploymentOperation.Properties.TargetResource.ResourceName, err))
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = deploymentOperations.Next(); err != nil {
|
|
|
|
ui.Error(fmt.Sprintf("Error deleting resources. Please delete them manually.\n\n"+
|
|
|
|
"Name: %s\n"+
|
|
|
|
"Error: %s", resourceGroupName, err))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-12 11:03:51 -04:00
|
|
|
// The disk is not defined as an operation in the template so it has to be deleted separately
|
|
|
|
if imageType == "" && imageName == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-08-10 12:57:06 -04:00
|
|
|
ui.Say(fmt.Sprintf(" -> %s : '%s'", imageType, imageName))
|
|
|
|
err = s.deleteDisk(context.TODO(), imageType, imageName, resourceGroupName)
|
|
|
|
if err != nil {
|
|
|
|
ui.Error(fmt.Sprintf("Error deleting resource. Please delete manually.\n\n"+
|
|
|
|
"Name: %s\n"+
|
|
|
|
"Error: %s", imageName, err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:12:58 -04:00
|
|
|
func (s *StepDeployTemplate) deployTemplate(ctx context.Context, resourceGroupName string, deploymentName string) error {
|
2016-05-21 02:01:16 -04:00
|
|
|
deployment, err := s.factory(s.config)
|
2016-03-04 05:14:55 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:12:58 -04:00
|
|
|
f, err := s.client.DeploymentsClient.CreateOrUpdate(ctx, resourceGroupName, deploymentName, *deployment)
|
|
|
|
if err == nil {
|
2019-05-30 17:25:43 -04:00
|
|
|
err = f.WaitForCompletionRef(ctx, s.client.DeploymentsClient.Client)
|
2018-04-06 04:12:58 -04:00
|
|
|
}
|
2017-06-08 20:57:59 -04:00
|
|
|
if err != nil {
|
|
|
|
s.say(s.client.LastError.Error())
|
|
|
|
}
|
2016-06-20 17:22:01 -04:00
|
|
|
return err
|
2016-03-04 05:14:55 -05:00
|
|
|
}
|
|
|
|
|
2020-01-22 17:58:26 -05:00
|
|
|
func (s *StepDeployTemplate) deleteTemplate(ctx context.Context, state multistep.StateBag) error {
|
|
|
|
var resourceGroupName = state.Get(constants.ArmResourceGroupName).(string)
|
|
|
|
var deploymentName = s.name
|
|
|
|
ui := state.Get("ui").(packer.Ui)
|
|
|
|
|
2020-08-10 12:57:06 -04:00
|
|
|
ui.Say(fmt.Sprintf("Removing the created Deployment object: '%s'", deploymentName))
|
2019-12-17 10:57:57 -05:00
|
|
|
f, err := s.client.DeploymentsClient.Delete(ctx, resourceGroupName, deploymentName)
|
|
|
|
if err == nil {
|
|
|
|
err = f.WaitForCompletionRef(ctx, s.client.DeploymentsClient.Client)
|
|
|
|
}
|
2020-08-10 12:57:06 -04:00
|
|
|
|
2019-12-17 10:57:57 -05:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:12:58 -04:00
|
|
|
func (s *StepDeployTemplate) getImageDetails(ctx context.Context, resourceGroupName string, computeName string) (string, string, error) {
|
2017-11-06 00:16:58 -05:00
|
|
|
//We can't depend on constants.ArmOSDiskVhd being set
|
2020-08-11 16:40:51 -04:00
|
|
|
var imageName, imageType string
|
2018-04-06 04:12:58 -04:00
|
|
|
vm, err := s.client.VirtualMachinesClient.Get(ctx, resourceGroupName, computeName, "")
|
2017-11-06 00:16:58 -05:00
|
|
|
if err != nil {
|
|
|
|
return imageName, imageType, err
|
2020-08-11 16:40:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if vm.StorageProfile.OsDisk.Vhd != nil {
|
|
|
|
imageType = "image"
|
|
|
|
imageName = *vm.StorageProfile.OsDisk.Vhd.URI
|
2017-11-06 00:16:58 -05:00
|
|
|
} else {
|
2020-08-11 16:40:51 -04:00
|
|
|
imageType = "Microsoft.Compute/disks"
|
|
|
|
imageName = *vm.StorageProfile.OsDisk.ManagedDisk.ID
|
2017-11-06 00:16:58 -05:00
|
|
|
}
|
2020-08-11 16:40:51 -04:00
|
|
|
|
2017-11-06 00:16:58 -05:00
|
|
|
return imageType, imageName, nil
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:12:58 -04:00
|
|
|
//TODO(paulmey): move to helpers file
|
|
|
|
func deleteResource(ctx context.Context, client *AzureClient, resourceType string, resourceName string, resourceGroupName string) error {
|
2017-11-06 00:16:58 -05:00
|
|
|
switch resourceType {
|
|
|
|
case "Microsoft.Compute/virtualMachines":
|
2018-04-06 04:12:58 -04:00
|
|
|
f, err := client.VirtualMachinesClient.Delete(ctx, resourceGroupName, resourceName)
|
|
|
|
if err == nil {
|
2019-05-30 17:25:43 -04:00
|
|
|
err = f.WaitForCompletionRef(ctx, client.VirtualMachinesClient.Client)
|
2017-11-06 00:16:58 -05:00
|
|
|
}
|
2018-04-06 04:12:58 -04:00
|
|
|
return err
|
2017-11-30 03:11:17 -05:00
|
|
|
case "Microsoft.KeyVault/vaults":
|
2020-03-18 16:38:38 -04:00
|
|
|
_, err := client.VaultClientDelete.Delete(ctx, resourceGroupName, resourceName)
|
2017-11-30 03:11:17 -05:00
|
|
|
return err
|
2017-11-06 00:16:58 -05:00
|
|
|
case "Microsoft.Network/networkInterfaces":
|
2018-04-06 04:12:58 -04:00
|
|
|
f, err := client.InterfacesClient.Delete(ctx, resourceGroupName, resourceName)
|
|
|
|
if err == nil {
|
2019-05-30 17:25:43 -04:00
|
|
|
err = f.WaitForCompletionRef(ctx, client.InterfacesClient.Client)
|
2018-04-06 04:12:58 -04:00
|
|
|
}
|
|
|
|
return err
|
2017-11-06 00:16:58 -05:00
|
|
|
case "Microsoft.Network/virtualNetworks":
|
2018-04-06 04:12:58 -04:00
|
|
|
f, err := client.VirtualNetworksClient.Delete(ctx, resourceGroupName, resourceName)
|
|
|
|
if err == nil {
|
2019-05-30 17:25:43 -04:00
|
|
|
err = f.WaitForCompletionRef(ctx, client.VirtualNetworksClient.Client)
|
2018-04-06 04:12:58 -04:00
|
|
|
}
|
|
|
|
return err
|
2019-10-08 17:56:43 -04:00
|
|
|
case "Microsoft.Network/networkSecurityGroups":
|
|
|
|
f, err := client.SecurityGroupsClient.Delete(ctx, resourceGroupName, resourceName)
|
|
|
|
if err == nil {
|
|
|
|
err = f.WaitForCompletionRef(ctx, client.SecurityGroupsClient.Client)
|
|
|
|
}
|
|
|
|
return err
|
2017-11-06 00:16:58 -05:00
|
|
|
case "Microsoft.Network/publicIPAddresses":
|
2018-04-06 04:12:58 -04:00
|
|
|
f, err := client.PublicIPAddressesClient.Delete(ctx, resourceGroupName, resourceName)
|
|
|
|
if err == nil {
|
2019-05-30 17:25:43 -04:00
|
|
|
err = f.WaitForCompletionRef(ctx, client.PublicIPAddressesClient.Client)
|
2017-11-06 00:16:58 -05:00
|
|
|
}
|
2018-04-06 04:12:58 -04:00
|
|
|
return err
|
2017-11-06 00:16:58 -05:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-04-06 04:12:58 -04:00
|
|
|
func (s *StepDeployTemplate) deleteImage(ctx context.Context, imageType string, imageName string, resourceGroupName string) error {
|
2017-11-06 00:16:58 -05:00
|
|
|
// Managed disk
|
|
|
|
if imageType == "Microsoft.Compute/disks" {
|
|
|
|
xs := strings.Split(imageName, "/")
|
|
|
|
diskName := xs[len(xs)-1]
|
2018-04-06 04:12:58 -04:00
|
|
|
f, err := s.client.DisksClient.Delete(ctx, resourceGroupName, diskName)
|
|
|
|
if err == nil {
|
2019-05-30 17:25:43 -04:00
|
|
|
err = f.WaitForCompletionRef(ctx, s.client.DisksClient.Client)
|
2018-04-06 04:12:58 -04:00
|
|
|
}
|
2017-11-06 00:16:58 -05:00
|
|
|
return err
|
|
|
|
}
|
2020-08-12 11:03:51 -04:00
|
|
|
|
2017-11-06 00:16:58 -05:00
|
|
|
// VHD image
|
|
|
|
u, err := url.Parse(imageName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
xs := strings.Split(u.Path, "/")
|
2017-12-12 04:12:53 -05:00
|
|
|
if len(xs) < 3 {
|
|
|
|
return errors.New("Unable to parse path of image " + imageName)
|
|
|
|
}
|
2017-11-06 00:16:58 -05:00
|
|
|
var storageAccountName = xs[1]
|
|
|
|
var blobName = strings.Join(xs[2:], "/")
|
|
|
|
|
|
|
|
blob := s.client.BlobStorageClient.GetContainerReference(storageAccountName).GetBlobReference(blobName)
|
2020-08-12 11:27:07 -04:00
|
|
|
_, err = blob.BreakLease(nil)
|
|
|
|
if err != nil && !strings.Contains(err.Error(), "LeaseNotPresentWithLeaseOperation") {
|
2020-08-12 11:03:51 -04:00
|
|
|
s.say(s.client.LastError.Error())
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-08-12 11:27:07 -04:00
|
|
|
return blob.Delete(nil)
|
2017-11-06 00:16:58 -05:00
|
|
|
}
|