duplicati/Duplicati/Library/WindowsModules/WmicShadowCopyManager.cs
Kenneth Skovhede 98deb8c576 Update support for VSS
This updates VSS to default use Vanara in favor of AlphaVSS which is no longer maintained.

The build for Vanara requires targeting `net8.0-windows7.0`, which will cause significant build overhead and complexity for cross platform builds.
To counter this, the setup is to have a single project, `Duplicati.Library.WindowsModules`, that is targeting `net8.0-windows7.0`.
The output from this project is then hoisted into the TrayIcon project for Windows builds so the files are available when debugging on Windows.
A top-level dummy executable project is added to ensure the project always builds.
The built modules are then loaded with reflection when requested.

With the use of Vanara there is now also support for using BackupRead to read files without making a VSS snapshot.
With BackupRead, it is possible to read locked files, but it still requires the SeBackupPrivilege as VSS does as well.

Unfortunately, the `vssapi.dll` file is not shipped for Arm64 on Windows, so even with Vanara this will not work, and only WMIC is supported on Arm64.
A workaround is to run Duplicati with x64 emulation if more advanced VSS features are needed (HyperV and MSSQL support).

This PR also updates options and filters out unsupported options for each operating system, so options that are not supported by the current OS are not reported and will give warnings if they are used, as opposed to just being ignored.

The release builder project has been updated to exclude the unused project, and purge unwanted outputs.
2025-07-01 16:32:19 +02:00

260 lines
No EOL
9.2 KiB
C#

