减少Azure VM OS磁盘的大小以进行下载/导出

时间:2015-06-30 16:49:39

标签: azure azure-storage-blobs azure-virtual-machine

我想创建一个Azure VM,其OS盘小于默认的127gb。我在Azure门户中找不到这样的选项,所以我试图缩小磁盘。我没有成功。

我知道我可以修剪(使用碎片整理工具)并缩小音量(使用磁盘管理),但这不会改变"物理"硬盘的大小。也就是说,如果我将磁盘缩小到40gb,那么将只有87gb未分配,并且blob仍会报告127gb。

我试图实现的是缩小blob以匹配分配的空间,从而促进VM映像的较小下载/导出(例如40 vs 127gb)。

感谢任何和所有帮助。

4 个答案:

答案 0 :(得分:2)

我写了一篇博文,详细说明了这个答案。但这里的主要问题是能够减少默认为127gb的Azure VM的大小,以便允许禁止导出/下载。我实现这一目标的方法是修剪硬盘驱动器,然后使用Disk2VHD创建正在运行的VM的VHD文件。 Disk2VHD将创建一个可扩展磁盘,该磁盘仅与光盘上的当前数据一样大,而不是整个可用磁盘。在我的情况下40gb vs 127gb。如果将此VHD文件保存到附加磁盘(读取:blob存储),则可以通过整个团队的HTTP轻松下载。因此,下载现在是40gbs而不是127gbs。有关更多信息,请阅读我的详细博客文章:

http://www.kevinmcloutier.com/?p=263

原始链接不再有效: https://web.archive.org/web/20161027213258/http://kevinmcloutier.com/post/4

答案 1 :(得分:1)

您必须创建自己的VM映像,然后使用它进行部署。此模板显示如何使用您自己的图像进行部署。

https://github.com/Azure/azure-quickstart-templates/tree/master/101-vm-from-user-image

目前,图库中的图片均为127gb。由于Azure VM仅使用固定大小的光盘,因此您无法选择大小。

答案 2 :(得分:0)

在Azure中,如果创建VM,它将使用某些默认配置创建。目前,不支持通过以下过程将Azure门户的Azure VM的OS磁盘(托管或非托管)磁盘大小减小/缩小(例如,从128Gb减小到32Gb),我们可以对其进行存档,并减少磁盘成本。

步骤1.打开VM,然后转到“磁盘管理”。 enter image description here

步骤2。打开PowerShell,然后执行以下命令。 enter image description here

成功执行后,您可以找到以下内容 enter image description here

第3步。现在从Azure门户取消分配VM enter image description here

第4步。现在转到磁盘的“ Properties”(属性)刀片并复制Resource ID enter image description here

第5步。现在从本地系统执行以下PowerShell脚本。必须用您的值更改$ DiskID,$ VMName,$ AzSubscription

    # Variables
$DiskID = ""# eg. "/subscriptions/203bdbf0-69bd-1a12-a894-a826cf0a34c8/resourcegroups/rg-server1-prod-1/providers/Microsoft.Compute/disks/Server1-Server1"
$VMName = "VM-Server1"
$DiskSizeGB = 32
$AzSubscription = "Prod Subscription"

# Script
# Provide your Azure admin credentials
Connect-AzAccount

#Provide the subscription Id of the subscription where snapshot is created
Select-AzSubscription -Subscription $AzSubscription

# VM to resize disk of
$VM = Get-AzVm | ? Name -eq $VMName

#Provide the name of your resource group where snapshot is created
$resourceGroupName = $VM.ResourceGroupName

# Get Disk from ID
$Disk = Get-AzDisk | ? Id -eq $DiskID

# Get VM/Disk generation from Disk
$HyperVGen = $Disk.HyperVGeneration

# Get Disk Name from Disk
$DiskName = $Disk.Name

# Get SAS URI for the Managed disk
$SAS = Grant-AzDiskAccess -ResourceGroupName $resourceGroupName -DiskName $DiskName -Access 'Read' -DurationInSecond 600000;

