For full change log and more information, visit my site.
Hyper-V Backup Utility is available from:
Please consider supporting my work:
- Support with Github Sponsors.
- Support with a one-time donation using PayPal.
Please report any problems via the ‘issues’ tab on GitHub.
-Mike
- Designed to be run on a Hyper-V host.
- The Hyper-V host must have the Hyper-V management PowerShell modules installed.
- Can be used to backup VMs to a device which the Hyper-V host does not have permission to run a regular export to.
- Supports Hyper-V hosts in a clustered configuration.
- The utility requires at least Windows PowerShell 5.0.
- Tested on Windows 11, Windows 10, Windows Server 2022, Windows Server 2019 and Windows Server 2016.
- The backup log can be sent via email and/or webhook.
An SMB share can be used as a backup location by specifying the UNC path (\\server\share) with the -BackupTo option. Please note that you must use either a local working directory (-Wd path) or the -NoPerms option if your SMB share host and Hyper-V host are not members of an Active Directory domain. Hyper-V cannot perform export VMs to a network share without the Hyper-V host having permissions to the share which are possible when using Windows servers and Active Directory and not available with a NAS appliance such as TrueNAS, QNAP or Synology.
The -NoPerms switch is intended as a workaround when used in an environment where the Hyper-V host cannot be given the required permissions to run a regular export to a remote device such as a NAS device. To copy all the files necessary for a complete backup, the VM must be in an offline state for the operation to be completed. Therefore the script will put the VM in a 'Saved' state (if it is running) so the files can be copied. In previous versions the VM would be shutdown but this is a faster and safer method as the VM does not require any integrations to be put in a saved state.
Hyper-V’s export operation requires that the computer account in Active Directory have access to the location where the exports are being stored. When a NAS is intended to be used as an export location, Hyper-V will not be able to complete the operation as the computer account cannot be given access to the share on the NAS.
I've implemented support for 7-Zip into the script. You should be able to use any option that 7-zip supports, although currently the only options I've tested fully are '-t' archive type, '-p' password and '-v' split files.
The password used for SMTP server authentication must be in an encrypted text file. To generate the password file, run the command below in PowerShell on the same computer and logged in with the user that will be running the utility. When you run the command, you will be prompted for a username and password. Enter the username and password you want to use to authenticate to your SMTP server.
Please note: This is only required if you need to authenticate to the SMTP server when sending the log via e-mail.
[path\]Hyper-V-Backup.ps1 -MakeCreds [filename.txt]
After running the commands, you will have a text file containing the encrypted password. When configuring the -Pwd switch enter the path and file name of this file.
The easiest and quickest way to restore a Virtual Machine that has been backed up using this script is to use Hyper-V's native import function.
- Copy the backup of the VM you want to restore to a location on the VM host server that the VM should run from. If the backup is compressed, uncompress the file.
- In the Hyper-V Manager, right-click on the VM host and select 'Import Virtual Machine'.
- Browse to the location of the VM backup folder and click Next.
- Select the VM you want to restore.
- Select 'Register the virtual machine in-place' option.
- The VM will be registered in Hyper-V and available for use.
Here’s a list of all the command line switches and example configurations.
Command Line Switch | Description | Example |
---|---|---|
-BackupTo | The path the virtual machines should be backed up to. Each VM will have its own folder inside this location. This can be an SMB share but you must also use either -Wd or -NoPerms. | [path] or [\\server\path] |
-SMBUsr | Enter the domain and user required to access the SMB share. Use only if required. | domain\user |
-SMBPwd | Enter the password for the above account required to access the SMB share. Use only if required. | password |
-AllVms | If this option is configured all VMs will be backed up. | N/A |
-Prefix | Use this option to specify VM names beginning with the string specified to backup. | [string] |
-List | Enter the path to a txt file with a list of Hyper-V VM names to backup. If this option, -Prefix and -AllVms are not configured, all running VMs will be backed up. | [path\vms.txt] |
-CaptureState | Enter a method to use when exporting the VM. If this option is not configured, the default method will be used. | CaptureCrashConsistentState, CaptureSavedState, CaptureDataConsistentState |
-Wd | The path to the working directory to use for the backup before copying it to the final backup directory. Use a directory on local fast media to improve performance. | [path] |
-NoPerms | Configures the utility to shut down running VMs to do the file-copy based backup instead of using the Hyper-V export function. | N/A |
-Keep | Instructs the utility to keep a specified number of days worth of backups. VM backups older than the number of days specified will be deleted. | [number] |
-Compress | This option will create a zip file of each Hyper-V VM backup. | N/A |
-Sz | Configure the utility to use 7-Zip to compress the VM backups. 7-Zip must be installed in the default location $env:ProgramFiles if it is not found, Windows compression will be used. |
N/A |
-SzOptions | Use this switch to configure options for 7-Zip. The switches must be comma separated. | "'-t7z,-v2G,-ppassword'" |
-ShortDate | Configure the script to use only the Year, Month and Day in backup filenames. | N/A |
-LowDisk | Remove old backups before new ones are created. For low disk space situations. | N/A |
-L | The path to output the log file to. | [path] |
-LogRotate | Remove logs produced by the utility older than X days | [number] |
-NoBanner | Use this option to hide the ASCII art title in the console. | N/A |
-Help | Display usage information. No arguments also displays help. | N/A |
-ProgCheck | Send notifications (email or webhook) after each VM is backed up. | N/A |
-OptimiseVHD | Optimise the VHDs and make them smaller before copy. Must be used with -NoPerms option. | N/A |
-Webhook | The txt file containing the URI for a webhook to send the log file to. | [path\webhook.txt] |
-Subject | Specify a subject line. If you leave this blank the default subject will be used | "'[Server: Notification]'" |
-SendTo | The e-mail address the log should be sent to. For multiple address, separate with a comma. | [example@contoso.com] |
-From | The e-mail address the log should be sent from. | [example@contoso.com] |
-Smtp | The DNS name or IP address of the SMTP server. | [smtp server address] |
-Port | The Port that should be used for the SMTP server. If none is specified then the default of 25 will be used. | [port number] |
-User | The user account to authenticate to the SMTP server. | [example@contoso.com] |
-Pwd | The txt file containing the encrypted password for SMTP authentication. | [path\ps-script-pwd.txt] |
-UseSsl | Configures the utility to connect to the SMTP server using SSL. | N/A |
-MakeCreds | Use this option to create a credentials file for SMTP authentication. The file will be created in the same directory as the script. | [filename.txt] |
[path\]Hyper-V-Backup.ps1 -BackupTo [path]
This will backup all the VMs running to the backup location specified.
- Added Sponsorship information to the projects Github.
- Added SMB Authentication, based on work from PR 37
- Added -MakeCreds option to help in creation of SMTP authentication file.
- Added -AllVms option to specify the backing up of all VMs on a Hyper-V server without the need for a list file.
- Added -Prefix option to backup all VMs beginning with the string specified.
- Added more information about Hyper-V's export limitations to the documentation and in app help.
- Fixed an issue where backup success would be reported when using a working directory and the final destination directory for the backup didn't have enough disk space. From Issue 27
- Added -CaptureState option for the user to specify the method that Export-VM uses to capture the state of the VM whilst running. From Issue 34
- Added fix for verifying password protected 7-Zip archives from Issue 33
- Fixed 7-Zip split files getting renamed and not keeping file extensions when short dates are used.
- Added a verify operation for 7-Zip created archives as per Issue 33
- Fixed an issue where failed backups where also listed as successful.
- Overhauled the backup success/fail checks. They now work a lot more reliably.
- Added check for the work dir/backup dir to exist before trying to remove as this caused a script error.
- Cleaned up console and log file output.
- Added new features from Issue 28
- Added -ProgCheck option. With this option set, notifications will be sent after each VM is backup is finished.
- Added backup time duration to the script output.
- Added -OptimiseVHD option to shrink the size of the VHDs. Can only be used the the -NoPerms option as the VM must be offline to optimise the VHDs.
- Minor improvement to update checker. If the internet is not reachable it silently errors out.
- Removed specific SMTP config info from config report. Issue 24
- Added a "simple auth edition" version of the script. Issue 25
- Removed SMTP authentication details from the 'Config' report. Now it just shows as 'configured' if SMTP user is configured. To be clear: no passwords were ever shown or stored in plain text.
- Added script update checker - shows if an update is available in the log and console.
- Added VM restore instructions to readme.md.
- Added "low disk space" mode. -LowDisk switch deletes previous backup files and folders before backup for systems with low disk space.
- Added webhook option to send log file to.
- Lot's of refactored code using functions. Simpler, easier to manage. Long overdue.
- Fixed bug that started VMs that were shutdown.
- Changed "VM not running" from an error state to an informational state.
- Changed "Backup Success" to a success state (green text in console).
- Changed -NoPerms so that VMs are now saved instead of shutdown (safer, faster, does not require Hyper-V integrations)
- Fixed an issue with the code checking for OS version too late.
- Fixed Get-Service check outputting to console.
- Fixed backup success/fail counter not working with -NoPerms switch.
- Fixed an issue with Windows Server 2012 R2 when checking for the Hyper-V service to be installed and running.
- Fixed Issue 19 on GitHub - All Virtual Hard Disk folders should now be called "Virtual Hard Disks" and not some with the name "VHD".
- Fixed Issue 20 on GitHub - If -L [path] not configured then a non fatal error would occur as no log path was specified for the log to be output to.
- Fixed an issue where a VM would not be backed up if it were in the state "saved" and was present in the user configured VM list text file.
- Added user feedback - make backup success or fail clear in the log and console.
- Added user feedback - add "VMs backed up x/x" to email subject for clear success/fail visibility.
- Added user feedback - Log can now be emailed to multiple addresses.
- Added checks and balances to help with configuration as I'm very aware that the initial configuration can be troublesome. Running the utility manually is a lot more friendly and step-by-step now.
- Added -Help to give usage instructions in the terminal. Running the script with no options will also trigger the -help switch.
- Cleaned user entered paths so that trailing slashes no longer break things or have otherwise unintended results.
- Added -LogRotate [days] to removed old logs created by the utility.
- Streamlined config report so non configured options are not shown.
- Added donation link to the ASCII banner.
- Cleaned up code, removed unneeded log noise.
- Made a small fix to the 'NoPerms' function: The VM will be left in the state it was found. For example, when a VM is found in an offline state, the script will not start the VM once the backup is complete. In the previous version the VM would be started regardless of what state it was in previously.
- Added fix for potential BSOD on Windows Server 2016 Hyper-V host when exporting VMs using VSS. The change to the registry will only happen if Windows Server 2016 is detected as the Hyper-V host and only if the registry value is in the default state. If it has been configured previously no change will be made. Issue 17 on GitHub
- When using -NoPerms the utility now waits for disk merging to complete before backing up.
- Utility now ignores blanks lines in VM list file.
- Added checks for success or failure in the backup, copy/compression process. If it fails none of the previous backups should be removed.
- Put checks in place so if a VM fails to backup the old backup for that VM is not removed and the error is logged.
- Added more logging info, clearer formatting.
- Fixed an error when moving compressed backup files from a working directory.
- Configured logs path now is created, if it does not exist.
- Added OS version info.
- Improved log output, added more information for each stage of the backup.
- Added an option to specify the Port for SMTP communication.
- Fixed many bugs introduced with implementing more 7-zip options. 7-zip options I've tested fully are '-t' archive type, '-p' password and '-v' split files.
- Implemented and automated a formal testing process.
- Replaced -Sz* specific options with -SzOptions which will support any option that 7-zip supports.
- Fixed an error where file types which are not .zip were not being moved from the working directory to the final backup location.
- Added additional 7-Zip options. -SzSplit to split archives into configuration volumes.
- Changed existing switches for 7-Zip options. Users must now add an additional hyphen '-' for 7-Zip options. This has been done to better support features that 7-Zip supports.
- Changed how old files are removed. Users should take extra care if they are storing non back-up files in the backup location. This has been done so that 7-Zip's split function can be supported.
- Added -ShortDate option. This will create backups with only the Year, Month, Day as the file name.
- Added pass through for 7-Zip options - CPU threads to use and compression level.
- Added proper error handling so errors are properly reported in the console, log and email.
- Bug fixes to create folders when paths are configured without the folders existing.
- Fixed e-mail report extra line breaks in Outlook 365, Version 2001.
- Config report matches design of Image Factory Utility.
- Improved and simplified code.
Current known issues:
- E-mail report has extra line breaks in Outlook 365, Version 2001.
New features:
- Refactored code.
- Fully backwards compatible.
- Added option to use a working directory to stage backups before moving them to final backup location.
- Added option to use 7-Zip for backup compression.
- Added ASCII banner art when run in the console.
- Added option to disable the ASCII banner art.
- Added custom subject line for e-mail.
- Added more feedback when the script is used interactively.
- Added the ability to specify the VMs to be backed up using a txt file.
- Improved logging slightly to be clearer about which VM's previous backups are being deleted.
- Added option to compress the VM backups to a zip file. This option will remove the original VM backup.
- Added option to keep a configurable number of days’ worth of backups, so you can keep a history/archive of previous backups. Every effort has been taken to only remove backup files or folders generated by this utility.
- Changed the script so that when backup is complete, the VM backup folders/zip files will be have the time and date append to them.
- The backup script no longer creates a folder named after the Host server. The VM backups are placed in the root of the specified backup location.
- Fixed a small issue with logging where the script completes the backup process, then states incorrectly "there are no VMs to backup".
- Fixed a small bug that occurred when there were no VMs to backup, the script incorrectly logged an error in exporting the VMs. It now states that that are no VMs to backup.
- The script has been tested performing backups of Virtual Machines running on a Hyper-V cluster.
- Minor update to documentation.
- Changed SMTP authentication to require an encrypted password file.
- Added instructions on how to generate an encrypted password file.
- Added necessary information to add the script to the PowerShell Gallery.
- Improved the log output to be easier to read.
- Improved commenting on the code for documentation purposes.
- Added authentication and SSL options for e-mail notification.
- Added configuration via command line switches.
- Added option to perform regular online export if destination allows it.
- Cleaned up the formatting and commented sections of the script.
- Added the ability to email the log file when the script completes.