]>
src.bluestatic.org Git - mailpopbox.git/blob - smtp/server.go
2 // Copyright 2020 Blue Static <https://www.bluestatic.org>
3 // This program is free software licensed under the GNU General Public License,
4 // version 3.0. The full text of the license can be found in LICENSE.txt.
5 // SPDX-License-Identifier: GPL-3.0-only
22 type ReplyLine
struct {
27 func (l ReplyLine
) String() string {
28 return fmt
.Sprintf("%d %s", l
.Code
, l
.Message
)
32 ReplyOK
= ReplyLine
{250, "OK"}
33 ReplyAuthOK
= ReplyLine
{235, "auth success"}
34 ReplyBadSyntax
= ReplyLine
{501, "syntax error"}
35 ReplyBadSequence
= ReplyLine
{503, "bad sequence of commands"}
36 ReplyBadMailbox
= ReplyLine
{550, "mailbox unavailable"}
37 ReplyMailboxUnallowed
= ReplyLine
{553, "mailbox name not allowed"}
40 func DomainForAddress(addr mail
.Address
) string {
41 return DomainForAddressString(addr
.Address
)
44 func DomainForAddressString(address
string) string {
45 domainIdx
:= strings
.LastIndex(address
, "@")
49 return address
[domainIdx
+1:]
52 type Envelope
struct {
62 func WriteEnvelopeForDelivery(w io
.Writer
, e Envelope
) {
63 fmt
.Fprintf(w
, "Delivered-To: <%s>\r\n", e
.RcptTo
[0].Address
)
64 fmt
.Fprintf(w
, "Return-Path: <%s>\r\n", e
.MailFrom
.Address
)
68 func generateEnvelopeId(prefix
string, t time
.Time
) string {
71 return fmt
.Sprintf("%s.%d.%x", prefix
, t
.UnixNano(), idBytes
)
74 // lookupRemoteHost attempts to reverse look-up the provided IP address. On
75 // success, it returns the hostname and the IP as formatted for a receive
76 // trace. If the lookup fails, it just returns the original IP.
77 func lookupRemoteHost(addr net
.Addr
) string {
78 rhost
, _
, err
:= net
.SplitHostPort(addr
.String())
83 rhosts
, err
:= net
.LookupAddr(rhost
)
85 rhost
= fmt
.Sprintf("%s [%s]", rhosts
[0], rhost
)
91 // Server provides an interface for handling incoming SMTP requests via
93 type Server
interface {
94 // Returns the name of the server, to use in HELO advertisements.
97 // If non-nil, enables STARTTLS support on the SMTP server with the given
99 TLSConfig() *tls
.Config
101 // Returns an status line indicating whether this server can receive
102 // mail for the specified email address.
103 VerifyAddress(mail
.Address
) ReplyLine
105 // Verify that the authc+passwd identity can send mail as authz on this
107 Authenticate(authz
, authc
, passwd
string) bool
109 // Delivers a valid incoming message to a recipient on this server. The
110 // addressee has been validated via VerifyAddress.
111 DeliverMessage(Envelope
) *ReplyLine
113 // RelayMessage instructs the server to send the Envelope to another
114 // MTA for outbound delivery. `authc` reports the authenticated username.
115 RelayMessage(en Envelope
, authc
string)
118 // MTA (Mail Transport Agent) allows a Server to interface with other SMTP
121 // RelayMessage will attempt to send the specified Envelope. It will ask the
122 // Server to dial the MX servers for the addresses in Envelope.RcptTo for
123 // delivery. If relaying fails, a failure notice will be sent to the sender
124 // via Server.DeliverMessage.
125 RelayMessage(Envelope
)
128 func NewDefaultMTA(server Server
, log
*zap
.Logger
) MTA
{
140 type EmptyServerCallbacks
struct{}
142 func (*EmptyServerCallbacks
) TLSConfig() *tls
.Config
{
146 func (*EmptyServerCallbacks
) VerifyAddress(mail
.Address
) ReplyLine
{
150 func (*EmptyServerCallbacks
) Authenticate(authz
, authc
, passwd
string) bool {
154 func (*EmptyServerCallbacks
) DeliverMessage(Envelope
) *ReplyLine
{
158 func (*EmptyServerCallbacks
) RelayMessage(Envelope
) {