-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathmain.go
157 lines (135 loc) · 3.97 KB
/
main.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
package main
import (
"bytes"
"encoding/binary"
"errors"
"fmt"
"io"
"log"
"os"
"github.com/davecgh/go-spew/spew"
"github.com/ghostiam/binstruct"
)
// .ZIP File Format Specification: https://pkware.cachefly.net/webdocs/casestudies/APPNOTE.TXT
func main() {
file, err := os.Open("sample.zip")
if err != nil {
log.Fatal(err)
}
var zip ZIP
decoder := binstruct.NewDecoder(file, binary.LittleEndian)
// decoder.SetDebug(true)
err = decoder.Decode(&zip)
if err != nil {
log.Fatal(err)
}
spew.Dump(zip)
}
type ZIP struct {
_ byte `bin:"ParseZIPSections"` // Call helper method for scan
LocalFileSections []ZIPLocalFileSection `bin:"-"`
CentralDirEntrySections []ZIPCentralDirEntrySection `bin:"-"`
EndOfCentralDirSection ZIPEndOfCentralDirSection `bin:"-"`
}
func (zip *ZIP) ParseZIPSections(r binstruct.Reader) error {
for {
// Find magic PK (0x50 0x4B)
var magicPrevByte byte
for {
b, err := r.ReadByte()
if errors.Is(err, io.EOF) {
return nil
}
if err != nil {
return fmt.Errorf("failed read magic PK: %w", err)
}
if magicPrevByte == 'P' && b == 'K' {
magicPrevByte = 0x00
break // exit from loop
}
magicPrevByte = b
}
// read section type
_, sectionType, err := r.ReadBytes(2)
if err != nil {
return fmt.Errorf("failed read section type: %w", err)
}
switch {
case bytes.Equal(sectionType, []byte{0x03, 0x04}):
// parse ZIPLocalFileSection
var localFileSection ZIPLocalFileSection
err = r.Unmarshal(&localFileSection)
if err != nil {
return fmt.Errorf("failed Unmarshal ZIPLocalFileSection: %w", err)
}
zip.LocalFileSections = append(zip.LocalFileSections, localFileSection)
case bytes.Equal(sectionType, []byte{0x01, 0x02}):
// parse CentralDirEntry
var centralDirEntrySections ZIPCentralDirEntrySection
err = r.Unmarshal(¢ralDirEntrySections)
if err != nil {
return fmt.Errorf("failed Unmarshal ZIPCentralDirEntrySection: %w", err)
}
zip.CentralDirEntrySections = append(zip.CentralDirEntrySections, centralDirEntrySections)
case bytes.Equal(sectionType, []byte{0x05, 0x06}):
// parse EndOfCentralDir
var endOfCentralDirSections ZIPEndOfCentralDirSection
err = r.Unmarshal(&endOfCentralDirSections)
if err != nil {
return fmt.Errorf("failed Unmarshal ZIPEndOfCentralDir: %w", err)
}
zip.EndOfCentralDirSection = endOfCentralDirSections
default:
log.Printf("unknown section type: %#x", sectionType)
}
}
}
type ZIPLocalFileSection struct {
LocalFileHeader
Body []byte `bin:"len:CompressedSize"`
}
type LocalFileHeader struct {
Version uint16
Flags [2]byte
CompressionMethod uint16
FileModTime uint16
FileModDate uint16
Crc32 [4]byte
CompressedSize uint32
UncompressedSize uint32
FileNameLen uint16
ExtraLen uint16
FileName string `bin:"len:FileNameLen"`
Extra []byte `bin:"len:ExtraLen"`
}
type ZIPCentralDirEntrySection struct {
VersionMadeBy int16
VersionNeededToExtract int16
Flags [2]byte
CompressionMethod int16
LastModFileTime int16
LastModFileDate int16
Crc32 [4]byte
CompressedSize int32
UncompressedSize int32
FileNameLen int16
ExtraLen int16
CommentLen int16
DiskNumberStart int16
IntFileAttr int16
ExtFileAttr int32
LocalHeaderOffset int32
FileName string `bin:"len:FileNameLen"`
Extra []byte `bin:"len:ExtraLen"`
Comment string `bin:"len:CommentLen"`
}
type ZIPEndOfCentralDirSection struct {
DiskOfEndOfCentralDir int16
DiskOfCentralDir int16
QtyCentralDirEntriesOnDisk int16
QtyCentralDirEntriesTotal int16
CentralDirSize int32
CentralDirOffset int32
CommentLen int16
Comment string `bin:"len:CommentLen"`
}