我想创建一个Azure VM,其OS盘小于默认的127gb。我在Azure门户中找不到这样的选项,所以我试图缩小磁盘。我没有成功。
我知道我可以修剪(使用碎片整理工具)并缩小音量(使用磁盘管理),但这不会改变"物理"硬盘的大小。也就是说,如果我将磁盘缩小到40gb,那么将只有87gb未分配,并且blob仍会报告127gb。
我试图实现的是缩小blob以匹配分配的空间,从而促进VM映像的较小下载/导出(例如40 vs 127gb)。
感谢任何和所有帮助。
答案 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),我们可以对其进行存档,并减少磁盘成本。
第4步。现在转到磁盘的“ Properties”(属性)刀片并复制Resource ID
第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 设置,脚本可以完美运行。