From 15a3295f1ef5d68a932a78b2c6905183c3165a63 Mon Sep 17 00:00:00 2001 From: Flashmob Date: Sat, 9 Jun 2018 01:32:31 +1000 Subject: [PATCH] Let Backends return their own error code (custom result or error) (#113) add ability for backends to specify a custom return code, fixes #78 --- api_test.go | 59 +++++++++++- backends/backend.go | 29 ++++-- backends/gateway.go | 54 ++++++----- client.go | 60 ++++++------ cmd/guerrillad/serve.go | 2 +- response/enhanced.go | 202 ++++++++++++++++++++++------------------ server.go | 21 +++-- tests/guerrilla_test.go | 4 +- util.go | 8 +- 9 files changed, 269 insertions(+), 170 deletions(-) diff --git a/api_test.go b/api_test.go index 0250e76d..8af4fcc2 100644 --- a/api_test.go +++ b/api_test.go @@ -2,10 +2,12 @@ package guerrilla import ( "bufio" + "errors" "fmt" "github.com/flashmob/go-guerrilla/backends" "github.com/flashmob/go-guerrilla/log" "github.com/flashmob/go-guerrilla/mail" + "github.com/flashmob/go-guerrilla/response" "io/ioutil" "net" "os" @@ -349,7 +351,7 @@ var funkyLogger = func() backends.Decorator { return backends.ProcessWith( func(e *mail.Envelope, task backends.SelectTask) (backends.Result, error) { if task == backends.TaskValidateRcpt { - // validate the last recipient appended to e.Rcpt + // log the last recipient appended to e.Rcpt backends.Log().Infof( "another funky recipient [%s]", e.RcptTo[len(e.RcptTo)-1]) @@ -556,3 +558,58 @@ func TestSkipAllowsHost(t *testing.T) { } } + +var customBackend2 = func() backends.Decorator { + + return func(p backends.Processor) backends.Processor { + return backends.ProcessWith( + func(e *mail.Envelope, task backends.SelectTask) (backends.Result, error) { + if task == backends.TaskValidateRcpt { + return p.Process(e, task) + } else if task == backends.TaskSaveMail { + backends.Log().Info("Another funky email!") + err := errors.New("system shock") + return backends.NewResult(response.Canned.FailReadErrorDataCmd, response.SP, err), err + } + return p.Process(e, task) + }) + } +} + +// Test a custom backend response +func TestCustomBackendResult(t *testing.T) { + os.Truncate("tests/testlog", 0) + cfg := &AppConfig{ + LogFile: "tests/testlog", + AllowedHosts: []string{"grr.la"}, + BackendConfig: backends.BackendConfig{ + "save_process": "HeadersParser|Debugger|Custom", + "validate_process": "Custom", + }, + } + d := Daemon{Config: cfg} + d.AddProcessor("Custom", customBackend2) + + if err := d.Start(); err != nil { + t.Error(err) + } + // lets have a talk with the server + talkToServer("127.0.0.1:2525") + + d.Shutdown() + + b, err := ioutil.ReadFile("tests/testlog") + if err != nil { + t.Error("could not read logfile") + return + } + // lets check for fingerprints + if strings.Index(string(b), "451 4.3.0 Error") < 0 { + t.Error("did not log: 451 4.3.0 Error") + } + + if strings.Index(string(b), "system shock") < 0 { + t.Error("did not log: system shock") + } + +} diff --git a/backends/backend.go b/backends/backend.go index 7a2e287e..7f9e554d 100644 --- a/backends/backend.go +++ b/backends/backend.go @@ -1,6 +1,7 @@ package backends import ( + "bytes" "fmt" "github.com/flashmob/go-guerrilla/log" "github.com/flashmob/go-guerrilla/mail" @@ -54,6 +55,7 @@ type BaseConfig interface{} type notifyMsg struct { err error queuedID string + result Result } // Result represents a response to an SMTP client after receiving DATA. @@ -66,16 +68,18 @@ type Result interface { } // Internal implementation of BackendResult for use by backend implementations. -type result string +type result struct { + bytes.Buffer +} -func (br result) String() string { - return string(br) +func (r *result) String() string { + return r.Buffer.String() } // Parses the SMTP code from the first 3 characters of the SMTP message. // Returns 554 if code cannot be parsed. -func (br result) Code() int { - trimmed := strings.TrimSpace(string(br)) +func (r *result) Code() int { + trimmed := strings.TrimSpace(r.String()) if len(trimmed) < 3 { return 554 } @@ -86,8 +90,19 @@ func (br result) Code() int { return code } -func NewResult(message string) Result { - return result(message) +func NewResult(r ...interface{}) Result { + buf := new(result) + for _, item := range r { + switch v := item.(type) { + case error: + buf.WriteString(v.Error()) + case fmt.Stringer: + buf.WriteString(v.String()) + case string: + buf.WriteString(v) + } + } + return buf } type processorInitializer interface { diff --git a/backends/gateway.go b/backends/gateway.go index 60e60241..5d65ca75 100644 --- a/backends/gateway.go +++ b/backends/gateway.go @@ -128,7 +128,7 @@ func (w *workerMsg) reset(e *mail.Envelope, task SelectTask) { // Process distributes an envelope to one of the backend workers with a TaskSaveMail task func (gw *BackendGateway) Process(e *mail.Envelope) Result { if gw.State != BackendStateRunning { - return NewResult(response.Canned.FailBackendNotRunning + gw.State.String()) + return NewResult(response.Canned.FailBackendNotRunning, response.SP, gw.State) } // borrow a workerMsg from the pool workerMsg := workerMsgPool.Get().(*workerMsg) @@ -139,11 +139,32 @@ func (gw *BackendGateway) Process(e *mail.Envelope) Result { // or timeout select { case status := <-workerMsg.notifyMe: - workerMsgPool.Put(workerMsg) // can be recycled since we used the notifyMe channel + // email saving transaction completed + if status.result == BackendResultOK && status.queuedID != "" { + return NewResult(response.Canned.SuccessMessageQueued, response.SP, status.queuedID) + } + + // A custom result, there was probably an error, if so, log it + if status.result != nil { + if status.err != nil { + Log().Error(status.err) + } + return status.result + } + + // if there was no result, but there's an error, then make a new result from the error if status.err != nil { - return NewResult(response.Canned.FailBackendTransaction + status.err.Error()) + if _, err := strconv.Atoi(status.err.Error()[:3]); err != nil { + return NewResult(response.Canned.FailBackendTransaction, response.SP, status.err) + } + return NewResult(status.err) } - return NewResult(response.Canned.SuccessMessageQueued + status.queuedID) + + // both result & error are nil (should not happen) + err := errors.New("no response from backend - processor did not return a result or an error") + Log().Error(err) + return NewResult(response.Canned.FailBackendTransaction, response.SP, err) + case <-time.After(gw.saveTimeout()): Log().Error("Backend has timed out while saving email") e.Lock() // lock the envelope - it's still processing here, we don't want the server to recycle it @@ -434,27 +455,12 @@ func (gw *BackendGateway) workDispatcher( return case msg = <-workIn: state = dispatcherStateWorking // recovers from panic if in this state + result, err := save.Process(msg.e, msg.task) + state = dispatcherStateNotify if msg.task == TaskSaveMail { - // process the email here - result, _ := save.Process(msg.e, TaskSaveMail) - state = dispatcherStateNotify - if result.Code() < 300 { - // if all good, let the gateway know that it was saved - msg.notifyMe <- ¬ifyMsg{nil, msg.e.QueuedId} - } else { - // notify the gateway about the error - msg.notifyMe <- ¬ifyMsg{err: errors.New(result.String())} - } - } else if msg.task == TaskValidateRcpt { - _, err := validate.Process(msg.e, TaskValidateRcpt) - state = dispatcherStateNotify - if err != nil { - // validation failed - msg.notifyMe <- ¬ifyMsg{err: err} - } else { - // all good. - msg.notifyMe <- ¬ifyMsg{err: nil} - } + msg.notifyMe <- ¬ifyMsg{err: err, result: result, queuedID: msg.e.QueuedId} + } else { + msg.notifyMe <- ¬ifyMsg{err: err, result: result} } } state = dispatcherStateIdle diff --git a/client.go b/client.go index a23bee32..17cf4548 100644 --- a/client.go +++ b/client.go @@ -38,8 +38,9 @@ type client struct { errors int state ClientState messagesSent int - // Response to be written to the client + // Response to be written to the client (for debugging) response bytes.Buffer + bufErr error conn net.Conn bufin *smtpBufferedReader bufout *bufio.Writer @@ -69,39 +70,38 @@ func NewClient(conn net.Conn, clientID uint64, logger log.Logger, envelope *mail return c } -// setResponse adds a response to be written on the next turn +// sendResponse adds a response to be written on the next turn +// the response gets buffered func (c *client) sendResponse(r ...interface{}) { c.bufout.Reset(c.conn) if c.log.IsDebug() { - // us additional buffer so that we can log the response in debug mode only + // an additional buffer so that we can log the response in debug mode only c.response.Reset() } + var out string + if c.bufErr != nil { + c.bufErr = nil + } for _, item := range r { switch v := item.(type) { - case string: - if _, err := c.bufout.WriteString(v); err != nil { - c.log.WithError(err).Error("could not write to c.bufout") - } - if c.log.IsDebug() { - c.response.WriteString(v) - } case error: - if _, err := c.bufout.WriteString(v.Error()); err != nil { - c.log.WithError(err).Error("could not write to c.bufout") - } - if c.log.IsDebug() { - c.response.WriteString(v.Error()) - } + out = v.Error() case fmt.Stringer: - if _, err := c.bufout.WriteString(v.String()); err != nil { - c.log.WithError(err).Error("could not write to c.bufout") - } - if c.log.IsDebug() { - c.response.WriteString(v.String()) - } + out = v.String() + case string: + out = v + } + if _, c.bufErr = c.bufout.WriteString(out); c.bufErr != nil { + c.log.WithError(c.bufErr).Error("could not write to c.bufout") + } + if c.log.IsDebug() { + c.response.WriteString(out) + } + if c.bufErr != nil { + return } } - c.bufout.WriteString("\r\n") + _, c.bufErr = c.bufout.WriteString("\r\n") if c.log.IsDebug() { c.response.WriteString("\r\n") } @@ -176,20 +176,20 @@ func (c *client) getID() uint64 { } // UpgradeToTLS upgrades a client connection to TLS -func (client *client) upgradeToTLS(tlsConfig *tls.Config) error { +func (c *client) upgradeToTLS(tlsConfig *tls.Config) error { var tlsConn *tls.Conn - // wrap client.conn in a new TLS server side connection - tlsConn = tls.Server(client.conn, tlsConfig) + // wrap c.conn in a new TLS server side connection + tlsConn = tls.Server(c.conn, tlsConfig) // Call handshake here to get any handshake error before reading starts err := tlsConn.Handshake() if err != nil { return err } // convert tlsConn to net.Conn - client.conn = net.Conn(tlsConn) - client.bufout.Reset(client.conn) - client.bufin.Reset(client.conn) - client.TLS = true + c.conn = net.Conn(tlsConn) + c.bufout.Reset(c.conn) + c.bufin.Reset(c.conn) + c.TLS = true return err } diff --git a/cmd/guerrillad/serve.go b/cmd/guerrillad/serve.go index e3cd0f7a..0a42a77b 100644 --- a/cmd/guerrillad/serve.go +++ b/cmd/guerrillad/serve.go @@ -136,7 +136,7 @@ func readConfig(path string, pidFile string) (*guerrilla.AppConfig, error) { // command line flags can override config values appConfig, err := d.LoadConfig(path) if err != nil { - return &appConfig, fmt.Errorf("Could not read config file: %s", err.Error()) + return &appConfig, fmt.Errorf("could not read config file: %s", err.Error()) } // override config pidFile with with flag from the command line if len(pidFile) > 0 { diff --git a/response/enhanced.go b/response/enhanced.go index 88975c39..b241d7ce 100644 --- a/response/enhanced.go +++ b/response/enhanced.go @@ -22,6 +22,9 @@ const ( ClassPermanentFailure = 5 ) +// space char +const SP = " " + // class is a type for ClassSuccess, ClassTransientFailure and ClassPermanentFailure constants type class int @@ -118,39 +121,39 @@ var ( type Responses struct { // The 500's - FailLineTooLong string - FailNestedMailCmd string - FailNoSenderDataCmd string - FailNoRecipientsDataCmd string - FailUnrecognizedCmd string - FailMaxUnrecognizedCmd string - FailReadLimitExceededDataCmd string - FailMessageSizeExceeded string - FailReadErrorDataCmd string - FailPathTooLong string - FailInvalidAddress string - FailLocalPartTooLong string - FailDomainTooLong string - FailBackendNotRunning string - FailBackendTransaction string - FailBackendTimeout string - FailRcptCmd string + FailLineTooLong *Response + FailNestedMailCmd *Response + FailNoSenderDataCmd *Response + FailNoRecipientsDataCmd *Response + FailUnrecognizedCmd *Response + FailMaxUnrecognizedCmd *Response + FailReadLimitExceededDataCmd *Response + FailMessageSizeExceeded *Response + FailReadErrorDataCmd *Response + FailPathTooLong *Response + FailInvalidAddress *Response + FailLocalPartTooLong *Response + FailDomainTooLong *Response + FailBackendNotRunning *Response + FailBackendTransaction *Response + FailBackendTimeout *Response + FailRcptCmd *Response // The 400's - ErrorTooManyRecipients string - ErrorRelayDenied string - ErrorShutdown string + ErrorTooManyRecipients *Response + ErrorRelayDenied *Response + ErrorShutdown *Response // The 200's - SuccessMailCmd string - SuccessRcptCmd string - SuccessResetCmd string - SuccessVerifyCmd string - SuccessNoopCmd string - SuccessQuitCmd string - SuccessDataCmd string - SuccessStartTLSCmd string - SuccessMessageQueued string + SuccessMailCmd *Response + SuccessRcptCmd *Response + SuccessResetCmd *Response + SuccessVerifyCmd *Response + SuccessNoopCmd *Response + SuccessQuitCmd *Response + SuccessDataCmd *Response + SuccessStartTLSCmd *Response + SuccessMessageQueued *Response } // Called automatically during package load to build up the Responses struct @@ -158,191 +161,195 @@ func init() { Canned = Responses{} - Canned.FailLineTooLong = (&Response{ + Canned.FailLineTooLong = &Response{ EnhancedCode: InvalidCommand, BasicCode: 554, Class: ClassPermanentFailure, Comment: "Line too long.", - }).String() + } - Canned.FailNestedMailCmd = (&Response{ + Canned.FailNestedMailCmd = &Response{ EnhancedCode: InvalidCommand, BasicCode: 503, Class: ClassPermanentFailure, Comment: "Error: nested MAIL command", - }).String() + } - Canned.SuccessMailCmd = (&Response{ + Canned.SuccessMailCmd = &Response{ EnhancedCode: OtherAddressStatus, Class: ClassSuccess, - }).String() + } - Canned.SuccessRcptCmd = (&Response{ + Canned.SuccessRcptCmd = &Response{ EnhancedCode: DestinationMailboxAddressValid, Class: ClassSuccess, - }).String() + } Canned.SuccessResetCmd = Canned.SuccessMailCmd - Canned.SuccessNoopCmd = (&Response{ + + Canned.SuccessNoopCmd = &Response{ EnhancedCode: OtherStatus, Class: ClassSuccess, - }).String() + } - Canned.SuccessVerifyCmd = (&Response{ + Canned.SuccessVerifyCmd = &Response{ EnhancedCode: OtherOrUndefinedProtocolStatus, BasicCode: 252, Class: ClassSuccess, Comment: "Cannot verify user", - }).String() + } - Canned.ErrorTooManyRecipients = (&Response{ + Canned.ErrorTooManyRecipients = &Response{ EnhancedCode: TooManyRecipients, BasicCode: 452, Class: ClassTransientFailure, Comment: "Too many recipients", - }).String() + } - Canned.ErrorRelayDenied = (&Response{ + Canned.ErrorRelayDenied = &Response{ EnhancedCode: BadDestinationMailboxAddress, BasicCode: 454, Class: ClassTransientFailure, - Comment: "Error: Relay access denied: ", - }).String() + Comment: "Error: Relay access denied:", + } - Canned.SuccessQuitCmd = (&Response{ + Canned.SuccessQuitCmd = &Response{ EnhancedCode: OtherStatus, BasicCode: 221, Class: ClassSuccess, Comment: "Bye", - }).String() + } - Canned.FailNoSenderDataCmd = (&Response{ + Canned.FailNoSenderDataCmd = &Response{ EnhancedCode: InvalidCommand, BasicCode: 503, Class: ClassPermanentFailure, Comment: "Error: No sender", - }).String() + } - Canned.FailNoRecipientsDataCmd = (&Response{ + Canned.FailNoRecipientsDataCmd = &Response{ EnhancedCode: InvalidCommand, BasicCode: 503, Class: ClassPermanentFailure, Comment: "Error: No recipients", - }).String() + } - Canned.SuccessDataCmd = "354 Enter message, ending with '.' on a line by itself" + Canned.SuccessDataCmd = &Response{ + BasicCode: 354, + Comment: "354 Enter message, ending with '.' on a line by itself", + } - Canned.SuccessStartTLSCmd = (&Response{ + Canned.SuccessStartTLSCmd = &Response{ EnhancedCode: OtherStatus, BasicCode: 220, Class: ClassSuccess, Comment: "Ready to start TLS", - }).String() + } - Canned.FailUnrecognizedCmd = (&Response{ + Canned.FailUnrecognizedCmd = &Response{ EnhancedCode: InvalidCommand, BasicCode: 554, Class: ClassPermanentFailure, Comment: "Unrecognized command", - }).String() + } - Canned.FailMaxUnrecognizedCmd = (&Response{ + Canned.FailMaxUnrecognizedCmd = &Response{ EnhancedCode: InvalidCommand, BasicCode: 554, Class: ClassPermanentFailure, Comment: "Too many unrecognized commands", - }).String() + } - Canned.ErrorShutdown = (&Response{ + Canned.ErrorShutdown = &Response{ EnhancedCode: OtherOrUndefinedMailSystemStatus, BasicCode: 421, Class: ClassTransientFailure, Comment: "Server is shutting down. Please try again later. Sayonara!", - }).String() + } - Canned.FailReadLimitExceededDataCmd = (&Response{ + Canned.FailReadLimitExceededDataCmd = &Response{ EnhancedCode: SyntaxError, BasicCode: 550, Class: ClassPermanentFailure, - Comment: "Error: ", - }).String() + Comment: "Error:", + } - Canned.FailMessageSizeExceeded = (&Response{ + Canned.FailMessageSizeExceeded = &Response{ EnhancedCode: SyntaxError, BasicCode: 550, Class: ClassPermanentFailure, - Comment: "Error: ", - }).String() + Comment: "Error:", + } - Canned.FailReadErrorDataCmd = (&Response{ + Canned.FailReadErrorDataCmd = &Response{ EnhancedCode: OtherOrUndefinedMailSystemStatus, BasicCode: 451, Class: ClassTransientFailure, - Comment: "Error: ", - }).String() + Comment: "Error:", + } - Canned.FailPathTooLong = (&Response{ + Canned.FailPathTooLong = &Response{ EnhancedCode: InvalidCommandArguments, BasicCode: 550, Class: ClassPermanentFailure, Comment: "Path too long", - }).String() + } - Canned.FailInvalidAddress = (&Response{ + Canned.FailInvalidAddress = &Response{ EnhancedCode: InvalidCommandArguments, BasicCode: 501, Class: ClassPermanentFailure, Comment: "Invalid address", - }).String() + } - Canned.FailLocalPartTooLong = (&Response{ + Canned.FailLocalPartTooLong = &Response{ EnhancedCode: InvalidCommandArguments, BasicCode: 550, Class: ClassPermanentFailure, Comment: "Local part too long, cannot exceed 64 characters", - }).String() + } - Canned.FailDomainTooLong = (&Response{ + Canned.FailDomainTooLong = &Response{ EnhancedCode: InvalidCommandArguments, BasicCode: 550, Class: ClassPermanentFailure, Comment: "Domain cannot exceed 255 characters", - }).String() + } - Canned.FailBackendNotRunning = (&Response{ + Canned.FailBackendNotRunning = &Response{ EnhancedCode: OtherOrUndefinedProtocolStatus, BasicCode: 554, Class: ClassPermanentFailure, - Comment: "Transaction failed - backend not running ", - }).String() + Comment: "Transaction failed - backend not running", + } - Canned.FailBackendTransaction = (&Response{ + Canned.FailBackendTransaction = &Response{ EnhancedCode: OtherOrUndefinedProtocolStatus, BasicCode: 554, Class: ClassPermanentFailure, - Comment: "Error: ", - }).String() + Comment: "Error:", + } - Canned.SuccessMessageQueued = (&Response{ + Canned.SuccessMessageQueued = &Response{ EnhancedCode: OtherStatus, BasicCode: 250, Class: ClassSuccess, - Comment: "OK : queued as ", - }).String() + Comment: "OK: queued as", + } - Canned.FailBackendTimeout = (&Response{ + Canned.FailBackendTimeout = &Response{ EnhancedCode: OtherOrUndefinedProtocolStatus, BasicCode: 554, Class: ClassPermanentFailure, Comment: "Error: transaction timeout", - }).String() + } - Canned.FailRcptCmd = (&Response{ + Canned.FailRcptCmd = &Response{ EnhancedCode: BadDestinationMailboxAddress, BasicCode: 550, Class: ClassPermanentFailure, Comment: "User unknown in local recipient table", - }).String() + } } @@ -409,6 +416,7 @@ type Response struct { Class class // Comment is optional Comment string + cached string } // it looks like this ".5.4" @@ -428,6 +436,14 @@ func (e EnhancedStatusCode) String() string { // String returns a custom Response as a string func (r *Response) String() string { + if r.cached != "" { + return r.cached + } + if r.EnhancedCode == "" { + r.cached = r.Comment + return r.Comment + } + basicCode := r.BasicCode comment := r.Comment if len(comment) == 0 && r.BasicCode == 0 { @@ -447,8 +463,8 @@ func (r *Response) String() string { if r.BasicCode == 0 { basicCode = getBasicStatusCode(e) } - - return fmt.Sprintf("%d %s %s", basicCode, e.String(), comment) + r.cached = fmt.Sprintf("%d %s %s", basicCode, e.String(), comment) + return r.cached } // getBasicStatusCode gets the basic status code from codeMap, or fallback code if not mapped diff --git a/server.go b/server.go index 0abec481..98702bf5 100644 --- a/server.go +++ b/server.go @@ -430,7 +430,7 @@ func (server *server) handleClient(client *client) { case strings.Index(cmd, "HELP") == 0: quote := response.GetQuote() - client.sendResponse("214-OK\r\n" + quote) + client.sendResponse("214-OK\r\n", quote) case sc.XClientOn && strings.Index(cmd, "XCLIENT ") == 0: if toks := strings.Split(input[8:], " "); len(toks) > 0 { @@ -482,13 +482,13 @@ func (server *server) handleClient(client *client) { client.sendResponse(err.Error()) } else { if !server.allowsHost(to.Host) { - client.sendResponse(response.Canned.ErrorRelayDenied, to.Host) + client.sendResponse(response.Canned.ErrorRelayDenied, " ", to.Host) } else { client.PushRcpt(to) rcptError := server.backend().ValidateRcpt(client.Envelope) if rcptError != nil { client.PopRcpt() - client.sendResponse(response.Canned.FailRcptCmd + " " + rcptError.Error()) + client.sendResponse(response.Canned.FailRcptCmd, " ", rcptError.Error()) } else { client.sendResponse(response.Canned.SuccessRcptCmd) } @@ -543,13 +543,13 @@ func (server *server) handleClient(client *client) { } if err != nil { if err == LineLimitExceeded { - client.sendResponse(response.Canned.FailReadLimitExceededDataCmd, LineLimitExceeded.Error()) + client.sendResponse(response.Canned.FailReadLimitExceededDataCmd, " ", LineLimitExceeded.Error()) client.kill() } else if err == MessageSizeExceeded { - client.sendResponse(response.Canned.FailMessageSizeExceeded, MessageSizeExceeded.Error()) + client.sendResponse(response.Canned.FailMessageSizeExceeded, " ", MessageSizeExceeded.Error()) client.kill() } else { - client.sendResponse(response.Canned.FailReadErrorDataCmd, err.Error()) + client.sendResponse(response.Canned.FailReadErrorDataCmd, " ", err.Error()) client.kill() } server.log().WithError(err).Warn("Error reading data") @@ -561,7 +561,7 @@ func (server *server) handleClient(client *client) { if res.Code() < 300 { client.messagesSent++ } - client.sendResponse(res.String()) + client.sendResponse(res) client.state = ClientCmd if server.isShuttingDown() { client.state = ClientShutdown @@ -589,13 +589,18 @@ func (server *server) handleClient(client *client) { client.kill() } + if client.bufErr != nil { + server.log().WithError(client.bufErr).Debug("client could not buffer a response") + return + } + // flush the response buffer if client.bufout.Buffered() > 0 { if server.log().IsDebug() { server.log().Debugf("Writing response to client: \n%s", client.response.String()) } err := server.flushResponse(client) if err != nil { - server.log().WithError(err).Debug("Error writing response") + server.log().WithError(err).Debug("error writing response") return } } diff --git a/tests/guerrilla_test.go b/tests/guerrilla_test.go index e3a380c4..2ae85b9a 100644 --- a/tests/guerrilla_test.go +++ b/tests/guerrilla_test.go @@ -1011,7 +1011,7 @@ func TestDataMaxLength(t *testing.T) { //expected := "500 Line too long" expected := "451 4.3.0 Error: Maximum DATA size exceeded" if strings.Index(response, expected) != 0 { - t.Error("Server did not respond with", expected, ", it said:"+response, err) + t.Error("Server did not respond with", expected, ", it said:"+response) } } @@ -1105,7 +1105,7 @@ func TestDataCommand(t *testing.T) { bufin, email+"\r\n.\r\n") //expected := "500 Line too long" - expected := "250 2.0.0 OK : queued as " + expected := "250 2.0.0 OK: queued as " if strings.Index(response, expected) != 0 { t.Error("Server did not respond with", expected, ", it said:"+response, err) } diff --git a/util.go b/util.go index db0420a5..0c7c0f20 100644 --- a/util.go +++ b/util.go @@ -15,7 +15,7 @@ func extractEmail(str string) (mail.Address, error) { email := mail.Address{} var err error if len(str) > RFC2821LimitPath { - return email, errors.New(response.Canned.FailPathTooLong) + return email, errors.New(response.Canned.FailPathTooLong.String()) } if matched := extractEmailRegex.FindStringSubmatch(str); len(matched) > 2 { email.User = matched[1] @@ -26,11 +26,11 @@ func extractEmail(str string) (mail.Address, error) { } err = nil if email.User == "" || email.Host == "" { - err = errors.New(response.Canned.FailInvalidAddress) + err = errors.New(response.Canned.FailInvalidAddress.String()) } else if len(email.User) > RFC2832LimitLocalPart { - err = errors.New(response.Canned.FailLocalPartTooLong) + err = errors.New(response.Canned.FailLocalPartTooLong.String()) } else if len(email.Host) > RFC2821LimitDomain { - err = errors.New(response.Canned.FailDomainTooLong) + err = errors.New(response.Canned.FailDomainTooLong.String()) } return email, err }