#Provide the managed disk name
#$managedDiskName = "yourManagedDiskName" 

#Provide Shared Access Signature (SAS) expiry duration in seconds e.g. 3600.
#$sasExpiryDuration = "3600"

#Provide storage account name where you want to copy the snapshot - the script will create a new one temporarily
$storageAccountName = "shrink" + [system.guid]::NewGuid().tostring().replace('-','').substring(1,18)

#Name of the storage container where the downloaded snapshot will be stored
$storageContainerName = $storageAccountName

#Provide the key of the storage account where you want to copy snapshot. 
#$storageAccountKey = "yourStorageAccountKey"

#Provide the name of the VHD file to which snapshot will be copied.
$destinationVHDFileName = "$($VM.StorageProfile.OsDisk.Name).vhd"

#Generate the SAS for the managed disk
#$sas = Grant-AzureRmDiskAccess -ResourceGroupName $resourceGroupName -DiskName $managedDiskName -Access Read -DurationInSecond $sasExpiryDuration

#Create the context for the storage account which will be used to copy snapshot to the storage account 
$StorageAccount = New-AzStorageAccount -ResourceGroupName $resourceGroupName -Name $storageAccountName -SkuName Standard_LRS -Location $VM.Location
$destinationContext = $StorageAccount.Context
$container = New-AzStorageContainer -Name $storageContainerName -Permission Off -Context $destinationContext

#Copy the snapshot to the storage account and wait for it to complete
Start-AzStorageBlobCopy -AbsoluteUri $SAS.AccessSAS -DestContainer $storageContainerName -DestBlob $destinationVHDFileName -DestContext $destinationContext
while(($state = Get-AzStorageBlobCopyState -Context $destinationContext -Blob $destinationVHDFileName -Container $storageContainerName).Status -ne "Success") { $state; Start-Sleep -Seconds 20 }
$state

# Revoke SAS token
Revoke-AzDiskAccess -ResourceGroupName $resourceGroupName -DiskName $DiskName

# Emtpy disk to get footer from
$emptydiskforfootername = "$($VM.StorageProfile.OsDisk.Name)-empty.vhd"

# Empty disk URI
#$EmptyDiskURI = $container.CloudBlobContainer.Uri.AbsoluteUri + "/" + $emptydiskforfooter

$diskConfig = New-AzDiskConfig `
    -Location $VM.Location `
    -CreateOption Empty `
    -DiskSizeGB $DiskSizeGB `
    -HyperVGeneration $HyperVGen

$dataDisk = New-AzDisk `
    -ResourceGroupName $resourceGroupName `
    -DiskName $emptydiskforfootername `
    -Disk $diskConfig

$VM = Add-AzVMDataDisk `
    -VM $VM `
    -Name $emptydiskforfootername `
    -CreateOption Attach `
    -ManagedDiskId $dataDisk.Id `
    -Lun 63

Update-AzVM -ResourceGroupName $resourceGroupName -VM $VM

$VM | Stop-AzVM -Force


# Get SAS token for the empty disk
$SAS = Grant-AzDiskAccess -ResourceGroupName $resourceGroupName -DiskName $emptydiskforfootername -Access 'Read' -DurationInSecond 600000;

# Copy the empty disk to blob storage
Start-AzStorageBlobCopy -AbsoluteUri $SAS.AccessSAS -DestContainer $storageContainerName -DestBlob $emptydiskforfootername -DestContext $destinationContext
while(($state = Get-AzStorageBlobCopyState -Context $destinationContext -Blob $emptydiskforfootername -Container $storageContainerName).Status -ne "Success") { $state; Start-Sleep -Seconds 20 }
$state

# Revoke SAS token
Revoke-AzDiskAccess -ResourceGroupName $resourceGroupName -DiskName $emptydiskforfootername

# Remove temp empty disk
Remove-AzVMDataDisk -VM $VM -DataDiskNames $emptydiskforfootername
Update-AzVM -ResourceGroupName $resourceGroupName -VM $VM

