wabbajack/Wabbajack.VFS/FileHashCache.cs

129 lines
3.9 KiB
C#
Raw Permalink Normal View History

2021-09-27 12:42:46 +00:00
using System.Data.SQLite;
using System.IO;
using System.Threading;
using System.Threading.Tasks;
using Wabbajack.Hashing.xxHash64;
using Wabbajack.Paths;
using Wabbajack.Paths.IO;
using Wabbajack.RateLimiter;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
namespace Wabbajack.VFS;
public class FileHashCache
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
private readonly SQLiteConnection _conn;
private readonly string _connectionString;
private readonly IResource<FileHashCache> _limiter;
private readonly AbsolutePath _location;
public FileHashCache(AbsolutePath location, IResource<FileHashCache> limiter)
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
_limiter = limiter;
_location = location;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
if (!_location.Parent.DirectoryExists())
_location.Parent.CreateDirectory();
_connectionString =
string.Intern($"URI=file:{_location};Pooling=True;Max Pool Size=100; Journal Mode=Memory;");
_conn = new SQLiteConnection(_connectionString);
_conn.Open();
using var cmd = new SQLiteCommand(_conn);
cmd.CommandText = @"CREATE TABLE IF NOT EXISTS HashCache (
2021-09-27 12:42:46 +00:00
Path TEXT PRIMARY KEY,
LastModified BIGINT,
Hash BIGINT)
WITHOUT ROWID";
2021-10-23 16:51:17 +00:00
cmd.ExecuteNonQuery();
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
private (AbsolutePath Path, long LastModified, Hash Hash) Get(AbsolutePath path)
{
using var cmd = new SQLiteCommand(_conn);
cmd.CommandText = "SELECT LastModified, Hash FROM HashCache WHERE Path = @path";
cmd.Parameters.AddWithValue("@path", path.ToString());
cmd.PrepareAsync();
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
using var reader = cmd.ExecuteReader();
while (reader.Read()) return (path, reader.GetInt64(0), Hash.FromLong(reader.GetInt64(1)));
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
return default;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public void Purge(AbsolutePath path)
{
using var cmd = new SQLiteCommand(_conn);
cmd.CommandText = "DELETE FROM HashCache WHERE Path = @path";
cmd.Parameters.AddWithValue("@path", path.ToString());
cmd.PrepareAsync();
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
cmd.ExecuteNonQuery();
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
private void Upsert(AbsolutePath path, long lastModified, Hash hash)
{
using var cmd = new SQLiteCommand(_conn);
cmd.CommandText = @"INSERT INTO HashCache (Path, LastModified, Hash) VALUES (@path, @lastModified, @hash)
2021-09-27 12:42:46 +00:00
ON CONFLICT(Path) DO UPDATE SET LastModified = @lastModified, Hash = @hash";
2021-10-23 16:51:17 +00:00
cmd.Parameters.AddWithValue("@path", path.ToString());
cmd.Parameters.AddWithValue("@lastModified", lastModified);
cmd.Parameters.AddWithValue("@hash", (long) hash);
cmd.PrepareAsync();
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
cmd.ExecuteNonQuery();
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public void VacuumDatabase()
{
using var cmd = new SQLiteCommand(_conn);
cmd.CommandText = @"VACUUM";
cmd.PrepareAsync();
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
cmd.ExecuteNonQuery();
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
public bool TryGetHashCache(AbsolutePath file, out Hash hash)
{
hash = default;
if (!file.FileExists()) return false;
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
var result = Get(file);
if (result == default)
2021-09-27 12:42:46 +00:00
return false;
2021-10-23 16:51:17 +00:00
if (result.LastModified == file.LastModifiedUtc().ToFileTimeUtc())
2021-09-27 12:42:46 +00:00
{
2021-10-23 16:51:17 +00:00
hash = result.Hash;
return true;
2021-09-27 12:42:46 +00:00
}
2021-10-23 16:51:17 +00:00
Purge(file);
return false;
}
2021-09-27 12:42:46 +00:00
2021-10-23 16:51:17 +00:00
private void WriteHashCache(AbsolutePath file, Hash hash)
{
if (!file.FileExists()) return;
Upsert(file, file.LastModifiedUtc().ToFileTimeUtc(), hash);
}
public void FileHashWriteCache(AbsolutePath file, Hash hash)
{
WriteHashCache(file, hash);
}
public async Task<Hash> FileHashCachedAsync(AbsolutePath file, CancellationToken token)
{
if (TryGetHashCache(file, out var foundHash)) return foundHash;
using var job = await _limiter.Begin($"Hasing {file.FileName}", file.Size(), token);
await using var fs = file.Open(FileMode.Open, FileAccess.Read, FileShare.Read);
var hash = await fs.HashingCopy(Stream.Null, token, job);
if (hash != default)
WriteHashCache(file, hash);
return hash;
2021-09-27 12:42:46 +00:00
}
}