forked from K0lb3/UnityPy
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathCompressionHelper.py
More file actions
137 lines (111 loc) · 3.43 KB
/
CompressionHelper.py
File metadata and controls
137 lines (111 loc) · 3.43 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
import gzip
import lzma
import struct
import brotli
import lz4.block
GZIP_MAGIC: bytes = b"\x1f\x8b"
BROTLI_MAGIC: bytes = b"brotli"
# LZMA
def decompress_lzma(data: bytes) -> bytes:
"""decompresses lzma-compressed data
:param data: compressed data
:type data: bytes
:raises _lzma.LZMAError: Compressed data ended before the end-of-stream marker was reached
:return: uncompressed data
:rtype: bytes
"""
props, dict_size = struct.unpack("<BI", data[:5])
lc = props % 9
props = props // 9
pb = props // 5
lp = props % 5
dec = lzma.LZMADecompressor(
format=lzma.FORMAT_RAW,
filters=[
{
"id": lzma.FILTER_LZMA1,
"dict_size": dict_size,
"lc": lc,
"lp": lp,
"pb": pb,
}
],
)
return dec.decompress(data[5:])
def compress_lzma(data: bytes) -> bytes:
"""compresses data via lzma (unity specific)
The current static settings may not be the best solution,
but they are the most commonly used values and should therefore be enough for the time being.
:param data: uncompressed data
:type data: bytes
:return: compressed data
:rtype: bytes
"""
ec = lzma.LZMACompressor(
format=lzma.FORMAT_RAW,
filters=[
{"id": lzma.FILTER_LZMA1, "dict_size": 524288, "lc": 3, "lp": 0, "pb": 2, }
],
)
ec.compress(data)
return b"]\x00\x00\x08\x00" + ec.flush()
# LZ4
def decompress_lz4(data: bytes, uncompressed_size: int) -> bytes: # LZ4M/LZ4HC
"""decompresses lz4-compressed data
:param data: compressed data
:type data: bytes
:param uncompressed_size: size of the uncompressed data
:type uncompressed_size: int
:raises _block.LZ4BlockError: Decompression failed: corrupt input or insufficient space in destination buffer.
:return: uncompressed data
:rtype: bytes
"""
return lz4.block.decompress(data, uncompressed_size)
def compress_lz4(data: bytes) -> bytes: # LZ4M/LZ4HC
"""compresses data via lz4.block
:param data: uncompressed data
:type data: bytes
:return: compressed data
:rtype: bytes
"""
return lz4.block.compress(
data, mode="high_compression", compression=9, store_size=False
)
# Brotli
def decompress_brotli(data: bytes) -> bytes:
"""decompresses brotli-compressed data
:param data: compressed data
:type data: bytes
:raises brotli.error: BrotliDecompress failed
:return: uncompressed data
:rtype: bytes
"""
return brotli.decompress(data)
def compress_brotli(data: bytes) -> bytes:
"""compresses data via brotli
:param data: uncompressed data
:type data: bytes
:return: compressed data
:rtype: bytes
"""
return brotli.compress(data)
# GZIP
def decompress_gzip(data: bytes) -> bytes:
"""decompresses gzip-compressed data
:param data: compressed data
:type data: bytes
:raises OSError: Not a gzipped file
:return: uncompressed data
:rtype: bytes
"""
return gzip.decompress(data)
def compress_gzip(data: bytes) -> bytes:
"""compresses data via gzip
The current static settings may not be the best solution,
but they are the most commonly used values and should therefore be enough for the time being.
:param data: uncompressed data
:type data: bytes
:return: compressed data
:rtype: bytes
"""
return gzip.compress(data)