# Delete temp disk
Remove-AzDisk -ResourceGroupName $resourceGroupName -DiskName $emptydiskforfootername -Force;

# Get the blobs
$emptyDiskblob = Get-AzStorageBlob -Context $destinationContext -Container $storageContainerName -Blob $emptydiskforfootername
$osdisk = Get-AzStorageBlob -Context $destinationContext -Container $storageContainerName -Blob $destinationVHDFileName

$footer = New-Object -TypeName byte[] -ArgumentList 512
write-output "Get footer of empty disk"

$downloaded = $emptyDiskblob.ICloudBlob.DownloadRangeToByteArray($footer, 0, $emptyDiskblob.Length - 512, 512)

$osDisk.ICloudBlob.Resize($emptyDiskblob.Length)
$footerStream = New-Object -TypeName System.IO.MemoryStream -ArgumentList (,$footer)
write-output "Write footer of empty disk to OSDisk"
$osDisk.ICloudBlob.WritePages($footerStream, $emptyDiskblob.Length - 512)

Write-Output -InputObject "Removing empty disk blobs"
$emptyDiskblob | Remove-AzStorageBlob -Force


#Provide the name of the Managed Disk
$NewDiskName = "$DiskName" + "-new"

#Create the new disk with the same SKU as the current one
$accountType = $Disk.Sku.Name

# Get the new disk URI
$vhdUri = $osdisk.ICloudBlob.Uri.AbsoluteUri

# Specify the disk options
$diskConfig = New-AzDiskConfig -AccountType $accountType -Location $VM.location -DiskSizeGB $DiskSizeGB -SourceUri $vhdUri -CreateOption Import -StorageAccountId $StorageAccount.Id -HyperVGeneration $HyperVGen

#Create Managed disk
$NewManagedDisk = New-AzDisk -DiskName $NewDiskName -Disk $diskConfig -ResourceGroupName $resourceGroupName

$VM | Stop-AzVM -Force

# Set the VM configuration to point to the new disk  
Set-AzVMOSDisk -VM $VM -ManagedDiskId $NewManagedDisk.Id -Name $NewManagedDisk.Name

# Update the VM with the new OS disk
Update-AzVM -ResourceGroupName $resourceGroupName -VM $VM

$VM | Start-AzVM

start-sleep 180
# Please check the VM is running before proceeding with the below tidy-up steps

# Delete old Managed Disk
Remove-AzDisk -ResourceGroupName $resourceGroupName -DiskName $DiskName -Force;

# Delete old blob storage
$osdisk | Remove-AzStorageBlob -Force

# Delete temp storage account
$StorageAccount | Remove-AzStorageAccount -Force

答案 3 :(得分:0)

Sapnandu 的解决方案有效。非常感谢! 我仍然不是 100% 确定让它工作的魔法标准是什么,但最后我有一个运行在减少磁盘上的虚拟机。使它成为 Gen1 绝对是其中之一。 我尝试了很多类似的东西,但它们都在启动时卡住了。

我的方法是:

从图库图像创建虚拟机。您将拥有一个 128Gb 的操作系统磁盘。

根据您的需要调整虚拟机。您可以在此处运行调整大小脚本。因此,您最终会得到大约 90Gb 的未分配空间。

经过几次失败的尝试后,我开始重视配置的虚拟机,我真的不想再次重复配置。

制作此磁盘的快照,然后从快照制作磁盘。这些选项将通过快照和磁盘呈现给您。

然后从磁盘创建一个新的虚拟机。检查它是否启动。启动诊断很快就会显示出来。

在停止新虚拟机后,我查找了 Sapnandu 脚本所需的 3 个(subscriptionID、diskID、vmName)参数(在他的帖子中的第 5 步)并在 azure cloud shell 中执行了该脚本。 (标题中的图标)

需要时间,大约 10 分钟左右。 脚本完成后,新虚拟机正在使用较小的磁盘运行。

所以,有一个 vm 设置,脚本可以完美运行。