2020-06-02 03:41:34 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Diagnostics.CodeAnalysis;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using RocksDbSharp;
|
|
|
|
|
|
|
|
|
|
namespace Wabbajack.Common
|
|
|
|
|
{
|
2020-08-14 12:16:09 +00:00
|
|
|
|
public static class PatchCache
|
2020-06-02 03:41:34 +00:00
|
|
|
|
{
|
2020-08-14 12:16:09 +00:00
|
|
|
|
// Keep rock DB out of Utils, as it causes lock problems for users of Wabbajack.Common that aren't interested in it, otherwise
|
2020-06-02 03:41:34 +00:00
|
|
|
|
private static RocksDb? _patchCache;
|
2020-08-14 12:16:09 +00:00
|
|
|
|
|
|
|
|
|
static PatchCache()
|
2020-06-02 03:41:34 +00:00
|
|
|
|
{
|
2020-08-14 12:16:09 +00:00
|
|
|
|
var options = new DbOptions().SetCreateIfMissing(true);
|
2020-06-02 03:41:34 +00:00
|
|
|
|
_patchCache = RocksDb.Open(options, (string)Consts.LocalAppDataPath.Combine("PatchCache.rocksDb"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static byte[] PatchKey(Hash src, Hash dest)
|
|
|
|
|
{
|
|
|
|
|
var arr = new byte[16];
|
|
|
|
|
Array.Copy(BitConverter.GetBytes((ulong)src), 0, arr, 0, 8);
|
|
|
|
|
Array.Copy(BitConverter.GetBytes((ulong)dest), 0, arr, 8, 8);
|
|
|
|
|
return arr;
|
|
|
|
|
}
|
2020-08-14 12:16:09 +00:00
|
|
|
|
|
2020-06-02 03:41:34 +00:00
|
|
|
|
public static async Task CreatePatchCached(byte[] a, byte[] b, Stream output)
|
|
|
|
|
{
|
|
|
|
|
var dataA = a.xxHash();
|
|
|
|
|
var dataB = b.xxHash();
|
|
|
|
|
var key = PatchKey(dataA, dataB);
|
|
|
|
|
var found = _patchCache!.Get(key);
|
|
|
|
|
|
|
|
|
|
if (found != null)
|
|
|
|
|
{
|
|
|
|
|
await output.WriteAsync(found);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
await using var patch = new MemoryStream();
|
|
|
|
|
|
2020-08-14 12:16:09 +00:00
|
|
|
|
Utils.Status("Creating Patch");
|
2020-06-02 03:41:34 +00:00
|
|
|
|
OctoDiff.Create(a, b, patch);
|
2020-08-14 12:16:09 +00:00
|
|
|
|
|
2020-06-02 03:41:34 +00:00
|
|
|
|
_patchCache.Put(key, patch.ToArray());
|
|
|
|
|
patch.Position = 0;
|
|
|
|
|
|
|
|
|
|
await patch.CopyToAsync(output);
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-05 14:01:32 +00:00
|
|
|
|
public static async Task<long> CreatePatchCached(Stream srcStream, Hash srcHash, Stream destStream, Hash destHash,
|
2020-06-30 23:09:59 +00:00
|
|
|
|
Stream? patchOutStream = null)
|
2020-06-02 03:41:34 +00:00
|
|
|
|
{
|
|
|
|
|
var key = PatchKey(srcHash, destHash);
|
|
|
|
|
var patch = _patchCache!.Get(key);
|
|
|
|
|
if (patch != null)
|
|
|
|
|
{
|
2020-06-30 23:09:59 +00:00
|
|
|
|
if (patchOutStream == null) return patch.Length;
|
2020-08-14 12:16:09 +00:00
|
|
|
|
|
2020-06-02 03:41:34 +00:00
|
|
|
|
await patchOutStream.WriteAsync(patch);
|
2020-06-30 23:09:59 +00:00
|
|
|
|
return patch.Length;
|
2020-06-02 03:41:34 +00:00
|
|
|
|
}
|
2020-08-14 12:16:09 +00:00
|
|
|
|
|
|
|
|
|
Utils.Status("Creating Patch");
|
2020-06-02 03:41:34 +00:00
|
|
|
|
await using var sigStream = new MemoryStream();
|
|
|
|
|
await using var patchStream = new MemoryStream();
|
|
|
|
|
OctoDiff.Create(srcStream, destStream, sigStream, patchStream);
|
|
|
|
|
_patchCache.Put(key, patchStream.ToArray());
|
|
|
|
|
|
2020-06-30 23:09:59 +00:00
|
|
|
|
if (patchOutStream == null) return patchStream.Position;
|
2020-08-14 12:16:09 +00:00
|
|
|
|
|
2020-06-02 03:41:34 +00:00
|
|
|
|
patchStream.Position = 0;
|
|
|
|
|
await patchStream.CopyToAsync(patchOutStream);
|
2020-06-30 23:09:59 +00:00
|
|
|
|
|
|
|
|
|
return patchStream.Position;
|
2020-06-02 03:41:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static bool TryGetPatch(Hash foundHash, Hash fileHash, [MaybeNullWhen(false)] out byte[] ePatch)
|
|
|
|
|
{
|
|
|
|
|
var key = PatchKey(foundHash, fileHash);
|
|
|
|
|
var patch = _patchCache!.Get(key);
|
|
|
|
|
|
|
|
|
|
if (patch != null)
|
|
|
|
|
{
|
|
|
|
|
ePatch = patch;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ePatch = null;
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-30 23:09:59 +00:00
|
|
|
|
public static bool HavePatch(Hash foundHash, Hash fileHash)
|
|
|
|
|
{
|
|
|
|
|
var key = PatchKey(foundHash, fileHash);
|
|
|
|
|
return _patchCache!.Get(key) != null;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-02 03:41:34 +00:00
|
|
|
|
public static void ApplyPatch(Stream input, Func<Stream> openPatchStream, Stream output)
|
|
|
|
|
{
|
|
|
|
|
using var ps = openPatchStream();
|
|
|
|
|
using var br = new BinaryReader(ps);
|
|
|
|
|
var bytes = br.ReadBytes(8);
|
|
|
|
|
var str = Encoding.ASCII.GetString(bytes);
|
|
|
|
|
switch (str)
|
|
|
|
|
{
|
|
|
|
|
case "BSDIFF40":
|
|
|
|
|
BSDiff.Apply(input, openPatchStream, output);
|
|
|
|
|
return;
|
|
|
|
|
case "OCTODELT":
|
|
|
|
|
OctoDiff.Apply(input, openPatchStream, output);
|
|
|
|
|
return;
|
|
|
|
|
default:
|
|
|
|
|
throw new Exception($"No diff dispatch for: {str}");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-08-14 12:16:09 +00:00
|
|
|
|
|
|
|
|
|
// Convenience hook ins to offer the API from Utils, without having the init fire until they're actually called
|
|
|
|
|
public static partial class Utils
|
|
|
|
|
{
|
|
|
|
|
public static void ApplyPatch(Stream input, Func<Stream> openPatchStream, Stream output) =>
|
|
|
|
|
PatchCache.ApplyPatch(input, openPatchStream, output);
|
|
|
|
|
|
|
|
|
|
public static Task CreatePatchCached(byte[] a, byte[] b, Stream output) =>
|
|
|
|
|
PatchCache.CreatePatchCached(a, b, output);
|
|
|
|
|
|
2020-09-05 14:01:32 +00:00
|
|
|
|
public static Task<long> CreatePatchCached(Stream srcStream, Hash srcHash, Stream destStream, Hash destHash, Stream? patchOutStream = null) =>
|
2020-08-14 12:16:09 +00:00
|
|
|
|
PatchCache.CreatePatchCached(srcStream, srcHash, destStream, destHash, patchOutStream);
|
|
|
|
|
|
|
|
|
|
public static bool TryGetPatch(Hash foundHash, Hash fileHash, [MaybeNullWhen(false)] out byte[] ePatch) =>
|
|
|
|
|
PatchCache.TryGetPatch(foundHash, fileHash, out ePatch);
|
|
|
|
|
}
|
2020-06-02 03:41:34 +00:00
|
|
|
|
}
|