1
0
mirror of https://github.com/chylex/Brotli-Builder.git synced 2024-12-21 22:42:48 +01:00
Brotli-Builder/BrotliBuilder/State/BrotliFileController.cs

266 lines
12 KiB
C#

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Threading;
using BrotliBuilder.Utils;
using BrotliLib.Brotli;
using BrotliLib.Brotli.Dictionary;
using BrotliLib.Brotli.Encode;
using BrotliLib.Brotli.Output;
using BrotliLib.Brotli.Parameters;
using BrotliLib.Markers;
using BrotliLib.Serialization;
namespace BrotliBuilder.State{
sealed class BrotliFileController{
public BrotliSerializationParameters SerializationParameters { get; set; } = BrotliSerializationParameters.Default;
public BrotliCompressionParameters CompressionParameters { get; set; } = BrotliCompressionParameters.Default;
public MarkerLevel BitMarkerLevel { get; set; } = MarkerLevel.Verbose;
public event EventHandler<StateChangedEventArgs>? StateChanged;
public BrotliFileStructure? CurrentFile => (state as BrotliFileState.Loaded)?.File;
private BrotliFileState state = new BrotliFileState.NoFile();
private readonly AsyncWorker worker;
private int currentToken;
private readonly List<BrotliFileState> replay = new List<BrotliFileState>();
public BrotliFileController(string name){
worker = new AsyncWorker(name);
}
// State handling
private void StartWorker(Action<CancellationToken, int> work){
int token = ++currentToken;
worker.Start(cancel => work((CancellationToken)cancel, token));
}
private void UpdateState(int token, BrotliFileState newState) => worker.Sync(() => {
if (token != currentToken){
Debug.WriteLine("Ignored state change due to mismatched token (" + token + " != " + currentToken + ")");
return;
}
BrotliFileState prevState = state;
state = newState;
if (state is BrotliFileState.Starting){
replay.Clear();
}
replay.Add(state);
StateChanged?.Invoke(this, new StateChangedEventArgs(prevState, newState));
});
private bool Check(CancellationToken cancel){
return cancel.IsCancellationRequested;
}
// Public triggers
public void ReplayOver(BrotliFileController target){
int token = ++target.currentToken;
foreach(var replayedState in replay){
target.UpdateState(token, replayedState);
}
}
public void ResetToNothing(){
worker.Abort();
UpdateState(++currentToken, new BrotliFileState.NoFile());
}
public void ResetToWaiting(){
worker.Abort();
UpdateState(++currentToken, new BrotliFileState.Waiting());
}
public void ResetToEmpty(){
LoadStructure(BrotliFileStructure.NewEmpty());
}
#pragma warning disable IDE0011 // Add braces
public void LoadFile(string path) => StartWorker((cancel, token) => {
UpdateState(token, new BrotliFileState.Starting());
if (!TryReadFile(token, path, out byte[] bytes) || Check(cancel)) return;
BitStream bits = new BitStream(bytes);
UpdateState(token, new BrotliFileState.HasBits(bits.ToString(), null));
if (!TryDeserialize(token, new BitStream(bytes), out BrotliFileStructure structure, out MarkerRoot markerRoot, out Stopwatch swDeserialize) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasMarkers(markerRoot));
UpdateState(token, new BrotliFileState.HasStructure(structure, swDeserialize));
if (!TryDecompress(token, structure, out BrotliOutputStored output, out Stopwatch swOutput) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasOutput(null, output.AsBytes, swOutput));
UpdateState(token, new BrotliFileState.Loaded(structure, bits, output, markerRoot));
});
public void LoadStructure(BrotliFileStructure structure, byte[]? checkAgainst = null) => StartWorker((cancel, token) => {
UpdateState(token, new BrotliFileState.Starting());
UpdateState(token, new BrotliFileState.HasStructure(structure, null));
if (!TrySerialize(token, structure, out BitStream bits, out Stopwatch swSerialization) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasBits(bits.ToString(), swSerialization));
if (!TryDeserialize(token, bits, out _, out MarkerRoot markerRoot, out _) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasMarkers(markerRoot));
if (!TryDecompress(token, structure, out BrotliOutputStored output, out Stopwatch swOutput) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasOutput(checkAgainst, output.AsBytes, swOutput));
UpdateState(token, new BrotliFileState.Loaded(structure, bits, output, markerRoot));
});
private void EncodeInternal(string path, Func<byte[], BrotliFileStructure> structureGenerator) => StartWorker((cancel, token) => {
UpdateState(token, new BrotliFileState.Starting());
if (!TryReadFile(token, path, out byte[] bytes) || Check(cancel)) return;
if (!TryEncode(token, bytes, structureGenerator, out BrotliFileStructure structure, out Stopwatch swEncode) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasStructure(structure, swEncode));
if (!TrySerialize(token, structure, out BitStream bits, out Stopwatch swSerialization) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasBits(bits.ToString(), swSerialization));
if (!TryDeserialize(token, bits, out _, out MarkerRoot markerRoot, out _) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasMarkers(markerRoot));
if (!TryDecompress(token, structure, out BrotliOutputStored output, out Stopwatch swOutput) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasOutput(bytes, output.AsBytes, swOutput));
UpdateState(token, new BrotliFileState.Loaded(structure, bits, output, markerRoot));
});
public void EncodeFile(string path, BrotliFileParameters parameters, IBrotliEncoder encoder){
EncodeInternal(path, bytes => BrotliFileStructure.FromEncoder(parameters, CompressionParameters, bytes, encoder));
}
public void EncodeFile(string path, BrotliEncodePipeline pipeline, BrotliDictionary dictionary){
EncodeInternal(path, bytes => pipeline.Apply(bytes, dictionary));
}
private void TransformInternal(BrotliFileStructure structure, IBrotliTransformer transformer) => StartWorker((cancel, token) => {
UpdateState(token, new BrotliFileState.Starting());
TryDecompress(token, structure, out BrotliOutputStored prevOutput, out _);
if (Check(cancel)) return;
if (!TryTransform(token, structure, transformer, out structure, out Stopwatch swTransform) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasStructure(structure, swTransform));
if (!TrySerialize(token, structure, out BitStream bits, out Stopwatch swSerialization) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasBits(bits.ToString(), swSerialization));
if (!TryDeserialize(token, bits, out _, out MarkerRoot markerRoot, out _) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasMarkers(markerRoot));
if (!TryDecompress(token, structure, out BrotliOutputStored output, out Stopwatch swOutput) || Check(cancel)) return;
UpdateState(token, new BrotliFileState.HasOutput(prevOutput?.AsBytes ?? Array.Empty<byte>(), output.AsBytes, swOutput));
UpdateState(token, new BrotliFileState.Loaded(structure, bits, output, markerRoot));
});
#pragma warning restore IDE0011 // Add braces
public bool Transform(IBrotliTransformer transformer){
if (state is BrotliFileState.Loaded loaded){
TransformInternal(loaded.File, transformer);
return true;
}
return false;
}
// Steps
private bool OnError(int token, ErrorType source, Exception exception){
UpdateState(token, new BrotliFileState.Error(source, exception));
return false;
}
private bool TryReadFile(int token, string path, out byte[] bytes){
try{
bytes = File.ReadAllBytes(path);
return true;
}catch(Exception ex){
bytes = null!;
return OnError(token, ErrorType.ReadingFile, ex);
}
}
private bool TryDeserialize(int token, BitStream bits, out BrotliFileStructure structure, out MarkerRoot markerRoot, out Stopwatch stopwatch){
try{
stopwatch = Stopwatch.StartNew();
(structure, markerRoot) = BrotliFileStructure.FromBytes(bits, BitMarkerLevel);
stopwatch.Stop();
return true;
}catch(Exception ex){
structure = null!;
markerRoot = null!;
stopwatch = null!;
return OnError(token, ErrorType.DeserializingFile, ex);
}
}
private bool TrySerialize(int token, BrotliFileStructure structure, out BitStream bits, out Stopwatch stopwatch){
try{
stopwatch = Stopwatch.StartNew();
bits = structure.Serialize(SerializationParameters);
stopwatch.Stop();
return true;
}catch(Exception ex){
bits = null!;
stopwatch = null!;
return OnError(token, ErrorType.SerializingStructure, ex);
}
}
private bool TryEncode(int token, byte[] bytes, Func<byte[], BrotliFileStructure> structureGenerator, out BrotliFileStructure file, out Stopwatch stopwatch){
try{
stopwatch = Stopwatch.StartNew();
file = structureGenerator(bytes);
stopwatch.Stop();
return true;
}catch(Exception ex){
file = null!;
stopwatch = null!;
return OnError(token, ErrorType.EncodingBytes, ex);
}
}
private bool TryTransform(int token, BrotliFileStructure structure, IBrotliTransformer transformer, out BrotliFileStructure transformed, out Stopwatch stopwatch){
try{
stopwatch = Stopwatch.StartNew();
transformed = structure.Transform(transformer, CompressionParameters);
stopwatch.Stop();
return true;
}catch(Exception ex){
transformed = null!;
stopwatch = null!;
return OnError(token, ErrorType.TransformingStructure, ex);
}
}
private bool TryDecompress(int token, BrotliFileStructure structure, out BrotliOutputStored output, out Stopwatch stopwatch){
try{
stopwatch = Stopwatch.StartNew();
output = structure.Decompress();
stopwatch.Stop();
return true;
}catch(Exception ex){
output = null!;
stopwatch = null!;
return OnError(token, ErrorType.DecompressingStructure, ex);
}
}
}
}