/
receiver.go
229 lines (188 loc) · 4.68 KB
/
receiver.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
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
// Copyright 2018, Shulhan <ms@kilabit.info>. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package smtp
import (
"bytes"
"fmt"
"io"
"log"
"net"
"strings"
liberrors "github.com/shuLhan/share/lib/errors"
)
type receiverMode int
const (
// receiverModeServer accept incoming email only from other server,
// without authentication, through port 25 on the server.
receiverModeServer receiverMode = iota
// receiverModeClient accept incoming email from client, with
// authentication, through port 465 on the server.
receiverModeClient
)
// receiver represent a connection that receive incoming email in server.
type receiver struct {
conn net.Conn
mail *MailTx
clientDomain string
clientAddress string
localAddress string
data []byte
buff bytes.Buffer
mode receiverMode
state CommandKind
authenticated bool
}
func newReceiver(conn net.Conn, mode receiverMode) (recv *receiver) {
recv = &receiver{
conn: conn,
mode: mode,
data: make([]byte, 4096),
mail: &MailTx{},
}
recv.clientAddress = conn.RemoteAddr().String()
recv.localAddress = conn.LocalAddr().String()
return recv
}
// close the receiving line.
func (recv *receiver) close() {
err := recv.conn.Close()
if err != nil {
log.Printf("receiver.close: %s\n", err)
}
}
// isAuthenticated will return true if receiver mode is client and user has
// authenticated to system.
func (recv *receiver) isAuthenticated() bool {
if recv.mode == receiverModeClient && recv.authenticated {
return true
}
return false
}
// readAuthData read AUTH initial response from client into Command Param.
func (recv *receiver) readAuthData(cmd *Command) (err error) {
recv.buff.Reset()
for {
recv.data = recv.data[0:]
n, err := recv.conn.Read(recv.data)
if n > 0 {
_, _ = recv.buff.Write(recv.data[:n])
}
if err != nil {
if err == io.EOF {
break
}
return err
}
if n == cap(recv.data) {
continue
}
break
}
cmd.Param = strings.TrimSpace(recv.buff.String())
return nil
}
// readCommand from client.
//
// Any error from command line (for example, unknown command, or syntax error)
// will be handled directly by this function by replying to client.
//
// An error returned from this function, MUST be considered error on system
// which should stop the receiver for further processing.
func (recv *receiver) readCommand() (cmd *Command, err error) {
recv.buff.Reset()
cmd = newCommand()
var n int
for {
recv.data = recv.data[0:]
n, err = recv.conn.Read(recv.data)
if n > 0 {
_, _ = recv.buff.Write(recv.data[:n])
}
if err != nil {
if err == io.EOF {
break
}
err = fmt.Errorf("smtp: recv: readCommand: " + err.Error())
return nil, err
}
if n == cap(recv.data) {
continue
}
break
}
err = cmd.unpack(recv.buff.Bytes())
if err != nil {
err = fmt.Errorf("smtp: cmd.unpack: " + err.Error())
return nil, err
}
return cmd, nil
}
// readDATA start mail input.
func (recv *receiver) readDATA() (err error) {
var n int
for {
recv.data = recv.data[0:]
n, err = recv.conn.Read(recv.data)
if err != nil {
return err
}
recv.mail.Data = append(recv.mail.Data, recv.data[:n]...)
if recv.mail.isTerminated() {
break
}
}
var l = len(recv.mail.Data)
// Remove the end-of-mail data indicator.
recv.mail.Data = recv.mail.Data[:l-5]
recv.mail.seal(recv.clientDomain, recv.clientAddress, recv.localAddress)
return nil
}
func (recv *receiver) reset() {
recv.state = CommandZERO
recv.mail.Reset()
}
func (recv *receiver) sendError(errRes error) (err error) {
reply, ok := errRes.(*liberrors.E)
if !ok {
reply = &liberrors.E{}
reply.Code = StatusLocalError
reply.Message = errRes.Error()
} else if reply.Code == 0 {
reply.Code = StatusLocalError
}
_, err = fmt.Fprintf(recv.conn, "%d %s\r\n", reply.Code, reply.Message)
if err != nil {
log.Println("sendError: ", err.Error())
return err
}
recv.reset()
return nil
}
// sendReply send single or multiple lines reply to client.
//
// An error returned from this function, MUST be considered error on system
// which should stop the receiver for further processing.
func (recv *receiver) sendReply(code int, msg string, body []string) (err error) {
recv.buff.Reset()
if len(body) == 0 {
_, err = fmt.Fprintf(&recv.buff, "%d %s\r\n", code, msg)
} else {
_, err = fmt.Fprintf(&recv.buff, "%d-%s\r\n", code, msg)
}
if err != nil {
return
}
for x, line := range body {
if x == len(body)-1 {
_, err = fmt.Fprintf(&recv.buff, "%d %s\r\n", code, line)
} else {
_, err = fmt.Fprintf(&recv.buff, "%d-%s\r\n", code, line)
}
if err != nil {
return
}
}
_, err = recv.conn.Write(recv.buff.Bytes())
return err
}