Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 package memory | |
| 6 | |
| 7 import ( | |
| 8 "fmt" | |
| 9 net_mail "net/mail" | |
| 10 "net/textproto" | |
| 11 "path/filepath" | |
| 12 "strings" | |
| 13 "sync" | |
| 14 | |
| 15 "github.com/luci/gae/service/mail" | |
| 16 "github.com/luci/gae/service/user" | |
| 17 "golang.org/x/net/context" | |
| 18 ) | |
| 19 | |
| 20 type mailData struct { | |
| 21 sync.Mutex | |
| 22 queue []*mail.TestMessage | |
| 23 admins []string | |
| 24 adminsPlain []string | |
| 25 } | |
| 26 | |
| 27 // mailImpl is a contextual pointer to the current mailData. | |
| 28 type mailImpl struct { | |
| 29 data *mailData | |
| 30 | |
| 31 c context.Context | |
| 32 } | |
| 33 | |
| 34 var _ mail.Interface = (*mailImpl)(nil) | |
| 35 | |
| 36 // useMail adds a mail.Interface implementation to context, accessible | |
| 37 // by mail.Get(c) | |
| 38 func useMail(c context.Context) context.Context { | |
| 39 data := &mailData{ | |
| 40 admins: []string{"admin@example.com"}, | |
| 41 adminsPlain: []string{"admin@example.com"}, | |
| 42 } | |
| 43 | |
| 44 return mail.SetFactory(c, func(c context.Context) mail.Interface { | |
| 45 return &mailImpl{data, c} | |
| 46 }) | |
| 47 } | |
| 48 | |
| 49 func parseEmails(emails ...string) error { | |
| 50 for _, e := range emails { | |
| 51 if _, err := net_mail.ParseAddress(e); err != nil { | |
| 52 return fmt.Errorf("invalid email (%q): %s", e, err) | |
| 53 } | |
| 54 } | |
| 55 return nil | |
| 56 } | |
| 57 | |
| 58 func checkMessage(msg *mail.TestMessage, adminsPlain []string, user string) erro r { | |
| 59 sender, err := net_mail.ParseAddress(msg.Sender) | |
| 60 if err != nil { | |
| 61 return fmt.Errorf("unparsable Sender address: %s: %s", msg.Sende r, err) | |
| 62 } | |
| 63 senderOK := user != "" && sender.Address == user | |
| 64 if !senderOK { | |
| 65 for _, a := range adminsPlain { | |
| 66 if sender.Address == a { | |
| 67 senderOK = true | |
| 68 break | |
| 69 } | |
| 70 } | |
| 71 } | |
| 72 if !senderOK { | |
| 73 return fmt.Errorf("invalid Sender: %s", msg.Sender) | |
| 74 } | |
| 75 | |
| 76 if len(msg.To) == 0 && len(msg.Cc) == 0 && len(msg.Bcc) == 0 { | |
| 77 return fmt.Errorf("one of To, Cc or Bcc must be non-empty") | |
| 78 } | |
| 79 | |
| 80 if err := parseEmails(msg.To...); err != nil { | |
| 81 return err | |
| 82 } | |
| 83 if err := parseEmails(msg.Cc...); err != nil { | |
| 84 return err | |
| 85 } | |
| 86 if err := parseEmails(msg.Bcc...); err != nil { | |
| 87 return err | |
| 88 } | |
| 89 | |
| 90 if len(msg.Body) == 0 && len(msg.HTMLBody) == 0 { | |
| 91 return fmt.Errorf("one of Body or HTMLBody must be non-empty") | |
| 92 } | |
| 93 | |
| 94 if len(msg.Attachments) > 0 { | |
| 95 msg.MIMETypes = make([]string, len(msg.Attachments)) | |
| 96 for i := range msg.Attachments { | |
| 97 n := msg.Attachments[i].Name | |
| 98 ext := strings.TrimLeft(strings.ToLower(filepath.Ext(n)) , ".") | |
| 99 if badExtensions.Has(ext) { | |
| 100 return fmt.Errorf("illegal attachment extension for %q", n) | |
| 101 } | |
| 102 mimetype := extensionMapping[ext] | |
| 103 if mimetype == "" { | |
| 104 mimetype = "application/octet-stream" | |
| 105 } | |
| 106 msg.MIMETypes[i] = mimetype | |
| 107 } | |
| 108 } | |
| 109 | |
| 110 fixKeys := map[string]string{} | |
| 111 for k := range msg.Headers { | |
| 112 canonK := textproto.CanonicalMIMEHeaderKey(k) | |
|
martiniss
2015/12/16 21:54:15
Is this the same way the app engine people do it?
iannucci
2015/12/17 00:01:07
yes, it's the same way that all mime implementatio
| |
| 113 if !okHeaders.Has(canonK) { | |
| 114 return fmt.Errorf("disallowed header: %s", k) | |
| 115 } | |
| 116 if canonK != k { | |
| 117 fixKeys[k] = canonK | |
| 118 } | |
| 119 } | |
| 120 for k, canonK := range fixKeys { | |
| 121 vals := msg.Headers[k] | |
| 122 delete(msg.Headers, k) | |
| 123 msg.Headers[canonK] = vals | |
| 124 } | |
| 125 | |
| 126 return nil | |
| 127 } | |
| 128 | |
| 129 func (m *mailImpl) sendImpl(msg *mail.Message) error { | |
| 130 email := "" | |
| 131 userSvc := user.Get(m.c) | |
| 132 if u := userSvc.Current(); u != nil { | |
| 133 email = u.Email | |
| 134 } | |
| 135 | |
| 136 m.data.Lock() | |
| 137 adminsPlain := m.data.adminsPlain[:] | |
| 138 m.data.Unlock() | |
| 139 | |
| 140 testMsg := &mail.TestMessage{Message: *msg} | |
| 141 | |
| 142 if err := checkMessage(testMsg, adminsPlain, email); err != nil { | |
| 143 return err | |
| 144 } | |
| 145 m.data.Lock() | |
| 146 m.data.queue = append(m.data.queue, testMsg) | |
| 147 m.data.Unlock() | |
| 148 return nil | |
| 149 } | |
| 150 | |
| 151 func (m *mailImpl) Send(msg *mail.Message) error { | |
| 152 return m.sendImpl(msg.Copy()) | |
| 153 } | |
| 154 | |
| 155 func (m *mailImpl) SendToAdmins(msg *mail.Message) error { | |
| 156 msg = msg.Copy() | |
| 157 m.data.Lock() | |
| 158 ads := m.data.admins[:] | |
| 159 m.data.Unlock() | |
| 160 | |
| 161 msg.To = make([]string, len(ads)) | |
| 162 copy(msg.To, ads) | |
| 163 | |
| 164 return m.sendImpl(msg) | |
| 165 } | |
| 166 | |
| 167 func (m *mailImpl) Testable() mail.Testable { | |
| 168 return m | |
| 169 } | |
| 170 | |
| 171 func (m *mailImpl) SetAdminEmails(emails ...string) { | |
| 172 adminsPlain := make([]string, len(emails)) | |
| 173 for i, e := range emails { | |
| 174 adr, err := net_mail.ParseAddress(e) | |
| 175 if err != nil { | |
| 176 panic(fmt.Errorf("invalid email (%q): %s", e, err)) | |
| 177 } | |
| 178 adminsPlain[i] = adr.Address | |
| 179 } | |
| 180 | |
| 181 m.data.Lock() | |
| 182 m.data.admins = emails | |
| 183 m.data.adminsPlain = adminsPlain | |
| 184 m.data.Unlock() | |
| 185 } | |
| 186 | |
| 187 func (m *mailImpl) SentMessages() []*mail.TestMessage { | |
| 188 m.data.Lock() | |
| 189 msgs := m.data.queue[:] | |
|
martiniss
2015/12/16 21:54:15
[:] means a new slice, but it's not a copy, so you
iannucci
2015/12/17 00:01:07
no, the queue is an replaceable slice of immutable
| |
| 190 m.data.Unlock() | |
| 191 | |
| 192 ret := make([]*mail.TestMessage, len(msgs)) | |
| 193 for i, m := range msgs { | |
| 194 ret[i] = m.Copy() | |
| 195 } | |
| 196 return ret | |
| 197 } | |
| 198 | |
| 199 func (m *mailImpl) Reset() { | |
| 200 m.data.Lock() | |
| 201 m.data.queue = nil | |
| 202 m.data.Unlock() | |
| 203 } | |
| OLD | NEW |