173 lines
5.5 KiB
C#
173 lines
5.5 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.IO;
|
|
using System.Threading.Tasks;
|
|
using Toolbox;
|
|
using System.Windows.Forms;
|
|
using Toolbox.Library;
|
|
using Toolbox.Library.IO;
|
|
using Toolbox.Library.Security.Cryptography;
|
|
using Newtonsoft.Json;
|
|
|
|
namespace FirstPlugin
|
|
{
|
|
public class PKG : IArchiveFile, IFileFormat
|
|
{
|
|
public FileType FileType { get; set; } = FileType.Archive;
|
|
|
|
public bool CanSave { get; set; } = true;
|
|
public string[] Description { get; set; } = new string[] { "PKG" };
|
|
public string[] Extension { get; set; } = new string[] { "*.pkg" };
|
|
public string FileName { get; set; }
|
|
public string FilePath { get; set; }
|
|
public IFileInfo IFileInfo { get; set; }
|
|
|
|
public bool CanAddFiles { get; set; } = true;
|
|
public bool CanRenameFiles { get; set; } = true;
|
|
public bool CanReplaceFiles { get; set; } = true;
|
|
public bool CanDeleteFiles { get; set; } = true;
|
|
|
|
public bool Identify(System.IO.Stream stream)
|
|
{
|
|
return FileName.EndsWith(".pkg");
|
|
}
|
|
|
|
public Type[] Types
|
|
{
|
|
get
|
|
{
|
|
List<Type> types = new List<Type>();
|
|
return types.ToArray();
|
|
}
|
|
}
|
|
|
|
public List<FileEntry> files = new List<FileEntry>();
|
|
|
|
public IEnumerable<ArchiveFileInfo> Files => files;
|
|
|
|
public void ClearFiles() { files.Clear(); }
|
|
|
|
static Dictionary<ulong, string> HashList = new Dictionary<ulong, string>();
|
|
|
|
static void CalculateHashes()
|
|
{
|
|
var mem = new MemoryStream(Properties.Resources.MetroidDread);
|
|
using (var reader = new StreamReader(mem)) {
|
|
//Thanks to UltiNaruto for the hash list
|
|
HashList = JsonConvert.DeserializeObject<Dictionary<ulong, string>>(reader.ReadToEnd());
|
|
}
|
|
}
|
|
|
|
public void Load(System.IO.Stream stream)
|
|
{
|
|
if (HashList.Count == 0)
|
|
CalculateHashes();
|
|
|
|
using (var reader = new FileReader(stream))
|
|
{
|
|
reader.SetByteOrder(false);
|
|
uint headerSize = reader.ReadUInt32();
|
|
uint fileSize = reader.ReadUInt32();
|
|
uint numFiles = reader.ReadUInt32();
|
|
|
|
for (int i = 0; i < numFiles; i++)
|
|
{
|
|
var file = new FileEntry();
|
|
ulong nameHash = reader.ReadUInt64();
|
|
uint fileStartOffset = reader.ReadUInt32();
|
|
uint fileEndOffset = reader.ReadUInt32();
|
|
|
|
uint size = fileEndOffset - fileStartOffset;
|
|
|
|
file.Hash = nameHash;
|
|
file.FileName = nameHash.ToString("X");
|
|
|
|
using (reader.TemporarySeek(fileStartOffset, SeekOrigin.Begin)) {
|
|
file.FileData = reader.ReadBytes((int)size);
|
|
}
|
|
|
|
if (HashList.ContainsKey(nameHash))
|
|
file.FileName = HashList[nameHash];
|
|
|
|
files.Add(file);
|
|
}
|
|
files = files.OrderBy(x => x.FileName).ToList();
|
|
}
|
|
}
|
|
|
|
public void Unload()
|
|
{
|
|
}
|
|
|
|
public void Save(System.IO.Stream stream)
|
|
{
|
|
using (var writer = new FileWriter(stream))
|
|
{
|
|
writer.Write(uint.MaxValue); //header size
|
|
writer.Write(uint.MaxValue); //file size
|
|
writer.Write(files.Count);
|
|
for (int i = 0; i < files.Count; i++)
|
|
{
|
|
writer.Write(files[i].Hash);
|
|
writer.Write(uint.MaxValue); //start offset
|
|
writer.Write(uint.MaxValue); //end offset
|
|
}
|
|
writer.Align(128);
|
|
|
|
writer.WriteUint32Offset(0, 4); //Size of header - 4
|
|
for (int i = 0; i < files.Count; i++)
|
|
{
|
|
writer.WriteUint32Offset(20 + (i * 16)); //start offset
|
|
writer.Write(files[i].FileData);
|
|
writer.WriteUint32Offset(24 + (i * 16)); //end offset
|
|
|
|
writer.Align(8);
|
|
}
|
|
using (writer.TemporarySeek(4, SeekOrigin.Begin)) {
|
|
writer.Write((int)writer.BaseStream.Length);
|
|
}
|
|
}
|
|
}
|
|
|
|
public bool AddFile(ArchiveFileInfo archiveFileInfo)
|
|
{
|
|
var hash = Crc64.Compute(archiveFileInfo.FileName);
|
|
|
|
files.Add(new FileEntry()
|
|
{
|
|
Hash = hash,
|
|
FileData = archiveFileInfo.FileData,
|
|
FileName = archiveFileInfo.FileName,
|
|
});
|
|
return true;
|
|
}
|
|
|
|
public bool DeleteFile(ArchiveFileInfo archiveFileInfo)
|
|
{
|
|
files.Remove((FileEntry)archiveFileInfo);
|
|
return true;
|
|
}
|
|
|
|
public class FileEntry : ArchiveFileInfo
|
|
{
|
|
public ulong Hash { get; set; }
|
|
|
|
public override string FileName
|
|
{
|
|
get => base.FileName;
|
|
set
|
|
{
|
|
if (base.FileName != value) {
|
|
//Filename changed (not empty previously)
|
|
bool isRenamed = !string.IsNullOrEmpty(base.FileName);
|
|
base.FileName = value;
|
|
if (isRenamed)
|
|
Hash = Crc64.Compute(value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|