Files
XRLib/Assets/ADL-Plugins/zip/Plugins/lzip.cs
2025-06-23 20:06:15 +09:00

3968 lines
167 KiB
C#

using System;
using System.IO;
using System.Runtime.InteropServices;
using System.Collections;
using System.Collections.Generic;
using System.Text;
using UnityEngine;
using UnityEngine.Networking;
public class lzip {
#if !UNITY_WEBPLAYER || UNITY_EDITOR
#if (UNITY_IPHONE || UNITY_IOS || UNITY_TVOS || UNITY_WEBGL) && !UNITY_EDITOR
#if (UNITY_IPHONE || UNITY_IOS) && !UNITY_WEBGL && !UNITY_TVOS
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zsetPermissions(string filePath, string _user, string _group, string _other);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipCD(int levelOfCompression, string zipArchive, string inFilePath, string fileName, string comment, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2, int diskSize, IntPtr bprog);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipEX(string zipArchive, string outPath, IntPtr progress, IntPtr FileBuffer, int fileBufferLength, IntPtr proc, [MarshalAs(UnmanagedType.LPStr)] string password);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipEntry(string zipArchive, string arcFilename, string outpath, IntPtr FileBuffer,int fileBufferLength, IntPtr proc, [MarshalAs(UnmanagedType.LPStr)] string password);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipEntryList(string zipArchive, IntPtr outpath, IntPtr filename, int arrayLength, IntPtr FileBuffer, int fileBufferLength, IntPtr proc, [MarshalAs(UnmanagedType.LPStr)] string password);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern bool zipBuf2File(int levelOfCompression, string zipArchive, string arcFilename, IntPtr buffer, int bufferSize, string comment,[MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipDeleteFile(string zipArchive, string arcFilename, string tempArchive);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipCDList(int levelOfCompression, string zipArchive, IntPtr filename, int arrayLength, IntPtr prog, IntPtr filenameForced, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2, int diskSize, IntPtr bprog);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int gzip_File(string inFile, string outFile, int level, IntPtr progress, bool addHeader);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int ungzip_File(string inFile, string outFile, IntPtr progress);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int readTarA(string zipArchive, IntPtr total);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr readTar(string zipArchive, int size, IntPtr unc);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int createTar(string outFile, IntPtr filePath, IntPtr filename, int arrayLength, IntPtr prog, IntPtr bprog);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int extractTar(string inFile, string outDir, string entry, IntPtr prog, IntPtr bprog, bool fullPaths);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int bz2(bool decompress, int level, string inFile, string outFile, IntPtr byteProgress);
#endif
#if (UNITY_IPHONE || UNITY_IOS || UNITY_TVOS || UNITY_WEBGL)
// Send cancel signal to the following operations:
// - compress a single file
// - compress directory or list of files
// - extract a single file to the file system
// - extract a zip to the file system.
// - entry2Buffer
//
// This function is useful when the zip operation is executed in a Thread.
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
public static extern void setCancel();
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
public static extern void releaseBuffer(IntPtr buffer);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
public static extern IntPtr createBuffer(int size);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern void addToBuffer(IntPtr destination, int offset, IntPtr buffer, int len);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr zipCompressBuffer(IntPtr source, int sourceLen, int levelOfCompression, ref int v);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr zipDecompressBuffer(IntPtr source, int sourceLen, ref int v);
//gzip section
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
internal static extern int zipGzip(IntPtr source, int sourceLen, IntPtr outBuffer, int levelOfCompression, bool addHeader, bool addFooter);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
internal static extern int zipUnGzip(IntPtr source, int sourceLen, IntPtr outBuffer, int outLen, bool hasHeader, bool hasFooter);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
internal static extern int zipUnGzip2(IntPtr source, int sourceLen, IntPtr outBuffer, int outLen);
//
//inMemory section
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int freeMemStruct(IntPtr buffer);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr zipCDMem(IntPtr info, IntPtr pnt, int levelOfCompression, IntPtr source, int sourceLen, string fileName, string comment, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr initMemStruct();
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr initFileStruct();
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int freeMemZ(IntPtr pointer);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int freeFileZ(IntPtr pointer);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr zipCDMemStart(IntPtr info, IntPtr pnt, IntPtr fileStruct, IntPtr memStruct);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipCDMemAdd(IntPtr zf, int levelOfCompression, IntPtr source, int sourceLen, string fileName, string comment, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr zipCDMemClose(IntPtr zf, IntPtr memStruct, IntPtr info, int err);
//
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern uint getEntryDateTime(string zipArchive, string arcFilename, IntPtr FileBuffer,int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipGetTotalFiles(string zipArchive, IntPtr FileBuffer, int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipGetTotalEntries(string zipArchive, IntPtr FileBuffer, int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipGetInfoA(string zipArchive, IntPtr total, IntPtr FileBuffer, int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern IntPtr zipGetInfo(string zipArchive, int size, IntPtr unc, IntPtr comp, IntPtr offs, IntPtr FileBuffer, int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern bool zipEntryExists(string zipArchive, string entry, IntPtr FileBuffer, int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern bool zipValidateFile(string zip_Archive, IntPtr FileBuffer, int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern ulong zipGetEntrySize(string zipArchive, string entry, IntPtr FileBuffer, int fileBufferLength);
[DllImport("__Internal", CallingConvention = CallingConvention.Cdecl)]
private static extern int zipEntry2Buffer(string zipArchive, string entry, IntPtr buffer, int bufferSize, IntPtr FileBuffer, int fileBufferLength, [MarshalAs(UnmanagedType.LPStr)] string password);
#endif
#endif
#if UNITY_5_4_OR_NEWER
#if (UNITY_ANDROID || UNITY_STANDALONE_LINUX || UNITY_WEBGL) && !UNITY_EDITOR || UNITY_EDITOR_LINUX
private const string libname = "zipw";
#elif UNITY_EDITOR || UNITY_STANDALONE_WIN || UNITY_STANDALONE_OSX
private const string libname = "libzipw";
#endif
#else
#if (UNITY_ANDROID || UNITY_STANDALONE_LINUX || UNITY_WEBGL) && !UNITY_EDITOR
private const string libname = "zipw";
#endif
#if UNITY_EDITOR || UNITY_STANDALONE_WIN || UNITY_STANDALONE_OSX
private const string libname = "libzipw";
#endif
#endif
#if UNITY_EDITOR || UNITY_STANDALONE_WIN || UNITY_STANDALONE_OSX || UNITY_ANDROID || UNITY_STANDALONE_LINUX
#if ((!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR)
#if (UNITY_STANDALONE_OSX || UNITY_STANDALONE_LINUX || UNITY_ANDROID || UNITY_EDITOR_OSX || UNITY_EDITOR_LINUX) && !UNITY_EDITOR_WIN
[DllImport(libname, EntryPoint = "zsetPermissions"
#if (UNITY_ANDROID && !UNITY_EDITOR)
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int zsetPermissions(string filePath, string _user, string _group, string _other);
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
// CP_ACP = 0
// CP_OEMCP/UNICODE = 1
// CP_UTF8 = 65001 (default)
// CP_WINUNICODE = 1200
// Chinese = 54936 or 936
// https://docs.microsoft.com/en-us/windows/win32/intl/code-page-identifiers
[DllImport(libname, EntryPoint = "setTarEncoding"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP)
, CallingConvention = CallingConvention.Cdecl
#endif
)]
public static extern void setTarEncoding(uint encoding);
// CP_ACP = 0
// CP_OEMCP/UNICODE = 1
// CP_UTF8 = 65001 (default)
// CP_WINUNICODE = 1200
// Chinese = 54936 or 936
// In general keeping the default utf-8 encoding should be ok. But in some cases where the application that created the zip didn't used utf-8, you should set your encoding with this function.
[DllImport(libname, EntryPoint = "setEncoding"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP)
, CallingConvention = CallingConvention.Cdecl
#endif
)]
public static extern void setEncoding(uint encoding);
#endif
[DllImport(libname, EntryPoint = "zipValidateFile"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern bool zipValidateFile(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, IntPtr FileBuffer,int fileBufferLength);
[DllImport(libname, EntryPoint = "zipGetTotalFiles"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipGetTotalFiles(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, IntPtr FileBuffer, int fileBufferLength);
[DllImport(libname, EntryPoint = "zipGetTotalEntries"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipGetTotalEntries(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, IntPtr FileBuffer, int fileBufferLength);
[DllImport(libname, EntryPoint = "zipGetInfoA"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipGetInfoA(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, IntPtr total, IntPtr FileBuffer, int fileBufferLength);
[DllImport(libname, EntryPoint = "zipGetInfo"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern IntPtr zipGetInfo(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, int size, IntPtr unc, IntPtr comp, IntPtr offs, IntPtr FileBuffer, int fileBufferLength);
[DllImport(libname, EntryPoint = "releaseBuffer"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
public static extern void releaseBuffer(IntPtr buffer);
[DllImport(libname, EntryPoint = "createBuffer"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
public static extern IntPtr createBuffer(int size);
[DllImport(libname, EntryPoint = "addToBuffer"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
private static extern void addToBuffer(IntPtr destination, int offset, IntPtr buffer, int len);
[DllImport(libname, EntryPoint = "zipGetEntrySize"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern ulong zipGetEntrySize(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string entry, IntPtr FileBuffer, int fileBufferLength);
[DllImport(libname, EntryPoint = "zipEntryExists"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern bool zipEntryExists(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string entry, IntPtr FileBuffer, int fileBufferLength);
[DllImport(libname, EntryPoint = "zipCD"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipCD(
int levelOfCompression,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
# endif
string inFilePath,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string fileName,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string comment, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2, int diskSize, IntPtr bprog
);
[DllImport(libname, EntryPoint = "zipCDList"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipCDList(
int levelOfCompression,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, IntPtr filename, int arrayLength, IntPtr prog, IntPtr filenameForced, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2, int diskSize, IntPtr bprog
);
[DllImport(libname, EntryPoint = "zipBuf2File"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern bool zipBuf2File(int levelOfCompression,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
# endif
string arcFilename, IntPtr buffer, int bufferSize,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string comment,[MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2
);
[DllImport(libname, EntryPoint = "zipDeleteFile"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipDeleteFile(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string arcFilename,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string tempArchive);
[DllImport(libname, EntryPoint = "zipEntry2Buffer"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipEntry2Buffer(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string entry, IntPtr buffer, int bufferSize, IntPtr FileBuffer, int fileBufferLength, [MarshalAs(UnmanagedType.LPStr)] string password);
[DllImport(libname, EntryPoint = "zipCompressBuffer"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern IntPtr zipCompressBuffer(IntPtr source, int sourceLen, int levelOfCompression, ref int v);
[DllImport(libname, EntryPoint = "zipDecompressBuffer"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern IntPtr zipDecompressBuffer(IntPtr source, int sourceLen, ref int v);
[DllImport(libname, EntryPoint = "zipEX"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipEX(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string outPath, IntPtr progress, IntPtr FileBuffer, int fileBufferLength, IntPtr proc, [MarshalAs(UnmanagedType.LPStr)] string password);
[DllImport(libname, EntryPoint = "zipEntry"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipEntry(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string arcFilename,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string outpath, IntPtr FileBuffer,int fileBufferLength, IntPtr proc, [MarshalAs(UnmanagedType.LPStr)] string password);
[DllImport(libname, EntryPoint = "zipEntryList"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipEntryList(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
IntPtr outpath,
IntPtr filename, int arrayLength, IntPtr FileBuffer,int fileBufferLength, IntPtr proc, [MarshalAs(UnmanagedType.LPStr)] string password);
[DllImport(libname, EntryPoint = "getEntryDateTime"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern uint getEntryDateTime(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string arcFilename, IntPtr FileBuffer,int fileBufferLength);
//inMemory section
[DllImport(libname, EntryPoint = "freeMemStruct"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int freeMemStruct(IntPtr buffer);
[DllImport(libname, EntryPoint = "zipCDMem"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern IntPtr zipCDMem(IntPtr info, IntPtr pnt, int levelOfCompression, IntPtr source, int sourceLen,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string fileName,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string comment, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2
);
[DllImport(libname, EntryPoint = "initMemStruct"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern IntPtr initMemStruct();
[DllImport(libname, EntryPoint = "initFileStruct"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern IntPtr initFileStruct();
[DllImport(libname, EntryPoint = "freeMemZ"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int freeMemZ(IntPtr pointer);
[DllImport(libname, EntryPoint = "freeFileZ"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int freeFileZ(IntPtr pointer);
[DllImport(libname, EntryPoint = "zipCDMemStart"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern IntPtr zipCDMemStart(IntPtr info, IntPtr pnt, IntPtr fileStruct, IntPtr memStruct);
[DllImport(libname, EntryPoint = "zipCDMemAdd"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int zipCDMemAdd(IntPtr zf, int levelOfCompression, IntPtr source, int sourceLen,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string fileName,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string comment, [MarshalAs(UnmanagedType.LPStr)] string password, bool useBz2
);
[DllImport(libname, EntryPoint = "zipCDMemClose"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern IntPtr zipCDMemClose(IntPtr zf, IntPtr memStruct, IntPtr info, int err);
//gzip section
[DllImport(libname, EntryPoint = "zipGzip"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int zipGzip(IntPtr source, int sourceLen, IntPtr outBuffer, int levelOfCompression, bool addHeader, bool addFooter);
[DllImport(libname, EntryPoint = "zipUnGzip"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int zipUnGzip(IntPtr source, int sourceLen, IntPtr outBuffer,int outLen, bool hasHeader, bool hasFooter);
[DllImport(libname, EntryPoint = "zipUnGzip2"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int zipUnGzip2(IntPtr source, int sourceLen, IntPtr outBuffer, int outLen);
[DllImport(libname, EntryPoint = "gzip_File"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int gzip_File(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string inFile,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string outFile, int level, IntPtr progress, bool addHeader);
[DllImport(libname, EntryPoint = "ungzip_File"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
internal static extern int ungzip_File(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string inFile,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string outFile, IntPtr progress);
// Send cancel signal to the following operations:
// - compress a single file
// - compress directory or list of files
// - extract a single file to the file system
// - extract a zip to the file system.
// - entry2Buffer
// - bz2/tar/gz file operations
//
// This function is useful when the zip operation is executed in a Thread.
[DllImport(libname, EntryPoint = "setCancel"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
)]
public static extern void setCancel();
[DllImport(libname, EntryPoint = "readTarA"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Ansi
#endif
)]
internal static extern int readTarA(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, IntPtr total);
[DllImport(libname, EntryPoint = "readTar"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Ansi
#endif
)]
internal static extern IntPtr readTar(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string zipArchive, int size, IntPtr unc);
[DllImport(libname, EntryPoint = "createTar"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Ansi
#endif
)]
internal static extern int createTar(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string outFile, IntPtr filePath, IntPtr filename, int arrayLength, IntPtr prog, IntPtr bprog
);
//int extractTar(char* inFile, char* outDir)
[DllImport(libname, EntryPoint = "extractTar"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Ansi
#endif
)]
internal static extern int extractTar(
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string inFile,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string outDir,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string entry, IntPtr prog, IntPtr bprog, bool fullPaths
);
[DllImport(libname, EntryPoint = "bz2"
#if (UNITY_STANDALONE_WIN && ENABLE_IL2CPP) || UNITY_ANDROID
, CallingConvention = CallingConvention.Cdecl
#endif
#if (UNITY_EDITOR_WIN || UNITY_STANDALONE_WIN)
, CharSet = CharSet.Auto
#endif
)]
internal static extern int bz2(bool decompress, int level,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string inFile,
#if (UNITY_STANDALONE_WIN || UNITY_EDITOR_WIN)
[MarshalAs(UnmanagedType.LPWStr)]
#endif
string outFile, IntPtr byteProgress
);
#endif
#endif
// A reusable native memory pointer for downloading files.
public static IntPtr nativeBuffer = IntPtr.Zero;
public static bool nativeBufferIsBeingUsed = false;
public static int nativeOffset = 0;
// Helper function
internal static GCHandle gcA(object o) {
return GCHandle.Alloc(o, GCHandleType.Pinned);
}
// Helper function
private static bool checkObject(object o, string zipArchive, ref int len, ref IntPtr ptr) {
if (o is IntPtr) {
len = Convert.ToInt32(zipArchive);
if (len <= 0) return false;
ptr = (IntPtr)o;
} else {
inMemory t = (inMemory)o;
len = t.info[0];
ptr = t.pointer;
}
return true;
}
// Lists get filled with filenames (including path if the file is in a folder) and uncompressed file sizes
// Call getFileInfo(string zipArchive, string path) to get them filled. After that you can iterate through them to get the info you want.
public static List<string> ninfo = new List<string>();//filenames
public static List<UInt64> uinfo = new List<UInt64>();//uncompressed file sizes
public static List<UInt64> cinfo = new List<UInt64>();//compressed file sizes
// Local offset file header. (usefull when the STORE method is used to grab a file from the zip.)
// For archives with the STORE method you should add + 30 bytes + name.length extra offset.
public static List<UInt64> localOffset = new List<UInt64>();
public static int zipFiles, zipFolders;// global integers that store the number of files and folders in a zip file.
// Global value of the compressed size of all the files in a zip archive. (gets updated when calling getFileinfo).
public static ulong totalCompressedSize;
// Global value of the uncompressed size of all the files in a zip archive. (gets updated when calling getFileinfo).
public static ulong totalUncompressedSize;
// This function fills the Lists with the filenames and file sizes that are in the zip file
// Returns : the total size of uncompressed bytes of the files in the zip archive
//
// zipArchive : the full path to the archive, including the archives name. (/myPath/myArchive.zip)
// fileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
//
// ERROR CODES : 0 = Input file not found or could not get info or fileBuffer didn't input correct zip file size.
//
public static UInt64 getFileInfo(string zipArchive, object fileBuffer = null) {
ninfo.Clear(); uinfo.Clear(); cinfo.Clear(); localOffset.Clear();
zipFiles = 0; zipFolders = 0;
totalCompressedSize = 0;
totalUncompressedSize = 0;
int res = 0;
int[] tt = new int[1];
var tb = gcA(tt);
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipGetInfoA(null,tb.AddrOfPinnedObject(), fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) { tb.Free(); return 0; }
res = zipGetInfoA(null,tb.AddrOfPinnedObject(), FileBuffer, fileBufferLength);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipGetInfoA(@zipArchive, tb.AddrOfPinnedObject(),IntPtr.Zero, 0);
#endif
}
tb.Free();
if (res <= 0) return 0;
IntPtr uni = IntPtr.Zero;
UInt64[] unc = new UInt64[tt[0]];
UInt64[] comp = new UInt64[tt[0]];
UInt64[] offs = new UInt64[tt[0]];
var uncb = gcA(unc);
var compb = gcA(comp);
var offsb = gcA(offs);
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
uni = zipGetInfo(null, res, uncb.AddrOfPinnedObject(), compb.AddrOfPinnedObject(), offsb.AddrOfPinnedObject(), fbuf.AddrOfPinnedObject() ,FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) { uncb.Free(); compb.Free(); offsb.Free(); return 0; }
uni = zipGetInfo(null, res, uncb.AddrOfPinnedObject(), compb.AddrOfPinnedObject(), offsb.AddrOfPinnedObject(), FileBuffer ,fileBufferLength);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
uni = zipGetInfo(@zipArchive, res, uncb.AddrOfPinnedObject(), compb.AddrOfPinnedObject(), offsb.AddrOfPinnedObject(), IntPtr.Zero, 0);
#endif
}
if(uni == IntPtr.Zero) { uncb.Free(); compb.Free(); offsb.Free(); return 0; }
string str = Marshal.PtrToStringAuto ( uni );
StringReader r = new StringReader(str);
string line;
UInt64 sum = 0;
for(int i=0; i<tt[0]; i++) {
if( ( line = r.ReadLine() ) != null) ninfo.Add(line);
if(unc != null) {
uinfo.Add(unc[i]);
sum += unc[i];
if (unc[i] > 0) zipFiles++; else zipFolders++;
}
if (comp != null) { cinfo.Add(comp[i]); totalCompressedSize += comp[i]; }
if (offs != null) localOffset.Add(offs[i]);
}
r.Close();
r.Dispose();
uncb.Free(); compb.Free(); offsb.Free();
releaseBuffer(uni);
tt = null;
unc = null;
comp = null;
offs = null;
totalUncompressedSize = sum;
return sum;
}
// This function returns the index of an entry assuming the getFileInfo function was called prior on a zip file.
//
// entry: the entry for which we want to get the index.
// Returns -1 if no entry was found in the List.
public static int getEntryIndex(string entry) {
if(ninfo == null || ninfo.Count == 0) return -1;
int index = -1;
for(int i = 0; i < ninfo.Count; i++) {
if(entry == ninfo[i]) { index = i; break; }
}
return index;
}
// A function that returns the total number of files in a zip archive (files only, no folders).
//
// zipArchive : the zip to be checked
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
//
// ERROR CODES
// : -1 = failed to access zip archive
// : -3 = non valid zip size when using native file buffer
// : any number>0 = the number of files in the zip archive
//
public static int getTotalFiles(string zipArchive, object fileBuffer = null) {
int res = 0;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipGetTotalFiles(null, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return -3;
res = zipGetTotalFiles(null, FileBuffer, fileBufferLength);
}
return res;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipGetTotalFiles(@zipArchive, IntPtr.Zero, 0);
#endif
return res;
}
// A function that will return the total entries in a zip arcive. (files + folders)
//
// zipArchive : the zip to be checked
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
//
// ERROR CODES
// : -2 = failed to access zip archive
// : -3 = non valid zip size when using native file buffer
// : any number>0 = the number of entries in the zip archive
//
public static int getTotalEntries(string zipArchive, object fileBuffer = null) {
int res = 0;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipGetTotalEntries(null, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return -3;
res = zipGetTotalEntries(null, FileBuffer, fileBufferLength);
}
return res;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipGetTotalEntries(@zipArchive, IntPtr.Zero, 0);
#endif
return res;
}
// A function that returns the uncompressed size of a file in a zip archive.
//
// zipArchive : the zip archive to get the info from.
// entry : the entry for which we want to know it uncompressed size.
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
//
public static ulong getEntrySize(string zipArchive, string entry, object fileBuffer = null) {
ulong res = 0;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipGetEntrySize(null, entry, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return 0;
res = zipGetEntrySize(null, entry, FileBuffer, fileBufferLength);
}
return res;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipGetEntrySize(@zipArchive, entry, IntPtr.Zero, 0);
#endif
return res;
}
// A function that tells if an entry in zip archive exists.
//
// Returns true or false.
//
// zipArchive : the zip archive to get the info from.
// entry : the entry for which we want to know if it exists.
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
//
public static bool entryExists(string zipArchive, string entry, object fileBuffer = null) {
bool res = false;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipEntryExists(null, entry, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return false;
res = zipEntryExists(null, entry, FileBuffer, fileBufferLength);
}
return res;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipEntryExists(@zipArchive, entry, IntPtr.Zero, 0);
#endif
return res;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
// set permissions of a file in user, group, other.
// Each string should contain any or all chars of "rwx".
// returns 0 on success
public static int setFilePermissions(string filePath, string _user, string _group, string _other){
#if (UNITY_STANDALONE_OSX || UNITY_STANDALONE_LINUX || UNITY_ANDROID || UNITY_EDITOR_OSX || UNITY_EDITOR_LINUX || UNITY_IOS || UNITY_IPHONE) && !UNITY_EDITOR_WIN
return zsetPermissions(filePath, _user, _group, _other);
#else
return -1;
#endif
}
// A function that compresses a byte buffer and writes it to a zip file. I you set the append flag to true, the output will get appended to an existing zip archive.
//
// levelOfCompression : (0-9) recommended 9 for maximum. (0 = Store method.)
// zipArchive : the full path to the zip archive to be created or append to.
// arcFilename : the name of the file that will be written to the archive.
// buffer : the buffer that will be compressed and will be put in the zip archive.
// append : set to true if you want the output to be appended to an existing zip archive.
// comment : an optional comment for this entry.
// password : an optional password to encrypt this entry.
// useBz2 : set to true if you want bz2 compression instead of zlib. (not available for MacOS/iOS/tvOS)
//
// ERROR CODES : true = success
// : false = failed
//
public static bool buffer2File(int levelOfCompression, string zipArchive, string arcFilename, byte[] buffer, bool append = false, string comment = null, string password = null, bool useBz2 = false
) {
if (!append) { if (File.Exists(@zipArchive)) File.Delete(@zipArchive); }
var sbuf = gcA(buffer);
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 9) levelOfCompression = 9;
if(password == "") password = null;
if(comment == "") comment = null;
bool res = zipBuf2File(levelOfCompression, @zipArchive, arcFilename, sbuf.AddrOfPinnedObject(), buffer.Length, comment, password, useBz2);
sbuf.Free();
return res;
}
// A function that deletes a file in a zip archive. It creates a temp file where the compressed data of the old archive is copied except the one that needs to be deleted.
// After that the old zip archive is deleted and the temp file gets renamed to the original zip archive.
// You can delete directories too if they are empty. Does not work on password protected archives.
//
// zipArchive : the full path to the zip archive
// arcFilename : the name of the file that will be deleted.
//
// ERROR CODES : 1 = success
// : -1 = failed to open zip
// : -2 = failed to locate the archive to be deleted in the zip file
// : -3 = error copying compressed data from original zip
// : -4 = failed to create temp zip file.
//
public static int delete_entry(string zipArchive, string arcFilename) {
string tmp = @zipArchive+".tmp";
int res = zipDeleteFile(@zipArchive, @arcFilename, tmp);
if(res>0) {
File.Delete(@zipArchive);
File.Move(tmp, @zipArchive);
}else {
if(File.Exists(tmp)) File.Delete(tmp);
}
return res;
}
// A function that replaces an entry in a zip archive with a file that lies in a path. The original name of the archive will be used. Does not work on password protected archives.
//
// zipArchive : the full path to the zip archive
// arcFilename : the name of the file that will be replaced.
// newFilePath : a path to the file that will replace the original entry.
// level: : the level of compression of the new entry. (0 = Store method.)
// comment : add a comment for the file in the zip file header.
// password : set the password to protect this file.
// useBz2 : use the bz2 compression algorithm. If false the zlib deflate algorithm will be used. (not available for MacOS/iOS/tvOS)
//
// ERROR CODES
// : -1 = could not create or append
// : -2 = error during operation
// : -3 = failed to delete original entry
//
public static int replace_entry(string zipArchive, string arcFilename, string newFilePath, int level = 9, string comment = null, string password = null, bool useBz2 = false) {
int res = delete_entry(@zipArchive, @arcFilename);
if(res < 0) return -3;
if(password == "") password = null;
if(comment == "") comment = null;
return zipCD(level, @zipArchive, newFilePath, @arcFilename, comment, password, useBz2, 0, IntPtr.Zero);
}
// A function that replaces an entry in a zip archive with a buffer. The original name of the archive will be used. Does not work on password protected archives.
//
// zipArchive : the full path to the zip archive
// arcFilename : the name of the file that will be replaced.
// newFileBuffer : a byte buffer that will replace the original entry.
// level: : the level of compression of the new entry. (0 = Store method.)
// password : set the password to protect this file.
// useBz2 : use the bz2 compression algorithm. If false the zlib deflate algorithm will be used. (not available for MacOS/iOS/tvOS)
//
// ERROR CODES
// : 1 = success
// : -5 = failed to delete the original file
// : -6 = failed to append the buffer to the zip
public static int replace_entry(string zipArchive, string arcFilename, byte[] newFileBuffer, int level = 9, string password = null, bool useBz2 = false) {
int res = delete_entry(@zipArchive, @arcFilename);
if(res<0) return -5;
if( buffer2File(level, @zipArchive, @arcFilename, newFileBuffer, true, null, password, useBz2) ) return 1; else return - 6;
}
// A function that will extract only the specified file that resides in a zip archive.
//
// zipArchive : the full path to the zip archive from which we want to extract the specific file.
// arcFilename : the specific file we want to extract. (If the file resides in a directory, the directory path should be included. like dir1/dir2/myfile.bin)
// : --> on some zip files the internal dir structure uses \\ instead of / characters for directories separators. In that case use the appropriate
// : --> chars that will allow the file to be extracted.
// outpath : the full path to where the file should be extracted + the desired name for it.
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
// proc: : a single item ulong array that gets updated with the progress of the decompression in bytes.
// (100% is reached when the decompressed size of the file is reached.)
// password : if needed, the password to decrypt the entry.
//
// ERROR CODES : -1 = extraction failed
// : -2 = could not initialize zip archive.
// : -3 = could not locate entry
// : -4 = could not get entry info
// : -5 = password error
// : -6 = non valid zip size when using native file buffer
// : -7 = output directory does not exist or could not be created
// : 1 = success
//
public static int extract_entry(string zipArchive, string arcFilename, string outpath, object fileBuffer = null, ulong[] proc = null, string password = null) {
if(!Directory.Exists( Path.GetDirectoryName(outpath) ) ) return -7;
int res=-1;
if(proc == null) proc = new ulong[1];
var pbuf = gcA(proc);
#if (UNITY_IPHONE || UNITY_IOS || UNITY_STANDALONE_OSX || UNITY_ANDROID || UNITY_STANDALONE_LINUX || UNITY_EDITOR || UNITY_STANDALONE_WIN)
if(fileBuffer != null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipEntry(null, arcFilename, @outpath, fbuf.AddrOfPinnedObject(), FileBuffer.Length, pbuf.AddrOfPinnedObject(), password);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return -6;
res = zipEntry(null, arcFilename, @outpath, FileBuffer, fileBufferLength, pbuf.AddrOfPinnedObject(), password);
}
pbuf.Free();
return res;
}
#endif
res = zipEntry(@zipArchive, arcFilename, @outpath, IntPtr.Zero, 0, pbuf.AddrOfPinnedObject(), password);
pbuf.Free();
return res;
}
// A function that will extract a list of entries from a zip file.
//
// zipArchive : the full path to the zip archive from which we want to extract the specific files.
// fileList[] : a string array list of the entries we want to exctract.
// outpath : the full path to where the entries should be extracted.
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
// proc : a single item ulong array that gets updated with the progress of the decompression in bytes.
// (100% is reached when the decompressed size of the file is reached.)
// password : if needed, the password to decrypt the entry.
//
// ERROR CODES : -2 = could not initialize zip archive.
// : -3 = fileList provided is null.
// : -4 = could not get entry info
// : -5 = password error
// : -6 = non valid zip size when using native file buffer
// : -7 = output directory does not exist or could not be created
// : 1 = success
//
public static int extract_entries(string zipArchive, string[] fileList, string outpath, object fileBuffer = null, ulong[] proc = null, string password = null) {
string odir = @outpath.Replace("\\", "/");
if(outpath.Substring(outpath.Length - 1) != "/") odir += "/";
if(!Directory.Exists( outpath )) Directory.CreateDirectory(@outpath);
if(!Directory.Exists( outpath )) return -7;
if(fileList == null) return -3;
int len = fileList.Length;
string[] outFiles = new string[len];
for(int i =0; i < len; i++) outFiles[i] = odir + fileList[i];
int res=-1;
if(proc == null) proc = new ulong[1];
var pbuf = gcA(proc);
IntPtr[] fp = new IntPtr[len];
IntPtr[] op = new IntPtr[len];
for(int i = 0; i < len; i++) {
#if UNITY_2019_3_OR_NEWER
fp[i] = Marshal.StringToCoTaskMemAuto(fileList[i]);
op[i] = Marshal.StringToCoTaskMemAuto(outFiles[i]);
#else
fp[i] = Marshal.StringToHGlobalAuto(fileList[i]);
op[i] = Marshal.StringToHGlobalAuto(outFiles[i]);
#endif
}
var faBuf = gcA(fp);
var ofBuf = gcA(op);
#if (UNITY_IPHONE || UNITY_IOS || UNITY_STANDALONE_OSX || UNITY_ANDROID || UNITY_STANDALONE_LINUX || UNITY_EDITOR || UNITY_STANDALONE_WIN)
if(fileBuffer != null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipEntryList(null, ofBuf.AddrOfPinnedObject(), faBuf.AddrOfPinnedObject(), len, fbuf.AddrOfPinnedObject(), FileBuffer.Length, pbuf.AddrOfPinnedObject(), password);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return -6;
res = zipEntryList(null, ofBuf.AddrOfPinnedObject(), faBuf.AddrOfPinnedObject(), len, FileBuffer, fileBufferLength, pbuf.AddrOfPinnedObject(), password);
}
pbuf.Free();
for (int i = 0; i < len; i++) {
#if UNITY_2019_3_OR_NEWER
Marshal.FreeCoTaskMem(fp[i]);
Marshal.FreeCoTaskMem(op[i]);
#else
Marshal.FreeHGlobal(fp[i]);
Marshal.FreeHGlobal(op[i]);
#endif
}
faBuf.Free(); fp = null;
ofBuf.Free(); op = null;
return res;
}
#endif
res = zipEntryList(@zipArchive, ofBuf.AddrOfPinnedObject(), faBuf.AddrOfPinnedObject(), len, IntPtr.Zero, 0, pbuf.AddrOfPinnedObject(), password);
pbuf.Free();
for (int i = 0; i < len; i++) {
#if UNITY_2019_3_OR_NEWER
Marshal.FreeCoTaskMem(fp[i]);
Marshal.FreeCoTaskMem(op[i]);
#else
Marshal.FreeHGlobal(fp[i]);
Marshal.FreeHGlobal(op[i]);
#endif
}
faBuf.Free(); fp = null;
ofBuf.Free(); op = null;
return res;
}
// A function that decompresses a zip file. If the zip contains directories, they will be created.
//
// zipArchive : the full path to the zip archive that will be decompressed.
// : In case we are going to pass an IntPtr as a file buffer, this parameter will be used as the length of the buffer passed to the function!
// outPath : the directory in which the zip contents will be extracted. If null the same path as the zip's path will be used.
// progress : a single item integer array that gets updated with the number of files decompressed. To use it in realtime, call
// : this function in a separate thread. Usefull for broad progress. Otherwise use the proc variable below.
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, then the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
// proc: : a single item ulong array that gets updated with the progress of the decompression in bytes.
// (100% is reached when the decompressed size of the files is reached.)
// password : if needed, the password to decrypt the archive.
//
// ERROR CODES
// : -1 = could not initialize zip archive.
// : -2 = failed extraction
// : -6 = IntPtr buffer had no valid buffer length parameter passed.
// : 1 = success
//
public static int decompress_File (string zipArchive, string outPath = null, int[] progress = null, object fileBuffer = null, ulong[] proc = null, string password = null) {
// if outPath == null use the same path as the zips's path.
if(outPath == null) outPath = Path.GetDirectoryName(zipArchive);
// make a check if the last '/' exists at the end of the exctractionPath and add it if it is missing
if (outPath.Substring(outPath.Length - 1, 1) != "/") { outPath += "/"; }
int res;
var ibuf = gcA(progress);
if(proc == null) proc = new ulong[1];
var pbuf = gcA(proc);
#if (UNITY_IPHONE || UNITY_IOS || UNITY_STANDALONE_OSX || UNITY_ANDROID || UNITY_STANDALONE_LINUX || UNITY_EDITOR || UNITY_STANDALONE_WIN)
if(fileBuffer != null) {
// Managed buffer
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipEX(null, @outPath, ibuf.AddrOfPinnedObject(), fbuf.AddrOfPinnedObject(), FileBuffer.Length, pbuf.AddrOfPinnedObject(), password);
fbuf.Free(); ibuf.Free(); pbuf.Free(); return res;
}
// When an IntPtr is used, then the zipArchive string parameter will be used to pass the Length of the IntPtr buffer!
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) { ibuf.Free(); pbuf.Free(); return -6; }
res = zipEX(null, @outPath, ibuf.AddrOfPinnedObject(), FileBuffer, fileBufferLength, pbuf.AddrOfPinnedObject(), password);
ibuf.Free(); pbuf.Free(); return res;
}
}
#endif
res = zipEX(@zipArchive, @outPath, ibuf.AddrOfPinnedObject(), IntPtr.Zero, 0, pbuf.AddrOfPinnedObject(), password);
ibuf.Free(); pbuf.Free();
return res;
}
// A function that compresses a file to a zip file. If the flag append is set to true then it will get appended to an existing zip file.
// (This function is slow when appending many files. Use compress_File_List instead.)
//
// levelOfCompression : (0-9) recommended 9 for maximum (0 = Store method.)
// zipArchive : the full path to the zip archive that will be created
// inFilePath : the full path to the file that should be compressed and added to the zip file.
// append : set to true if you want the input file to get appended to an existing zip file. (if the zip file does not exist it will be created.)
// filename : if you want the name of your file to be different then the one it has, set it here. If you add a folder structure to it,
// like (dir1/dir2/myfile.bin) the directories will be created in the zip file.
// comment : add a comment for the file in the zip file header.
// password : set the password to protect this file.
// useBz2 : use the bz2 compression algorithm. If false the zlib deflate algorithm will be used. (not available for MacOS/iOS/tvOS)
// disksize : if a disksize is used > 0 then the zip archive will be split to the assigned disksize (in bytes).
// byteProgress : this variable is a single ulong array that keeps track of all the uncompressed bytes that have been processed. (set to 0 after finish).
// : To get a progress of the compression, store the sum of the files that will get zipped and compare it to byteProgress.
// ERROR CODES
// : 1 = success
// : -1 = could not create or append
// : -2 = error during operation
//
public static int compress_File(int levelOfCompression, string zipArchive, string inFilePath, bool append = false, string fileName = "", string comment = null, string password = null, bool useBz2 = false, int diskSize = 0, ulong[] byteProgress = null) {
if (!File.Exists(@inFilePath)) return -10;
if (!append) { if (File.Exists(@zipArchive)) File.Delete(@zipArchive); }
if (fileName == null || fileName == "") fileName = Path.GetFileName(@inFilePath);
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 9) levelOfCompression = 9;
if (password == "") password = null;
if (comment == "") comment = null;
int res = 0;
if(byteProgress == null) res = zipCD(levelOfCompression, @zipArchive, @inFilePath, fileName, comment, password, useBz2, diskSize, IntPtr.Zero);
else {
var prog = gcA(byteProgress);
res = zipCD(levelOfCompression, @zipArchive, @inFilePath, fileName, comment, password, useBz2, diskSize, prog.AddrOfPinnedObject());
prog.Free();
}
return res;
}
// A function that compresses a list of files to a zip file. Use this function to compress multiple files fast instead of appending to existing files with the compress_File function.
//
// levelOfCompression : (0-9) recommended 9 for maximum (0 = Store method.)
// zipArchive : the full path to the zip archive that will be created.
// inFilePath[] : an array of the full paths to the files that should be compressed and added to the zip file.
// progress : this var will increment until the number of the input files and this are equal.
// append : set to true if you want the input files to get appended to an existing zip file. (if the zip file does not exist it will be created.)
// filename[] : if you want the names of your files to be different then the one they have, set it here. If you add a folder structure to it,
// like (dir1/dir2/myfile.bin) the directories will be created in the zip file.
// password : set the password to protect this file.
// useBz2 : use the bz2 compression algorithm. If false the zlib deflate algorithm will be used. (not available for MacOS/iOS/tvOS)
// diskSize : if a disksize is used > 0 then the zip archive will be split to the assigned disksize (in bytes).
// byteProgress : this variable is a single ulong array that keeps track of all the uncompressed bytes that have been processed. (set to 0 after finish).
// : To get a progress of the compression, store the sum of the files that will get zipped and compare it to byteProgress.
// ERROR CODES
// : 1 = success
// : -1 = could not create or append
// : -2 = error during operation
// : -3 = file list is null
// : -4 = custom file names length does not match the original file names length
// : -10 = some or all input files are missing
public static int compress_File_List(int levelOfCompression, string zipArchive, string[] inFilePath, int[] progress = null, bool append = false, string[] fileName = null, string password = null, bool useBz2 = false, int diskSize = 0, ulong[] byteProgress = null) {
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 9) levelOfCompression = 9;
if(password == "") password = null;
if (!append) { if (File.Exists(@zipArchive)) File.Delete(@zipArchive); }
if (inFilePath == null) return -3;
if(fileName !=null && fileName.Length != inFilePath.Length) return -4;
for(int i=0; i<inFilePath.Length; i++) {
if (!File.Exists(inFilePath[i])) return -10;
}
IntPtr[] fp = new IntPtr[inFilePath.Length];
IntPtr[] np = new IntPtr[inFilePath.Length];
int res = 0;
fillPointers(zipArchive, fileName, inFilePath, ref fp, ref np);
if(byteProgress == null) byteProgress = new ulong[1];
if(progress == null) progress = new int[1];
var faBuf = gcA(fp);
var naBuf = gcA(np);
var prog = gcA(progress);
var bProg = gcA(byteProgress);
res = zipCDList(levelOfCompression, @zipArchive, faBuf.AddrOfPinnedObject(), inFilePath.Length, prog.AddrOfPinnedObject(), naBuf.AddrOfPinnedObject(), password, useBz2, diskSize, bProg.AddrOfPinnedObject());
for (int i = 0; i < inFilePath.Length; i++) {
#if UNITY_2019_3_OR_NEWER
Marshal.FreeCoTaskMem(fp[i]);
Marshal.FreeCoTaskMem(np[i]);
#else
Marshal.FreeHGlobal(fp[i]);
Marshal.FreeHGlobal(np[i]);
#endif
}
faBuf.Free(); fp = null;
naBuf.Free(); np = null;
prog.Free(); bProg.Free();
return res;
}
// Compress a directory with all its files and subfolders to a zip file. This function is way faster when adding manually multiple files to a zip with the compress_File function.
//
// sourceDir : the directory you want to compress
// levelOfCompression : the level of compression (0-9). (0 = Store method.)
// zipArchive : the full path+name to the zip file to be created. If null the directory + ".zip" will be used.
// includeRoot : set to true if you want the root folder of the directory to be included in the zip archive. Otherwise leave it to false.
// progress : provide a single item integer array to write the current index of the files getting compressed.
// password : set the password to protect this file.
// useBz2 : use the bz2 compression algorithm. If false the zlib deflate algorithm will be used. (not available for MacOS/iOS/tvOS)
// disksize : if a disksize is used > 0 then the zip archive will be split to the assigned disksize (in bytes).
// append : set to true if you want the input files to get appended to an existing zip file. (if the zip file does not exist it will be created.)
// byteProgress : this variable is a single ulong array that keeps track of all the uncompressed bytes that have been processed. (set to 0 after finish).
// : To get a progress of the compression, store the sum of the files that will get zipped and compare it to byteProgress.
//
// If you want to get the progress of compression, call the getAllFiles function to get the total number of files
// in a directory and its subdirectories. The compressDir when called from a separate thread will update the progress[0] provided parameter.
// Divide this with the total number of files (as floats) and you have the % of the procedure.
//
// ERROR CODES
// : 1 = success
// : -1 = could not create or append
// : -2 = error during operation
//
public static int compressDir(string sourceDir, int levelOfCompression, string zipArchive = null, bool includeRoot = false, int[] progress = null, string password = null, bool useBz2 = false, int diskSize = 0, bool append = false, ulong[] byteProgress = null) {
if(!Directory.Exists(sourceDir)) return 0;
string fdir = @sourceDir.Replace("\\", "/");
if(sourceDir.Substring(sourceDir.Length - 1) != "/") fdir += "/";
if(zipArchive == null) zipArchive = sourceDir.Substring(0, sourceDir.Length) + ".zip";
int totalFiles = getAllFiles(fdir);
if (totalFiles == 0) return 0;
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 9) levelOfCompression = 9;
int res = 0;
if(Directory.Exists(fdir)) {
List<string> inFilePath = new List<string>();
List<string> fileName = new List<string>();
fillLists(fdir, includeRoot, ref inFilePath, ref fileName);
res = compress_File_List(levelOfCompression, zipArchive, inFilePath.ToArray(), progress, append, fileName.ToArray(), password, useBz2, diskSize, byteProgress);
inFilePath.Clear(); inFilePath = null;
fileName.Clear(); fileName = null;
}
return res;
}
private static void fillPointers(string outFile, string[] fileName, string[] inFilePath, ref IntPtr[] fp, ref IntPtr[] np) {
string[] fna = null;
string path = Path.GetDirectoryName(@outFile);
if(fileName == null) {
fna = new string[inFilePath.Length];
for(int i = 0; i < inFilePath.Length; i++) { fna[i] = inFilePath[i].Replace(path, ""); }
} else {
fna = fileName;
}
for(int i = 0; i < inFilePath.Length; i++) { if(fna[i] == null) fna[i] = inFilePath[i].Replace(path, ""); }
for(int i = 0; i < inFilePath.Length; i++) {
inFilePath[i] = inFilePath[i].Replace("\\", "/");
fna[i] = fna[i].Replace("\\", "/");
#if UNITY_2019_3_OR_NEWER
fp[i] = Marshal.StringToCoTaskMemAuto(inFilePath[i]);
np[i] = Marshal.StringToCoTaskMemAuto(fna[i]);
#else
fp[i] = Marshal.StringToHGlobalAuto(inFilePath[i]);
np[i] = Marshal.StringToHGlobalAuto(fna[i]);
#endif
}
path = null;
}
private static void fillLists(string fdir, bool includeRoot, ref List<string> inFilePath, ref List<string> fileName) {
string[] ss = fdir.Split('/');
string rdir = ss[ss.Length - 1];
string root = rdir;
if(ss.Length > 1 && includeRoot) root = rdir = ss[ss.Length - 2] + "/";
foreach (string f in Directory.GetFiles(fdir, "*", SearchOption.AllDirectories)) {
string s = f.Replace(fdir, rdir).Replace("\\", "/").Replace("//", "/");
if (!includeRoot) {
s = s.Substring(root.Length);
if(s.Substring(0, 1) == "/") s = s.Substring(1, s.Length-1);
}
inFilePath.Add(f);
fileName.Add(s);
}
}
// Use this function to get the total files of a directory and its subdirectories.
public static int getAllFiles(string dir) {
string[] filePaths = Directory.GetFiles(@dir, "*", SearchOption.AllDirectories);
int res = filePaths.Length;
filePaths = null;
return res;
}
// Use this function to get the size of a file in the file system.
public static long getFileSize(string file) {
FileInfo fi = new FileInfo(file);
if(fi.Exists) return fi.Length; else return 0;
}
// Use this function to get the size of the files in a directory.
public static ulong getDirSize(string dir) {
string[] filePaths = Directory.GetFiles(@dir, "*", SearchOption.AllDirectories);
ulong size = 0;
for(int i = 0; i < filePaths.Length; i++) {
FileInfo fi = new FileInfo(filePaths[i]);
if(fi.Exists) size += (ulong)fi.Length;
}
return size;
}
//---------------------------------------------------------------------------------------------------------------------------
// TAR SECTION
//---------------------------------------------------------------------------------------------------------------------------
// Untar a .tar archive
//
// inFile : the full path to the tar archive.
// outPath : the path where the extraction will take place. If null, the same path as the one of the inFile will be used.
// progress : a single item integer array that will get updated with the number of the entries that get extracted. Use in a Thread for real time report.
// byteProgress : a single item ulong array that will get updated with the bytes that have been extracted so far. Use in a Thread for real time report.
//
// Error codes : -1 could not find input file.
// : -3 could not write output file.
// : -8 canceled
// : 1 Success
public static int tarExtract(string inFile, string outPath = null, int[] progress = null, ulong[] byteProgress = null) {
if(outPath == null) outPath = Path.GetDirectoryName(inFile);
if (outPath.Substring(outPath.Length - 1, 1) != "/") { outPath += "/"; }
var prog = gcA(progress);
var bProg = gcA(byteProgress);
int res = 0;
res = extractTar(inFile, outPath, null, prog.AddrOfPinnedObject(), bProg.AddrOfPinnedObject(), true);
prog.Free(); bProg.Free();
return res;
}
// Extract an entry from a tar archive
//
// inFile : the full path to our tar archive
// entry : the entry we want to extract. (If the file resides in a directory, the directory path should be included. like dir1/dir2/myfile.bin)
// outPath : the path in which want to extract our entry. If null the same path as the inFile will be used.
// fullPaths : if the entry resides in a directory, use this flag to create the directory structure or not.
// ! If it is set to false, you can use an absolute path in the outPath parameter to extract with a different filename !
// byteProgress : a sigle item ulong array that will get updated with the bytes of the extraction progress. Use in a Thread for real time report.
//
// Error codes : -1 could not find input file.
// : -3 could not write output file.
// : -5 could not find entry
// : -8 canceled
// : 1 Success
public static int tarExtractEntry(string inFile, string entry, string outPath = null, bool fullPaths = true, ulong[] byteProgress = null) {
if(outPath == null) outPath = Path.GetDirectoryName(inFile);
if (fullPaths && outPath.Substring(outPath.Length - 1, 1) != "/") { outPath += "/"; }
if(fullPaths && File.Exists(outPath)) { Debug.Log("There is a file with the same name in the path!"); return -7; }
if(!fullPaths && Directory.Exists(outPath)) { Debug.Log("There is a directory with the same name in the path!"); return -8; }
var bProg = gcA(byteProgress);
int res = 0;
res = extractTar(inFile, outPath, entry, IntPtr.Zero, bProg.AddrOfPinnedObject(), fullPaths);
bProg.Free();
return res;
}
// GetOrCreate a tar file out of a directory containing files.
//
// sourceDir : the directory that contains our files
// outFile : the full path to the tar archive that will be created. If null the same name as the sourceDir will be used.
// includeRoot : if the root directory should be included in the filenames.
// progress : a single item integer array that will increment with each file added to the tar archive. Use in a Thread for real time report.
// byteProgress : a sigle item ulong array that will get updated with the bytes that get added to the tar archive. Call the function in a thread to get real time progress.
//
// Error codes : -1 could not write output file.
// : -3 could not find input file.
// : -8 canceled
// : 1 Success
// : 0 no files found in Dir.
public static int tarDir(string sourceDir, string outFile = null, bool includeRoot = false, int[] progress = null, ulong[] byteProgress = null) {
if(!Directory.Exists(sourceDir)) return 0;
string fdir = @sourceDir.Replace("\\", "/");
if(sourceDir.Substring(sourceDir.Length - 1) != "/") fdir += "/";
if(outFile == null) outFile = sourceDir.Substring(0,sourceDir.Length - 1) + ".tar";
int totalFiles = getAllFiles(fdir);
if (totalFiles == 0) return 0;
int res = 0;
if(Directory.Exists(fdir)) {
List<string> inFilePath = new List<string>();
List<string> fileName = new List<string>();
fillLists(fdir, includeRoot, ref inFilePath, ref fileName);
res = tarList(outFile, inFilePath.ToArray(), fileName.ToArray(), progress, byteProgress);
inFilePath.Clear(); inFilePath = null;
fileName.Clear(); fileName = null;
}
return res;
}
// This function creates a tar archive from a list of file paths provided.
//
// outFile : the full path to the tar archive that will be created.
// inFilePath[] : an array of the full paths to the files that should be added to the tar archive.
// fileName[] : if you want the names of your files to be different then the one they have, set it here.
// progress : this var will increment until the number of the input files and this are equal. Use in a Thread for real time report.
// byteProgress : this variable is a single ulong array that keeps track of all the bytes that have been processed. Use in a Thread for real time report.
//
//
// Error codes : -1 could not write output file.
// : -3 could not find input file.
// : -4 input files number different than filenames number
// : -8 canceled
// : -10 an input file was not found.
// : 1 Success
public static int tarList(string outFile, string[] inFilePath, string[] fileName = null, int[] progress = null, ulong[] byteProgress = null) {
if (inFilePath == null) return -3;
if(fileName !=null && fileName.Length != inFilePath.Length) return -4;
for(int i=0; i<inFilePath.Length; i++) {
if (!File.Exists(inFilePath[i])) return -10;
}
if (File.Exists(@outFile)) File.Delete(@outFile);
int res = 0;
IntPtr[] fp = new IntPtr[inFilePath.Length];
IntPtr[] np = new IntPtr[inFilePath.Length];
fillPointers(outFile, fileName, inFilePath, ref fp, ref np);
var faBuf = gcA(fp);
var naBuf = gcA(np);
var prog = gcA(progress);
var bProg = gcA(byteProgress);
res = createTar( @outFile, faBuf.AddrOfPinnedObject(), naBuf.AddrOfPinnedObject(),inFilePath.Length, prog.AddrOfPinnedObject(), bProg.AddrOfPinnedObject());
for (int i = 0; i < inFilePath.Length; i++) {
#if UNITY_2019_3_OR_NEWER
Marshal.FreeCoTaskMem(fp[i]);
Marshal.FreeCoTaskMem(np[i]);
#else
Marshal.FreeHGlobal(fp[i]);
Marshal.FreeHGlobal(np[i]);
#endif
}
faBuf.Free(); fp = null;
naBuf.Free(); np = null;
prog.Free(); bProg.Free();
return res;
}
// This function fills the same Lists as the getFileInfo for zip, with the filenames and file sizes that are in the tar file.
// Returns : the total size of uncompressed bytes of the files in the tar archive
//
// tarArchive : the full path to the archive, including the archives name. (/myPath/myArchive.tar)
//
// ERROR CODES : 0 = Input file not found or Could not get info.
//
public static UInt64 getTarInfo(string tarArchive) {
ninfo.Clear(); uinfo.Clear(); cinfo.Clear(); localOffset.Clear();
zipFiles = 0; zipFolders = 0;
totalCompressedSize = 0;
totalUncompressedSize = 0;
int res = 0;
int[] tt = new int[1];
var tb = gcA(tt);
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = readTarA(@tarArchive, tb.AddrOfPinnedObject());
#endif
tb.Free();
if (res <= 0) return 0;
IntPtr uni = IntPtr.Zero;
UInt64[] unc = new UInt64[tt[0]];
var uncb = gcA(unc);
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
uni = readTar(@tarArchive, res, uncb.AddrOfPinnedObject());
#endif
if(uni == IntPtr.Zero) {uncb.Free(); return 0; }
string str = Marshal.PtrToStringAuto ( uni );
StringReader r = new StringReader(str);
string line;
UInt64 sum = 0;
for(int i=0; i<tt[0]; i++) {
if( ( line = r.ReadLine() ) != null) ninfo.Add(line);
if(unc != null) {
uinfo.Add(unc[i]);
sum += unc[i];
if (unc[i] > 0) zipFiles++; else zipFolders++;
}
}
r.Close();
r.Dispose();
uncb.Free();
releaseBuffer(uni);
tt = null;
unc = null;
totalUncompressedSize = sum;
return sum;
}
//---------------------------------------------------------------------------------------------------------------------------
// END TAR SECTION
//---------------------------------------------------------------------------------------------------------------------------
#endif
// get the DateTime of an entry in a zip archive
//
// zipArchive : the full path to the zip archive from which we want to extract the specific file.
// entry : the specific entry we want to get the DateTime of. (If the entry resides in a directory, the directory path should be included. like dir1/dir2/myfile.bin)
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the zipArchive path.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, then the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
//
// Returns the date and time of the entry in DateTime format.
//
public static DateTime entryDateTime(string zipArchive, string entry, object fileBuffer = null) {
uint dosDateTime = 0;
if (fileBuffer != null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
dosDateTime = getEntryDateTime(null, entry, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) { Debug.Log("Error in getting DateTime: " + dosDateTime); return DateTime.Now; }
dosDateTime = getEntryDateTime(null, entry, FileBuffer, fileBufferLength);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
dosDateTime = getEntryDateTime(zipArchive, entry, IntPtr.Zero, 0);
#endif
}
var date = (dosDateTime & 0xFFFF0000) >> 16;
var time = (dosDateTime & 0x0000FFFF);
var year = (date >> 9) + 1980;
var month = (date & 0x01e0) >> 5;
var day = date & 0x1F;
var hour = time >> 11;
var minute = (time & 0x07e0) >> 5;
var second = (time & 0x1F) * 2;
if (dosDateTime == 0 || dosDateTime == 1 || dosDateTime == 2) { Debug.Log("Error in getting DateTime: " + dosDateTime); return DateTime.Now; }
return new DateTime((int)year, (int)month, (int)day, (int)hour, (int)minute, (int)second);
}
//---------------------------------------------------------------------------------------------------------------------------
// In Memory Functions
//---------------------------------------------------------------------------------------------------------------------------
// The inMemory class the holds the pointer to our in memory zip archive.
// Use the size() function to get the size of the memory it occupies.
// The getZipBuffer() function returns a new byte[] buffer that contains the inmemory zip.
//
public class inMemory
{
public IntPtr pointer = IntPtr.Zero;
public IntPtr zf = IntPtr.Zero;
public IntPtr memStruct = IntPtr.Zero;
public IntPtr fileStruct = IntPtr.Zero;
public int[] info = new int[3];
public int lastResult = 0;
// A bool that tells if an inMemory zip archive is open or closed. (Used with the low level functions only.)
public bool isClosed = true;
public int size() { return info[0]; }
public IntPtr memoryPointer() { return pointer; }
public byte[] getZipBuffer() {
if (pointer != IntPtr.Zero && info[0] > 0 ) {
byte [] p = new byte[ info[0] ];
Marshal.Copy(pointer, p, 0, info[0]);
return p;
} else {
return null;
}
}
}
// Use this function to free the pointer and the object of the inMemory zip archive.
// It is important to call this function after you don't need the in memory zip any more!
//
public static void free_inmemory(inMemory t) {
if(t.info == null) { Debug.Log("inMemory object is null"); return; }
if(freeMemStruct(t.pointer) != 1) Debug.Log("In memory pointer was not freed");
t.info = null;
if(t.memStruct != IntPtr.Zero && freeMemZ(t.memStruct) != 1) Debug.Log("MemStruct was not freed");
if(t.fileStruct != IntPtr.Zero && freeFileZ(t.fileStruct) != 1) Debug.Log("FileStruct was not freed");
t = null;
}
// Low level inMemory functions ----------------------------------------------------------------------------------------------------
// A function that creates an inMemory zip archive
//
// t : the inMemory class that holds the pointer to our inMemory zip file. You can call this function again for more buffers with the same
// : inMemory object and the next buffers will get appended to the in memory zip.
//
// Returns true on success.
public static bool inMemoryZipStart(inMemory t) {
if(t.info == null) { Debug.Log("inMemory object is null"); return false; }
if(t.fileStruct == IntPtr.Zero) t.fileStruct = initFileStruct();
if(t.memStruct == IntPtr.Zero) t.memStruct = initMemStruct();
if(!t.isClosed) inMemoryZipClose(t);
var inf = gcA(t.info);
t.zf = zipCDMemStart(inf.AddrOfPinnedObject(), t.pointer, t.fileStruct, t.memStruct);
inf.Free();
t.isClosed = false;
if(t.zf != IntPtr.Zero) return true; else return false;
}
// A function that adds a buffer as a zip entry in an opened inMemory zip archive with the inMemoryZipStart function.
//
// t : the inMemory class that holds the pointer to our inMemory zip file.
// levelOfCompression : (0-9) recommended 9 for maximum (0 = Store method.)
// buffer : The byte[] buffer that should be added to the zip.
// filename : The name of the file added. If you add a folder structure to it,
// like (dir1/dir2/myfile.bin) the directories will be created in the zip file.
// comment : add a comment for the file in the zip file header.
// password : set the password to protect this file.
// useBz2 : use the bz2 compression algorithm. If false the zlib deflate algorithm will be used. (not available for MacOS/iOS/tvOS)
//
// Returns 0 on success.
public static int inMemoryZipAdd(inMemory t, int levelOfCompression, byte[] buffer, string fileName, string comment = null, string password = null, bool useBz2 = false) {
if(t.info == null) { Debug.Log("inMemory object is null"); return -1; }
if (t.isClosed) { Debug.Log("Can't add entry. inMemory zip is closed."); return -2; }
if (password == "") password = null;
if (comment == "") comment = null;
if (fileName == null) fileName = "";
var fbuf = gcA(buffer);
int res = zipCDMemAdd(t.zf, levelOfCompression, fbuf.AddrOfPinnedObject(), buffer.Length, fileName, comment, password, useBz2);
fbuf.Free();
t.lastResult = res;
return res;
}
// A function to close the inMemory zip archive that has been created with the inMemoryZipStart function.
//
// t : the inMemory class that holds the pointer to our inMemory zip file.
//
// If t.lastResult is different then 0 a null pointer will get returned.
//
public static IntPtr inMemoryZipClose(inMemory t) {
if(t.info == null) { Debug.Log("inMemory object is null"); return IntPtr.Zero; }
if (t.isClosed) { Debug.Log("Can't close zip. inMemory zip is closed."); return t.pointer; }
var inf = gcA(t.info);
t.pointer = zipCDMemClose(t.zf, t.memStruct, inf.AddrOfPinnedObject(), t.lastResult);
inf.Free();
t.isClosed = true;
return t.pointer;
}
// End low level inMemory functions ------------------------------------------------------------------------------------------------
// A function that compresses a buffer to an inMemory zip file. Appending using this function can be slow. Use the low level functions for way faster proccessing.
//
// t : the inMemory class that holds the pointer to our inMemory zip file. You can call this function again for more buffers with the same
// : inMemory object and the next buffers will get appended to the in memory zip.
// levelOfCompression : (0-9) recommended 9 for maximum (0 = Store method.)
// buffer : The byte[] buffer that should be added to the zip.
// filename : The name of the file added. If you add a folder structure to it,
// like (dir1/dir2/myfile.bin) the directories will be created in the zip file.
// comment : add a comment for the file in the zip file header.
// password : set the password to protect this file.
// useBz2 : use the bz2 compression algorithm. If false the zlib deflate algorithm will be used. (not available for MacOS/iOS/tvOS)
//
/// Although the inMemory t.pointer gets internally updated, the function returns an IntPtr of the inMemory zip file buffer.
/// So to check if the operation was successful, check if the pointer returned is a non IntPtr.Zero pointer.
public static IntPtr compress_Buf2Mem(inMemory t, int levelOfCompression, byte[] buffer, string fileName, string comment = null, string password = null, bool useBz2 = false) {
if(t.info == null) { Debug.Log("inMemory object is null"); return IntPtr.Zero; }
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 9) levelOfCompression = 9;
if (password == "") password = null;
if (comment == "") comment = null;
if (fileName == null) fileName = "";
if (buffer == null || buffer.Length == 0) { Debug.Log("Buffer was null or zero size !"); return t.pointer; }
var fbuf = gcA(buffer);
var inf = gcA(t.info);
t.pointer = zipCDMem(inf.AddrOfPinnedObject(), t.pointer, levelOfCompression, fbuf.AddrOfPinnedObject(), buffer.Length, fileName, comment, password, useBz2);
fbuf.Free();
inf.Free();
return t.pointer;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
// A function that decompresses a zip file from an inMemory pointer. If the zip contains directories, they will be created.
//
// t : the inMemory class that holds the pointer to our inMemory zip file.
// outPath : the directory in which the zip contents will be extracted.
// progress : a single item integer array that increments with the archives that have been extracted. To use it in realtime, call
// : this function in a separate thread.
// proc: : a single item ulong array that gets updated with the progress of the decompression in bytes.
// (100% is reached when the compressed size of the file is reached.)
// password : if needed, the password to decrypt the archive.
//
// ERROR CODES
// : -1 = could not initialize zip archive.
// : -2 = failed extraction
// : 1 = success
//
public static int decompress_Mem2File(inMemory t, string outPath, int[] progress = null, ulong[] proc = null, string password = null) {
if(t.info == null) { Debug.Log("inMemory object is null"); return -1; }
//make a check if the last '/' exists at the end of the exctractionPath and add it if it is missing
if (outPath.Substring(outPath.Length - 1, 1) != "/") { outPath += "/"; }
int res = 0;
var ibuf = gcA(progress);
if(progress == null) progress = new int[1];
if (proc == null) proc = new ulong[1];
var pbuf = gcA(proc);
if (t != null) {
res = zipEX(null, @outPath, ibuf.AddrOfPinnedObject(), t.pointer, t.info[0], pbuf.AddrOfPinnedObject(), password);
ibuf.Free(); pbuf.Free(); return res;
} else return 0;
}
#endif
// A function that will decompress a file from an inmemory zip file pointer directly in a provided byte buffer.
//
// t : the inMemory class that holds the pointer to our inMemory zip file.
// entry : the file we want to extract to a buffer. (If the file resides in a directory, the directory should be included.)
// buffer : a referenced byte buffer that will be resized and will be filled with the extraction data.
// password : If the archive is encrypted use a password.
//
// ERROR CODES : 1 = success
// : -2 = could not find/open zip file
// : -3 = could not locate entry
// : -4 = could not get entry info
// : -5 = password error
// : -18 = the entry has no size
// : -104 = internal memory error
//
public static int entry2BufferMem(inMemory t, string entry, ref byte[] buffer, string password = null) {
if(t.info == null) return -2;
int siz = 0;
if (password == "") password = null;
if (t != null) siz = (int)zipGetEntrySize(null, entry, t.pointer, t.info[0]);
if (siz <= 0) return -18;
if(buffer == null) buffer = new byte[0];
Array.Resize(ref buffer, (int)siz);
var sbuf = gcA(buffer);
int res = 0;
if (t != null) res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), (int)siz, t.pointer, t.info[0], password);
sbuf.Free();
return res;
}
// A function that will decompress a file from an inmemory zip file pointer to a new created and returned byte buffer.
//
// t : the inMemory class that holds the pointer to our inMemory zip file.
// entry : the file we want to extract to a buffer. (If the file resides in a directory, the directory should be included.
// password : If the archive is encrypted use a password.
//
// ERROR CODES : non-null = success
// : null = failed
//
public static byte[] entry2BufferMem(inMemory t, string entry, string password = null) {
if(t.info == null) return null;
int siz = 0;
if (password == "") password = null;
if (t != null) siz = (int)zipGetEntrySize(null, entry, t.pointer, t.info[0]);
if (siz <= 0) return null;
byte[] buffer = new byte[siz];
var sbuf = gcA(buffer);
int res = 0;
if (t != null) res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), (int)siz, t.pointer, t.info[0], password);
sbuf.Free();
if (res != 1) return null;
else return buffer;
}
// A function that will decompress a file from an inmemory zip file pointer directly to a provided fixed size byte buffer.
//
// Returns the uncompressed size of the entry.
//
// t : the inMemory class that holds the pointer to our inMemory zip file.
// entry : the file we want to extract to a buffer. (If the file resides in a directory, the directory should be included.)
// buffer : a referenced fixed size byte buffer that will be filled with the extraction data. It should be large enough to store the data.
// password : if the archive is encrypted use a password.
//
// ERROR CODES : 1 = success
// : -2 = could not find/open zip archive
// : -3 = could not locate entry
// : -4 = could not get entry info
// : -5 = password error
// : -18 = the entry has no size
// : -19 = the fixed size buffer is not big enough to store the uncompressed data
// : -104 = internal memory error
//
public static int entry2FixedBufferMem(inMemory t, string entry, ref byte[] fixedBuffer, string password = null) {
if(t.info == null) return -2;
int siz = 0;
if (password == "") password = null;
if (t != null) siz = (int)zipGetEntrySize(null, entry, t.pointer, t.info[0]);
if (siz <= 0) return -18;
if (fixedBuffer.Length < (int)siz) return -19;
var sbuf = gcA(fixedBuffer);
int res = 0;
if (t != null) res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), siz, t.pointer, t.info[0], password);
sbuf.Free();
if (res != 1) return res;
return siz;
}
// This function fills the Lists with the filenames and file sizes that are in the inMemory zip file.
// Returns : the total size of uncompressed bytes of the files in the zip archive.
//
// t : the inMemory class that holds the pointer to our inMemory zip file.
//
// ERROR CODES : 0 = Input file not found or Could not get info
//
public static UInt64 getFileInfoMem(inMemory t) {
if(t.info == null) return 0;
ninfo.Clear(); uinfo.Clear(); cinfo.Clear(); localOffset.Clear();
zipFiles = 0; zipFolders = 0;
totalCompressedSize = 0;
totalUncompressedSize = 0;
int res = 0;
int[] tt = new int[1];
var tb = gcA(tt);
if (t != null) res = zipGetInfoA(null, tb.AddrOfPinnedObject(), t.pointer, t.info[0]);
tb.Free();
if (res <= 0) return 0;
IntPtr uni = IntPtr.Zero;
UInt64[] unc = new UInt64[tt[0]];
UInt64[] comp = new UInt64[tt[0]];
UInt64[] offs = new UInt64[tt[0]];
var uncb = gcA(unc);
var compb = gcA(comp);
var offsb = gcA(offs);
if (t != null) uni = zipGetInfo(null, res, uncb.AddrOfPinnedObject(), compb.AddrOfPinnedObject(), offsb.AddrOfPinnedObject(), t.pointer, t.info[0]);
if (uni == IntPtr.Zero) { uncb.Free(); compb.Free(); offsb.Free(); return 0; }
string str = Marshal.PtrToStringAuto(uni);
StringReader r = new StringReader(str);
string line;
UInt64 sum = 0;
for (int i = 0; i < tt[0]; i++) {
if ((line = r.ReadLine()) != null) ninfo.Add(line);
if (unc != null) {
uinfo.Add(unc[i]);
sum += unc[i];
if (unc[i] > 0) zipFiles++; else zipFolders++;
}
if (comp != null) { cinfo.Add(comp[i]); totalCompressedSize += comp[i]; }
if (offs != null) localOffset.Add(offs[i]);
}
r.Close();
r.Dispose();
uncb.Free(); compb.Free(); offsb.Free();
releaseBuffer(uni);
tt = null;
unc = null;
comp = null;
offs = null;
totalUncompressedSize = sum;
return sum;
}
//---------------------------------------------------------------------------------------------------------------------------
// End in Memory Functions
//---------------------------------------------------------------------------------------------------------------------------
// A function that will decompress a file in a zip archive directly to a provided byte buffer.
//
// zipArchive : the full path to the zip archive from which a specific file will be extracted to a byte buffer.
// entry : the file we want to extract to a buffer. (If the file resides in a directory, the directory should be included.)
// buffer : a referenced byte buffer that will be resized and will be filled with the extraction data.
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, then the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
// password : If the archive is encrypted use a password.
//
// ERROR CODES : 1 = success
// : -2 = could not find/open zip archive
// : -3 = could not locate entry
// : -4 = could not get entry info
// : -5 = password error
// : -6 = non valid zip size when using native file buffer
// : -18 = the entry has no size
// : -104 = internal memory error
//
public static int entry2Buffer(string zipArchive, string entry, ref byte[] buffer, object fileBuffer = null, string password = null) {
int siz = 0;
if(password == "") password = null;
if(fileBuffer != null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
siz = (int)zipGetEntrySize(null, entry, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return -6;
siz = (int)zipGetEntrySize(null, entry, FileBuffer, fileBufferLength);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
siz = (int)zipGetEntrySize( @zipArchive, entry, IntPtr.Zero, 0);
#endif
}
if (siz <= 0) return -18;
Array.Resize(ref buffer, siz);
var sbuf = gcA(buffer);
int res = 0;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), siz, fbuf.AddrOfPinnedObject(), FileBuffer.Length, password);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) { sbuf.Free(); return -6; }
res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), siz, FileBuffer, fileBufferLength, password);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipEntry2Buffer( @zipArchive, entry, sbuf.AddrOfPinnedObject(), siz, IntPtr.Zero, 0, password);
#endif
}
sbuf.Free();
return res;
}
// A function that will decompress a file in a zip archive directly to a provided fixed size byte buffer.
//
// Returns the uncompressed size of the entry.
//
// zipArchive : the full path to the zip archive from which a specific file will be extracted to a byte buffer.
// entry : the file we want to extract to a buffer. (If the file resides in a directory, the directory should be included.)
// buffer : a referenced fixed size byte buffer that will be filled with the extraction data. It should be large enough to store the data.
// FileBuffer : a buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, then the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
// password : if the archive is encrypted use a password.
//
// ERROR CODES : number of extracted bytes = success
// : -2 = could not find/open zip archive
// : -3 = could not locate entry
// : -4 = could not get entry info
// : -5 = password error
// : -6 = non valid zip size when using native file buffer
// : -18 = the entry has no size
// : -19 = the fixed size buffer is not big enough to store the uncompressed data
// : -104 = internal memory error
//
public static int entry2FixedBuffer(string zipArchive, string entry, ref byte[] fixedBuffer, object fileBuffer = null, string password = null) {
int siz = 0;
if(password == "") password = null;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
siz = (int)zipGetEntrySize(null, entry, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return -6;
siz = (int)zipGetEntrySize(null, entry, FileBuffer, fileBufferLength);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
siz = (int)zipGetEntrySize( @zipArchive, entry, IntPtr.Zero, 0);
#endif
}
if (siz <= 0) return -18;
if(fixedBuffer.Length < siz) return -19;
var sbuf = gcA(fixedBuffer);
int res = 0;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), siz, fbuf.AddrOfPinnedObject(), FileBuffer.Length, password);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) { sbuf.Free(); return -6; }
res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), siz, FileBuffer, fileBufferLength, password);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipEntry2Buffer( @zipArchive, entry, sbuf.AddrOfPinnedObject(), siz, IntPtr.Zero, 0, password);
#endif
}
sbuf.Free();
if(res != 1) return res;
return siz;
}
// A function that will decompress a file in a zip archive to a new created and returned byte buffer.
//
// zipArchive : the full path to the zip archive from which a specific file will be extracted to a byte buffer.
// entry : the file we want to extract to a buffer. (If the file resides in a directory, the directory should be included.
// FileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, then the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
// password : If the archive is encrypted use a password.
//
// ERROR CODES : non-null = success
// : null = failed
//
public static byte[] entry2Buffer(string zipArchive, string entry, object fileBuffer = null, string password = null) {
int siz = 0;
if(password == "") password = null;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
siz = (int)zipGetEntrySize(null, entry, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return null;
siz = (int)zipGetEntrySize(null, entry, FileBuffer, fileBufferLength);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
siz = (int)zipGetEntrySize( @zipArchive, entry, IntPtr.Zero, 0);
#endif
}
if (siz <= 0) return null;
byte[] buffer = new byte[siz];
var sbuf = gcA(buffer);
int res = 0;
if(fileBuffer!= null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), siz, fbuf.AddrOfPinnedObject(), FileBuffer.Length, password);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) { sbuf.Free(); return null; }
res = zipEntry2Buffer(null, entry, sbuf.AddrOfPinnedObject(), siz, FileBuffer, fileBufferLength, password);
}
} else {
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipEntry2Buffer( @zipArchive, entry, sbuf.AddrOfPinnedObject(), siz, IntPtr.Zero, 0, password);
#endif
}
sbuf.Free();
if (res != 1) return null;
else return buffer;
}
// A function that will validate a zip archive.
//
// zipArchive : the zip to be checked
// fileBuffer : A buffer that holds a zip file. When assigned the function will read from this buffer and will ignore the filePath.
// : Can be a c# byte[] buffer or an unmanaged Inptr buffer or an inMemory class! When an IntPtr is used, then the zipArchive string parameter will be used to pass the Length of the Intptr buffer!
// ERROR CODES
//
// : true. The archive is ok.
// : false. The archive could not be validated.
//
public static bool validateFile(string zipArchive, object fileBuffer = null) {
bool res = false;
if(fileBuffer != null) {
if (fileBuffer is byte[]) {
byte[] FileBuffer = (byte[])fileBuffer;
var fbuf = gcA(FileBuffer);
res = zipValidateFile(null, fbuf.AddrOfPinnedObject(), FileBuffer.Length);
fbuf.Free();
}
if (fileBuffer is IntPtr || fileBuffer is inMemory) {
IntPtr FileBuffer = IntPtr.Zero;
int fileBufferLength = 0;
if (!checkObject(fileBuffer, zipArchive, ref fileBufferLength, ref FileBuffer)) return false;
res = zipValidateFile(null, FileBuffer, fileBufferLength);
}
return res;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
res = zipValidateFile(@zipArchive, IntPtr.Zero, 0);
#endif
return res;
}
// ---------------------------------------------------------------------------------------------------------------------------------
//
// Hidden/merged zip and zipInfo functions
//
// ---------------------------------------------------------------------------------------------------------------------------------
// a struct to store zip entry information
public struct zipInfo{
public short VersionMadeBy;
public short MinimumVersionToExtract;
public short BitFlag;
public short CompressionMethod;
public short FileLastModificationTime;
public short FileLastModificationDate;
public int CRC;
public int CompressedSize;
public int UncompressedSize;
public short DiskNumberWhereFileStarts;
public short InternalFileAttributes;
public int ExternalFileAttributes;
public int RelativeOffsetOfLocalFileHeader;
public int AbsoluteOffsetOfLocalFileHeaderStore;
public string filename;
public string extraField;
public string fileComment;
};
// a list that will be filled with info of all the zip archive entries.
public static List<zipInfo> zinfo;
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
// This function is used to get extended info of the entries in a zip archive in the file system.
// Use this as an alternative function to get zip info with more information about entries.
// For now it does not work with zip64.
// Returns true on success.
public static bool getZipInfo(string fileName) {
if (!File.Exists(fileName)) { Debug.Log("File not found: "+ fileName); return false; }
int pos =0, size = 0;
using (var file = File.OpenRead(fileName)) {
using (var reader = new BinaryReader(file)) {
if (findPK(reader)) {
int entryCount = findEnd(reader, ref pos, ref size);
if (entryCount > 0) {
getCentralDir(reader, entryCount);
return true;
} else {
Debug.Log("No Entries in zip");
return false;
}
}
}
}
return false;
}
// Get position, size and/or entry info of a zip archive in the file system.
// This function is mainly useful to discover a zip archive hidden or merged in another bigger archive.
// filename: the path to the archive.
// ref pos: the position in bytes of the zip archive.
// ref size: the size of the zip archive.
// If getCentralDirectory is set to true it will fill the zinfo List with extended entry information.
// For now it does not work with zip64.
// Returns true on success.
public static bool getZipInfoMerged(string fileName, ref int pos, ref int size, bool getCentralDirectory = false) {
if (!File.Exists(fileName)) { Debug.Log("File not found: "+ fileName); return false; }
using (var file = File.OpenRead(fileName)) {
using (var reader = new BinaryReader(file)) {
if (findPK(reader)) {
int entryCount = findEnd(reader, ref pos, ref size);
if (entryCount > 0) {
if(getCentralDirectory) getCentralDir(reader, entryCount);
return true;
} else {
Debug.Log("No Entries in zip");
return false;
}
}
}
}
return false;
}
#endif
// Get position, size and/or entry info of a zip archive in a buffer.
// This function is mainly useful to discover a zip archive hidden or merged in another bigger buffer.
// buffer: the buffer where the merged zip archive resides.
// ref pos: the position in bytes of the zip archive.
// ref size: the size of the zip archive.
// If getCentralDirectory is set to true it will fill the zinfo List with extended entry information.
// For now it does not work with zip64.
// Returns true on success.
public static bool getZipInfoMerged(byte[] buffer, ref int pos, ref int size, bool getCentralDirectory = false) {
if (buffer == null) { Debug.Log("Buffer is null"); return false; }
using (var file = new MemoryStream(buffer)) {
using (var reader = new BinaryReader(file)) {
if (findPK(reader)) {
int entryCount = findEnd(reader, ref pos, ref size);
if (entryCount > 0) {
if (getCentralDirectory) getCentralDir(reader, entryCount);
return true;
} else {
Debug.Log("No Entries in zip");
return false;
}
}
}
}
return false;
}
// This function is used to get extended info of the entries in a zip archive in a buffer.
// buffer: the buffer where the merged zip archive resides.
// For now it does not work with zip64.
// Returns true on success.
public static bool getZipInfoMerged(byte[] buffer) {
if (buffer == null) { Debug.Log("Buffer is null"); return false; }
int pos =0, size = 0;
using (var file = new MemoryStream(buffer)) {
using (var reader = new BinaryReader(file)) {
if (findPK(reader)) {
int entryCount = findEnd(reader, ref pos, ref size);
if (entryCount > 0) {
getCentralDir(reader, entryCount);
return true;
} else {
Debug.Log("No Entries in zip");
return false;
}
}
}
}
return false;
}
// Find Central directory position
private static bool findPK(BinaryReader reader) {
var p = reader.ReadByte();
bool res = false;
int i = 0;
while (reader.BaseStream.Position < reader.BaseStream.Length-3) {
i++;
if (p == 0x50) {
if(reader.ReadByte() == 0x4b && reader.ReadByte() == 0x05 && reader.ReadByte() == 0x06) { reader.BaseStream.Seek(reader.BaseStream.Position - 4, 0); res = true; break; }
else{ reader.BaseStream.Seek(i, 0);}
}
p = reader.ReadByte();
}
return res;
}
// Read central directory basics and find position and size of the zip archive in stream. Returns entry count.
private static int findEnd(BinaryReader reader, ref int pos, ref int size) {
long origin = reader.BaseStream.Position;
int result = 0;
while (result == 0 && reader.BaseStream.Position < reader.BaseStream.Length) {
var b = reader.ReadByte();
while (b != 0x50 ) {
if (reader.BaseStream.Position < reader.BaseStream.Length)
b = reader.ReadByte();
else
break;
}
if (reader.BaseStream.Position >= reader.BaseStream.Length) break;
if (reader.ReadByte() == 0x4b && reader.ReadByte() == 0x05 && reader.ReadByte() == 0x06) {
/*int diskNumber =*/ reader.ReadInt16();
/*int centralDirectoryStartDiskNumber =*/ reader.ReadInt16();
/*int centralDirectoryCount =*/ reader.ReadInt16();
int centralDirectoryTotal = reader.ReadInt16();
result = centralDirectoryTotal;
int centralDirectorySize = reader.ReadInt32();
int centralDirectoryOffset = reader.ReadInt32();
int commentLength = reader.ReadInt16();
reader.ReadBytes(commentLength);
//string comment = Encoding.UTF8.GetString(reader.ReadBytes(commentLength));
pos = (int)reader.BaseStream.Position - (centralDirectoryOffset + centralDirectorySize + 22);
size = (int)reader.BaseStream.Position - pos;
reader.BaseStream.Seek(pos + centralDirectoryOffset, 0);
break;
}
}
return result;
}
// Read info for each entry in the zip archive and store it in zinfo List.
private static void getCentralDir(BinaryReader reader, int count) {
if(zinfo != null && zinfo.Count > 0) zinfo.Clear();
zinfo = new List<zipInfo>();
for (int i = 0; i < count; i++) {
var magic = reader.ReadInt32();
if (magic == 0x02014b50) {
zipInfo z = new zipInfo();
z.VersionMadeBy = reader.ReadInt16();
z.MinimumVersionToExtract = reader.ReadInt16();
z.BitFlag = reader.ReadInt16();
z.CompressionMethod = reader.ReadInt16();
z.FileLastModificationTime = reader.ReadInt16();
z.FileLastModificationDate = reader.ReadInt16();
z.CRC = reader.ReadInt32();
z.CompressedSize = reader.ReadInt32();
z.UncompressedSize = reader.ReadInt32();
short fileNameLength = reader.ReadInt16();
short extraFieldLength = reader.ReadInt16();
short fileCommentLength = reader.ReadInt16();
z.DiskNumberWhereFileStarts = reader.ReadInt16();
z.InternalFileAttributes = reader.ReadInt16();
z.ExternalFileAttributes = reader.ReadInt32();
z.RelativeOffsetOfLocalFileHeader = reader.ReadInt32();
z.filename = Encoding.UTF8.GetString(reader.ReadBytes(fileNameLength));
z.AbsoluteOffsetOfLocalFileHeaderStore = z.RelativeOffsetOfLocalFileHeader + 30 + z.filename.Length;
var extra = reader.ReadBytes(extraFieldLength);
z.extraField = Encoding.ASCII.GetString(extra);
z.fileComment = Encoding.UTF8.GetString(reader.ReadBytes(fileCommentLength));
zinfo.Add(z);
}
}
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
// Get the merged zip archive in a file system archive as a byte buffer and provide postion and size.
// filePath: the path to the archive.
// ref position: the position in bytes of the zip archive.
// ref size: the size of the zip archive.
public static byte[] getMergedZip(string filePath, ref int position, ref int siz) {
int pos = 0, size = 0;
if (!File.Exists(filePath)) return null;
getZipInfoMerged(filePath, ref pos, ref size);
position = pos; siz = size;
if(size == 0) return null;
byte[] tempBuffer = new byte[size];
using (var file = File.OpenRead(filePath)) {
using (var reader = new BinaryReader(file)) {
reader.BaseStream.Seek(pos, SeekOrigin.Begin);
reader.Read(tempBuffer, 0, size);
}
}
return tempBuffer;
}
// Get the merged zip archive in a file system archive as a byte buffer.
// filePath: the path to the archive.
public static byte[] getMergedZip(string filePath) {
int pos = 0, size = 0;
if (!File.Exists(filePath)) return null;
getZipInfoMerged(filePath, ref pos, ref size);
if(size == 0) return null;
byte[] tempBuffer = new byte[size];
using (var file = File.OpenRead(filePath)) {
using (var reader = new BinaryReader(file)) {
reader.BaseStream.Seek(pos, SeekOrigin.Begin);
reader.Read(tempBuffer, 0, size);
}
}
return tempBuffer;
}
#endif
// Get the merged zip archive in a buffer as a byte buffer.
// buffer: the buffer where the zip archive resides.
// ref position: the position in bytes of the zip archive.
// ref size: the size of the zip archive.
public static byte[] getMergedZip(byte[] buffer, ref int position, ref int siz) {
int pos = 0, size = 0;
if (buffer == null) return null;
getZipInfoMerged(buffer, ref pos, ref size);
position = pos; siz = size;
if(size == 0) return null;
byte[] tempBuffer = new byte[size];
using (var file = new MemoryStream(buffer)) {
using (var reader = new BinaryReader(file)) {
reader.BaseStream.Seek(pos, SeekOrigin.Begin);
reader.Read(tempBuffer, 0, size);
}
}
return tempBuffer;
}
// Get the merged zip archive in a buffer as a byte buffer.
// buffer: the buffer where the zip archive resides.
public static byte[] getMergedZip(byte[] buffer) {
int pos = 0, size = 0;
if (buffer == null) return null;
getZipInfoMerged(buffer, ref pos, ref size);
if(size == 0) return null;
byte[] tempBuffer = new byte[size];
using (var file = new MemoryStream(buffer)) {
using (var reader = new BinaryReader(file)) {
reader.BaseStream.Seek(pos, SeekOrigin.Begin);
reader.Read(tempBuffer, 0, size);
}
}
return tempBuffer;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
// A function that extracts all contents of a zip file that is merged in another file in the file system, to disk.
// file : the path to the file where the zip archive resides.
// outPath : the directory in which the zip contents will be extracted.
// progress : provide a single item integer array to write the current index of the file getting extracted. To use it in realtime, call
// : this function in a separate thread.
// proc: : a single item ulong array that gets updated with the progress of the decompression in bytes.
// (100% is reached when the compressed size of the file is reached.)
// password : if needed, the password to decrypt the archive.
// ERROR CODES
// : -1 = could not initialize zip archive.
// : -2 = failed extraction
// : 1 = success
public static int decompressZipMerged(string file, string outPath, int[] progress = null, ulong[] proc = null, string password = null) {
if (!File.Exists(file)) return 0;
outPath = outPath.Replace("//", "/");
if (!Directory.Exists(outPath)) Directory.CreateDirectory(outPath);
int pos = 0, size = 0, res = 0;
var tempBuffer = getMergedZip(file, ref pos, ref size);
if(tempBuffer != null) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(tempBuffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64() );
t2.info[0] = size;
res = decompress_Mem2File(t2, outPath, progress, proc, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
tempBuffer = null;
}
return res;
}
// A function that extracts all contents of a merged zip file that resides in a buffer to disk.
// buffer : the buffer where the zip archive resides.
// outPath : the directory in which the zip contents will be extracted.
// progress : a single item integer array that increments with the archives that have been extracted. To use it in realtime, call
// : this function in a separate thread.
// proc: : a single item ulong array that gets updated with the progress of the decompression in bytes.
// (100% is reached when the compressed size of the file is reached.)
// password : if needed, the password to decrypt the archive.
// ERROR CODES
// : -1 = could not initialize zip archive.
// : -2 = failed extraction
// : 1 = success
public static int decompressZipMerged(byte[] buffer, string outPath, int[] progress = null, ulong[] proc = null, string password = null) {
if (buffer == null) return 0;
outPath = outPath.Replace("//", "/");
if (!Directory.Exists(outPath)) Directory.CreateDirectory(outPath);
int pos = 0, size = 0, res = 0;
if( getZipInfoMerged(buffer, ref pos, ref size)) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(buffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64() + pos);
t2.info[0] = size;
res = decompress_Mem2File(t2, outPath, progress, proc, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
}
return res;
}
private static void writeFile(byte[] tb, string entry, string outPath, string overrideEntryName, ref int res) {
if(tb != null) {
string fn;
if(overrideEntryName == null){
if(entry.Contains("/")) {
string[] fileName = entry.Split('/');
fn = fileName[fileName.Length-1];
} else{
fn = entry;
}
} else {
fn = overrideEntryName;
}
File.WriteAllBytes(outPath + "/" + fn, tb);
res = 1;
} else {
Debug.Log("Could not extract entry.");
}
}
// Extract an entry from a merged zip that resides in the file system to disk.
// Returns 1 on success.
//
// file : the path to the file where the zip archive resides.
// entry : the entry to extract.
// outPath : the path where the entry will be extracted.
// overrideEntryName : with this you can override the default entry name.
// password : if needed, the password to decrypt the archive.
public static int entry2FileMerged(string file, string entry, string outPath, string overrideEntryName = null, string password = null) {
if (!File.Exists(file)) return -10;
outPath = outPath.Replace("//", "/");
int pos = 0, size = 0, res = 0;
var tempBuffer = getMergedZip(file, ref pos, ref size);
if(tempBuffer != null) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(tempBuffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64() );
t2.info[0] = size;
var tb = entry2BufferMem(t2, entry, password);
pinnedArray.Free();
writeFile(tb, entry, outPath, overrideEntryName, ref res);
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
tempBuffer = null;
tb = null;
}
return res;
}
// Extract an entry from a merged zip that resides in a buffer to disk.
// Returns 1 on success.
//
// buffer : the buffer where the zip archive resides.
// entry : the entry to extract.
// outPath : the path where the entry will be extracted.
// overrideEntryName : with this you can override the default entry name.
// password : if needed, the password to decrypt the archive.
public static int entry2FileMerged(byte[] buffer, string entry, string outPath, string overrideEntryName = null, string password = null) {
if (buffer == null) return -10;
outPath = outPath.Replace("//", "/");
int pos = 0, size = 0, res = 0;
if( getZipInfoMerged(buffer, ref pos, ref size)) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(buffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64() + pos);
t2.info[0] = size;
var tb = entry2BufferMem(t2, entry, password);
pinnedArray.Free();
writeFile(tb, entry, outPath, overrideEntryName, ref res);
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
tb = null;
}
return res;
}
// A function that extracts an entry from a zip archive that is merged/hidden in the file system and returns the extracted data in a new buffer.
//
// file : the path to the file where the zip archive resides.
// entry : the entry to extract.
// password : if needed, the password to decrypt the archive.
public static byte[] entry2BufferMerged(string file, string entry, string password = null) {
if (!File.Exists(file)) return null;
int pos = 0, size = 0;
var tempBuffer = getMergedZip(file, ref pos, ref size);
if(tempBuffer != null) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(tempBuffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64());
t2.info[0] = size;
var res = entry2BufferMem(t2, entry, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
tempBuffer = null;
return res;
}
return null;
}
// A function that extracts an entry from a zip archive that is merged/hidden in the file system and returns the extracted data in a referenced buffer that will get resized to fit the data.
//
// file : the path to the file where the zip archive resides.
// entry : the entry to extract.
// refBuffer : the referenced buffer that will get resized to store the decompressed data.
// password : if needed, the password to decrypt the archive.
public static int entry2BufferMerged(string file, string entry, ref byte[] refBuffer, string password = null) {
if (!File.Exists(file)) return 0;
int pos = 0, size = 0;
var tempBuffer = getMergedZip(file, ref pos, ref size);
if(tempBuffer != null) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(tempBuffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64());
t2.info[0] = size;
var res = entry2BufferMem(t2, entry,ref refBuffer, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
tempBuffer = null;
return res;
}
return 0;
}
// A function that extracts an entry from a zip archive that is merged/hidden in the file system and writes the extracted data in a fixed size buffer.
//
// file : the path to the file where the zip archive resides.
// entry : the entry to extract.
// fixedBuffer : the fixed sized buffer where the data will be written.
// password : if needed, the password to decrypt the archive.
public static int entry2FixedBufferMerged(string file, string entry, ref byte[] fixedBuffer, string password = null) {
if (!File.Exists(file)) return 0;
int pos = 0, size = 0;
var tempBuffer = getMergedZip(file, ref pos, ref size);
if(tempBuffer != null) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(tempBuffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64());
t2.info[0] = size;
int res = entry2FixedBufferMem(t2, entry, ref fixedBuffer, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
tempBuffer = null;
return res;
}
return 0;
}
#endif
// A function that extracts an entry from a zip archive that is merged/hidden in a buffer and returns the extracted data in a new buffer.
//
// buffer : the buffer where the zip archive resides.
// entry : the entry to extract.
// password : if needed, the password to decrypt the archive.
public static byte[] entry2BufferMerged(byte[] buffer, string entry, string password = null) {
if (buffer == null) return null;
int pos = 0, size = 0;
if( getZipInfoMerged(buffer, ref pos, ref size)) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(buffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64() + pos);
t2.info[0] = size;
var res = entry2BufferMem(t2, entry, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
return res;
}
return null;
}
// A function that extracts an entry from a zip archive that is merged/hidden in a buffer and returns the extracted data in a referenced buffer that will get resized to fit the data.
//
// buffer : the buffer where the zip archive resides.
// entry : the entry to extract.
// refBuffer : the referenced buffer that will get resized to store the decompressed data.
// password : if needed, the password to decrypt the archive.
public static int entry2BufferMerged(byte[] buffer, string entry, ref byte[] refBuffer, string password = null) {
if (buffer == null) return 0;
int pos = 0, size = 0;
if( getZipInfoMerged(buffer, ref pos, ref size)) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(buffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64() + pos);
t2.info[0] = size;
var res = entry2BufferMem(t2, entry, ref refBuffer, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
return res;
}
return 0;
}
// A function that extracts an entry from a zip archive that is merged/hidden in a buffer and writes the extracted data in a fixed size buffer.
// Returns the size of the uncompressed data.
//
// buffer : the buffer where the zip archive resides.
// entry : the entry to extract.
// fixedBuffer : the fixed sized buffer where the data will be written.
// password : if needed, the password to decrypt the archive.
// ERROR CODES
// : -1 = could not initialize zip archive.
// : -2 = failed extraction
// : 1 = success
public static int entry2FixedBufferMerged(byte[] buffer, string entry, ref byte[] fixedBuffer, string password = null) {
if (buffer == null) return 0;
int pos = 0, size = 0;
if( getZipInfoMerged(buffer, ref pos, ref size)) {
inMemory t2 = new inMemory();
var pinnedArray = gcA(buffer);
t2.pointer = new IntPtr(pinnedArray.AddrOfPinnedObject().ToInt64() + pos);
t2.info[0] = size;
int res = entry2FixedBufferMem(t2, entry, ref fixedBuffer, password);
pinnedArray.Free();
t2.info = null;
t2.pointer = IntPtr.Zero;
t2 = null;
return res;
}
return 0;
}
// ---------------------------------------------------------------------------------------------------------------------------------
//
// END hidden/merged zip and zipInfo functions
//
// ---------------------------------------------------------------------------------------------------------------------------------
//---------------------------------------------------------------------------------------------------------------------------
// ZLIB BUFFER SECTION
//---------------------------------------------------------------------------------------------------------------------------
// A function that compresses a byte buffer to a zlib stream compressed buffer. Provide a reference buffer to write to. This buffer will be resized.
//
// source : the input buffer
// outBuffer : the referenced output buffer
// levelOfCompression : (0-10) recommended 9 for maximum (10 is highest but slower and not zlib compatible)
//
// ERROR CODES : true = success
// : false = failed
//
public static bool compressBuffer(byte[] source, ref byte[] outBuffer, int levelOfCompression) {
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 10) levelOfCompression = 10;
var sbuf = gcA(source);
IntPtr ptr;
int siz = 0;
ptr = zipCompressBuffer(sbuf.AddrOfPinnedObject(), source.Length, levelOfCompression, ref siz);
if (siz == 0 || ptr == IntPtr.Zero) { sbuf.Free(); releaseBuffer(ptr); return false; }
Array.Resize(ref outBuffer, siz);
Marshal.Copy(ptr, outBuffer, 0, siz);
sbuf.Free();
releaseBuffer(ptr);
return true;
}
// same as the compressBuffer function, only this function will put the result in a fixed size buffer to avoid memory allocations.
// the compressed size is returned so you can manipulate it at will.
//
// safe: if set to true the function will abort if the compressed resut is larger the the fixed size output buffer.
// Otherwise compressed data will be written only until the end of the fixed output buffer.
//
public static int compressBufferFixed(byte[] source, ref byte[] outBuffer, int levelOfCompression, bool safe = true) {
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 10) levelOfCompression = 10;
var sbuf = gcA(source);
IntPtr ptr;
int siz = 0;
ptr = zipCompressBuffer(sbuf.AddrOfPinnedObject(), source.Length, levelOfCompression, ref siz);
if (siz == 0 || ptr == IntPtr.Zero) { sbuf.Free(); releaseBuffer(ptr); return 0; }
if (siz>outBuffer.Length) {
if(safe) { sbuf.Free(); releaseBuffer(ptr); return 0; } else { siz = outBuffer.Length; }
}
Marshal.Copy(ptr, outBuffer, 0, siz);
sbuf.Free();
releaseBuffer(ptr);
return siz;
}
// A function that compresses a byte buffer to a zlib stream compressed buffer. Returns a new buffer with the compressed data.
//
// source : the input buffer
// levelOfCompression : (0-10) recommended 9 for maximum (10 is highest but slower and not zlib compatible)
//
// ERROR CODES : a valid byte buffer = success
// : null = failed
//
public static byte[] compressBuffer(byte[] source, int levelOfCompression) {
if (levelOfCompression < 0) levelOfCompression = 0; if (levelOfCompression > 10) levelOfCompression = 10;
var sbuf = gcA(source);
IntPtr ptr;
int siz = 0;
ptr = zipCompressBuffer(sbuf.AddrOfPinnedObject(), source.Length, levelOfCompression, ref siz);
if (siz == 0 || ptr == IntPtr.Zero) { sbuf.Free(); releaseBuffer(ptr); return null; }
byte[] buffer = new byte[siz];
Marshal.Copy(ptr, buffer, 0, siz);
sbuf.Free();
releaseBuffer(ptr);
return buffer;
}
// A function that decompresses a zlib compressed buffer to a referenced outBuffer. The outbuffer will be resized.
//
// source : a zlib compressed buffer.
// outBuffer : a referenced out buffer provided to extract the data. This buffer will be resized to fit the uncompressed data.
//
// ERROR CODES : true = success
// : false = failed
//
public static bool decompressBuffer(byte[] source, ref byte[] outBuffer) {
var sbuf = gcA(source);
IntPtr ptr;
int siz = 0;
ptr = zipDecompressBuffer(sbuf.AddrOfPinnedObject(), source.Length, ref siz);
if (siz == 0 || ptr == IntPtr.Zero) { sbuf.Free(); releaseBuffer(ptr); return false; }
Array.Resize(ref outBuffer, siz);
Marshal.Copy(ptr, outBuffer, 0, siz);
sbuf.Free();
releaseBuffer(ptr);
return true;
}
// same as the decompressBuffer function. Only this one outputs to a buffer of fixed which size isn't resized to avoid memory allocations.
// The fixed buffer should have a size that will be able to hold the incoming decompressed data.
// Returns the uncompressed size.
//
// safe: if set to true the function will abort if the decompressed resut is larger the the fixed size output buffer.
// Otherwise decompressed data will be written only until the end of the fixed output buffer.
//
public static int decompressBufferFixed(byte[] source, ref byte[] outBuffer, bool safe = true) {
var sbuf = gcA(source);
IntPtr ptr;
int siz = 0;
ptr = zipDecompressBuffer(sbuf.AddrOfPinnedObject(), source.Length, ref siz);
if (siz == 0 || ptr == IntPtr.Zero) { sbuf.Free(); releaseBuffer(ptr); return 0; }
if (siz>outBuffer.Length) {
if(safe) { sbuf.Free(); releaseBuffer(ptr); return 0; } else { siz = outBuffer.Length; }
}
Marshal.Copy(ptr, outBuffer, 0, siz);
sbuf.Free();
releaseBuffer(ptr);
return siz;
}
// A function that decompresses a zlib compressed buffer and creates a new buffer. Returns a new buffer with the uncompressed data.
//
// source : a zlib compressed buffer.
//
// ERROR CODES : a valid byte buffer = success
// : null = failed
//
public static byte[] decompressBuffer(byte[] source) {
var sbuf = gcA(source);
IntPtr ptr;
int siz = 0;
ptr = zipDecompressBuffer(sbuf.AddrOfPinnedObject(), source.Length, ref siz);
if (siz == 0 || ptr == IntPtr.Zero) { sbuf.Free(); releaseBuffer(ptr); return null; }
byte[] buffer = new byte[siz];
Marshal.Copy(ptr, buffer, 0, siz);
sbuf.Free();
releaseBuffer(ptr);
return buffer;
}
//---------------------------------------------------------------------------------------------------------------------------
// END ZLIB BUFFER SECTION
//---------------------------------------------------------------------------------------------------------------------------
//---------------------------------------------------------------------------------------------------------------------------
// GZIP SECTION
//---------------------------------------------------------------------------------------------------------------------------
// compress a byte buffer to gzip format.
//
// returns the size of the compressed buffer.
//
// source: the uncompressed input buffer.
// outBuffer: the provided output buffer where the compressed data will be stored (it should be at least the size of the input buffer +18 bytes).
// level: the level of compression (0-10). (0 = Store method.)
// addHeader: if a gzip header should be added. (recommended if you want to write out a gzip file)
// addFooter: if a gzip footer should be added. (recommended if you want to write out a gzip file)
// overrideDateTimeWithLength: use the bytes 5-8 of the header to store the gzip length instead of DateTime modification. This is useful when you want to know the
// compressed size of a gzip buffer. Then use the gzipCompressedSize function to get this size.
public static int gzip(byte[] source, byte[] outBuffer, int level, bool addHeader = true, bool addFooter = true, bool overrideDateTimeWithLength = false) {
if(source == null || outBuffer == null) return 0;
var sbuf = gcA(source);
var dbuf = gcA(outBuffer);
if(level < 0) level = 0; if(level > 10) level = 10;
int res = zipGzip(sbuf.AddrOfPinnedObject(), source.Length, dbuf.AddrOfPinnedObject(), level, addHeader, addFooter);
sbuf.Free(); dbuf.Free();
int hf = 0;
if(addHeader) hf += 10;
if(addFooter) hf += 8;
int compressedSize = res + hf;
if(addHeader && overrideDateTimeWithLength) {
outBuffer[4] = (byte)(((uint)compressedSize >> 0) & 0xff);
outBuffer[5] = (byte)(((uint)compressedSize >> 8) & 0xff);
outBuffer[6] = (byte)(((uint)compressedSize >> 16) & 0xff);
outBuffer[7] = (byte)(((uint)compressedSize >> 24) & 0xff);
// use the operating system flag to mark this gzip that it holds the compressed data size.
outBuffer[9] = 0xfe;
}
return compressedSize;
}
// get the uncompressed size from a gzip buffer that has a footer included
//
// source: the gzip compressed input buffer. (it should have at least a gzip footer)
public static int gzipUncompressedSize(byte[] source) {
if (source == null) return 0;
int res = source.Length;
uint size = ((uint)source[res-4] & 0xff) |
((uint)source[res-3] & 0xff) << 8 |
((uint)source[res-2] & 0xff) << 16 |
((uint)source[res-1]& 0xff) << 24;
return (int)size;
}
// get the compressed size of a gzip, if the compressed size of it has been written in the date header bytes and marked as such, with the gzip function above.
//
// source: the gzip compressed input buffer.
public static int gzipCompressedSize(byte[] source, int offset = 0) {
if (source == null) return 0;
if(source[offset + 9] != 0xfe) {
Debug.Log("Gzip has not been marked to have compressed size stored.");
return 0;
}
int res = offset + 8;
uint size = ((uint)source[res-4] & 0xff) |
((uint)source[res-3] & 0xff) << 8 |
((uint)source[res-2] & 0xff) << 16 |
((uint)source[res-1]& 0xff) << 24;
return (int)size;
}
// find where the merged gzip starts in a buffer.
//
// buffer: a memory buffer that has a gzip merged at the end of it.
public static int findGzStart(byte[] buffer) {
if(buffer == null) return 0;
int res = 0;
int i = 0;
while (i < buffer.Length - 2) {
if (buffer[i] == 0x1f) {
if(buffer[i+1] == 0x8b && buffer[i+2] == 0x08 ) { res = i; break; }
}
i++;
}
return res;
}
// decompress a gzip buffer
//
// returns: uncompressed size. negative error code on error.
//
// source: the gzip compressed input buffer.
// outBuffer: the provided output buffer where the uncompressed data will be stored.
// hasHeader: if the buffer has a header.
// hasFooter: if the buffer has a footer.
//
public static int unGzip(byte[] source, byte[] outBuffer, bool hasHeader = true, bool hasFooter = true) {
if(source == null || outBuffer == null) return 0;
var sbuf = gcA(source);
var dbuf = gcA(outBuffer);
int res = zipUnGzip( sbuf.AddrOfPinnedObject(), source.Length, dbuf.AddrOfPinnedObject(), outBuffer.Length, hasHeader, hasFooter );
sbuf.Free(); dbuf.Free();
return res;
}
// decompress a gzip buffer (This function assumes that the gzip buffer has a gzip header !!!)
//
// returns: uncompressed size. negative error code on error. -11 when a native pointer is passed as source and no source length defined.
//
// source: the gzip compressed input buffer. It can be a managed byte[] buffer or a native/unmanaged IntPtr buffer.
// outBuffer: the provided output buffer where the uncompressed data will be stored.
// intPtrLength: when passing an IntPtr pointer as the input gzip, we need to pass the size of this buffer to the function. Not needed for byte[] source gz.
//
public static int unGzip2(object source, byte[] outBuffer, int intPtrLength = 0) {
if(source == null || outBuffer == null) return 0;
int res = 0;
if (source is byte[]) {
byte[] Source = (byte[])source;
var sbuf = gcA(Source);
var dbuf = gcA(outBuffer);
res = zipUnGzip2( sbuf.AddrOfPinnedObject(), Source.Length, dbuf.AddrOfPinnedObject(), outBuffer.Length );
sbuf.Free(); dbuf.Free();
}
if (source is IntPtr && intPtrLength > 0) {
IntPtr Source = (IntPtr)source;
var dbuf = gcA(outBuffer);
res = zipUnGzip2( Source, intPtrLength, dbuf.AddrOfPinnedObject(), outBuffer.Length );
dbuf.Free();
} else { return -11; }
return res;
}
// decompress a gzip buffer that is merged in the end of a buffer (This function assumes that the gzip buffer has a gzip header !!!)
//
// returns: uncompressed size. negative error code on error.
//
// source: the buffer where the gzip compressed input buffer resides. (at the end of it, or anywhere if you know the length of it)
// offset: the offset in bytes where the gzip starts.
// bufferLength: the length of the gzip buffer.
// outBuffer: the provided output buffer where the uncompressed data will be stored.
//
public static int unGzip2Merged(byte[] source, int offset, int bufferLength, byte[] outBuffer) {
if(source == null || outBuffer == null) return 0;
if(bufferLength == 0 ) return 0;
var sbuf = gcA(source);
var dbuf = gcA(outBuffer);
IntPtr p = new IntPtr(sbuf.AddrOfPinnedObject().ToInt64() + offset);
int res = zipUnGzip2( p, bufferLength, dbuf.AddrOfPinnedObject(), outBuffer.Length );
sbuf.Free(); dbuf.Free();
return res;
}
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
// GetOrCreate a gzip file in the file system.
//
// returns 1 on success. -8 when canceled.
//
// inFile : the input file to compress.
// outFile : the output path of the created gzip. If null, the input file + ".gz" will be used.
// level : level of compression (1 - 10).
// progress : a ulong single item array that will report how many bytes have been processed. It should equal the uncompressed size.
// addHeader : if gzip header and footer should be added. If set to false the gzip can still be extracted but not opened from other decompression apps.
//
// error codes : -10 could not open input file
// : -11 could not write output file
// : -1 general error
// : -8 canceled. (Make sure you delete the canceled gz file)
public static int gzipFile(string inFile, string outFile = null, int level = 9, ulong[] progress = null, bool addHeader = true) {
int res = -1;
if(level < 1) level = 1; if(level > 10) level = 10;
if(outFile == null) outFile = inFile + ".gz";
if(progress != null) {
var prog = gcA(progress);
res = gzip_File(@inFile.Replace("//", "/"), @outFile.Replace("//", "/"), level, prog.AddrOfPinnedObject(), addHeader);
prog.Free();
} else {
res = gzip_File(@inFile.Replace("//", "/"), @outFile.Replace("//", "/"), level, IntPtr.Zero, addHeader);
}
if(res == 0) return 1; else return res;
}
// Decompress a gzip file
//
// returns 1 on success. -8 when canceled.
//
// inFile : the gzip file to extract.
// outFile : the output path of the extracted file. If null, the original name will be used.
// progress : a ulong single item array that will report how many bytes have been processed. It should equal the compressed size of the gz file.
//
// error codes : -11 could not open input file
// : -12 could not write output file
// : -3 error reading gz file
// : -4 error writing output
// : -8 canceled. (Make sure you delete the canceled output file)
public static int ungzipFile(string inFile, string outFile = null, ulong[] progress = null) {
int res = -1;
if(outFile == null) {
if(inFile.Substring(inFile.Length -3, 3).ToLower() != ".gz") { Debug.Log("Input file does not have a .gz extension"); return -2; }
outFile = inFile.Substring(0, inFile.Length -3);
}
if(progress != null) {
var prog = gcA(progress);
res = ungzip_File(@inFile.Replace("//", "/"), @outFile.Replace("//", "/"), prog.AddrOfPinnedObject());
prog.Free();
} else {
res = ungzip_File(@inFile.Replace("//", "/"), @outFile.Replace("//", "/"), IntPtr.Zero);
}
return res;
}
#endif
//---------------------------------------------------------------------------------------------------------------------------
// END GZIP SECTION
//---------------------------------------------------------------------------------------------------------------------------
//---------------------------------------------------------------------------------------------------------------------------
// START BZ2 SECTION
//---------------------------------------------------------------------------------------------------------------------------
// GetOrCreate a bz2 file in the file system.
//
// returns 1 on success.
//
// inFile: the input file to compress.
// outFile: the output path of the created bz2. If null, the input file + ".bz2" will be used.
// level: level of compression (0 - 9).
// progress: a ulong single item array that will report how many bytes have been processed. It should equal the uncompressed size.
//
// error codes : 1 success
// : -3 could not read input file
// : -4 could not create output file
// : -8 canceled
#if (!UNITY_WEBGL && !UNITY_TVOS) || UNITY_EDITOR
public static int bz2Create(string inFile, string outFile = null, int level = 9, ulong[] byteProgress = null) {
int res = -10;
if(outFile == null) outFile = inFile + ".bz2";
if(byteProgress != null) {
var prog = gcA(byteProgress);
res = bz2(false, level, @inFile.Replace("//", "/"), @outFile.Replace("//", "/"), prog.AddrOfPinnedObject());
prog.Free();
} else {
res = bz2(false, level, @inFile.Replace("//", "/"), @outFile.Replace("//", "/"), IntPtr.Zero);
}
return res;
}
// Decompress a bz2 file
//
// returns 1 on success.
//
// inFile : the bz2 file to extract.
// outFile : the output path of the extracted file. If null, the original name will be used.
// progress : a ulong single item array that will report how many bytes have been processed. It should equal the compressed size of the bz2 file.
//
// error codes : 1 success
// : -1 could not create output file
// : -2 could not read input file
// : -8 canceled
public static int bz2Decompress(string inFile, string outFile = null, ulong[] byteProgress = null) {
int res = -10;
if(outFile == null) {
if(inFile.Substring(inFile.Length -4, 4).ToLower() != ".bz2") { Debug.Log("Input file does not have a .bz2 extension"); return -2; }
outFile = inFile.Substring(0, inFile.Length -4);
}
if(byteProgress != null) {
var prog = gcA(byteProgress);
res = bz2(true, 0, @inFile.Replace("//", "/"), @outFile.Replace("//", "/"), prog.AddrOfPinnedObject());
prog.Free();
} else {
res = bz2(true, 0, @inFile.Replace("//", "/"), @outFile.Replace("//", "/"), IntPtr.Zero);
}
return res;
}
#endif
//---------------------------------------------------------------------------------------------------------------------------
// END BZ2 SECTION
//---------------------------------------------------------------------------------------------------------------------------
// A Coroutine to download a file to a native/unmaged memory buffer.
// You can call it for an lzip.inMemory class or for an IntPtr.
// See testZip.cs or testWebGLtvOS.cs for usage examples.
//
// This function can only be called for one file at a time. Don't use it to call multiple files at once.
//
// This is useful to avoid memory spikes when downloading large files and intend to decompress from memory.
// With the old method, a copy of the downloaded file to memory would be produced by pinning the buffer to memory.
// Now with this method, it is downloaded to memory and an inMemory zip structure is created. You can decompress from there, with no memory spikes and
// delete the inMemory zip, or keep it for further usage.
//
// In any case, if you don't need the created inMemory zip, you should use the lzip.free_inmemory or lzip.releaseBuffer functions to free the memory!
//
// (You can modify it or replace it with one that suits you better)
//
// Keep in mind that the server where your files are should allow reporting of file sizes as defined in the .htaccess server file.
//
// Parameters:
//
// url: The url of the file you want to download to a native memory buffer.
// downloadDone: Informs a bool that the download of the file to memory is done.
// inmem: An lzip.inMemory class to get the data.
// pointer: An IntPtr for a native memory buffer
// fileSize: The size of the downloaded file will be returned here.
public static IEnumerator downloadZipFileNative(string url, Action<bool> downloadDone, Action<lzip.inMemory> inmem, Action<IntPtr> pointer = null, Action<int> fileSize = null) {
// Get the file lenght first, so we create a correct size native memory buffer.
UnityWebRequest wr = UnityWebRequest.Head(url);
lzip.nativeBufferIsBeingUsed = true;
yield return wr.SendWebRequest();
string size = wr.GetResponseHeader("Content-Length");
lzip.nativeBufferIsBeingUsed = false;
#if UNITY_2020_1_OR_NEWER
if (wr.result == UnityWebRequest.Result.ConnectionError || wr.result == UnityWebRequest.Result.ProtocolError) {
#else
if (wr.isNetworkError || wr.isHttpError) {
#endif
Debug.LogError("Error While Getting Length: " + wr.error);
} else {
if (!lzip.nativeBufferIsBeingUsed) {
//get the size of the zip
int zipSize = Convert.ToInt32(size);
// If the zip size is larger then 0
if (zipSize > 0) {
lzip.nativeBuffer = lzip.createBuffer(zipSize);
lzip.nativeBufferIsBeingUsed = true;
// buffer for the download
byte[] bytes = new byte[2048];
lzip.nativeOffset = 0;
using (UnityWebRequest wwwSK = UnityWebRequest.Get(url)) {
// Here we call our custom webrequest function to download our archive to a native memory buffer.
wwwSK.downloadHandler = new lzip.CustomWebRequest(bytes);
yield return wwwSK.SendWebRequest();
if (wwwSK.error != null) {
Debug.Log(wwwSK.error);
} else {
downloadDone(true);
if(inmem != null) {
// create a new inMemory struct to pass as output to the Action param.
lzip.inMemory t = new lzip.inMemory();
t.pointer = lzip.nativeBuffer;
t.info[0] = zipSize;
inmem(t);
}
if(pointer != null) { pointer(lzip.nativeBuffer); fileSize(zipSize); }
//reset lzip intermediate buffer params.
lzip.nativeBufferIsBeingUsed = false;
lzip.nativeOffset = 0;
lzip.nativeBuffer = IntPtr.Zero;
//Debug.Log("Custom download done");
}
}
}
} else { Debug.LogError("Native buffer is being used, or not yet freed!"); }
}
}
// A custom WebRequest Override to download data to a native-unmanaged memory buffer.
public class CustomWebRequest : DownloadHandlerScript {
public CustomWebRequest()
: base()
{
}
public CustomWebRequest(byte[] buffer)
: base(buffer)
{
}
protected override byte[] GetData() { return null; }
protected override bool ReceiveData(byte[] bytesFromServer, int dataLength)
{
if (bytesFromServer == null || bytesFromServer.Length < 1)
{
Debug.Log("CustomWebRequest: Received a null/empty buffer");
return false;
}
var pbuf = gcA(bytesFromServer);
//Process byteFromServer
addToBuffer(nativeBuffer, nativeOffset, pbuf.AddrOfPinnedObject(), dataLength );
nativeOffset += dataLength;
pbuf.Free();
return true;
}
// Use the below functions on ly when needed. You get the same functionality from the main coroutine.
/*
// If all data has been received from the server
protected override void CompleteContent()
{
//Debug.Log("Download Complete.");
}
// If a Content-Length header is received from the server.
protected override void ReceiveContentLength(int fileLength)
{
//Debug.Log("ReceiveContentLength: " + fileLength);
}
*/
}
#endif
}