Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

go-guerrilla doesn’t support “RCPT TO: <Postmaster>”. #201

Closed
issuefiler opened this issue Nov 24, 2019 · 3 comments
Closed

go-guerrilla doesn’t support “RCPT TO: <Postmaster>”. #201

issuefiler opened this issue Nov 24, 2019 · 3 comments
Labels

Comments

@issuefiler
Copy link

issuefiler commented Nov 24, 2019

RFC 5321

   Any system that includes an SMTP server supporting mail relaying or
   delivery MUST support the reserved mailbox "postmaster" as a case-
   insensitive local name.  This postmaster address is not strictly
   necessary if the server always returns 554 on connection opening (as
   described in Section 3.1).  The requirement to accept mail for
   postmaster implies that RCPT commands that specify a mailbox for
   postmaster at any of the domains for which the SMTP server provides
   mail service, as well as the special case of "RCPT TO:<Postmaster>"
   (with no domain specification), MUST be supported.
   o  The reserved mailbox name "postmaster" may be used in a RCPT
      command without domain qualification (see Section 4.1.1.3) and
      MUST be accepted if so used.
   Syntax:

      rcpt = "RCPT TO:" ( "<Postmaster@" Domain ">" / "<Postmaster>" /
                  Forward-path ) [SP Rcpt-parameters] CRLF

                  Note that, in a departure from the usual rules for
                  local-parts, the "Postmaster" string shown above is
                  treated as case-insensitive.

Test

Preset

"allowed_hosts": ["a.com"]

Expectation

Case # 1

RCPT TO: <postMaStER@a.com>
250 2.1.5 OK

Case # 2

RCPT TO: <Postmaster@not-a.com>
454 4.1.1 Error: Relay access denied: not-a.com

Case # 3

RCPT TO: <poSTmAsteR>
250 2.1.5 OK

Case # 4

EDITED

RCPT TO: <"po\ST\mAs\t\eR">
250 2.1.5 OK

Current

Case # 1 ✅

RCPT TO: <postMaStER@a.com>
250 2.1.5 OK

Case # 2 ✅

RCPT TO: <Postmaster@not-a.com>
454 4.1.1 Error: Relay access denied: not-a.com

Case # 3 ❌

RCPT TO: <poSTmAsteR>
454 4.1.1 Error: Relay access denied: 

Case # 4 ❌ EDITED

RCPT TO: <"po\ST\mAs\t\eR">
501 5.5.4 Invalid address

Fix (not tested)

  1. Validate the syntax per the ABNF. This must be done prior to the step 2, or it would break on the case # 4.
  2. If the Host is empty and the User is case-insensitively Postmaster, say OK.

go-guerrilla/server.go

Lines 492 to 503 in 51f7dda

if !s.allowsHost(to.Host) {
client.sendResponse(r.ErrorRelayDenied, " ", to.Host)
} else {
client.PushRcpt(to)
rcptError := s.backend().ValidateRcpt(client.Envelope)
if rcptError != nil {
client.PopRcpt()
client.sendResponse(r.FailRcptCmd, " ", rcptError.Error())
} else {
client.sendResponse(r.SuccessRcptCmd)
}
}

				if s.allowsHost(to.Host) || (to.Host == "" && strings.toLower(to.User) == "postmaster") {
					client.PushRcpt(to)
					rcptError := s.backend().ValidateRcpt(client.Envelope)
					if rcptError != nil {
						client.PopRcpt()
						client.sendResponse(r.FailRcptCmd, " ", rcptError.Error())
					} else {
						client.sendResponse(r.SuccessRcptCmd)
					}
				} else {
					client.sendResponse(r.ErrorRelayDenied, " ", to.Host)
				}
@flashmob
Copy link
Owner

flashmob commented Nov 25, 2019

The ABNF has already been parsed in parse.go

TODO:

  • modify the parser to evaluate the quoted local-part values as the expression is being parsed.
  • add a new string to the Address struct to store the evaluated result (we want to store both the raw value and the evaluated one)
  • modify the if s.allowsHost line as suggested above. However, we should also modify the allowsHost to match the evaluated value. (The case 4 above is a valid address <"po\ST\mAs\t\eR"> - no?)
  • add a test case for RCPT TO: <poSTmAsteR> and <"po\ST\mAs\t\eR">

There's a question - what Address.host should we default to? Should we fill any value there? Leave blank, or add a new config option, or just use 'localhost' ?

@issuefiler
Copy link
Author

The RFC seems to be a bit unclear for "po\ST\mAs\t\eR" and "po\ST\mAs\t\eR"@example.com inputs, but what you’ve said would be the right path to go, I guess; edited.

And for the Address.Host of <Postmaster>: we could simply leave the Host blank (as is) and modify the stringifying steps for use by processors (#199).

@flashmob flashmob added the todo label Nov 27, 2019
flashmob added a commit that referenced this issue Dec 13, 2019
* ensure <postmaster> RCPT TO is correctly handled, for #201
@flashmob
Copy link
Owner

flashmob commented Dec 13, 2019

Hi @issuefiler, Please see PR #202 - it's ready for your acceptance test :-)

It works a little different from what was specified above (#201 (comment))
It simply captures quoted local-parts without the escape characters, and then the String() function knows if it was quoted or not. So if the local-part is "test" " then it will be stored as "test " ".
So, allowsHost didn't need any modification after all.

When testing:

Note that if <postmaster> is used in the RCPT TO (without a host), then new functionality was added to assume that the host is assumed to be the Hostname setting for the Server. Therefore, we must ensure that the Hostname setting also exists in the AllowedHosts config setting. (The server will log a warning if it detects this case)

@flashmob flashmob added testing and removed todo labels Dec 13, 2019
flashmob added a commit that referenced this issue Dec 25, 2019
@flashmob flashmob added lgtm and removed testing labels Dec 25, 2019
flashmob added a commit that referenced this issue Dec 27, 2019
- Parser captures quoted local-parts without the escape characters
- mail.Address.String() know when to quote local-part, 
- server's `allowsHost` function is ipv6 address aware (addresses specified in the config will get normalized to their ipv6 simplest form, addresses parsed from RCPT and MAIL commands will have ipv6 normalized)
- if `<postmaster>` is used in the RCPT TO (without a host), then new functionality was added to assume that the host is assumed to be the Hostname setting for the Server
- HELO/EHLO argument validation. #200
- The “header” processor populates “Received:” headers wrongly. #198
-  tiny bug in “p_redis.go”. #196
- “MimeHeaderDecode” (envelope.go) returns an incorrectly-spaced string. #195
- go-guerrilla cannot properly handle some valid addresses. #199
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

2 participants