-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathpaktool.go
398 lines (324 loc) · 8.43 KB
/
paktool.go
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
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
package main
import (
"bytes"
"crypto/sha1"
"encoding/binary"
"encoding/hex"
"encoding/json"
"flag"
"fmt"
"io"
"io/ioutil"
"log"
"os"
"path/filepath"
)
const (
PakVersion = 5
PakEncoding = 1
)
type PakHeader struct {
Version uint32
Encodeing uint32
ResourceCount uint16
AliasCount uint16
}
type PakEntryRaw struct {
ResourceId uint16
FileOffset uint32
}
type PakEntry struct {
ResourceId uint16
FileOffset uint32
NextResourceId uint16
NextFileOffset uint32
}
type PakAlias struct {
ResourceId uint16
EntryIndex uint16
}
func SHA1(data []byte) string {
a := sha1.Sum(data)
return hex.EncodeToString(a[:])
}
type EntryNode struct {
ResourceId uint16 `json:"id"`
Path string `json:"path"`
// Gzip bool `json:"gzip"`
}
type AliasNode struct {
ResourceId uint16 `json:"id"`
EntryIndex uint16 `json:"index"`
}
type MetaRecord struct {
Entry []EntryNode `json:"entry"`
Alias []AliasNode `json:"alias"`
}
type LangNode struct {
ResourceId uint16 `json:"id"`
Text string `json:"text"`
// Gzip bool `json:"gzip"`
}
type LangRecord struct {
Entry []LangNode `json:"entry"`
Alias []AliasNode `json:"alias"`
}
func ToJson(data interface{}) []byte {
buf := new(bytes.Buffer)
encoder := json.NewEncoder(buf)
encoder.SetEscapeHTML(false)
encoder.SetIndent("", "\t")
_ = encoder.Encode(data)
return buf.Bytes()
}
func SaveToFile(path string, buf []byte) {
os.MkdirAll(filepath.Dir(path), os.ModePerm)
ioutil.WriteFile(path, []byte(buf), os.ModePerm)
}
func unpack(path string) {
name := filepath.Ext(path)
name = path[0 : len(path)-len(name)]
f, err := os.Open(path)
if err != nil {
log.Fatal(err)
}
defer f.Close()
// 检查文件头
var h PakHeader
err = binary.Read(f, binary.LittleEndian, &h)
if h.Version != PakVersion || h.Encodeing != PakEncoding {
log.Fatal("invalid pak file")
}
// fmt.Printf("%+v\n", h)
// 加载识别数据库
db, err := Asset("assets/res_sha1.json")
var json_db map[string]string
json.Unmarshal(db, &json_db)
write_files := make(map[string]bool)
var mr MetaRecord
var r PakEntryRaw
for i := 0; i < int(h.ResourceCount); i++ {
// 读取索引
f.Seek(int64(binary.Size(h)+i*binary.Size(r)), io.SeekStart)
var e PakEntry
binary.Read(f, binary.LittleEndian, &e)
// 读取内容
f.Seek(int64(e.FileOffset), io.SeekStart)
data := make([]byte, e.NextFileOffset-e.FileOffset)
f.Read(data)
// 自动判断名称
path, err := json_db[SHA1(data)]
if !err {
path = fmt.Sprintf("unknown/%d", e.ResourceId)
}
// 检查重复名称
_, ok := write_files[path]
if ok {
path = fmt.Sprintf("unknown/%d", e.ResourceId)
}
write_files[path] = true
// 写入文件
fmt.Println("write " + path)
path = name + "/" + path
os.MkdirAll(filepath.Dir(path), os.ModePerm)
ioutil.WriteFile(path, data, os.ModePerm)
en := EntryNode{e.ResourceId, path}
mr.Entry = append(mr.Entry, en)
}
for i := 0; i < int(h.AliasCount); i++ {
var a PakAlias
// 读取索引
f.Seek(int64(binary.Size(h)+int(h.ResourceCount+1)*binary.Size(r)+i*binary.Size(a)), io.SeekStart)
binary.Read(f, binary.LittleEndian, &a)
an := AliasNode{a.ResourceId, a.EntryIndex}
mr.Alias = append(mr.Alias, an)
}
output := name + ".json"
SaveToFile(output, ToJson(mr))
fmt.Printf("\nwrite %s ok\n", output)
}
func ReadFromFile(path string) []byte {
data, err := ioutil.ReadFile(path)
if err != nil {
log.Fatal(err)
}
return data
}
func GetFileSize(path string) uint32 {
fi, err := os.Stat(path)
if err != nil {
log.Fatal(err)
}
return uint32(fi.Size())
}
func repack(path string) {
result := MetaRecord{}
err := json.Unmarshal(ReadFromFile(path), &result)
if err != nil {
fmt.Println(err)
return
}
name := filepath.Ext(path)
name = path[0 : len(path)-len(name)]
output := name + ".pak"
f, err := os.Create(output)
if err != nil {
log.Fatal(err)
}
defer f.Close()
// 写入文件头
var h PakHeader
h.Version = PakVersion
h.Encodeing = PakEncoding
h.ResourceCount = uint16(len(result.Entry))
h.AliasCount = uint16(len(result.Alias))
binary.Write(f, binary.LittleEndian, &h)
var r PakEntryRaw
var a PakAlias
//写入entry
var offset = uint32(binary.Size(h)) + uint32(h.ResourceCount+1)*uint32(binary.Size(r)) + uint32(h.AliasCount)*uint32(binary.Size(a))
for _, v := range result.Entry {
r.ResourceId = v.ResourceId
r.FileOffset = offset
offset += GetFileSize(v.Path)
binary.Write(f, binary.LittleEndian, &r)
// fmt.Printf("%+v %+v %+v \n", i, offset, GetFileSize(v.Path))
}
//写入entry末尾
r.ResourceId = 0
r.FileOffset = offset
binary.Write(f, binary.LittleEndian, &r)
//写入alias
for _, v := range result.Alias {
a.ResourceId = v.ResourceId
a.EntryIndex = v.EntryIndex
binary.Write(f, binary.LittleEndian, &a)
// fmt.Printf("%+v %+v\n", a.ResourceId, a.EntryIndex)
}
//写入文件内容
for _, v := range result.Entry {
f.Write(ReadFromFile(v.Path))
}
fmt.Printf("repack %s ok\n", output)
}
func lang_unpack(path string) {
name := filepath.Ext(path)
name = path[0 : len(path)-len(name)]
f, err := os.Open(path)
if err != nil {
log.Fatal(err)
}
defer f.Close()
// 检查文件头
var h PakHeader
err = binary.Read(f, binary.LittleEndian, &h)
if h.Version != PakVersion || h.Encodeing != PakEncoding {
log.Fatal("invalid pak file")
}
var lr LangRecord
var r PakEntryRaw
for i := 0; i < int(h.ResourceCount); i++ {
// 读取索引
f.Seek(int64(binary.Size(h)+i*binary.Size(r)), io.SeekStart)
var e PakEntry
binary.Read(f, binary.LittleEndian, &e)
// 读取内容
f.Seek(int64(e.FileOffset), io.SeekStart)
data := make([]byte, e.NextFileOffset-e.FileOffset)
f.Read(data)
en := LangNode{e.ResourceId, string(data)}
lr.Entry = append(lr.Entry, en)
}
for i := 0; i < int(h.AliasCount); i++ {
var a PakAlias
// 读取索引
f.Seek(int64(binary.Size(h)+int(h.ResourceCount+1)*binary.Size(r)+i*binary.Size(a)), io.SeekStart)
binary.Read(f, binary.LittleEndian, &a)
an := AliasNode{a.ResourceId, a.EntryIndex}
lr.Alias = append(lr.Alias, an)
}
output := name + ".json"
SaveToFile(output, ToJson(lr))
fmt.Printf("write %s ok\n", output)
}
func lang_repack(path string) {
result := LangRecord{}
err := json.Unmarshal(ReadFromFile(path), &result)
if err != nil {
fmt.Println(err)
return
}
name := filepath.Ext(path)
name = path[0 : len(path)-len(name)]
output := name + ".pak"
f, err := os.Create(output)
if err != nil {
log.Fatal(err)
}
defer f.Close()
// 写入文件头
var h PakHeader
h.Version = PakVersion
h.Encodeing = PakEncoding
h.ResourceCount = uint16(len(result.Entry))
h.AliasCount = uint16(len(result.Alias))
binary.Write(f, binary.LittleEndian, &h)
var r PakEntryRaw
var a PakAlias
//写入entry
var offset = uint32(binary.Size(h)) + uint32(h.ResourceCount+1)*uint32(binary.Size(r)) + uint32(h.AliasCount)*uint32(binary.Size(a))
for _, v := range result.Entry {
r.ResourceId = v.ResourceId
r.FileOffset = offset
offset += uint32(len(v.Text))
binary.Write(f, binary.LittleEndian, &r)
// fmt.Printf("%+v %+v %+v \n", i, offset, GetFileSize(v.Path))
}
//写入entry末尾
r.ResourceId = 0
r.FileOffset = offset
binary.Write(f, binary.LittleEndian, &r)
//写入alias
for _, v := range result.Alias {
a.ResourceId = v.ResourceId
a.EntryIndex = v.EntryIndex
binary.Write(f, binary.LittleEndian, &a)
// fmt.Printf("%+v %+v\n", a.ResourceId, a.EntryIndex)
}
//写入文件内容
for _, v := range result.Entry {
f.Write([]byte(v.Text))
}
fmt.Printf("repack %s ok\n", output)
}
func usage() {
fmt.Println("usage:")
fmt.Println("pak_tools -c=unpack -f=resources.pak")
fmt.Println("\tunpack files in resources.pak to resources.json and resources folder")
fmt.Println("pak_tools -c=repack -f=resources.json")
fmt.Println("\trepack files to resources.pak according to resources.json")
fmt.Println("pak_tools -c=lang_unpack -f=zh-CN.pak")
fmt.Println("\textract the text in zh-CN.pak to zh-CN.json")
fmt.Println("pak_tools -c=lang_repack -f=zh-CN.json")
fmt.Println("\trepack text to zh-CN.pak from zh-CN.json")
var input string
fmt.Scanln(&input)
}
func main() {
if len(os.Args) < 3 {
usage()
return
}
command := flag.String("c", "", "command")
file := flag.String("f", "", "file path")
flag.Parse()
if *command == "unpack" {
unpack(*file)
} else if *command == "repack" {
repack(*file)
} else if *command == "lang_unpack" {
lang_unpack(*file)
} else if *command == "lang_repack" {
lang_repack(*file)
}
}