updating port to use xdelta, fixed line item progress not updating to 100%, styling WIP
This commit is contained in:
parent
eb27dc51a6
commit
3cb2b9cfae
@ -9,4 +9,20 @@
|
||||
<Application.Styles>
|
||||
<FluentTheme Mode="Light"/>
|
||||
</Application.Styles>
|
||||
|
||||
<Application.Resources>
|
||||
<!-- Colors -->
|
||||
<Color x:Key="AKI_DarkGray">#121212</Color>
|
||||
<Color x:Key="AKI_Yellow">#FFC107</Color>
|
||||
<Color x:Key="AKI_White">#FFFFFF</Color>
|
||||
<Color x:Key="AKI_Gray">#282828</Color>
|
||||
<Color x:Key="AKI_DarkGrayBlue">#323947</Color>
|
||||
|
||||
<!-- Brushes -->
|
||||
<SolidColorBrush x:Key="AKI_Foreground_Light" Color="{StaticResource AKI_White}"/>
|
||||
<SolidColorBrush x:Key="AKI_Background_Light" Color="{StaticResource AKI_Gray}"/>
|
||||
<SolidColorBrush x:Key="AKI_Background_Dark" Color="{StaticResource AKI_DarkGray}"/>
|
||||
<SolidColorBrush x:Key="AKI_Background_Yellow" Color="{StaticResource AKI_Yellow}"/>
|
||||
<SolidColorBrush x:Key="AKI_Background_DarkGrayBlue" Color="{StaticResource AKI_DarkGrayBlue}"/>
|
||||
</Application.Resources>
|
||||
</Application>
|
||||
|
@ -1,12 +1,19 @@
|
||||
<Styles xmlns="https://github.com/avaloniaui"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
|
||||
<Design.PreviewWith>
|
||||
<Border Padding="20">
|
||||
<!-- Add Controls for Previewer Here -->
|
||||
</Border>
|
||||
</Design.PreviewWith>
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:cc="using:PatchClient.CustomControls">
|
||||
<Design.PreviewWith>
|
||||
<Border Padding="20">
|
||||
<!-- Add Controls for Previewer Here -->
|
||||
</Border>
|
||||
</Design.PreviewWith>
|
||||
|
||||
<!-- Add Styles Here -->
|
||||
|
||||
<Style Selector="cc|TitleBar">
|
||||
<Setter Property="Background" Value="{StaticResource AKI_Background_Dark}"/>
|
||||
<Setter Property="Foreground" Value="{StaticResource AKI_Foreground_Light}"/>
|
||||
</Style>
|
||||
|
||||
<!-- Add Styles Here -->
|
||||
<Style Selector="Label.blue">
|
||||
<Setter Property="Foreground" Value="DodgerBlue"/>
|
||||
</Style>
|
||||
@ -19,7 +26,7 @@
|
||||
</Transitions>
|
||||
</Setter>
|
||||
</Style>
|
||||
|
||||
|
||||
<Style Selector="ProgressBar.Done /template/ Border">
|
||||
<Setter Property="CornerRadius" Value="0"/>
|
||||
<Setter Property="Width" Value="5"/>
|
||||
|
@ -0,0 +1,45 @@
|
||||
<UserControl xmlns="https://github.com/avaloniaui"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
|
||||
x:Class="PatchClient.CustomControls.TitleBar">
|
||||
|
||||
<Grid ColumnDefinitions="AUTO,*,AUTO">
|
||||
<Rectangle Grid.ColumnSpan="3" IsHitTestVisible="False"
|
||||
Fill="{Binding Background, RelativeSource={
|
||||
RelativeSource AncestorType=UserControl}}"
|
||||
/>
|
||||
<Label Content="{Binding Title, RelativeSource={
|
||||
RelativeSource
|
||||
AncestorType=UserControl}}"
|
||||
IsHitTestVisible="False"
|
||||
Foreground="{Binding Foreground, RelativeSource={
|
||||
RelativeSource
|
||||
AncestorType=UserControl}}"
|
||||
Background="Transparent"
|
||||
VerticalContentAlignment="Center"
|
||||
/>
|
||||
<Button Content="X" Grid.Column="2"
|
||||
Command="{Binding XButtonCommand, RelativeSource={
|
||||
RelativeSource
|
||||
AncestorType=UserControl}}"
|
||||
Background="Transparent"
|
||||
HorizontalContentAlignment="Center"
|
||||
VerticalContentAlignment="Center"
|
||||
VerticalAlignment="Stretch"
|
||||
CornerRadius="0"
|
||||
Width="35"
|
||||
>
|
||||
<Button.Styles>
|
||||
<Style Selector="Button:pointerover /template/ ContentPresenter">
|
||||
<Setter Property="Background" Value="IndianRed"/>
|
||||
</Style>
|
||||
<Style Selector="Button:pressed /template/ ContentPresenter">
|
||||
<Setter Property="Background" Value="Crimson"/>
|
||||
</Style>
|
||||
</Button.Styles>
|
||||
</Button>
|
||||
</Grid>
|
||||
|
||||
</UserControl>
|
@ -0,0 +1,58 @@
|
||||
using Avalonia;
|
||||
using Avalonia.Controls;
|
||||
using Avalonia.Markup.Xaml;
|
||||
using Avalonia.Media;
|
||||
using System.Windows.Input;
|
||||
|
||||
namespace PatchClient.CustomControls
|
||||
{
|
||||
public partial class TitleBar : UserControl
|
||||
{
|
||||
public TitleBar()
|
||||
{
|
||||
InitializeComponent();
|
||||
}
|
||||
|
||||
private void InitializeComponent()
|
||||
{
|
||||
AvaloniaXamlLoader.Load(this);
|
||||
}
|
||||
|
||||
public static readonly StyledProperty<string> TitleProperty =
|
||||
AvaloniaProperty.Register<TitleBar, string>(nameof(Title));
|
||||
|
||||
public string Title
|
||||
{
|
||||
get => GetValue(TitleProperty);
|
||||
set => SetValue(TitleProperty, value);
|
||||
}
|
||||
|
||||
public static new readonly StyledProperty<IBrush> ForegroundProperty =
|
||||
AvaloniaProperty.Register<TitleBar, IBrush>(nameof(Foreground));
|
||||
|
||||
public new IBrush Foreground
|
||||
{
|
||||
get => GetValue(ForegroundProperty);
|
||||
set => SetValue(ForegroundProperty, value);
|
||||
}
|
||||
|
||||
public static new readonly StyledProperty<IBrush> BackgroundProperty =
|
||||
AvaloniaProperty.Register<TitleBar, IBrush>(nameof(Background));
|
||||
|
||||
public new IBrush Background
|
||||
{
|
||||
get => GetValue(BackgroundProperty);
|
||||
set => SetValue(BackgroundProperty, value);
|
||||
}
|
||||
|
||||
//Close Button Command (X Button) Property
|
||||
public static readonly StyledProperty<ICommand> XButtonCommandProperty =
|
||||
AvaloniaProperty.Register<TitleBar, ICommand>(nameof(XButtonCommand));
|
||||
|
||||
public ICommand XButtonCommand
|
||||
{
|
||||
get => GetValue(XButtonCommandProperty);
|
||||
set => SetValue(XButtonCommandProperty, value);
|
||||
}
|
||||
}
|
||||
}
|
@ -13,7 +13,7 @@ namespace PatchClient.Models
|
||||
set => this.RaiseAndSetIfChanged(ref _Completed, value);
|
||||
}
|
||||
|
||||
private int total = 0;
|
||||
public int Total { get; private set; } = 0;
|
||||
|
||||
private string _Info = "";
|
||||
public string Info
|
||||
@ -29,11 +29,22 @@ namespace PatchClient.Models
|
||||
set => this.RaiseAndSetIfChanged(ref _Progress, value);
|
||||
}
|
||||
|
||||
private string _ProgressInfo = "";
|
||||
public string ProgressInfo
|
||||
{
|
||||
get => _ProgressInfo;
|
||||
set => this.RaiseAndSetIfChanged(ref _ProgressInfo, value);
|
||||
}
|
||||
|
||||
public void UpdateProgress(int RemainingCount)
|
||||
{
|
||||
if (Completed) return; //this doesn't work right ... need to look at it.
|
||||
if (Completed) return;
|
||||
|
||||
Progress = (int)Math.Floor((double)RemainingCount / total * 100);
|
||||
int processed = Total - RemainingCount;
|
||||
|
||||
Progress = (int)Math.Floor((double)processed / Total * 100);
|
||||
|
||||
ProgressInfo = $"{processed} / {Total}";
|
||||
|
||||
if (Progress == 100) Completed = true;
|
||||
}
|
||||
@ -42,9 +53,9 @@ namespace PatchClient.Models
|
||||
{
|
||||
Info = Item.ItemText;
|
||||
|
||||
total = Item.ItemValue;
|
||||
Total = Item.ItemValue;
|
||||
|
||||
Progress = (int)Math.Floor((double)Item.ItemValue / total * 100);
|
||||
Progress = (int)Math.Floor((double)Item.ItemValue / Total * 100);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -8,6 +8,10 @@
|
||||
<AvaloniaResource Include="Assets\**" />
|
||||
<AvaloniaResource Remove="Assets\Styles.axaml" />
|
||||
<None Remove=".gitignore" />
|
||||
<None Remove="Resources\xdelta3.exe" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<EmbeddedResource Include="Resources\xdelta3.exe" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
|
@ -0,0 +1,9 @@
|
||||
{
|
||||
"profiles": {
|
||||
"PatchClient": {
|
||||
"commandName": "Executable",
|
||||
"executablePath": "Z:\\SPTarkov\\Patcher\\Patcher\\_port\\Patcher\\PatchClient\\bin\\Debug\\net5.0\\PatchClient.exe",
|
||||
"workingDirectory": "C:\\Users\\JohnO\\Desktop\\12.12.2.16165"
|
||||
}
|
||||
}
|
||||
}
|
BIN
Patcher/_port/Patcher/PatchClient/Resources/xdelta3.exe
Normal file
BIN
Patcher/_port/Patcher/PatchClient/Resources/xdelta3.exe
Normal file
Binary file not shown.
@ -1,10 +1,21 @@
|
||||
using Avalonia;
|
||||
using PatchClient.Models;
|
||||
using ReactiveUI;
|
||||
using Splat;
|
||||
using System.Windows.Input;
|
||||
|
||||
namespace PatchClient.ViewModels
|
||||
{
|
||||
public class MainWindowViewModel : ViewModelBase
|
||||
{
|
||||
public ICommand CloseCommand => ReactiveCommand.Create(() =>
|
||||
{
|
||||
if(Application.Current.ApplicationLifetime is Avalonia.Controls.ApplicationLifetimes.IClassicDesktopStyleApplicationLifetime desktopApp)
|
||||
{
|
||||
desktopApp.MainWindow.Close();
|
||||
}
|
||||
});
|
||||
|
||||
public ViewNavigator navigator { get; set; } = new ViewNavigator();
|
||||
public MainWindowViewModel()
|
||||
{
|
||||
|
@ -49,9 +49,9 @@ namespace PatchClient.ViewModels
|
||||
{
|
||||
Task.Run(() =>
|
||||
{
|
||||
LineItem x = new LineItem("test 1", 100);
|
||||
LineItem x = new LineItem("test 1", 30);
|
||||
LineItem xx = new LineItem("test 2", 100);
|
||||
LineItem xxx = new LineItem("test 3", 100);
|
||||
LineItem xxx = new LineItem("test 3", 70);
|
||||
|
||||
LineItems.Add(new LineItemProgress(x));
|
||||
LineItems.Add(new LineItemProgress(xx));
|
||||
@ -65,11 +65,11 @@ namespace PatchClient.ViewModels
|
||||
|
||||
foreach(var item in LineItems)
|
||||
{
|
||||
item.UpdateProgress(i);
|
||||
item.UpdateProgress(item.Total - i);
|
||||
}
|
||||
}
|
||||
|
||||
//navigator.SelectedViewModel = new MessageViewModel("Patch completed without issues");
|
||||
navigator.SelectedViewModel = new MessageViewModel("Test Run Complete").WithDelay(400);
|
||||
});
|
||||
}
|
||||
|
||||
@ -77,40 +77,24 @@ namespace PatchClient.ViewModels
|
||||
{
|
||||
Task.Run(() =>
|
||||
{
|
||||
FilePatcher bp = new FilePatcher()
|
||||
{
|
||||
TargetBase = Environment.CurrentDirectory,
|
||||
PatchBase = LazyOperations.PatchFolder.FromCwd()
|
||||
};
|
||||
PatchHelper patcher = new PatchHelper(Environment.CurrentDirectory, null, LazyOperations.PatchFolder);
|
||||
|
||||
bp.ProgressChanged += Bp_ProgressChanged;
|
||||
patcher.ProgressChanged += patcher_ProgressChanged;
|
||||
|
||||
try
|
||||
{
|
||||
if (bp.Run())
|
||||
{
|
||||
//navigator.SelectedViewModel = new MessageViewModel("Patch completed without issues");
|
||||
}
|
||||
else
|
||||
{
|
||||
navigator.SelectedViewModel = new MessageViewModel("Failed to patch client");
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
navigator.SelectedViewModel = new MessageViewModel(ex.Message);
|
||||
}
|
||||
string message = patcher.ApplyPatches();
|
||||
|
||||
navigator.SelectedViewModel = new MessageViewModel(message).WithDelay(400);
|
||||
});
|
||||
}
|
||||
|
||||
private void Bp_ProgressChanged(object Sender, int Progress, int Total, int Percent, string Message = "", params LineItem[] AdditionalLineItems)
|
||||
private void patcher_ProgressChanged(object Sender, int Progress, int Total, int Percent, string Message = "", params LineItem[] AdditionalLineItems)
|
||||
{
|
||||
foreach (LineItem item in AdditionalLineItems)
|
||||
{
|
||||
if (item.ItemValue <= 0) continue;
|
||||
|
||||
if(initLineItemProgress)
|
||||
{
|
||||
if (item.ItemValue <= 0) continue;
|
||||
|
||||
LineItems.Add(new LineItemProgress(item));
|
||||
}
|
||||
|
@ -2,10 +2,25 @@ using ReactiveUI;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using PatchClient.Models;
|
||||
|
||||
namespace PatchClient.ViewModels
|
||||
{
|
||||
public class ViewModelBase : ReactiveObject
|
||||
{
|
||||
/// <summary>
|
||||
/// Delay the return of the viewmodel
|
||||
/// </summary>
|
||||
/// <param name="Milliseconds">The amount of time in milliseconds to delay</param>
|
||||
/// <returns>The viewmodel after the delay time</returns>
|
||||
/// <remarks>Useful to delay the navigation to another view via the <see cref="ViewNavigator"/>. For instance, to allow an animation to complete.</remarks>
|
||||
public ViewModelBase WithDelay(int Milliseconds)
|
||||
{
|
||||
System.Threading.Thread.Sleep(Milliseconds);
|
||||
|
||||
return this;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -3,19 +3,33 @@
|
||||
xmlns:vm="using:PatchClient.ViewModels"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
xmlns:cc="using:PatchClient.CustomControls"
|
||||
mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
|
||||
x:Class="PatchClient.Views.MainWindow"
|
||||
Icon="/Assets/avalonia-logo.ico"
|
||||
Title="Patch Client"
|
||||
Height="300" Width="600"
|
||||
WindowStartupLocation="CenterScreen"
|
||||
ExtendClientAreaToDecorationsHint="True"
|
||||
ExtendClientAreaChromeHints="NoChrome"
|
||||
ExtendClientAreaTitleBarHeightHint="-1"
|
||||
>
|
||||
<Window.Styles>
|
||||
<StyleInclude Source="/Assets/Styles.axaml"/>
|
||||
</Window.Styles>
|
||||
|
||||
<Design.DataContext>
|
||||
<vm:MainWindowViewModel/>
|
||||
</Design.DataContext>
|
||||
|
||||
<DockPanel LastChildFill="True">
|
||||
<ContentControl Content="{Binding navigator.SelectedViewModel}"/>
|
||||
</DockPanel>
|
||||
<Grid RowDefinitions="AUTO,*">
|
||||
|
||||
|
||||
<cc:TitleBar Title="Patch Client"
|
||||
XButtonCommand="{Binding CloseCommand}"/>
|
||||
|
||||
<DockPanel LastChildFill="True" Grid.Row="1">
|
||||
<ContentControl Content="{Binding navigator.SelectedViewModel}"/>
|
||||
</DockPanel>
|
||||
</Grid>
|
||||
</Window>
|
||||
|
@ -36,6 +36,7 @@
|
||||
<DataTemplate DataType="{x:Type model:LineItemProgress}">
|
||||
<StackPanel Orientation="Horizontal">
|
||||
<Label Content="{Binding Info}"/>
|
||||
<Label Content="{Binding ProgressInfo}"/>
|
||||
<ProgressBar Value="{Binding Progress}"
|
||||
Margin="0 0 0 4"
|
||||
VerticalAlignment="Center"
|
||||
|
@ -1,4 +1,4 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<PropertyGroup>
|
||||
<OutputType>WinExe</OutputType>
|
||||
<TargetFramework>net5.0</TargetFramework>
|
||||
@ -10,15 +10,9 @@
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<Reference Include="Aki.ByteBanger">
|
||||
<HintPath>References\Aki.ByteBanger.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="Aki.Common">
|
||||
<HintPath>References\Aki.Common.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="ComponentAce.Compression.Libs.zlib">
|
||||
<HintPath>References\ComponentAce.Compression.Libs.zlib.dll</HintPath>
|
||||
</Reference>
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
|
@ -1,185 +0,0 @@
|
||||
// NOTES:
|
||||
// - redo search pattern;
|
||||
// - compare both directories against eachother, not just one to the other
|
||||
// - add ability to handle missing directories
|
||||
|
||||
using System.IO;
|
||||
using Aki.Common.Utils;
|
||||
using Aki.ByteBanger;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
|
||||
namespace PatcherUtils
|
||||
{
|
||||
public class FileCompare
|
||||
{
|
||||
public string PatchBase;
|
||||
public string TargetBase;
|
||||
public string CompareBase;
|
||||
private int fileCount;
|
||||
private int fileIt;
|
||||
|
||||
private int diffCount = 0;
|
||||
private int newCount = 0;
|
||||
private int delCount = 0;
|
||||
private int matchCount = 0;
|
||||
|
||||
private List<FileInfo> TargetPaths;
|
||||
private List<FileInfo> ComparePaths;
|
||||
private List<LineItem> AdditionalInfo = new List<LineItem>();
|
||||
|
||||
/// <summary>
|
||||
/// Provides patch generation progress changes
|
||||
/// </summary>
|
||||
public event ProgressChangedHandler ProgressChanged;
|
||||
protected virtual void RaiseProgressChanged(int progress, int total, string Message = "", params LineItem[] AdditionalLineItems)
|
||||
{
|
||||
int percent = (int)Math.Floor((double)progress / total * 100);
|
||||
|
||||
ProgressChanged?.Invoke(this, progress, total, percent, Message, AdditionalLineItems);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Compare a target file to an assumed compareable file.
|
||||
/// </summary>
|
||||
/// <param name="targetFile">The known target path</param>
|
||||
/// <param name="assumedCompareFile">The assumed comparable file path</param>
|
||||
/// <returns>True if a comparison was made | False if a comparison could not be made</returns>
|
||||
private bool Compare(string targetFile, string assumedCompareFile)
|
||||
{
|
||||
string patchFilePath = targetFile.Replace(TargetBase, PatchBase);
|
||||
//we know our target file exists
|
||||
byte[] targetData = VFS.ReadFile(targetFile);
|
||||
|
||||
if(!File.Exists(assumedCompareFile))
|
||||
{
|
||||
//save the data we won't have in our target as new
|
||||
VFS.WriteFile($"{patchFilePath}.new", Zlib.Compress(targetData, ZlibCompression.Maximum));
|
||||
newCount++;
|
||||
return true;
|
||||
}
|
||||
|
||||
//now our compare file is known to exist
|
||||
byte[] compareData = VFS.ReadFile(assumedCompareFile);
|
||||
|
||||
// get diffs
|
||||
DiffResult result = PatchUtil.Diff(compareData, targetData);
|
||||
|
||||
switch (result.Result)
|
||||
{
|
||||
case DiffResultType.Success:
|
||||
VFS.WriteFile($"{patchFilePath}.bpf", result.PatchInfo.ToBytes());
|
||||
diffCount++;
|
||||
return true;
|
||||
|
||||
case DiffResultType.FilesMatch:
|
||||
matchCount++;
|
||||
return true;
|
||||
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Compares the base folders and generates patch files.
|
||||
/// </summary>
|
||||
/// <returns>True if patches were generated successfully | False if patch generation failed</returns>
|
||||
public bool CompareAll()
|
||||
{
|
||||
DirectoryInfo targetDirInfo = new DirectoryInfo(TargetBase);
|
||||
DirectoryInfo compareDirInfo = new DirectoryInfo(CompareBase);
|
||||
|
||||
AdditionalInfo.Add(new LineItem("Diff Patch", 0));
|
||||
AdditionalInfo.Add(new LineItem("New Patch", 0));
|
||||
AdditionalInfo.Add(new LineItem("Del Patch", 0));
|
||||
AdditionalInfo.Add(new LineItem("Files Match", 0));
|
||||
|
||||
if (!targetDirInfo.Exists || !compareDirInfo.Exists)
|
||||
{
|
||||
Console.WriteLine("Target or Compare folder does not exist");
|
||||
return false;
|
||||
}
|
||||
|
||||
//Get all the files recursively
|
||||
TargetPaths = new List<FileInfo>(targetDirInfo.GetFiles("*.*", SearchOption.AllDirectories));
|
||||
ComparePaths = new List<FileInfo>(compareDirInfo.GetFiles("*.*", SearchOption.AllDirectories));
|
||||
|
||||
RaiseProgressChanged(0, fileCount, "Generating diffs...");
|
||||
|
||||
/* Comparing Target files -> Compare files
|
||||
* - Exists = Diff (.bfd file)
|
||||
* - Doesn't Exist = New (.new file)
|
||||
*
|
||||
* Once everything has been compared from one side, any remaining paths in our ComparePaths
|
||||
* are things that don't exist in our target and can be deleted (.del file)
|
||||
*/
|
||||
|
||||
for (int x = 0; x < TargetPaths.Count; x++)
|
||||
{
|
||||
FileInfo file = TargetPaths[x];
|
||||
|
||||
string assumedComparePath = file.DirectoryName.Replace(TargetBase, CompareBase);
|
||||
|
||||
if (!Compare(file.FullName, VFS.Combine(assumedComparePath, file.Name)))
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
//remove any existing files from our ComparePaths
|
||||
FileInfo assumedFile = new FileInfo(VFS.Combine(assumedComparePath, file.Name));
|
||||
if (assumedFile.Exists && ComparePaths.Exists(x => x.FullName == assumedFile.FullName))
|
||||
{
|
||||
ComparePaths.Remove(ComparePaths.Where(x => x.FullName == assumedFile.FullName).FirstOrDefault());
|
||||
}
|
||||
|
||||
|
||||
AdditionalInfo[0].ItemValue = diffCount;
|
||||
AdditionalInfo[1].ItemValue = newCount;
|
||||
AdditionalInfo[3].ItemValue = matchCount;
|
||||
|
||||
fileIt++;
|
||||
RaiseProgressChanged(fileIt, fileCount, file.Name, AdditionalInfo.ToArray());
|
||||
}
|
||||
|
||||
|
||||
if (ComparePaths.Count == 0)
|
||||
{
|
||||
//if there are no files to delete, just return true
|
||||
return true;
|
||||
}
|
||||
|
||||
//progress reset for files that need to be deleted
|
||||
RaiseProgressChanged(0, ComparePaths.Count, "Processing .del files...");
|
||||
fileIt = 0;
|
||||
fileCount = ComparePaths.Count;
|
||||
|
||||
//the paths remaining in ComparePaths don't exist in our target and need to be removed during patching.
|
||||
foreach (FileInfo file in ComparePaths)
|
||||
{
|
||||
//add del files replace root dir with patch base
|
||||
string patchFilePath = file.FullName.Replace(CompareBase, PatchBase);
|
||||
VFS.WriteFile($"{patchFilePath}.del", new byte[0]);
|
||||
|
||||
delCount++;
|
||||
AdditionalInfo[2].ItemValue = delCount;
|
||||
|
||||
fileIt++;
|
||||
RaiseProgressChanged(fileIt, fileCount, "", AdditionalInfo.ToArray());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public FileCompare(string TargetBase, string CompareBase, string PatchBase)
|
||||
{
|
||||
this.TargetBase = TargetBase;
|
||||
this.CompareBase = CompareBase;
|
||||
this.PatchBase = PatchBase;
|
||||
|
||||
fileCount = VFS.GetFilesCount(TargetBase);
|
||||
fileIt = 0;
|
||||
}
|
||||
}
|
||||
}
|
@ -1,147 +0,0 @@
|
||||
using System;
|
||||
using System.IO;
|
||||
using Aki.Common.Utils;
|
||||
using Aki.ByteBanger;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
|
||||
namespace PatcherUtils
|
||||
{
|
||||
public class FilePatcher
|
||||
{
|
||||
public string TargetBase;
|
||||
public string PatchBase;
|
||||
private int fileCount;
|
||||
private int fileIt;
|
||||
|
||||
private int diffCount;
|
||||
private int newCount;
|
||||
private int delCount;
|
||||
|
||||
private List<LineItem> AdditionalInfo;
|
||||
|
||||
|
||||
public event ProgressChangedHandler ProgressChanged;
|
||||
|
||||
protected virtual void RaiseProgressChanged(int progress, int total, string Message = "", params LineItem[] AdditionalLineItems)
|
||||
{
|
||||
int percent = (int)Math.Floor((double)progress / total * 100);
|
||||
|
||||
ProgressChanged?.Invoke(this, progress, total, percent, Message, AdditionalLineItems);
|
||||
}
|
||||
|
||||
public bool Patch(string targetfile, string patchfile)
|
||||
{
|
||||
byte[] target = VFS.ReadFile(targetfile);
|
||||
byte[] patch = VFS.ReadFile(patchfile);
|
||||
|
||||
PatchResult result = PatchUtil.Patch(target, PatchInfo.FromBytes(patch));
|
||||
|
||||
switch (result.Result)
|
||||
{
|
||||
case PatchResultType.Success:
|
||||
VFS.WriteFile(targetfile, result.PatchedData);
|
||||
return true;
|
||||
|
||||
case PatchResultType.AlreadyPatched:
|
||||
case PatchResultType.InputChecksumMismatch:
|
||||
case PatchResultType.InputLengthMismatch:
|
||||
return true;
|
||||
|
||||
case PatchResultType.OutputChecksumMismatch:
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private bool PatchAll(string targetpath, string patchpath)
|
||||
{
|
||||
DirectoryInfo di = new DirectoryInfo(patchpath);
|
||||
|
||||
foreach (FileInfo file in di.GetFiles())
|
||||
{
|
||||
FileInfo target = null;
|
||||
|
||||
switch (file.Extension)
|
||||
{
|
||||
// patch
|
||||
case ".bpf":
|
||||
{
|
||||
target = new FileInfo(VFS.Combine(targetpath, file.Name.Replace(".bpf", "")));
|
||||
|
||||
if (!Patch(target.FullName, file.FullName))
|
||||
{
|
||||
// patch failed
|
||||
return false;
|
||||
}
|
||||
|
||||
diffCount--;
|
||||
}
|
||||
break;
|
||||
|
||||
// add new files
|
||||
case ".new":
|
||||
{
|
||||
target = new FileInfo(VFS.Combine(targetpath, file.Name.Replace(".new", "")));
|
||||
VFS.WriteFile(target.FullName, Zlib.Decompress(VFS.ReadFile(file.FullName)));
|
||||
newCount--;
|
||||
}
|
||||
break;
|
||||
|
||||
// delete old files
|
||||
case ".del":
|
||||
{
|
||||
target = new FileInfo(VFS.Combine(targetpath, file.Name.Replace(".del", "")));
|
||||
target.IsReadOnly = false;
|
||||
target.Delete();
|
||||
delCount--;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
AdditionalInfo[0].ItemValue = diffCount;
|
||||
AdditionalInfo[1].ItemValue = newCount;
|
||||
AdditionalInfo[2].ItemValue = delCount;
|
||||
|
||||
++fileIt;
|
||||
RaiseProgressChanged(fileIt, fileCount, target.Name, AdditionalInfo.ToArray());
|
||||
}
|
||||
|
||||
foreach (DirectoryInfo directory in di.GetDirectories())
|
||||
{
|
||||
PatchAll(VFS.Combine(targetpath, directory.Name), directory.FullName);
|
||||
}
|
||||
|
||||
di.Refresh();
|
||||
|
||||
if (di.GetFiles().Length == 0 && di.GetDirectories().Length == 0)
|
||||
{
|
||||
// remove empty folders
|
||||
di.Delete();
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public bool Run()
|
||||
{
|
||||
fileCount = VFS.GetFilesCount(PatchBase);
|
||||
|
||||
FileInfo[] files = new DirectoryInfo(PatchBase).GetFiles("*.*", SearchOption.AllDirectories);
|
||||
|
||||
diffCount = files.Where(x => x.Extension == ".bpf").Count();
|
||||
newCount = files.Where(x => x.Extension == ".new").Count();
|
||||
delCount = files.Where(x => x.Extension == ".del").Count();
|
||||
|
||||
AdditionalInfo = new List<LineItem>()
|
||||
{
|
||||
new LineItem("Patches Remaining", diffCount),
|
||||
new LineItem("New Files to Inflate", newCount),
|
||||
new LineItem("Files to Delete", delCount)
|
||||
};
|
||||
|
||||
fileIt = 0;
|
||||
return PatchAll(TargetBase, PatchBase);
|
||||
}
|
||||
}
|
||||
}
|
@ -12,12 +12,13 @@ namespace PatcherUtils
|
||||
/// </summary>
|
||||
public static string TempDir = "PATCHER_TEMP".FromCwd();
|
||||
|
||||
private static string SevenZExe = "7za.exe";
|
||||
|
||||
/// <summary>
|
||||
/// The folder that the patches will be stored in
|
||||
/// </summary>
|
||||
public static string PatchFolder = "Aki_Data\\Patcher";
|
||||
public static string PatchFolder = "Aki_Patches";
|
||||
|
||||
private static string SevenZExe = "7za.exe";
|
||||
|
||||
/// <summary>
|
||||
/// The path to the 7za.exe file in the <see cref="TempDir"/>
|
||||
@ -30,6 +31,13 @@ namespace PatcherUtils
|
||||
/// </summary>
|
||||
public static string PatcherClientPath = $"{TempDir}\\{PatcherClient}";
|
||||
|
||||
private static string XDelta3EXE = "xdelta3.exe";
|
||||
|
||||
/// <summary>
|
||||
/// The path to the xdelta3.exe flie in the <see cref="TempDir"/>
|
||||
/// </summary>
|
||||
public static string XDelta3Path = $"{TempDir}\\{XDelta3EXE}";
|
||||
|
||||
/// <summary>
|
||||
/// Streams embedded resources out of the assembly
|
||||
/// </summary>
|
||||
@ -73,6 +81,11 @@ namespace PatcherUtils
|
||||
StreamResourceOut(resource, PatcherClientPath);
|
||||
break;
|
||||
}
|
||||
case string a when a.EndsWith(XDelta3EXE):
|
||||
{
|
||||
StreamResourceOut(resource, XDelta3Path);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
301
Patcher/_port/Patcher/PatcherUtils/PatchHelper.cs
Normal file
301
Patcher/_port/Patcher/PatcherUtils/PatchHelper.cs
Normal file
@ -0,0 +1,301 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Security.Cryptography;
|
||||
|
||||
namespace PatcherUtils
|
||||
{
|
||||
public class PatchHelper
|
||||
{
|
||||
private string SourceFolder = "";
|
||||
private string TargetFolder = "";
|
||||
private string DeltaFolder = "";
|
||||
|
||||
private int fileCountTotal;
|
||||
private int filesProcessed;
|
||||
|
||||
private int deltaCount;
|
||||
private int newCount;
|
||||
private int delCount;
|
||||
|
||||
private List<LineItem> AdditionalInfo = new List<LineItem>();
|
||||
|
||||
public event ProgressChangedHandler ProgressChanged;
|
||||
|
||||
protected virtual void RaiseProgressChanged(int progress, int total, string Message = "", params LineItem[] AdditionalLineItems)
|
||||
{
|
||||
int percent = (int)Math.Floor((double)progress / total * 100);
|
||||
|
||||
ProgressChanged?.Invoke(this, progress, total, percent, Message, AdditionalLineItems);
|
||||
}
|
||||
|
||||
public PatchHelper(string SourceFolder, string TargetFolder, string DeltaFolder)
|
||||
{
|
||||
this.SourceFolder = SourceFolder;
|
||||
this.TargetFolder = TargetFolder;
|
||||
this.DeltaFolder = DeltaFolder;
|
||||
}
|
||||
|
||||
private string GetDeltaPath(string SourceFilePath, string SourceFolderPath, string FileExtension)
|
||||
{
|
||||
return Path.Join(DeltaFolder, $"{SourceFilePath.Replace(SourceFolderPath, "")}.{FileExtension}");
|
||||
}
|
||||
|
||||
private bool CompareFileHashes(string SourceFilePath, string TargetFilePath)
|
||||
{
|
||||
using (MD5 md5Service = MD5.Create())
|
||||
using (var sourceStream = File.OpenRead(SourceFilePath))
|
||||
using (var targetStream = File.OpenRead(TargetFilePath))
|
||||
{
|
||||
byte[] sourceHash = md5Service.ComputeHash(sourceStream);
|
||||
byte[] targetHash = md5Service.ComputeHash(targetStream);
|
||||
|
||||
return Enumerable.SequenceEqual(sourceHash, targetHash);
|
||||
}
|
||||
}
|
||||
|
||||
private void ApplyDelta(string SourceFilePath, string DeltaFilePath)
|
||||
{
|
||||
string decodedPath = SourceFilePath + ".decoded";
|
||||
|
||||
Process.Start(new ProcessStartInfo
|
||||
{
|
||||
FileName = LazyOperations.XDelta3Path,
|
||||
Arguments = $"-d -f -s \"{SourceFilePath}\" \"{DeltaFilePath}\" \"{decodedPath}\"",
|
||||
CreateNoWindow = true
|
||||
})
|
||||
.WaitForExit();
|
||||
|
||||
if(File.Exists(decodedPath))
|
||||
{
|
||||
File.Delete(SourceFilePath);
|
||||
File.Move(decodedPath, SourceFilePath);
|
||||
}
|
||||
}
|
||||
|
||||
private void CreateDelta(string SourceFilePath, string TargetFilePath)
|
||||
{
|
||||
FileInfo sourceFileInfo = new FileInfo(SourceFilePath);
|
||||
|
||||
string deltaPath = GetDeltaPath(SourceFilePath, SourceFolder, "delta");
|
||||
|
||||
Directory.CreateDirectory(deltaPath.Replace(sourceFileInfo.Name+".delta", ""));
|
||||
|
||||
//TODO - don't hardcode FileName
|
||||
|
||||
Process.Start(new ProcessStartInfo
|
||||
{
|
||||
FileName = LazyOperations.XDelta3Path,
|
||||
Arguments = $"-0 -e -f -s \"{SourceFilePath}\" \"{TargetFilePath}\" \"{deltaPath}\"",
|
||||
CreateNoWindow = true
|
||||
})
|
||||
.WaitForExit();
|
||||
}
|
||||
|
||||
private void CreateDelFile(string SourceFile)
|
||||
{
|
||||
FileInfo sourceFileInfo = new FileInfo(SourceFile);
|
||||
|
||||
string deltaPath = GetDeltaPath(SourceFile, SourceFolder, "del");
|
||||
|
||||
Directory.CreateDirectory(deltaPath.Replace(sourceFileInfo.Name+".del", ""));
|
||||
|
||||
File.Create(deltaPath);
|
||||
}
|
||||
|
||||
private void CreateNewFile(string TargetFile)
|
||||
{
|
||||
FileInfo targetSourceInfo = new FileInfo(TargetFile);
|
||||
|
||||
string deltaPath = GetDeltaPath(TargetFile, TargetFolder, "new");
|
||||
|
||||
Directory.CreateDirectory(deltaPath.Replace(targetSourceInfo.Name+".new", ""));
|
||||
|
||||
targetSourceInfo.CopyTo(deltaPath, true);
|
||||
}
|
||||
|
||||
public bool GeneratePatches()
|
||||
{
|
||||
//get all directory information needed
|
||||
DirectoryInfo sourceDir = new DirectoryInfo(SourceFolder);
|
||||
DirectoryInfo targetDir = new DirectoryInfo(TargetFolder);
|
||||
DirectoryInfo deltaDir = Directory.CreateDirectory(DeltaFolder);
|
||||
|
||||
//make sure all directories exist
|
||||
if (!sourceDir.Exists || !targetDir.Exists || !deltaDir.Exists)
|
||||
{
|
||||
//One of the directories doesn't exist
|
||||
return false;
|
||||
}
|
||||
|
||||
List<FileInfo> SourceFiles = sourceDir.GetFiles("*", SearchOption.AllDirectories).ToList();
|
||||
|
||||
fileCountTotal = SourceFiles.Count;
|
||||
|
||||
AdditionalInfo.Clear();
|
||||
AdditionalInfo.Add(new LineItem("Delta Patch", 0));
|
||||
AdditionalInfo.Add(new LineItem("New Patch", 0));
|
||||
AdditionalInfo.Add(new LineItem("Del Patch", 0));
|
||||
|
||||
filesProcessed = 0;
|
||||
|
||||
RaiseProgressChanged(0, fileCountTotal, "Generating deltas...");
|
||||
|
||||
foreach (FileInfo targetFile in targetDir.GetFiles("*", SearchOption.AllDirectories))
|
||||
{
|
||||
//find a matching source file based on the relative path of the file
|
||||
FileInfo sourceFile = SourceFiles.Find(f => f.FullName.Replace(sourceDir.FullName, "") == targetFile.FullName.Replace(targetDir.FullName, ""));
|
||||
|
||||
//if the target file doesn't exist in the source files, the target file needs to be added.
|
||||
if (sourceFile == null)
|
||||
{
|
||||
CreateNewFile(targetFile.FullName);
|
||||
|
||||
newCount++;
|
||||
filesProcessed++;
|
||||
|
||||
RaiseProgressChanged(filesProcessed, fileCountTotal, targetFile.Name, AdditionalInfo.ToArray());
|
||||
|
||||
continue;
|
||||
}
|
||||
|
||||
//if a matching source file was found, check the file hashes and get the delta.
|
||||
if(CompareFileHashes(sourceFile.FullName, targetFile.FullName))
|
||||
{
|
||||
CreateDelta(sourceFile.FullName, targetFile.FullName);
|
||||
deltaCount++;
|
||||
}
|
||||
|
||||
SourceFiles.Remove(sourceFile);
|
||||
|
||||
filesProcessed++;
|
||||
|
||||
AdditionalInfo[0].ItemValue = deltaCount;
|
||||
AdditionalInfo[1].ItemValue = newCount;
|
||||
|
||||
RaiseProgressChanged(filesProcessed, fileCountTotal, targetFile.Name, AdditionalInfo.ToArray());
|
||||
}
|
||||
|
||||
//Any remaining source files do not exist in the target folder and can be removed.
|
||||
//reset progress info
|
||||
RaiseProgressChanged(0, SourceFiles.Count, "Processing .del files...");
|
||||
filesProcessed = 0;
|
||||
fileCountTotal = SourceFiles.Count;
|
||||
|
||||
foreach (FileInfo delFile in SourceFiles)
|
||||
{
|
||||
CreateDelFile(delFile.FullName);
|
||||
|
||||
delCount++;
|
||||
|
||||
AdditionalInfo[2].ItemValue = delCount;
|
||||
|
||||
filesProcessed++;
|
||||
RaiseProgressChanged(filesProcessed, fileCountTotal, "", AdditionalInfo.ToArray());
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public string ApplyPatches()
|
||||
{
|
||||
//get needed directory information
|
||||
DirectoryInfo sourceDir = new DirectoryInfo(SourceFolder);
|
||||
DirectoryInfo deltaDir = new DirectoryInfo(DeltaFolder);
|
||||
|
||||
//check directories exist
|
||||
if (!sourceDir.Exists || !deltaDir.Exists)
|
||||
{
|
||||
return "One of the supplied directories doesn't exist";
|
||||
}
|
||||
|
||||
LazyOperations.CleanupTempDir();
|
||||
LazyOperations.PrepTempDir();
|
||||
|
||||
List<FileInfo> SourceFiles = sourceDir.GetFiles("*", SearchOption.AllDirectories).ToList();
|
||||
|
||||
List<FileInfo> deltaFiles = deltaDir.GetFiles("*", SearchOption.AllDirectories).ToList();
|
||||
|
||||
deltaCount = deltaFiles.Where(x => x.Extension == ".delta").Count();
|
||||
newCount = deltaFiles.Where(x => x.Extension == ".new").Count();
|
||||
delCount = deltaFiles.Where(x => x.Extension == ".del").Count();
|
||||
|
||||
|
||||
AdditionalInfo = new List<LineItem>()
|
||||
{
|
||||
new LineItem("Patches Remaining", deltaCount),
|
||||
new LineItem("New Files to Add", newCount),
|
||||
new LineItem("Files to Delete", delCount)
|
||||
};
|
||||
|
||||
filesProcessed = 0;
|
||||
|
||||
fileCountTotal = deltaFiles.Count;
|
||||
|
||||
foreach (FileInfo deltaFile in deltaDir.GetFiles("*", SearchOption.AllDirectories))
|
||||
{
|
||||
switch(deltaFile.Extension)
|
||||
{
|
||||
case ".delta":
|
||||
{
|
||||
//apply delta
|
||||
FileInfo sourceFile = SourceFiles.Find(f => f.FullName.Replace(sourceDir.FullName, "") == deltaFile.FullName.Replace(deltaDir.FullName, "").Replace(".delta", ""));
|
||||
|
||||
if(sourceFile == null)
|
||||
{
|
||||
return $"Failed to find matching source file for '{deltaFile.FullName}'";
|
||||
}
|
||||
|
||||
ApplyDelta(sourceFile.FullName, deltaFile.FullName);
|
||||
|
||||
deltaCount--;
|
||||
|
||||
break;
|
||||
}
|
||||
case ".new":
|
||||
{
|
||||
if(newCount == 2 || newCount == 1 || newCount == 0)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
//copy new file
|
||||
string destination = Path.Join(sourceDir.FullName, deltaFile.FullName.Replace(deltaDir.FullName, "").Replace(".new", ""));
|
||||
|
||||
File.Copy(deltaFile.FullName, destination);
|
||||
|
||||
newCount--;
|
||||
|
||||
break;
|
||||
}
|
||||
case ".del":
|
||||
{
|
||||
//remove unneeded file
|
||||
string delFilePath = Path.Join(sourceDir.FullName, deltaFile.FullName.Replace(deltaDir.FullName, "").Replace(".del", ""));
|
||||
|
||||
File.Delete(delFilePath);
|
||||
|
||||
delCount--;
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
AdditionalInfo[0].ItemValue = deltaCount;
|
||||
AdditionalInfo[1].ItemValue = newCount;
|
||||
AdditionalInfo[2].ItemValue = delCount;
|
||||
|
||||
++filesProcessed;
|
||||
RaiseProgressChanged(filesProcessed, fileCountTotal, deltaFile.Name, AdditionalInfo.ToArray());
|
||||
}
|
||||
|
||||
LazyOperations.CleanupTempDir();
|
||||
|
||||
Directory.Delete(LazyOperations.PatchFolder, true);
|
||||
|
||||
return $"Patching Complete. You can delete the patcher.exe file.";
|
||||
}
|
||||
}
|
||||
}
|
@ -7,23 +7,19 @@
|
||||
<ItemGroup>
|
||||
<None Remove="Resources\7za.exe" />
|
||||
<None Remove="Resources\PatchClient.exe" />
|
||||
<None Remove="Resources\xdelta3.exe" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<EmbeddedResource Include="Resources\7za.exe" />
|
||||
<EmbeddedResource Include="Resources\PatchClient.exe" />
|
||||
<EmbeddedResource Include="Resources\xdelta3.exe" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<Reference Include="Aki.ByteBanger">
|
||||
<HintPath>..\PatchGenerator\References\Aki.ByteBanger.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="Aki.Common">
|
||||
<HintPath>..\PatchGenerator\References\Aki.Common.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="ComponentAce.Compression.Libs.zlib">
|
||||
<HintPath>..\PatchGenerator\References\ComponentAce.Compression.Libs.zlib.dll</HintPath>
|
||||
</Reference>
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
BIN
Patcher/_port/Patcher/PatcherUtils/Resources/xdelta3.exe
Normal file
BIN
Patcher/_port/Patcher/PatcherUtils/Resources/xdelta3.exe
Normal file
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user