mirror of
https://github.com/Cleanuparr/Cleanuparr.git
synced 2025-12-26 23:48:30 -05:00
114 lines
3.8 KiB
C#
114 lines
3.8 KiB
C#
using System.Collections.Concurrent;
|
|
using System.Runtime.InteropServices;
|
|
using Cleanuparr.Infrastructure.Features.Files;
|
|
using Microsoft.Extensions.Logging;
|
|
using Microsoft.Win32.SafeHandles;
|
|
|
|
namespace Infrastructure.Verticals.Files;
|
|
|
|
public class WindowsHardLinkFileService : IHardLinkFileService, IDisposable
|
|
{
|
|
private readonly ILogger<WindowsHardLinkFileService> _logger;
|
|
private readonly ConcurrentDictionary<ulong, int> _fileIndexCounts = new();
|
|
|
|
public WindowsHardLinkFileService(ILogger<WindowsHardLinkFileService> logger)
|
|
{
|
|
_logger = logger;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public long GetHardLinkCount(string filePath, bool ignoreRootDir)
|
|
{
|
|
try
|
|
{
|
|
using SafeFileHandle fileStream = File.OpenHandle(filePath);
|
|
|
|
if (!GetFileInformationByHandle(fileStream, out var file))
|
|
{
|
|
_logger.LogDebug("failed to get file handle {file}", filePath);
|
|
return -1;
|
|
}
|
|
|
|
if (!ignoreRootDir)
|
|
{
|
|
_logger.LogDebug("stat file | hardlinks: {nlink} | {file}", file.NumberOfLinks, filePath);
|
|
return file.NumberOfLinks == 1 ? 0 : 1;
|
|
}
|
|
|
|
// Get unique file ID (combination of high and low indices)
|
|
ulong fileIndex = ((ulong)file.FileIndexHigh << 32) | file.FileIndexLow;
|
|
|
|
// get the number of hardlinks in the same root directory
|
|
int linksInIgnoredDir = _fileIndexCounts.TryGetValue(fileIndex, out int count)
|
|
? count
|
|
: 1; // default to 1 if not found
|
|
|
|
_logger.LogDebug("stat file | hardlinks: {links} | ignored: {ignored} | {file}", file.NumberOfLinks, linksInIgnoredDir, filePath);
|
|
return file.NumberOfLinks - linksInIgnoredDir;
|
|
}
|
|
catch (Exception exception)
|
|
{
|
|
_logger.LogError(exception, "failed to stat file {file}", filePath);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public void PopulateFileCounts(string directoryPath)
|
|
{
|
|
try
|
|
{
|
|
// traverse all files in the ignored path and subdirectories
|
|
foreach (string file in Directory.EnumerateFiles(directoryPath, "*", SearchOption.AllDirectories))
|
|
{
|
|
AddFileIndexToCount(file);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogError(ex, "Failed to populate file index counts from {dir}", directoryPath);
|
|
}
|
|
}
|
|
|
|
private void AddFileIndexToCount(string path)
|
|
{
|
|
try
|
|
{
|
|
using SafeFileHandle fileStream = File.OpenHandle(path);
|
|
if (GetFileInformationByHandle(fileStream, out var file))
|
|
{
|
|
ulong fileIndex = ((ulong)file.FileIndexHigh << 32) | file.FileIndexLow;
|
|
_fileIndexCounts.AddOrUpdate(fileIndex, 1, (_, count) => count + 1);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogWarning(ex, "Couldn't stat {path} during file index counting", path);
|
|
}
|
|
}
|
|
|
|
[DllImport("kernel32.dll", SetLastError = true)]
|
|
private static extern bool GetFileInformationByHandle(
|
|
SafeFileHandle hFile,
|
|
out BY_HANDLE_FILE_INFORMATION lpFileInformation
|
|
);
|
|
|
|
private struct BY_HANDLE_FILE_INFORMATION
|
|
{
|
|
public uint FileAttributes;
|
|
public System.Runtime.InteropServices.ComTypes.FILETIME CreationTime;
|
|
public System.Runtime.InteropServices.ComTypes.FILETIME LastAccessTime;
|
|
public System.Runtime.InteropServices.ComTypes.FILETIME LastWriteTime;
|
|
public uint VolumeSerialNumber;
|
|
public uint FileSizeHigh;
|
|
public uint FileSizeLow;
|
|
public uint NumberOfLinks;
|
|
public uint FileIndexHigh;
|
|
public uint FileIndexLow;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
_fileIndexCounts.Clear();
|
|
}
|
|
} |