| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363 |
- package tools
- import (
- "bytes"
- "fmt"
- "github.com/axgle/mahonia"
- "github.com/emersion/go-imap"
- "github.com/emersion/go-imap/client"
- "github.com/emersion/go-message/mail"
- "io"
- "io/ioutil"
- "log"
- "mime"
- "strconv"
- "strings"
- )
- //验证邮箱密码
- func CheckEmailPassword(server string, email string, password string) bool {
- if !strings.Contains(server, ":") {
- return false
- }
- var c *client.Client
- serverSlice := strings.Split(server, ":")
- port, _ := strconv.Atoi(serverSlice[1])
- if port != 993 && port != 143 {
- return false
- }
- // 不要忘了退出
- //defer c.Logout()
- // 登陆
- c = connect(server, email, password)
- if c == nil {
- return false
- }
- return true
- }
- //获取连接
- func connect(server string, email string, password string) *client.Client {
- var c *client.Client
- var err error
- serverSlice := strings.Split(server, ":")
- uri := serverSlice[0]
- port, _ := strconv.Atoi(serverSlice[1])
- if port != 993 && port != 143 {
- return nil
- }
- if port == 993 {
- c, err = client.DialTLS(fmt.Sprintf("%s:%d", uri, port), nil)
- } else {
- c, err = client.Dial(fmt.Sprintf("%s:%d", uri, port))
- }
- if err != nil {
- return nil
- }
- // 登陆
- if err := c.Login(email, password); err != nil {
- return nil
- }
- return c
- }
- //获取邮件总数
- func GetMailNum(server string, email string, password string) map[string]int {
- var c *client.Client
- //defer c.Logout()
- c = connect(server, email, password)
- if c == nil {
- return nil
- }
- // 列邮箱
- mailboxes := make(chan *imap.MailboxInfo, 10)
- done := make(chan error, 1)
- go func() {
- done <- c.List("", "*", mailboxes)
- }()
- //// 存储邮件夹
- var folders = make(map[string]int)
- for m := range mailboxes {
- folders[m.Name] = 0
- }
- for m, _ := range folders {
- mbox, _ := c.Select(m, true)
- if mbox != nil {
- folders[m] = int(mbox.Messages)
- }
- }
- return folders
- }
- //获取邮件夹
- func GetFolders(server string, email string, password string, folder string) map[string]int {
- var c *client.Client
- //defer c.Logout()
- c = connect(server, email, password)
- if c == nil {
- return nil
- }
- // 列邮箱
- mailboxes := make(chan *imap.MailboxInfo, 10)
- done := make(chan error, 1)
- go func() {
- done <- c.List("", "*", mailboxes)
- }()
- // 存储邮件夹
- var folders = make(map[string]int)
- for m := range mailboxes {
- folders[m.Name] = 0
- }
- for m, _ := range folders {
- if m == folder {
- mbox, _ := c.Select(m, true)
- if mbox != nil {
- folders[m] = int(mbox.Messages)
- }
- break
- }
- }
- //log.Println(folders)
- return folders
- }
- //获取邮件夹邮件
- func GetFolderMail(server string, email string, password string, folder string, currentPage int, pagesize int) []*MailItem {
- var c *client.Client
- //defer c.Logout()
- c = connect(server, email, password)
- if c == nil {
- return nil
- }
- mbox, _ := c.Select(folder, true)
- to := mbox.Messages - uint32((currentPage-1)*pagesize)
- from := to - uint32(pagesize)
- if to <= uint32(pagesize) {
- from = 1
- }
- seqset := new(imap.SeqSet)
- seqset.AddRange(from, to)
- messages := make(chan *imap.Message, pagesize)
- done := make(chan error, 1)
- fetchItem := imap.FetchItem(imap.FetchEnvelope)
- items := make([]imap.FetchItem, 0)
- items = append(items, fetchItem)
- go func() {
- done <- c.Fetch(seqset, items, messages)
- }()
- var mailPagelist = new(MailPageList)
- dec := GetDecoder()
- for msg := range messages {
- log.Println(msg.Envelope.Date)
- ret, err := dec.Decode(msg.Envelope.Subject)
- if err != nil {
- ret, _ = dec.DecodeHeader(msg.Envelope.Subject)
- }
- var mailitem = new(MailItem)
- mailitem.Subject = ret
- mailitem.Id = msg.SeqNum
- mailitem.Fid = folder
- mailitem.Date = msg.Envelope.Date.String()
- from := ""
- for _, s := range msg.Envelope.Sender {
- from += s.Address()
- }
- mailitem.From = from
- mailPagelist.MailItems = append(mailPagelist.MailItems, mailitem)
- }
- return mailPagelist.MailItems
- }
- func GetMessage(server string, email string, password string, folder string, id uint32) *MailItem {
- var c *client.Client
- //defer c.Logout()
- c = connect(server, email, password)
- if c == nil {
- //return nil
- }
- // Select INBOX
- mbox, err := c.Select(folder, false)
- if err != nil {
- log.Fatal(err)
- }
- // Get the last message
- if mbox.Messages == 0 {
- log.Fatal("No message in mailbox")
- }
- seqSet := new(imap.SeqSet)
- seqSet.AddNum(id)
- // Get the whole message body
- section := &imap.BodySectionName{}
- items := []imap.FetchItem{section.FetchItem()}
- messages := make(chan *imap.Message, 1)
- go func() {
- if err := c.Fetch(seqSet, items, messages); err != nil {
- log.Fatal(err)
- }
- }()
- msg := <-messages
- if msg == nil {
- log.Fatal("Server didn't returned message")
- }
- r := msg.GetBody(section)
- if r == nil {
- log.Fatal("Server didn't returned message body")
- }
- var mailitem = new(MailItem)
- // Create a new mail reader
- mr, _ := mail.CreateReader(r)
- // Print some info about the message
- header := mr.Header
- date, _ := header.Date()
- mailitem.Date = date.String()
- var f string
- dec := GetDecoder()
- if from, err := header.AddressList("From"); err == nil {
- for _, address := range from {
- fromStr := address.String()
- temp, _ := dec.DecodeHeader(fromStr)
- f += " " + temp
- }
- }
- mailitem.From = f
- log.Println("From:", mailitem.From)
- var t string
- if to, err := header.AddressList("To"); err == nil {
- log.Println("To:", to)
- for _, address := range to {
- toStr := address.String()
- temp, _ := dec.DecodeHeader(toStr)
- t += " " + temp
- }
- }
- mailitem.To = t
- subject, _ := header.Subject()
- s, err := dec.Decode(subject)
- if err != nil {
- s, _ = dec.DecodeHeader(subject)
- }
- log.Println("Subject:", s)
- mailitem.Subject = s
- // Process each message's part
- var bodyMap = make(map[string]string)
- bodyMap["text/plain"] = ""
- bodyMap["text/html"] = ""
- for {
- p, err := mr.NextPart()
- if err == io.EOF {
- break
- } else if err != nil {
- //log.Fatal(err)
- }
- switch h := p.Header.(type) {
- case *mail.InlineHeader:
- // This is the message's text (can be plain-text or HTML)
- b, _ := ioutil.ReadAll(p.Body)
- ct := p.Header.Get("Content-Type")
- if strings.Contains(ct, "text/plain") {
- bodyMap["text/plain"] += Encoding(string(b), ct)
- } else {
- bodyMap["text/html"] += Encoding(string(b), ct)
- }
- //body,_:=dec.Decode(string(b))
- case *mail.AttachmentHeader:
- // This is an attachment
- filename, _ := h.Filename()
- log.Println("Got attachment: ", filename)
- }
- }
- if bodyMap["text/html"] != "" {
- mailitem.Body = bodyMap["text/html"]
- } else {
- mailitem.Body = bodyMap["text/plain"]
- }
- //log.Println(mailitem.Body)
- return mailitem
- }
- func GetDecoder() *mime.WordDecoder {
- dec := new(mime.WordDecoder)
- dec.CharsetReader = func(charset string, input io.Reader) (io.Reader, error) {
- charset = strings.ToLower(charset)
- switch charset {
- case "gb2312":
- content, err := ioutil.ReadAll(input)
- if err != nil {
- return nil, err
- }
- //ret:=bytes.NewReader(content)
- //ret:=transform.NewReader(bytes.NewReader(content), simplifiedchinese.HZGB2312.NewEncoder())
- utf8str := ConvertToStr(string(content), "gbk", "utf-8")
- t := bytes.NewReader([]byte(utf8str))
- //ret:=utf8.DecodeRune(t)
- //log.Println(ret)
- return t, nil
- case "gbk":
- content, err := ioutil.ReadAll(input)
- if err != nil {
- return nil, err
- }
- //ret:=bytes.NewReader(content)
- //ret:=transform.NewReader(bytes.NewReader(content), simplifiedchinese.HZGB2312.NewEncoder())
- utf8str := ConvertToStr(string(content), "gbk", "utf-8")
- t := bytes.NewReader([]byte(utf8str))
- //ret:=utf8.DecodeRune(t)
- //log.Println(ret)
- return t, nil
- case "gb18030":
- content, err := ioutil.ReadAll(input)
- if err != nil {
- return nil, err
- }
- //ret:=bytes.NewReader(content)
- //ret:=transform.NewReader(bytes.NewReader(content), simplifiedchinese.HZGB2312.NewEncoder())
- utf8str := ConvertToStr(string(content), "gbk", "utf-8")
- t := bytes.NewReader([]byte(utf8str))
- //ret:=utf8.DecodeRune(t)
- //log.Println(ret)
- return t, nil
- default:
- return nil, fmt.Errorf("unhandle charset:%s", charset)
- }
- }
- return dec
- }
- // 任意编码转特定编码
- func ConvertToStr(src string, srcCode string, tagCode string) string {
- result := mahonia.NewDecoder(srcCode).ConvertString(src)
- //srcCoder := mahonia.NewDecoder(srcCode)
- //srcResult := srcCoder.ConvertString(src)
- //tagCoder := mahonia.NewDecoder(tagCode)
- //_, cdata, _ := tagCoder.Translate([]byte(srcResult), true)
- //result := string(cdata)
- return result
- }
|