forked from flashmob/go-guerrilla
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathclient.go
143 lines (127 loc) · 3.02 KB
/
client.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
package guerrilla
import (
"bufio"
"net"
"strings"
"sync"
"time"
)
// ClientState indicates which part of the SMTP transaction a given client is in.
type ClientState int
const (
// The client has connected, and is awaiting our first response
ClientGreeting = iota
// We have responded to the client's connection and are awaiting a command
ClientCmd
// We have received the sender and recipient information
ClientData
// We have agreed with the client to secure the connection over TLS
ClientStartTLS
// Server will shutdown, client to shutdown on next command turn
ClientShutdown
)
type client struct {
*Envelope
ID uint64
ConnectedAt time.Time
KilledAt time.Time
// Number of errors encountered during session with this client
errors int
state ClientState
messagesSent int
// Response to be written to the client
response string
conn net.Conn
bufin *smtpBufferedReader
bufout *bufio.Writer
timeoutMu sync.Mutex
}
// Email represents a single SMTP message.
type Envelope struct {
// Remote IP address
RemoteAddress string
// Message sent in EHLO command
Helo string
// Sender
MailFrom *EmailAddress
// Recipients
RcptTo []EmailAddress
Data string
Subject string
TLS bool
}
func NewClient(conn net.Conn, clientID uint64) *client {
return &client{
conn: conn,
Envelope: &Envelope{
RemoteAddress: conn.RemoteAddr().String(),
},
ConnectedAt: time.Now(),
bufin: newSMTPBufferedReader(conn),
bufout: bufio.NewWriter(conn),
ID: clientID,
}
}
func (c *client) responseAdd(r string) {
c.response = c.response + r + "\r\n"
}
func (c *client) resetTransaction() {
c.MailFrom = &EmailAddress{}
c.RcptTo = []EmailAddress{}
c.Data = ""
c.Subject = ""
}
func (c *client) isInTransaction() bool {
isMailFromEmpty := *c.MailFrom == (EmailAddress{})
if isMailFromEmpty {
return false
}
return true
}
func (c *client) kill() {
c.KilledAt = time.Now()
}
func (c *client) isAlive() bool {
return c.KilledAt.IsZero()
}
func (c *client) scanSubject(reply string) {
if c.Subject == "" && (len(reply) > 8) {
test := strings.ToUpper(reply[0:9])
if i := strings.Index(test, "SUBJECT: "); i == 0 {
// first line with \r\n
c.Subject = reply[9:]
}
} else if strings.HasSuffix(c.Subject, "\r\n") {
// chop off the \r\n
c.Subject = c.Subject[0 : len(c.Subject)-2]
if (strings.HasPrefix(reply, " ")) || (strings.HasPrefix(reply, "\t")) {
// subject is multi-line
c.Subject = c.Subject + reply[1:]
}
}
}
func (c *client) setTimeout(t time.Duration) {
defer c.timeoutMu.Unlock()
c.timeoutMu.Lock()
if c.conn != nil {
c.conn.SetDeadline(time.Now().Add(t * time.Second))
}
}
func (c *client) init(conn net.Conn, clientID uint64) {
c.conn = conn
// reset our reader & writer
c.bufout.Reset(conn)
c.bufin.Reset(conn)
// reset session data
c.state = 0
c.KilledAt = time.Time{}
c.ConnectedAt = time.Now()
c.ID = clientID
c.TLS = false
c.errors = 0
c.response = ""
c.Helo = ""
}
func (c *client) getID() uint64 {
return c.ID
}