more abstractions for errors
This commit is contained in:
parent
c80dca4a59
commit
c35e01f640
|
@ -92,7 +92,7 @@ type TestKeys struct {
|
||||||
Queries []*model.ArchivalDNSLookupResult `json:"queries"`
|
Queries []*model.ArchivalDNSLookupResult `json:"queries"`
|
||||||
TCPConnect []*model.ArchivalTCPConnectResult `json:"tcp_connect"`
|
TCPConnect []*model.ArchivalTCPConnectResult `json:"tcp_connect"`
|
||||||
TLSHandshakes []*model.ArchivalTLSOrQUICHandshakeResult `json:"tls_handshakes"`
|
TLSHandshakes []*model.ArchivalTLSOrQUICHandshakeResult `json:"tls_handshakes"`
|
||||||
SMTPErrors []*string `json:"smtp"`
|
SMTPErrors map[string][]*string `json:"smtp"`
|
||||||
NoOpCounter uint8 `json:"successful_noops"`
|
NoOpCounter uint8 `json:"successful_noops"`
|
||||||
// Used for global failure (DNS resolution)
|
// Used for global failure (DNS resolution)
|
||||||
Failure string `json:"failure"`
|
Failure string `json:"failure"`
|
||||||
|
@ -117,12 +117,28 @@ func (m Measurer) ExperimentVersion() string {
|
||||||
return testVersion
|
return testVersion
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Manages sequential SMTP sessions to the same hostname (over different IPs)
|
||||||
|
// don't use in parallel!
|
||||||
type SMTPRunner struct {
|
type SMTPRunner struct {
|
||||||
trace *measurexlite.Trace
|
trace *measurexlite.Trace
|
||||||
logger model.Logger
|
logger model.Logger
|
||||||
ctx context.Context
|
ctx context.Context
|
||||||
tk *TestKeys
|
tk *TestKeys
|
||||||
tlsconfig *tls.Config
|
tlsconfig *tls.Config
|
||||||
|
host string
|
||||||
|
port string
|
||||||
|
// addr is changed everytime SMTPRunner.conn(addr) is called
|
||||||
|
addr string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r SMTPRunner) smtp_error(err error) {
|
||||||
|
key := net.JoinHostPort(r.addr, r.port)
|
||||||
|
errors, exists := r.tk.SMTPErrors[key]
|
||||||
|
if exists {
|
||||||
|
r.tk.SMTPErrors[key] = append(errors, tracex.NewFailure(err))
|
||||||
|
} else {
|
||||||
|
r.tk.SMTPErrors[key] = []*string{tracex.NewFailure(err)}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r SMTPRunner) resolve(host string) ([]string, bool) {
|
func (r SMTPRunner) resolve(host string) ([]string, bool) {
|
||||||
|
@ -139,44 +155,49 @@ func (r SMTPRunner) resolve(host string) ([]string, bool) {
|
||||||
return addrs, true
|
return addrs, true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r SMTPRunner) conn(addr string, port string) (net.Conn, bool) {
|
func (r SMTPRunner) conn(addr string) (net.Conn, bool) {
|
||||||
|
r.addr = addr
|
||||||
dialer := r.trace.NewDialerWithoutResolver(r.logger)
|
dialer := r.trace.NewDialerWithoutResolver(r.logger)
|
||||||
conn, err := dialer.DialContext(r.ctx, "tcp", net.JoinHostPort(addr, port))
|
conn, err := dialer.DialContext(r.ctx, "tcp", net.JoinHostPort(r.addr, r.port))
|
||||||
r.tk.TCPConnect = append(r.tk.TCPConnect, r.trace.TCPConnects()...)
|
r.tk.TCPConnect = append(r.tk.TCPConnect, r.trace.TCPConnects()...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
r.smtp_error(err)
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
return conn, true
|
return conn, true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r SMTPRunner) handshake(conn net.Conn, host string, port string) (net.Conn, bool) {
|
func (r SMTPRunner) handshake(conn net.Conn) (net.Conn, bool) {
|
||||||
r.logger.Infof("Starting TLS handshake with %s:%s", host, port)
|
r.logger.Infof("Starting TLS handshake with %s:%s (%s)", r.host, r.port, r.addr)
|
||||||
thx := r.trace.NewTLSHandshakerStdlib(r.logger)
|
thx := r.trace.NewTLSHandshakerStdlib(r.logger)
|
||||||
tconn, _, err := thx.Handshake(r.ctx, conn, r.tlsconfig)
|
tconn, _, err := thx.Handshake(r.ctx, conn, r.tlsconfig)
|
||||||
r.tk.TLSHandshakes = append(r.tk.TLSHandshakes, r.trace.FirstTLSHandshakeOrNil())
|
r.tk.TLSHandshakes = append(r.tk.TLSHandshakes, r.trace.FirstTLSHandshakeOrNil())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
r.smtp_error(err)
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
r.logger.Infof("Handshake succeeded")
|
r.logger.Infof("Handshake succeeded")
|
||||||
return tconn, true
|
return tconn, true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r SMTPRunner) starttls(conn net.Conn, host string, port string, message string) (net.Conn, bool) {
|
func (r SMTPRunner) starttls(conn net.Conn, message string) (net.Conn, bool) {
|
||||||
r.logger.Infof("Asking for StartTLS upgrade")
|
if message != "" {
|
||||||
conn.Write([]byte(message))
|
r.logger.Infof("Asking for StartTLS upgrade")
|
||||||
tconn, success := r.handshake(conn, host, port)
|
conn.Write([]byte(message))
|
||||||
|
}
|
||||||
|
tconn, success := r.handshake(conn)
|
||||||
return tconn, success
|
return tconn, success
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r SMTPRunner) smtp(conn net.Conn, ehlo string, noop uint8) bool {
|
func (r SMTPRunner) smtp(conn net.Conn, ehlo string, noop uint8) bool {
|
||||||
client, err := smtp.NewClient(conn, ehlo)
|
client, err := smtp.NewClient(conn, ehlo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.tk.SMTPErrors = append(r.tk.SMTPErrors, []*string{tracex.NewFailure(err)}...)
|
r.smtp_error(err)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
err = client.Hello(ehlo)
|
err = client.Hello(ehlo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.tk.SMTPErrors = append(r.tk.SMTPErrors, []*string{tracex.NewFailure(err)}...)
|
r.smtp_error(err)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -189,7 +210,7 @@ func (r SMTPRunner) smtp(conn net.Conn, ehlo string, noop uint8) bool {
|
||||||
r.logger.Infof("NoOp Iteration %d", r.tk.NoOpCounter)
|
r.logger.Infof("NoOp Iteration %d", r.tk.NoOpCounter)
|
||||||
err = client.Noop()
|
err = client.Noop()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
r.tk.SMTPErrors = append(r.tk.SMTPErrors, []*string{tracex.NewFailure(err)}...)
|
r.smtp_error(err)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -222,6 +243,8 @@ func (m Measurer) Run(
|
||||||
}
|
}
|
||||||
|
|
||||||
tk := new(TestKeys)
|
tk := new(TestKeys)
|
||||||
|
// TODO: make it so we don't forget it
|
||||||
|
tk.SMTPErrors = make(map[string][]*string)
|
||||||
measurement.TestKeys = tk
|
measurement.TestKeys = tk
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(ctx, 60*time.Second)
|
ctx, cancel := context.WithTimeout(ctx, 60*time.Second)
|
||||||
|
@ -238,6 +261,9 @@ func (m Measurer) Run(
|
||||||
ctx: ctx,
|
ctx: ctx,
|
||||||
tk: tk,
|
tk: tk,
|
||||||
tlsconfig: &tlsconfig,
|
tlsconfig: &tlsconfig,
|
||||||
|
host: config.host,
|
||||||
|
port: config.port,
|
||||||
|
addr: "",
|
||||||
}
|
}
|
||||||
|
|
||||||
// First resolve DNS
|
// First resolve DNS
|
||||||
|
@ -247,7 +273,7 @@ func (m Measurer) Run(
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, addr := range addrs {
|
for _, addr := range addrs {
|
||||||
conn, success := runner.conn(addr, config.port)
|
conn, success := runner.conn(addr)
|
||||||
if !success {
|
if !success {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -255,7 +281,7 @@ func (m Measurer) Run(
|
||||||
|
|
||||||
if config.forced_tls {
|
if config.forced_tls {
|
||||||
// Direct TLS connection
|
// Direct TLS connection
|
||||||
tconn, success := runner.handshake(conn, config.host, config.port)
|
tconn, success := runner.handshake(conn)
|
||||||
if !success {
|
if !success {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -272,7 +298,7 @@ func (m Measurer) Run(
|
||||||
}
|
}
|
||||||
|
|
||||||
// Upgrade via StartTLS and try EHLO + NoOps
|
// Upgrade via StartTLS and try EHLO + NoOps
|
||||||
tconn, success := runner.starttls(conn, config.host, config.port, "STARTTLS\n")
|
tconn, success := runner.starttls(conn, "STARTTLS\n")
|
||||||
if !success {
|
if !success {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user