/
decode.go
110 lines (100 loc) 路 2.59 KB
/
decode.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
package imap
import (
"bytes"
"crypto/md5"
"encoding/base64"
"fmt"
"io"
"io/ioutil"
"mime"
"mime/multipart"
"mime/quotedprintable"
"net/mail"
log "github.com/sirupsen/logrus"
"github.com/yesnault/go-imap/imap"
"github.com/ovh/venom"
)
func decodeHeader(msg *mail.Message, headerName string) (string, error) {
dec := new(mime.WordDecoder)
s, err := dec.DecodeHeader(msg.Header.Get(headerName))
if err != nil {
return msg.Header.Get(headerName), fmt.Errorf("Error while decode header %s:%s", headerName, msg.Header.Get(headerName))
}
return s, nil
}
func hash(in string) string {
h2 := md5.New()
io.WriteString(h2, in)
return fmt.Sprintf("%x", h2.Sum(nil))
}
func extract(rsp imap.Response, l venom.Logger) (*Mail, error) {
tm := &Mail{}
header := imap.AsBytes(rsp.MessageInfo().Attrs["RFC822.HEADER"])
tm.UID = imap.AsNumber((rsp.MessageInfo().Attrs["UID"]))
body := imap.AsBytes(rsp.MessageInfo().Attrs["RFC822.TEXT"])
mmsg, err := mail.ReadMessage(bytes.NewReader(header))
if err != nil {
return nil, err
}
tm.Subject, err = decodeHeader(mmsg, "Subject")
if err != nil {
log.Warnf("Cannot decode Subject header: %s", err)
return nil, nil
}
tm.From, err = decodeHeader(mmsg, "From")
if err != nil {
log.Warnf("Cannot decode From header: %s", err)
return nil, nil
}
tm.To, err = decodeHeader(mmsg, "To")
if err != nil {
log.Warnf("Cannot decode To header: %s", err)
return nil, nil
}
encoding := mmsg.Header.Get("Content-Transfer-Encoding")
var r io.Reader = bytes.NewReader(body)
switch encoding {
case "7bit", "8bit", "binary":
// noop, reader already initialized.
case "quoted-printable":
r = quotedprintable.NewReader(r)
case "base64":
r = base64.NewDecoder(base64.StdEncoding, r)
}
l.Debugf("Mail Content-Transfer-Encoding is %s ", encoding)
contentType, params, err := mime.ParseMediaType(mmsg.Header.Get("Content-Type"))
if err != nil {
return nil, fmt.Errorf("Error while reading Content-Type:%s", err)
}
if contentType == "multipart/mixed" || contentType == "multipart/alternative" {
if boundary, ok := params["boundary"]; ok {
mr := multipart.NewReader(r, boundary)
for {
p, errm := mr.NextPart()
if errm == io.EOF {
continue
}
if errm != nil {
l.Debugf("Error while read Part:%s", err)
break
}
slurp, errm := ioutil.ReadAll(p)
if errm != nil {
l.Debugf("Error while ReadAll Part:%s", err)
continue
}
tm.Body = string(slurp)
break
}
}
} else {
body, err = ioutil.ReadAll(r)
if err != nil {
return nil, err
}
}
if tm.Body == "" {
tm.Body = string(body)
}
return tm, nil
}