-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmime.go
102 lines (90 loc) · 2.4 KB
/
mime.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
package main
import (
"bytes"
"encoding/base64"
"fmt"
"io"
"io/ioutil"
"mime"
"mime/multipart"
"mime/quotedprintable"
"net/mail"
"strings"
)
type MimeFile struct {
Filename string
Content []byte
}
type MimeFiles []MimeFile
func BuildFileName(part *multipart.Part, radix string, index int) (filename string) {
filename = part.FileName()
if len(filename) > 0 {
return
}
mediaType, _, err := mime.ParseMediaType(part.Header.Get("Content-Type"))
if err == nil {
mime_type, e := mime.ExtensionsByType(mediaType)
if e == nil {
return fmt.Sprintf("%s-%d%s", radix, index, mime_type[0])
}
}
return
}
// WitePart decodes the data of MIME part and writes it to the file filename.
func WritePart(part *multipart.Part) []byte {
part_data, err := ioutil.ReadAll(part)
if err != nil {
return nil
}
content_transfer_encoding := strings.ToUpper(part.Header.Get("Content-Transfer-Encoding"))
switch {
case strings.Compare(content_transfer_encoding, "BASE64") == 0:
if decoded_content, err := base64.StdEncoding.DecodeString(string(part_data)); err != nil {
return decoded_content
}
case strings.Compare(content_transfer_encoding, "QUOTED-PRINTABLE") == 0:
if decoded_content, err := ioutil.ReadAll(quotedprintable.NewReader(bytes.NewReader(part_data))); err != nil {
return decoded_content
}
default:
return part_data
}
return nil
}
func ParsePart(out *MimeFiles, mime_data io.Reader, boundary string, index int) {
if boundary == "" {
bytes, _ := io.ReadAll(mime_data)
*out = append(*out, MimeFile{Filename: "file", Content: bytes})
return
}
reader := multipart.NewReader(mime_data, boundary)
if reader == nil {
return
}
for {
new_part, err := reader.NextPart()
if err == io.EOF || err != nil {
break
}
mediaType, params, err := mime.ParseMediaType(new_part.Header.Get("Content-Type"))
if err == nil && strings.HasPrefix(mediaType, "multipart/") {
ParsePart(out, new_part, params["boundary"], index+1)
} else {
filename := BuildFileName(new_part, boundary, 1)
*out = append(*out, MimeFile{Filename: filename, Content: WritePart(new_part)})
}
}
}
func ParseMail(r io.Reader) (MimeFiles, error) {
m, err := mail.ReadMessage(r)
if err != nil {
return nil, err
}
_, params, err := mime.ParseMediaType(m.Header.Get("Content-Type"))
if err != nil {
return nil, err
}
var files MimeFiles
ParsePart(&files, m.Body, params["boundary"], 1)
return files, nil
}