Merge pull request #6798 from mbearup/mb-sharedgallery
Consume images from Azure Shared Image Gallery
This commit is contained in:
commit
520dea23fd
|
@ -65,6 +65,14 @@ type PlanInformation struct {
|
||||||
PlanPromotionCode string `mapstructure:"plan_promotion_code"`
|
PlanPromotionCode string `mapstructure:"plan_promotion_code"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type SharedImageGallery struct {
|
||||||
|
Subscription string `mapstructure:"subscription"`
|
||||||
|
ResourceGroup string `mapstructure:"resource_group"`
|
||||||
|
GalleryName string `mapstructure:"gallery_name"`
|
||||||
|
ImageName string `mapstructure:"image_name"`
|
||||||
|
ImageVersion string `mapstructure:"image_version"`
|
||||||
|
}
|
||||||
|
|
||||||
type Config struct {
|
type Config struct {
|
||||||
common.PackerConfig `mapstructure:",squash"`
|
common.PackerConfig `mapstructure:",squash"`
|
||||||
|
|
||||||
|
@ -79,6 +87,9 @@ type Config struct {
|
||||||
CaptureNamePrefix string `mapstructure:"capture_name_prefix"`
|
CaptureNamePrefix string `mapstructure:"capture_name_prefix"`
|
||||||
CaptureContainerName string `mapstructure:"capture_container_name"`
|
CaptureContainerName string `mapstructure:"capture_container_name"`
|
||||||
|
|
||||||
|
// Shared Gallery
|
||||||
|
SharedGallery SharedImageGallery `mapstructure:"shared_image_gallery"`
|
||||||
|
|
||||||
// Compute
|
// Compute
|
||||||
ImagePublisher string `mapstructure:"image_publisher"`
|
ImagePublisher string `mapstructure:"image_publisher"`
|
||||||
ImageOffer string `mapstructure:"image_offer"`
|
ImageOffer string `mapstructure:"image_offer"`
|
||||||
|
@ -572,19 +583,36 @@ func assertRequiredParametersSet(c *Config, errs *packer.MultiError) {
|
||||||
|
|
||||||
isImageUrl := c.ImageUrl != ""
|
isImageUrl := c.ImageUrl != ""
|
||||||
isCustomManagedImage := c.CustomManagedImageName != "" || c.CustomManagedImageResourceGroupName != ""
|
isCustomManagedImage := c.CustomManagedImageName != "" || c.CustomManagedImageResourceGroupName != ""
|
||||||
|
isSharedGallery := c.SharedGallery.GalleryName != ""
|
||||||
isPlatformImage := c.ImagePublisher != "" || c.ImageOffer != "" || c.ImageSku != ""
|
isPlatformImage := c.ImagePublisher != "" || c.ImageOffer != "" || c.ImageSku != ""
|
||||||
|
|
||||||
countSourceInputs := toInt(isImageUrl) + toInt(isCustomManagedImage) + toInt(isPlatformImage)
|
countSourceInputs := toInt(isImageUrl) + toInt(isCustomManagedImage) + toInt(isPlatformImage) + toInt(isSharedGallery)
|
||||||
|
|
||||||
if countSourceInputs > 1 {
|
if countSourceInputs > 1 {
|
||||||
errs = packer.MultiErrorAppend(errs, fmt.Errorf("Specify either a VHD (image_url), Image Reference (image_publisher, image_offer, image_sku) or a Managed Disk (custom_managed_disk_image_name, custom_managed_disk_resource_group_name"))
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("Specify either a VHD (image_url), Image Reference (image_publisher, image_offer, image_sku), a Managed Disk (custom_managed_disk_image_name, custom_managed_disk_resource_group_name), or a Shared Gallery Image (shared_image_gallery)"))
|
||||||
}
|
}
|
||||||
|
|
||||||
if isImageUrl && c.ManagedImageResourceGroupName != "" {
|
if isImageUrl && c.ManagedImageResourceGroupName != "" {
|
||||||
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A managed image must be created from a managed image, it cannot be created from a VHD."))
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A managed image must be created from a managed image, it cannot be created from a VHD."))
|
||||||
}
|
}
|
||||||
|
|
||||||
if c.ImageUrl == "" && c.CustomManagedImageName == "" {
|
if c.SharedGallery.GalleryName != "" {
|
||||||
|
if c.SharedGallery.Subscription == "" {
|
||||||
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A shared_image_gallery.subscription must be specified"))
|
||||||
|
}
|
||||||
|
if c.SharedGallery.ResourceGroup == "" {
|
||||||
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A shared_image_gallery.resource_group must be specified"))
|
||||||
|
}
|
||||||
|
if c.SharedGallery.ImageName == "" {
|
||||||
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("A shared_image_gallery.image_name must be specified"))
|
||||||
|
}
|
||||||
|
if c.CaptureContainerName != "" {
|
||||||
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("VHD Target [capture_container_name] is not supported when using Shared Image Gallery as source. Use managed_image_resource_group_name instead."))
|
||||||
|
}
|
||||||
|
if c.CaptureNamePrefix != "" {
|
||||||
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("VHD Target [capture_name_prefix] is not supported when using Shared Image Gallery as source. Use managed_image_name instead."))
|
||||||
|
}
|
||||||
|
} else if c.ImageUrl == "" && c.CustomManagedImageName == "" {
|
||||||
if c.ImagePublisher == "" {
|
if c.ImagePublisher == "" {
|
||||||
errs = packer.MultiErrorAppend(errs, fmt.Errorf("An image_publisher must be specified"))
|
errs = packer.MultiErrorAppend(errs, fmt.Errorf("An image_publisher must be specified"))
|
||||||
}
|
}
|
||||||
|
|
|
@ -1294,6 +1294,51 @@ func TestConfigShouldAllowAsyncResourceGroupOverrideBadValue(t *testing.T) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
func TestConfigShouldAllowSharedImageGalleryOptions(t *testing.T) {
|
||||||
|
config := map[string]interface{}{
|
||||||
|
"location": "ignore",
|
||||||
|
"subscription_id": "ignore",
|
||||||
|
"os_type": "linux",
|
||||||
|
"shared_image_gallery": map[string]string{
|
||||||
|
"subscription": "ignore",
|
||||||
|
"resource_group": "ignore",
|
||||||
|
"gallery_name": "ignore",
|
||||||
|
"image_name": "ignore",
|
||||||
|
"image_version": "ignore",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, _, err := newConfig(config, getPackerConfiguration())
|
||||||
|
if err == nil {
|
||||||
|
t.Log("expected config to accept Shared Image Gallery options", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestConfigShouldRejectSharedImageGalleryWithVhdTarget(t *testing.T) {
|
||||||
|
config := map[string]interface{}{
|
||||||
|
"location": "ignore",
|
||||||
|
"subscription_id": "ignore",
|
||||||
|
"os_type": "linux",
|
||||||
|
"shared_image_gallery": map[string]string{
|
||||||
|
"subscription": "ignore",
|
||||||
|
"resource_group": "ignore",
|
||||||
|
"gallery_name": "ignore",
|
||||||
|
"image_name": "ignore",
|
||||||
|
"image_version": "ignore",
|
||||||
|
},
|
||||||
|
"resource_group_name": "ignore",
|
||||||
|
"storage_account": "ignore",
|
||||||
|
"capture_container_name": "ignore",
|
||||||
|
"capture_name_prefix": "ignore",
|
||||||
|
}
|
||||||
|
|
||||||
|
_, _, err := newConfig(config, getPackerConfiguration())
|
||||||
|
if err != nil {
|
||||||
|
t.Log("expected an error if Shared Image Gallery source is used with VHD target", err)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func getArmBuilderConfiguration() map[string]string {
|
func getArmBuilderConfiguration() map[string]string {
|
||||||
|
|
|
@ -72,6 +72,18 @@ func GetVirtualMachineDeployment(config *Config) (*resources.Deployment, error)
|
||||||
config.ImageVersion)
|
config.ImageVersion)
|
||||||
|
|
||||||
builder.SetManagedMarketplaceImage(config.Location, config.ImagePublisher, config.ImageOffer, config.ImageSku, config.ImageVersion, imageID, config.managedImageStorageAccountType)
|
builder.SetManagedMarketplaceImage(config.Location, config.ImagePublisher, config.ImageOffer, config.ImageSku, config.ImageVersion, imageID, config.managedImageStorageAccountType)
|
||||||
|
} else if config.SharedGallery.Subscription != "" {
|
||||||
|
imageID := fmt.Sprintf("/subscriptions/%s/resourceGroups/%s/providers/Microsoft.Compute/galleries/%s/images/%s",
|
||||||
|
config.SharedGallery.Subscription,
|
||||||
|
config.SharedGallery.ResourceGroup,
|
||||||
|
config.SharedGallery.GalleryName,
|
||||||
|
config.SharedGallery.ImageName)
|
||||||
|
if config.SharedGallery.ImageVersion != "" {
|
||||||
|
imageID += fmt.Sprintf("/versions/%s",
|
||||||
|
config.SharedGallery.ImageVersion)
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.SetSharedGalleryImage(config.Location, imageID)
|
||||||
} else {
|
} else {
|
||||||
builder.SetMarketPlaceImage(config.ImagePublisher, config.ImageOffer, config.ImageSku, config.ImageVersion)
|
builder.SetMarketPlaceImage(config.ImagePublisher, config.ImageOffer, config.ImageSku, config.ImageVersion)
|
||||||
}
|
}
|
||||||
|
|
|
@ -145,6 +145,21 @@ func (s *TemplateBuilder) SetManagedMarketplaceImage(location, publisher, offer,
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *TemplateBuilder) SetSharedGalleryImage(location, imageID string) error {
|
||||||
|
resource, err := s.getResourceByType(resourceVirtualMachine)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
s.setVariable("apiVersion", "2018-04-01") // Required for Shared Image Gallery
|
||||||
|
profile := resource.Properties.StorageProfile
|
||||||
|
profile.ImageReference = &compute.ImageReference{ID: &imageID}
|
||||||
|
profile.OsDisk.OsType = s.osType
|
||||||
|
profile.OsDisk.Vhd = nil
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *TemplateBuilder) SetMarketPlaceImage(publisher, offer, sku, version string) error {
|
func (s *TemplateBuilder) SetMarketPlaceImage(publisher, offer, sku, version string) error {
|
||||||
resource, err := s.getResourceByType(resourceVirtualMachine)
|
resource, err := s.getResourceByType(resourceVirtualMachine)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -0,0 +1,170 @@
|
||||||
|
{
|
||||||
|
"$schema": "http://schema.management.azure.com/schemas/2014-04-01-preview/deploymentTemplate.json",
|
||||||
|
"contentVersion": "1.0.0.0",
|
||||||
|
"parameters": {
|
||||||
|
"adminPassword": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"adminUsername": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"dnsNameForPublicIP": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"nicName": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"osDiskName": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"publicIPAddressName": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"storageAccountBlobEndpoint": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"subnetName": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"virtualNetworkName": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"vmName": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"vmSize": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"resources": [
|
||||||
|
{
|
||||||
|
"apiVersion": "[variables('publicIPAddressApiVersion')]",
|
||||||
|
"location": "[variables('location')]",
|
||||||
|
"name": "[parameters('publicIPAddressName')]",
|
||||||
|
"properties": {
|
||||||
|
"dnsSettings": {
|
||||||
|
"domainNameLabel": "[parameters('dnsNameForPublicIP')]"
|
||||||
|
},
|
||||||
|
"publicIPAllocationMethod": "[variables('publicIPAddressType')]"
|
||||||
|
},
|
||||||
|
"type": "Microsoft.Network/publicIPAddresses"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"apiVersion": "[variables('virtualNetworksApiVersion')]",
|
||||||
|
"location": "[variables('location')]",
|
||||||
|
"name": "[variables('virtualNetworkName')]",
|
||||||
|
"properties": {
|
||||||
|
"addressSpace": {
|
||||||
|
"addressPrefixes": [
|
||||||
|
"[variables('addressPrefix')]"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"subnets": [
|
||||||
|
{
|
||||||
|
"name": "[variables('subnetName')]",
|
||||||
|
"properties": {
|
||||||
|
"addressPrefix": "[variables('subnetAddressPrefix')]"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"type": "Microsoft.Network/virtualNetworks"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"apiVersion": "[variables('networkInterfacesApiVersion')]",
|
||||||
|
"dependsOn": [
|
||||||
|
"[concat('Microsoft.Network/publicIPAddresses/', parameters('publicIPAddressName'))]",
|
||||||
|
"[concat('Microsoft.Network/virtualNetworks/', variables('virtualNetworkName'))]"
|
||||||
|
],
|
||||||
|
"location": "[variables('location')]",
|
||||||
|
"name": "[parameters('nicName')]",
|
||||||
|
"properties": {
|
||||||
|
"ipConfigurations": [
|
||||||
|
{
|
||||||
|
"name": "ipconfig",
|
||||||
|
"properties": {
|
||||||
|
"privateIPAllocationMethod": "Dynamic",
|
||||||
|
"publicIPAddress": {
|
||||||
|
"id": "[resourceId('Microsoft.Network/publicIPAddresses', parameters('publicIPAddressName'))]"
|
||||||
|
},
|
||||||
|
"subnet": {
|
||||||
|
"id": "[variables('subnetRef')]"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"type": "Microsoft.Network/networkInterfaces"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"apiVersion": "[variables('apiVersion')]",
|
||||||
|
"dependsOn": [
|
||||||
|
"[concat('Microsoft.Network/networkInterfaces/', parameters('nicName'))]"
|
||||||
|
],
|
||||||
|
"location": "[variables('location')]",
|
||||||
|
"name": "[parameters('vmName')]",
|
||||||
|
"properties": {
|
||||||
|
"diagnosticsProfile": {
|
||||||
|
"bootDiagnostics": {
|
||||||
|
"enabled": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"hardwareProfile": {
|
||||||
|
"vmSize": "[parameters('vmSize')]"
|
||||||
|
},
|
||||||
|
"networkProfile": {
|
||||||
|
"networkInterfaces": [
|
||||||
|
{
|
||||||
|
"id": "[resourceId('Microsoft.Network/networkInterfaces', parameters('nicName'))]"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"osProfile": {
|
||||||
|
"adminPassword": "[parameters('adminPassword')]",
|
||||||
|
"adminUsername": "[parameters('adminUsername')]",
|
||||||
|
"computerName": "[parameters('vmName')]",
|
||||||
|
"linuxConfiguration": {
|
||||||
|
"ssh": {
|
||||||
|
"publicKeys": [
|
||||||
|
{
|
||||||
|
"keyData": "--test-ssh-authorized-key--",
|
||||||
|
"path": "[variables('sshKeyPath')]"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"storageProfile": {
|
||||||
|
"imageReference": {
|
||||||
|
"id": "/subscriptions/ignore/resourceGroups/ignore/providers/Microsoft.Compute/galleries/ignore/images/ignore"
|
||||||
|
},
|
||||||
|
"osDisk": {
|
||||||
|
"caching": "ReadWrite",
|
||||||
|
"createOption": "FromImage",
|
||||||
|
"name": "[parameters('osDiskName')]",
|
||||||
|
"osType": "Linux"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"type": "Microsoft.Compute/virtualMachines"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"variables": {
|
||||||
|
"addressPrefix": "10.0.0.0/16",
|
||||||
|
"apiVersion": "2018-04-01",
|
||||||
|
"location": "[resourceGroup().location]",
|
||||||
|
"managedDiskApiVersion": "2017-03-30",
|
||||||
|
"networkInterfacesApiVersion": "2017-04-01",
|
||||||
|
"publicIPAddressApiVersion": "2017-04-01",
|
||||||
|
"publicIPAddressType": "Dynamic",
|
||||||
|
"sshKeyPath": "[concat('/home/',parameters('adminUsername'),'/.ssh/authorized_keys')]",
|
||||||
|
"subnetAddressPrefix": "10.0.0.0/24",
|
||||||
|
"subnetName": "[parameters('subnetName')]",
|
||||||
|
"subnetRef": "[concat(variables('vnetID'),'/subnets/',variables('subnetName'))]",
|
||||||
|
"virtualNetworkName": "[parameters('virtualNetworkName')]",
|
||||||
|
"virtualNetworkResourceGroup": "[resourceGroup().name]",
|
||||||
|
"virtualNetworksApiVersion": "2017-04-01",
|
||||||
|
"vmStorageAccountContainerName": "images",
|
||||||
|
"vnetID": "[resourceId(variables('virtualNetworkResourceGroup'), 'Microsoft.Network/virtualNetworks', variables('virtualNetworkName'))]"
|
||||||
|
}
|
||||||
|
}
|
|
@ -181,3 +181,32 @@ func TestBuildWindows02(t *testing.T) {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Shared Image Gallery Build
|
||||||
|
func TestSharedImageGallery00(t *testing.T) {
|
||||||
|
testSubject, err := NewTemplateBuilder(BasicTemplate)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = testSubject.BuildLinux("--test-ssh-authorized-key--")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
imageID := "/subscriptions/ignore/resourceGroups/ignore/providers/Microsoft.Compute/galleries/ignore/images/ignore"
|
||||||
|
err = testSubject.SetSharedGalleryImage("westcentralus", imageID)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
doc, err := testSubject.ToJSON()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = approvaltests.VerifyJSONBytes(t, []byte(*doc))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -186,6 +186,19 @@ Providing `temp_resource_group_name` or `location` in combination with `build_re
|
||||||
1. PlanPublisher
|
1. PlanPublisher
|
||||||
1. PlanPromotionCode
|
1. PlanPromotionCode
|
||||||
|
|
||||||
|
- `shared_image_gallery` (object) Use a [Shared Gallery image](https://azure.microsoft.com/en-us/blog/announcing-the-public-preview-of-shared-image-gallery/) as the source for this build. *VHD targets are incompatible with this build type* - the target must be a *Managed Image*.
|
||||||
|
```
|
||||||
|
"shared_image_gallery": {
|
||||||
|
"subscription": "00000000-0000-0000-0000-00000000000",
|
||||||
|
"resource_group": "ResourceGroup",
|
||||||
|
"gallery_name": "GalleryName",
|
||||||
|
"image_name": "ImageName",
|
||||||
|
"image_version": "1.0.0"
|
||||||
|
}
|
||||||
|
"managed_image_name": "TargetImageName",
|
||||||
|
"managed_image_resource_group_name": "TargetResourceGroup"
|
||||||
|
```
|
||||||
|
|
||||||
- `temp_compute_name` (string) temporary name assigned to the VM. If this value is not set, a random value will be
|
- `temp_compute_name` (string) temporary name assigned to the VM. If this value is not set, a random value will be
|
||||||
assigned. Knowing the resource group and VM name allows one to execute commands to update the VM during a Packer
|
assigned. Knowing the resource group and VM name allows one to execute commands to update the VM during a Packer
|
||||||
build, e.g. attach a resource disk to the VM.
|
build, e.g. attach a resource disk to the VM.
|
||||||
|
|
Loading…
Reference in New Issue