mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-27 19:10:29 +08:00
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.
202 lines
8.7 KiB
C#
202 lines
8.7 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.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Runtime.Versioning;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using Duplicati.Library.Common.IO;
|
|
using Duplicati.Library.Interface;
|
|
using Duplicati.Library.Snapshots.Windows;
|
|
|
|
namespace Duplicati.Library.Snapshots
|
|
{
|
|
/// <summary>
|
|
/// Handler for providing a snapshot like access to files and folders
|
|
/// </summary>
|
|
[SupportedOSPlatform("windows")]
|
|
public sealed class NoSnapshotWindows : SnapshotBase
|
|
{
|
|
/// <summary>
|
|
/// The list of folders to create snapshots of
|
|
/// </summary>
|
|
private readonly IEnumerable<string> m_sources;
|
|
/// <summary>
|
|
/// The SeBackupPrivilege token, if used
|
|
/// </summary>
|
|
private readonly IDisposable m_seBackupPrivilege;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="NoSnapshotWindows"/> class.
|
|
/// </summary>
|
|
/// <param name="sources">The list of entries to create snapshots of</param>
|
|
/// <param name="followSymlinks">A flag indicating if symlinks should be followed</param>
|
|
/// <param name="useSeBackup">A flag indicating if the SeBackupPrivilege should be used</param>
|
|
public NoSnapshotWindows(IEnumerable<string> sources, bool followSymlinks, bool useSeBackup)
|
|
: base(followSymlinks)
|
|
{
|
|
m_sources = sources;
|
|
if (useSeBackup)
|
|
m_seBackupPrivilege = WindowsShimLoader.NewSeBackupPrivilegeScope();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the symlink target if the entry is a symlink, and null otherwise
|
|
/// </summary>
|
|
/// <param name="localPath">The file or folder to examine</param>
|
|
/// <returns>The symlink target</returns>
|
|
public override string GetSymlinkTarget(string localPath)
|
|
=> SystemIO.IO_OS.GetSymlinkTarget(localPath);
|
|
|
|
/// <summary>
|
|
/// Gets the attributes for the given file or folder
|
|
/// </summary>
|
|
/// <returns>The file attributes</returns>
|
|
/// <param name="localPath">The file or folder to examine</param>
|
|
public override System.IO.FileAttributes GetAttributes(string localPath)
|
|
=> SystemIO.IO_OS.GetFileAttributes(localPath);
|
|
|
|
/// <summary>
|
|
/// Returns the size of a file
|
|
/// </summary>
|
|
/// <param name="localPath">The full path to the file in non-snapshot format</param>
|
|
/// <returns>The length of the file</returns>
|
|
public override long GetFileSize(string localPath)
|
|
=> SystemIO.IO_OS.FileLength(localPath);
|
|
|
|
/// <summary>
|
|
/// Gets the source folders
|
|
/// </summary>
|
|
public override IEnumerable<string> SourceEntries
|
|
=> m_sources;
|
|
|
|
/// <summary>
|
|
/// Enumerates the root source files and folders
|
|
/// </summary>
|
|
/// <returns>The source files and folders</returns>
|
|
public override IEnumerable<ISourceProviderEntry> EnumerateFilesystemEntries()
|
|
{
|
|
foreach (var folder in m_sources.Select(SystemIOWindows.RemoveExtendedDevicePathPrefix))
|
|
{
|
|
if (DirectoryExists(folder) || folder.EndsWith(System.IO.Path.DirectorySeparatorChar))
|
|
yield return new SnapshotSourceFileEntry(this, Util.AppendDirSeparator(folder), true, true);
|
|
else
|
|
yield return new SnapshotSourceFileEntry(this, folder, false, true);
|
|
}
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Gets the last write time of a given file in UTC
|
|
/// </summary>
|
|
/// <param name="localPath">The full path to the file in non-snapshot format</param>
|
|
/// <returns>The last write time of the file</returns>
|
|
public override DateTime GetLastWriteTimeUtc(string localPath)
|
|
=> SystemIO.IO_OS.FileGetLastWriteTimeUtc(localPath);
|
|
|
|
/// <summary>
|
|
/// Gets the last write time of a given file in UTC
|
|
/// </summary>
|
|
/// <param name="localPath">The full path to the file in non-snapshot format</param>
|
|
/// <returns>The last write time of the file</returns>
|
|
public override DateTime GetCreationTimeUtc(string localPath)
|
|
=> SystemIO.IO_OS.FileGetCreationTimeUtc(localPath);
|
|
|
|
/// <summary>
|
|
/// Opens a file for reading
|
|
/// </summary>
|
|
/// <param name="localPath">The full path to the file in non-snapshot format</param>
|
|
/// <returns>An open filestream that can be read</returns>
|
|
public override System.IO.Stream OpenRead(string localPath)
|
|
=> SystemIO.IO_OS.FileOpenRead(localPath);
|
|
|
|
/// <summary>
|
|
/// Lists all files in the given folder
|
|
/// </summary>
|
|
/// <returns>All folders found in the folder</returns>
|
|
/// <param name='localFolderPath'>The folder to examinate</param>
|
|
protected override string[] ListFiles(string localFolderPath)
|
|
{
|
|
string[] tmp = SystemIO.IO_OS.GetFiles(localFolderPath);
|
|
string[] res = new string[tmp.Length];
|
|
for (int i = 0; i < tmp.Length; i++)
|
|
res[i] = SystemIOWindows.RemoveExtendedDevicePathPrefix(tmp[i]);
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Lists all folders in the given folder
|
|
/// </summary>
|
|
/// <returns>All folders found in the folder</returns>
|
|
/// <param name='localFolderPath'>The folder to examinate</param>
|
|
protected override string[] ListFolders(string localFolderPath)
|
|
{
|
|
string[] tmp = SystemIO.IO_OS.GetDirectories(SystemIOWindows.AddExtendedDevicePathPrefix(localFolderPath));
|
|
string[] res = new string[tmp.Length];
|
|
for (int i = 0; i < tmp.Length; i++)
|
|
res[i] = SystemIOWindows.RemoveExtendedDevicePathPrefix(tmp[i]);
|
|
|
|
return res;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the metadata for the given file or folder
|
|
/// </summary>
|
|
/// <returns>The metadata for the given file or folder</returns>
|
|
/// <param name="localPath">The file or folder to examine</param>
|
|
/// <param name="isSymlink">A flag indicating if the target is a symlink</param>
|
|
public override Dictionary<string, string> GetMetadata(string localPath, bool isSymlink)
|
|
=> SystemIO.IO_OS.GetMetadata(localPath, isSymlink, FollowSymlinks);
|
|
|
|
/// <inheritdoc />
|
|
public override string ConvertToLocalPath(string snapshotPath)
|
|
{
|
|
return snapshotPath;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override string ConvertToSnapshotPath(string localPath) =>
|
|
// For Windows, ensure we don't store paths with extended device path prefixes (i.e., @"\\?\" or @"\\?\UNC\")
|
|
SystemIOWindows.RemoveExtendedDevicePathPrefix(localPath);
|
|
|
|
/// <inheritdoc />
|
|
public override Task<Stream> OpenReadAsync(string localPath, CancellationToken cancellationToken)
|
|
{
|
|
if (m_seBackupPrivilege != null)
|
|
return Task.FromResult(WindowsShimLoader.NewBackupDataStream(localPath));
|
|
|
|
return base.OpenReadAsync(localPath, cancellationToken);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
m_seBackupPrivilege?.Dispose();
|
|
base.Dispose(disposing);
|
|
}
|
|
}
|
|
}
|
|
|