// Copyright (C) 2025, The Duplicati Team
// https://duplicati.com, hello@duplicati.com
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
#nullable enable
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Runtime.Versioning;
using System.Text.RegularExpressions;
namespace Duplicati.Library.WindowsModules;
/// <summary>
/// A shadow copy manager using the wmic commandline tool
/// </summary>
[SupportedOSPlatform("windows")]
internal class WmicShadowCopyManager : IDisposable
{
/// <summary>
/// The tag used for logging messages
/// </summary>
private static readonly string LOGTAG = Logging.Log.LogTagFromType<WmicShadowCopyManager>();
/// <summary>
/// A single shadow copy
/// </summary>
/// <param name="shadowId">The shadow ID as a string</param>
/// <param name="parsedId">The shadow ID as a GUID</param>
/// <param name="originalDrive">The drive that the snapshot is for</param>
/// <param name="mappedPath">The path that contains the snapshot</param>
public class WmicShadowCopy(string shadowId, Guid parsedId, string originalDrive, string mappedPath) : IDisposable
{
/// <summary>
/// Gets the shadow ID
/// </summary>
public string ShadowID { get; } = shadowId;
/// <summary>
/// Gets the shadow ID
/// </summary>
public Guid ParsedId { get; } = parsedId;
/// <summary>
/// Gets the drive that was originally mapped
/// </summary>
public string OriginalDrive { get; } = originalDrive;
/// <summary>
/// Gets the path where the snapshot is found
/// </summary>
public string MappedPath { get; } = mappedPath;
/// <summary>
/// Flag keeping track of the snapshot deletion state
/// </summary>
private bool _snapshotDeleted;
/// <inheritdoc/>
public void Dispose()
{
DeleteShadowCopy();
}
/// <summary>
/// Deletes the shadow copy
/// </summary>
private void DeleteShadowCopy()
{
if (_snapshotDeleted)
return;
if (!string.IsNullOrEmpty(ShadowID))
{
_snapshotDeleted = true;
Logging.Log.WriteVerboseMessage(LOGTAG, "DeleteShadowCopy", $"Deleting Shadow Copy: {ShadowID}");
DeleteShadow(ShadowID);
}
}
}
/// <summary>
/// The list of the currently registered shadow copies
/// </summary>
private List<WmicShadowCopy> _shadowCopies = new List<WmicShadowCopy>();
/// <summary>
/// Gets the list of the currently registered shadow copies
/// </summary>
public IEnumerable<WmicShadowCopy> ShadowCopies => _shadowCopies;
/// <summary>
/// Creates a new snapshot for the given drive
/// </summary>
/// <param name="drive">The drive to create the snapshot for</param>
/// <returns>The created snapshot</returns>
public WmicShadowCopy Add(string drive)
{
var shadowId = CreateShadowCopy(drive);
if (string.IsNullOrEmpty(shadowId))
throw new InvalidOperationException("Failed to create shadow copy");
var shadowPath = GetShadowPath(shadowId);
if (string.IsNullOrEmpty(shadowPath))
{
DeleteShadow(shadowId);
throw new InvalidOperationException("Failed to get shadow copy path");
}
var snapshot = new WmicShadowCopy(shadowId, Guid.Parse(shadowId), drive, shadowPath);
_shadowCopies.Add(snapshot);
return snapshot;
}
/// <inheritdoc />
public void Dispose()
{
foreach (var shadow in ShadowCopies)
{
shadow.Dispose();
}
_shadowCopies.Clear();
}
/// <summary>
/// Creates a shadow copy
/// </summary>
/// <param name="drive">The drive to create the snapshot for</param>
/// <returns>The shadow id</returns>
private static string? CreateShadowCopy(string drive)
{
var output = ExecuteCommand("wmic", $"shadowcopy call create Volume='{drive}'", 10000);
// Extract ShadowID using regex
var match = Regex.Match(output, @"ShadowID\s*=\s*""({[0-9A-F\-]+})""");
if (match.Success)
{
return match.Groups[1].Value;
}
Logging.Log.WriteErrorMessage(LOGTAG, "ShadowCopyFailed", null, "Failed to create shadow copy for {0}: {1}", drive, output);
return null;
}
/// <summary>
/// Gets the path where the shadow copy is mounted
/// </summary>
/// <param name="shadowId">The shadow copy id</param>
/// <returns>The path where the copy is mounted</returns>
private static string? GetShadowPath(string shadowId)
{
string output = ExecuteCommand("wmic", "shadowcopy get ID, DeviceObject", 5000);
// Extract DeviceObject corresponding to the ShadowID
string pattern = $@"(\\\\\?\\GLOBALROOT\\Device\\HarddiskVolumeShadowCopy\d+)\s+{shadowId}";
Match match = Regex.Match(output, pattern);
if (match.Success)
{
return match.Groups[1].Value;
}
Logging.Log.WriteErrorMessage(LOGTAG, "ShadowCopyFailed", null, "Failed to get shadow copy path for {0}: {1}", shadowId, output);
return null;
}
/// <summary>
/// Returns the drives that are vss enabled
/// </summary>
/// <returns></returns>
public static HashSet<string> GetVssCapableDrivesViaVssadmin()
{
var vssDrives = new HashSet<string>(StringComparer.OrdinalIgnoreCase);
try
{
var output = ExecuteCommand("vssadmin", "list volumes", 5000);
// Regex to match drive letters in output
var matches = Regex.Matches(output, @"Volume\s+path:\s*([A-Z]:)\\?\s", RegexOptions.IgnoreCase);
foreach (Match match in matches)
vssDrives.Add(match.Groups[1].Value.Substring(0, 1));
}
catch (Exception ex)
{
Logging.Log.WriteErrorMessage(LOGTAG, "ShadowCopyListFailed", ex, "Failed to list volumes");
}
return vssDrives;
}
/// <summary>
/// Delete a shadow copy
/// </summary>
/// <param name="shadowId">The shadow id</param>
private static void DeleteShadow(string shadowId)
=> ExecuteCommand("wmic", $"shadowcopy where ID=\"{shadowId}\" delete", 5000);
/// <summary>
/// Executes a command
/// </summary>
/// <param name="fileName">The binary to execute</param>
/// <param name="arguments">The arguments to use</param>
/// <param name="timeoutMs">The timeout</param>
/// <returns>The output of the command</returns>
private static string ExecuteCommand(string fileName, string arguments, int timeoutMs)
{
try
{
using (var process = new Process())
{
process.StartInfo.FileName = fileName;
process.StartInfo.Arguments = arguments;
process.StartInfo.RedirectStandardOutput = true;
process.StartInfo.RedirectStandardError = true;
process.StartInfo.UseShellExecute = false;
process.StartInfo.CreateNoWindow = true;
process.Start();
var outputTask = process.StandardOutput.ReadToEndAsync();
var errorTask = process.StandardError.ReadToEndAsync();
if (!process.WaitForExit(timeoutMs))
{
process.Kill();
throw new TimeoutException($"Command '{fileName} {arguments}' timed out after {timeoutMs}ms.");
}
var output = outputTask.Result;
var error = errorTask.Result;
if (!string.IsNullOrWhiteSpace(error))
Logging.Log.WriteWarningMessage(LOGTAG, "ShadowCopyFailed", null, "Failed to execute command: {0} {1}: {2}", fileName, arguments, error);
return output;
}
}
catch (Exception ex)
{
Logging.Log.WriteErrorMessage(LOGTAG, "ShadowCopyFailed", ex, "Failed to execute command: {0} {1}", fileName, arguments);
return string.Empty;
}
}
}