duplicati/Duplicati/Library/Snapshots/SnapshotBase.cs
Kenneth Skovhede eb870e16dc Strip null characters from path
Due to a bug in .NET Core, it will sometimes return a trailing null character in the paths. Since null characters are not allowed on any OS, this PR ensures that null characters are removed before returning the path string.

This fixes #6500
2025-09-22 12:09:59 +02:00

239 lines
No EOL
10 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 Duplicati.Library.Interface;
using System.Threading.Tasks;
using System.Threading;
using Duplicati.Library.Common.IO;
namespace Duplicati.Library.Snapshots
{
public abstract class SnapshotBase : ISnapshotService
{
/// <summary>
/// Releases unmanaged resources and performs other cleanup operations before the
/// <see cref="SnapshotBase"/> is reclaimed by garbage collection.
/// </summary>
/// <remarks>We must implement a finalizer to guarantee that our native handle is cleaned up</remarks>
~SnapshotBase()
{
// Our finalizer should call our Dispose(bool) method with false
Dispose(false);
}
/// <summary>
/// Initializes a new instance of the <see cref="SnapshotBase"/> class.
/// </summary>
/// <param name="followSymlinks">A flag indicating if symlinks should be followed</param>
protected SnapshotBase(bool followSymlinks)
{
FollowSymlinks = followSymlinks;
}
#region ISnapshotService
protected bool FollowSymlinks { get; }
/// <summary>
/// Gets the source entries
/// </summary>
public abstract IEnumerable<string> SourceEntries { get; }
/// <summary>
/// Enuemrates the root source files and folders
/// </summary>
/// <returns>The source files and folders</returns>
public abstract IEnumerable<ISourceProviderEntry> EnumerateFilesystemEntries();
/// <summary>
/// Gets a filesystem entry for a given path
/// </summary>
/// <param name="path">The path to get the entry for</param>
/// <param name="isFolder">A flag indicating if the path is a folder</param>
/// <returns>The filesystem entry</returns>
public virtual ISourceProviderEntry GetFilesystemEntry(string path, bool isFolder)
{
if (isFolder && !DirectoryExists(path))
return null;
else if (!isFolder && !FileExists(path))
return null;
return new SnapshotSourceFileEntry(this, isFolder ? Util.AppendDirSeparator(path) : path, isFolder, false);
}
/// <summary>
/// Enumerates files and folders in the given folder
/// </summary>
/// <param name="source">Source to enumerate</param>
/// <returns>The files and folders in the given folder</returns>
public virtual IEnumerable<ISourceProviderEntry> EnumerateFilesystemEntries(ISourceProviderEntry source)
{
if (source.IsFolder)
{
// Removing the null characters is a workaround for a bug in .NET core:
// https://github.com/dotnet/runtime/issues/49803
foreach (var f in ListFiles(source.Path))
yield return new SnapshotSourceFileEntry(this, f.Trim('\0'), false, false);
foreach (var d in ListFolders(source.Path))
yield return new SnapshotSourceFileEntry(this, Util.AppendDirSeparator(d.Trim('\0')), true, false);
}
}
/// <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 virtual DateTime GetLastWriteTimeUtc(string localPath)
=> File.GetLastWriteTimeUtc(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 virtual DateTime GetCreationTimeUtc(string localPath)
=> File.GetCreationTimeUtc(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 virtual Stream OpenRead(string localPath)
=> File.Open(localPath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite);
/// <summary>
/// Opens a file for reading
/// </summary>
/// <param name="localPath">The full path to the file in non-snapshot format</param>
/// <param name="cancellationToken">A token that can be used to cancel the operation</param>
/// <returns>An open filestream that can be read</returns>
public virtual Task<Stream> OpenReadAsync(string localPath, CancellationToken cancellationToken)
=> Task.FromResult(OpenRead(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 virtual long GetFileSize(string localPath)
=> new FileInfo(localPath).Length;
/// <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 abstract string GetSymlinkTarget(string 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 virtual FileAttributes GetAttributes(string localPath)
=> File.GetAttributes(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 abstract Dictionary<string, string> GetMetadata(string localPath, bool isSymlink);
/// <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 virtual bool IsBlockDevice(string localPath)
=> false;
/// <summary>
/// Gets a unique hardlink target ID
/// </summary>
/// <returns>The hardlink ID</returns>
/// <param name="localPath">The file or folder to examine</param>
public virtual string HardlinkTargetID(string localPath)
=> null;
/// <inheritdoc />
public abstract string ConvertToLocalPath(string snapshotPath);
/// <inheritdoc />
public abstract string ConvertToSnapshotPath(string localPath);
/// <inheritdoc />
public virtual bool FileExists(string localFilePath)
=> File.Exists(ConvertToSnapshotPath(localFilePath));
/// <inheritdoc />
public virtual bool DirectoryExists(string localFolderPath)
=> Directory.Exists(ConvertToSnapshotPath(localFolderPath));
#endregion
/// <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 virtual string[] ListFolders(string localFolderPath)
=> Directory.GetDirectories(localFolderPath);
/// <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 virtual string[] ListFiles(string localFolderPath)
=> Directory.GetFiles(localFolderPath);
#region IDisposable interface
/// <inheritdoc />
public void Dispose()
{
// We start by calling Dispose(bool) with true
Dispose(true);
// Now suppress finalization for this object, since we've already handled our resource cleanup tasks
GC.SuppressFinalize(this);
}
#endregion
/// <summary>
/// Releases unmanaged and - optionally - managed resources
/// </summary>
/// <param name="disposing"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
protected virtual void Dispose(bool disposing)
{
// nothing to dispose of at this level
}
}
}