wabbajack/Wabbajack.Common/FileExtractor.cs

173 lines
5.7 KiB
C#
Raw Normal View History

2019-09-14 04:35:42 +00:00
using System;
2019-08-25 23:52:03 +00:00
using System.Diagnostics;
using System.Linq;
2019-08-25 23:52:03 +00:00
using System.Reflection;
using System.Threading.Tasks;
2019-08-28 03:22:57 +00:00
using Alphaleonis.Win32.Filesystem;
2019-09-14 04:35:42 +00:00
using Compression.BSA;
2019-08-25 23:52:03 +00:00
using ICSharpCode.SharpZipLib.GZip;
2019-10-23 16:14:34 +00:00
using OMODFramework;
using Wabbajack.Common.CSP;
namespace Wabbajack.Common
{
public class FileExtractor
{
static FileExtractor()
{
2019-08-25 23:52:03 +00:00
ExtractResource("Wabbajack.Common.7z.dll.gz", "7z.dll");
ExtractResource("Wabbajack.Common.7z.exe.gz", "7z.exe");
2019-10-01 02:47:46 +00:00
//ExtractResource("Wabbajack.Common.innounp.exe.gz", "innounp.exe");
2019-08-25 23:52:03 +00:00
}
private static void ExtractResource(string from, string to)
{
if (File.Exists(to))
File.Delete(to);
using (var ous = File.OpenWrite(to))
using (var ins = new GZipInputStream(Assembly.GetExecutingAssembly().GetManifestResourceStream(from)))
{
ins.CopyTo(ous);
}
}
2019-11-12 05:33:32 +00:00
public static async Task ExtractAll(string source, string dest)
{
2019-09-08 22:44:15 +00:00
try
{
2019-10-11 23:31:36 +00:00
if (Consts.SupportedBSAs.Any(b => source.ToLower().EndsWith(b)))
2019-11-12 05:33:32 +00:00
await ExtractAllWithBSA(source, dest);
2019-09-24 16:03:50 +00:00
else if (source.EndsWith(".omod"))
2019-11-12 05:33:32 +00:00
await ExtractAllWithOMOD(source, dest);
2019-09-08 22:44:15 +00:00
else
2019-11-12 05:33:32 +00:00
await ExtractAllWith7Zip(source, dest);
}
2019-09-08 22:44:15 +00:00
catch (Exception ex)
{
2019-09-08 22:44:15 +00:00
Utils.Log($"Error while extracting {source}");
throw ex;
}
}
2019-11-12 05:33:32 +00:00
private static Task ExtractAllWithOMOD(string source, string dest)
2019-09-24 16:03:50 +00:00
{
2019-11-12 05:33:32 +00:00
return CSPExtensions.ThreadedTask(() =>
{
Utils.Log($"Extracting {Path.GetFileName(source)}");
var f = new Framework();
f.SetTempDirectory(dest);
var omod = new OMOD(source, ref f);
omod.ExtractDataFiles();
omod.ExtractPlugins();
return dest;
});
2019-09-24 16:03:50 +00:00
}
private static async Task ExtractAllWithBSA(string source, string dest)
{
try
{
using (var arch = await BSADispatch.OpenRead(source))
{
await arch.Files.ToChannel()
.UnorderedPipeline(
Channel.CreateSink<IFile>(),
async f =>
{
var path = f.Path;
if (f.Path.StartsWith("\\"))
path = f.Path.Substring(1);
Utils.Status($"Extracting {path}");
var out_path = Path.Combine(dest, path);
var parent = Path.GetDirectoryName(out_path);
if (!Directory.Exists(parent))
Directory.CreateDirectory(parent);
using (var fs = File.OpenWrite(out_path))
{
await f.CopyDataToAsync(fs);
}
return f;
});
}
}
catch (Exception ex)
{
Utils.Log($"While Extracting {source}");
throw ex;
}
}
2019-11-12 05:33:32 +00:00
private static async Task ExtractAllWith7Zip(string source, string dest)
{
2019-08-25 23:52:03 +00:00
Utils.Log($"Extracting {Path.GetFileName(source)}");
2019-08-25 23:52:03 +00:00
var info = new ProcessStartInfo
{
2019-08-25 23:52:03 +00:00
FileName = "7z.exe",
2019-08-29 22:49:48 +00:00
Arguments = $"x -bsp1 -y -o\"{dest}\" \"{source}\"",
2019-08-25 23:52:03 +00:00
RedirectStandardError = true,
RedirectStandardInput = true,
RedirectStandardOutput = true,
UseShellExecute = false,
2019-09-14 04:35:42 +00:00
CreateNoWindow = true
2019-08-25 23:52:03 +00:00
};
2019-08-25 23:52:03 +00:00
var p = new Process
{
2019-09-14 04:35:42 +00:00
StartInfo = info
2019-08-25 23:52:03 +00:00
};
2019-08-25 23:52:03 +00:00
p.Start();
ChildProcessTracker.AddProcess(p);
2019-08-28 03:22:57 +00:00
try
{
p.PriorityClass = ProcessPriorityClass.BelowNormal;
}
2019-09-14 04:35:42 +00:00
catch (Exception)
{
}
2019-08-29 22:49:48 +00:00
var name = Path.GetFileName(source);
try
{
while (!p.HasExited)
{
2019-11-12 05:33:32 +00:00
var line = await p.StandardOutput.ReadLineAsync();
if (line == null)
break;
2019-11-12 05:33:32 +00:00
if (line.Length <= 4 || line[3] != '%') continue;
int.TryParse(line.Substring(0, 3), out var percent);
Utils.Status($"Extracting {name} - {line.Trim()}", percent);
2019-08-29 22:49:48 +00:00
}
}
2019-10-16 21:36:14 +00:00
catch (Exception)
2019-08-29 22:49:48 +00:00
{
}
2019-08-25 23:52:03 +00:00
p.WaitForExit();
if (p.ExitCode != 0)
{
Utils.Log(p.StandardOutput.ReadToEnd());
Utils.Log($"Extraction error extracting {source}");
}
}
/// <summary>
2019-09-14 04:35:42 +00:00
/// Returns true if the given extension type can be extracted
/// </summary>
/// <param name="v"></param>
/// <returns></returns>
public static bool CanExtract(string v)
{
2019-10-11 23:31:36 +00:00
v = v.ToLower();
return Consts.SupportedArchives.Contains(v) || Consts.SupportedBSAs.Contains(v);
}
}
2019-09-14 04:35:42 +00:00
}