mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-28 03:20:25 +08:00
This fixes the restore issues as all sources will now appear as if they were originally part of the source file system. The backend implementation has now been simplified so the backends only return names of entries with no path information. Most backends already do this, but the S3 backends did not, so this was fixed as well. This also clears up the interfaces, so the backend does not know about the `ISourceProviderEntry`. All mapping is done by the `BackendSourceProvider`, making it simpler to support other backends as sources.
565 lines
23 KiB
C#
565 lines
23 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.Diagnostics;
|
|
using System.Linq;
|
|
using System.Runtime.Versioning;
|
|
using System.Text;
|
|
using Duplicati.Library.Common.IO;
|
|
using Duplicati.Library.Interface;
|
|
|
|
namespace Duplicati.Library.Snapshots
|
|
{
|
|
/// <summary>
|
|
/// This class encapsulates all access to the Linux LVM snapshot feature,
|
|
/// implementing the disposable patterns to ensure correct release of resources.
|
|
///
|
|
/// The class presents all files and folders with their regular filenames to the caller,
|
|
/// and internally handles the conversion to the snapshot path.
|
|
/// </summary>
|
|
[SupportedOSPlatform("linux")]
|
|
[SupportedOSPlatform("macOS")]
|
|
public sealed class LinuxSnapshot : SnapshotBase
|
|
{
|
|
/// <summary>
|
|
/// The tag used for logging messages
|
|
/// </summary>
|
|
public static readonly string LOGTAG = Logging.Log.LogTagFromType(typeof(WindowsSnapshot));
|
|
|
|
/// <summary>
|
|
/// This is a lookup, mapping each source folder to the corresponding snapshot
|
|
/// </summary>
|
|
private readonly List<KeyValuePair<string, SnapShot>> m_entries;
|
|
|
|
/// <summary>
|
|
/// This is the list of the snapshots we have created, which must be disposed
|
|
/// </summary>
|
|
private List<SnapShot> m_snapShots;
|
|
|
|
/// <summary>
|
|
/// Constructs a new snapshot module using LVM
|
|
/// </summary>
|
|
/// <param name="sources">The list of folders to create snapshots for</param>
|
|
/// <param name="followSymlinks">A flag indicating if symlinks should be followed</param>
|
|
public LinuxSnapshot(IEnumerable<string> sources, bool followSymlinks)
|
|
: base(followSymlinks)
|
|
{
|
|
try
|
|
{
|
|
m_entries = new List<KeyValuePair<string, SnapShot>>();
|
|
|
|
// Make sure we do not create more snapshots than we have to
|
|
var snaps = new Dictionary<string, SnapShot>();
|
|
foreach (var path in sources)
|
|
{
|
|
var tmp = new SnapShot(path);
|
|
if (!snaps.TryGetValue(tmp.DeviceName, out var snap))
|
|
{
|
|
snaps.Add(tmp.DeviceName, tmp);
|
|
snap = tmp;
|
|
}
|
|
|
|
m_entries.Add(new KeyValuePair<string, SnapShot>(path, snap));
|
|
}
|
|
|
|
m_snapShots = [.. snaps.Values];
|
|
|
|
// We have all the snapshots that we need, lets activate them
|
|
foreach (var snap in m_snapShots)
|
|
{
|
|
snap.CreateSnapshotVolume();
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
// If something goes wrong, try to clean up
|
|
try
|
|
{
|
|
Dispose();
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logging.Log.WriteVerboseMessage(LOGTAG, "SnapshotCleanupError", ex, "Failed to clean up after error");
|
|
}
|
|
|
|
throw;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the source folders
|
|
/// </summary>
|
|
public override IEnumerable<string> SourceEntries => m_entries.Select(x => x.Key);
|
|
|
|
/// <summary>
|
|
/// Enumerates the root source files and folders
|
|
/// </summary>
|
|
/// <returns>The source files and folders</returns>
|
|
public override IEnumerable<ISourceProviderEntry> EnumerateFilesystemEntries()
|
|
{
|
|
foreach (var sourceEntry in SourceEntries)
|
|
{
|
|
if (DirectoryExists(sourceEntry) || sourceEntry.EndsWith(System.IO.Path.DirectorySeparatorChar))
|
|
yield return new SnapshotSourceFileEntry(this, Util.AppendDirSeparator(sourceEntry), true, true);
|
|
else
|
|
yield return new SnapshotSourceFileEntry(this, sourceEntry, false, true);
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
if (m_snapShots != null)
|
|
{
|
|
if (disposing)
|
|
{
|
|
// Attempt to clean out as many as possible
|
|
foreach (var s in m_snapShots)
|
|
{
|
|
try { s.Dispose(); }
|
|
catch (Exception ex) { Logging.Log.WriteVerboseMessage(LOGTAG, "SnapshotCloseError", ex, "Failed to close a snapshot"); }
|
|
}
|
|
}
|
|
|
|
// Don't try this again
|
|
m_snapShots = null;
|
|
}
|
|
|
|
base.Dispose(disposing);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Internal helper class for keeping track of a single snapshot volume
|
|
/// </summary>
|
|
private sealed class SnapShot : IDisposable
|
|
{
|
|
/// <summary>
|
|
/// The unique id of the snapshot
|
|
/// </summary>
|
|
private readonly string m_name;
|
|
|
|
/// <summary>
|
|
/// Constructs a new snapshot for the given folder
|
|
/// </summary>
|
|
/// <param name="path"></param>
|
|
public SnapShot(string path)
|
|
{
|
|
m_name = $"duplicati-{Guid.NewGuid().ToString()}";
|
|
LocalPath = System.IO.Directory.Exists(path) ? Util.AppendDirSeparator(path) : path;
|
|
Initialize(LocalPath);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the path of the folder that this snapshot represents
|
|
/// </summary>
|
|
public string LocalPath { get; }
|
|
|
|
/// <summary>
|
|
/// Gets a value representing the volume on which the folder resides
|
|
/// </summary>
|
|
public string DeviceName { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Gets the path where the snapshot is mounted
|
|
/// </summary>
|
|
public string SnapshotPath { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Gets the path the source disk is originally mounted
|
|
/// </summary>
|
|
public string MountPoint { get; private set; }
|
|
|
|
#region IDisposable Members
|
|
|
|
/// <summary>
|
|
/// Cleanup any used resources
|
|
/// </summary>
|
|
public void Dispose()
|
|
{
|
|
if (SnapshotPath != null && System.IO.Directory.Exists(SnapshotPath))
|
|
{
|
|
var output = ExecuteCommand("remove-lvm-snapshot.sh", $"\"{m_name}\" \"{DeviceName}\" \"{SnapshotPath}\"", 0);
|
|
if (System.IO.Directory.Exists(SnapshotPath))
|
|
throw new Exception(Strings.LinuxSnapshot.MountFolderNotRemovedError(SnapshotPath, output));
|
|
|
|
SnapshotPath = null;
|
|
DeviceName = null;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
/// <summary>
|
|
/// Converts a local path to a snapshot path
|
|
/// </summary>
|
|
/// <param name="localPath">The local path</param>
|
|
/// <returns>The snapshot path</returns>
|
|
public string ConvertToSnapshotPath(string localPath)
|
|
{
|
|
if (!localPath.StartsWith(MountPoint, StringComparison.Ordinal))
|
|
throw new InvalidOperationException();
|
|
|
|
return SystemIOLinux.NormalizePath(SnapshotPath + localPath.Substring(MountPoint.Length));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts a snapshot path to a local path
|
|
/// </summary>
|
|
/// <param name="snapshotPath">The snapshot path</param>
|
|
/// <returns>The local path</returns>
|
|
public string ConvertToLocalPath(string snapshotPath)
|
|
{
|
|
if (!snapshotPath.StartsWith(SnapshotPath, StringComparison.Ordinal))
|
|
throw new InvalidOperationException();
|
|
|
|
return MountPoint + snapshotPath.Substring(SnapshotPath.Length);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Helper function to execute a script
|
|
/// </summary>
|
|
/// <param name="program">The name of the lvm-script to execute</param>
|
|
/// <param name="commandline">The arguments to pass to the executable</param>
|
|
/// <param name="expectedExitCode">The exitcode that is expected</param>
|
|
/// <returns>A string with the combined output of the stdout and stderr</returns>
|
|
private static string ExecuteCommand(string program, string commandline, int expectedExitCode)
|
|
{
|
|
program = System.IO.Path.Combine(System.IO.Path.Combine(Duplicati.Library.AutoUpdater.UpdaterManager.INSTALLATIONDIR, "lvm-scripts"), program);
|
|
var inf = new ProcessStartInfo(program, commandline)
|
|
{
|
|
CreateNoWindow = true,
|
|
RedirectStandardError = true,
|
|
RedirectStandardOutput = true,
|
|
RedirectStandardInput = false,
|
|
WindowStyle = ProcessWindowStyle.Hidden,
|
|
UseShellExecute = false
|
|
};
|
|
|
|
try
|
|
{
|
|
var p = Process.Start(inf);
|
|
|
|
//Allow up 20 seconds for the execution
|
|
if (!p.WaitForExit(30 * 1000))
|
|
{
|
|
//Attempt to close down semi-nicely
|
|
p.Kill();
|
|
p.WaitForExit(5 * 1000); //This should work, and if it does, prevents a race with any cleanup invocations
|
|
|
|
throw new Interface.UserInformationException(Strings.LinuxSnapshot.ExternalProgramTimeoutError(program, commandline), "LvmScriptTimeout");
|
|
}
|
|
|
|
//Build the output string. Since the process has exited, these cannot block
|
|
var output = string.Format("Exit code: {1}{0}{2}{0}{3}", Environment.NewLine, p.ExitCode, p.StandardOutput.ReadToEnd(), p.StandardError.ReadToEnd());
|
|
|
|
//Throw an exception if something went wrong
|
|
if (p.ExitCode != expectedExitCode)
|
|
throw new Interface.UserInformationException(Strings.LinuxSnapshot.ScriptExitCodeError(p.ExitCode, expectedExitCode, output), "LvmScriptWrongExitCode");
|
|
|
|
return output;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
throw new Exception(Strings.LinuxSnapshot.ExternalProgramLaunchError(ex.ToString(), program, commandline));
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Finds the LVM id of the volume id where the folder is placed
|
|
/// </summary>
|
|
private void Initialize(string folder)
|
|
{
|
|
//Figure out what logical volume the path is located on
|
|
var output = ExecuteCommand("find-volume.sh", $"\"{folder}\"", 0);
|
|
|
|
var rex = new System.Text.RegularExpressions.Regex("device=\"(?<device>[^\"]+)\"");
|
|
var m = rex.Match(output);
|
|
|
|
if (!m.Success)
|
|
throw new Exception(Strings.LinuxSnapshot.ScriptOutputError("device", output));
|
|
|
|
DeviceName = rex.Match(output).Groups["device"].Value;
|
|
|
|
if (string.IsNullOrEmpty(DeviceName) || DeviceName.Trim().Length == 0)
|
|
throw new Exception(Strings.LinuxSnapshot.ScriptOutputError("device", output));
|
|
|
|
rex = new System.Text.RegularExpressions.Regex("mountpoint=\"(?<mountpoint>[^\"]+)\"");
|
|
m = rex.Match(output);
|
|
|
|
if (!m.Success)
|
|
throw new Exception(Strings.LinuxSnapshot.ScriptOutputError("mountpoint", output));
|
|
|
|
MountPoint = rex.Match(output).Groups["mountpoint"].Value;
|
|
|
|
if (string.IsNullOrEmpty(MountPoint) || MountPoint.Trim().Length == 0)
|
|
throw new Exception(Strings.LinuxSnapshot.ScriptOutputError("mountpoint", output));
|
|
|
|
MountPoint = Util.AppendDirSeparator(MountPoint);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create the snapshot and mount it, this is not done in the constructor,
|
|
/// because we want to see if some folders are on the same volume
|
|
/// </summary>
|
|
public void CreateSnapshotVolume()
|
|
{
|
|
if (DeviceName == null)
|
|
throw new InvalidOperationException();
|
|
if (SnapshotPath != null)
|
|
throw new InvalidOperationException();
|
|
|
|
//Create the snapshot volume
|
|
var output = ExecuteCommand("create-lvm-snapshot.sh", $"\"{m_name}\" \"{DeviceName}\" \"{Util.AppendDirSeparator(Utility.TempFolder.SystemTempPath)}\"", 0);
|
|
|
|
var rex = new System.Text.RegularExpressions.Regex("tmpdir=\"(?<tmpdir>[^\"]+)\"");
|
|
var m = rex.Match(output);
|
|
|
|
if (!m.Success)
|
|
throw new Exception(Strings.LinuxSnapshot.ScriptOutputError("tmpdir", output));
|
|
|
|
SnapshotPath = rex.Match(output).Groups["tmpdir"].Value;
|
|
|
|
if (!System.IO.Directory.Exists(SnapshotPath))
|
|
throw new Exception(Strings.LinuxSnapshot.MountFolderMissingError(SnapshotPath, output));
|
|
|
|
SnapshotPath = Util.AppendDirSeparator(SnapshotPath);
|
|
}
|
|
}
|
|
|
|
#region Private functions
|
|
|
|
/// <summary>
|
|
/// A callback function that takes a non-snapshot path to a folder,
|
|
/// and returns all folders found in a non-snapshot path format.
|
|
/// </summary>
|
|
/// <param name="localFolderPath">The non-snapshot path of the folder to list</param>
|
|
/// <returns>A list of non-snapshot paths</returns>
|
|
protected override string[] ListFolders(string localFolderPath)
|
|
{
|
|
var snap = FindSnapshotByLocalPath(localFolderPath);
|
|
|
|
var tmp = System.IO.Directory.GetDirectories(snap.ConvertToSnapshotPath(localFolderPath));
|
|
for (var i = 0; i < tmp.Length; i++)
|
|
tmp[i] = snap.ConvertToLocalPath(tmp[i]);
|
|
|
|
return tmp;
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// A callback function that takes a non-snapshot path to a folder,
|
|
/// and returns all files found in a non-snapshot path format.
|
|
/// </summary>
|
|
/// <param name="localFolderPath">The non-snapshot path of the folder to list</param>
|
|
/// <returns>A list of non-snapshot paths</returns>
|
|
protected override string[] ListFiles(string localFolderPath)
|
|
{
|
|
var snap = FindSnapshotByLocalPath(localFolderPath);
|
|
|
|
var tmp = System.IO.Directory.GetFiles(snap.ConvertToSnapshotPath(localFolderPath));
|
|
for (var i = 0; i < tmp.Length; i++)
|
|
tmp[i] = snap.ConvertToLocalPath(tmp[i]);
|
|
return tmp;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Locates the snapshot instance that maps the path
|
|
/// </summary>
|
|
/// <param name="localPath">The file or folder name to match</param>
|
|
/// <returns>The matching snapshot</returns>
|
|
private SnapShot FindSnapshotByLocalPath(string localPath)
|
|
{
|
|
KeyValuePair<string, SnapShot>? best = null;
|
|
foreach (var s in m_entries)
|
|
{
|
|
if (localPath.StartsWith(s.Key, StringComparison.Ordinal) && (best == null || s.Key.Length > best.Value.Key.Length))
|
|
{
|
|
best = s;
|
|
}
|
|
}
|
|
|
|
if (best != null)
|
|
return best.Value.Value;
|
|
|
|
var sb = new StringBuilder();
|
|
sb.Append(Environment.NewLine);
|
|
|
|
foreach (var s in m_entries)
|
|
{
|
|
sb.Append($"{s.Key} ({s.Value.MountPoint} -> {s.Value.SnapshotPath}){Environment.NewLine}");
|
|
}
|
|
|
|
throw new InvalidOperationException(Strings.LinuxSnapshot.InvalidFilePathError(localPath, sb.ToString()));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Locates the snapshot containing the snapshot path
|
|
/// </summary>
|
|
/// <param name="snapshotPath"></param>
|
|
/// <returns>Snapshot containing snapshotPath</returns>
|
|
private SnapShot FindSnapshotBySnapshotPath(string snapshotPath)
|
|
{
|
|
foreach (var snap in m_snapShots)
|
|
{
|
|
if (snapshotPath.StartsWith(snap.SnapshotPath, StringComparison.Ordinal))
|
|
return snap;
|
|
}
|
|
|
|
throw new InvalidOperationException();
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ISnapshotService Members
|
|
|
|
/// <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)
|
|
{
|
|
return System.IO.File.GetLastWriteTimeUtc(ConvertToSnapshotPath(localPath));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the creation 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)
|
|
{
|
|
return System.IO.File.GetLastWriteTimeUtc(ConvertToSnapshotPath(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)
|
|
{
|
|
return System.IO.File.OpenRead(ConvertToSnapshotPath(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)
|
|
{
|
|
return new System.IO.FileInfo(ConvertToSnapshotPath(localPath)).Length;
|
|
}
|
|
|
|
/// <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)
|
|
{
|
|
return System.IO.File.GetAttributes(ConvertToSnapshotPath(localPath));
|
|
}
|
|
|
|
/// <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)
|
|
{
|
|
return SystemIO.IO_OS.GetSymlinkTarget(ConvertToSnapshotPath(localPath));
|
|
}
|
|
|
|
/// <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)
|
|
{
|
|
return SystemIO.IO_OS.GetMetadata(ConvertToSnapshotPath(localPath), isSymlink, FollowSymlinks);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a value indicating if the path points to a block device
|
|
/// </summary>
|
|
/// <returns><c>true</c> if this instance is a block device; otherwise, <c>false</c>.</returns>
|
|
/// <param name="localPath">The file or folder to examine</param>
|
|
public override bool IsBlockDevice(string localPath)
|
|
{
|
|
try
|
|
{
|
|
var n = PosixFile.GetFileType(SystemIOLinux.NormalizePath(localPath));
|
|
switch (n)
|
|
{
|
|
case PosixFile.FileType.Directory:
|
|
case PosixFile.FileType.Symlink:
|
|
case PosixFile.FileType.File:
|
|
return false;
|
|
default:
|
|
return true;
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
if (!System.IO.File.Exists(SystemIOLinux.NormalizePath(localPath)))
|
|
return false;
|
|
|
|
throw;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a unique hardlink target ID
|
|
/// </summary>
|
|
/// <returns>The hardlink ID</returns>
|
|
/// <param name="localPath">The file or folder to examine</param>
|
|
public override string HardlinkTargetID(string localPath)
|
|
{
|
|
var snapshotPath = ConvertToSnapshotPath(localPath);
|
|
|
|
if (PosixFile.GetHardlinkCount(snapshotPath) <= 1)
|
|
return null;
|
|
|
|
return PosixFile.GetInodeTargetID(snapshotPath);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override string ConvertToLocalPath(string snapshotPath)
|
|
{
|
|
return FindSnapshotBySnapshotPath(snapshotPath).ConvertToLocalPath(snapshotPath);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override string ConvertToSnapshotPath(string localPath)
|
|
{
|
|
return FindSnapshotByLocalPath(localPath).ConvertToSnapshotPath(localPath);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|