package smtpserver import ( "bytes" "crypto/ed25519" "encoding/hex" "fmt" "io" "time" "github.com/emersion/go-message" "github.com/emersion/go-smtp" "github.com/jxskiss/base62" "github.com/neilalexander/yggmail/internal/utils" ) type SessionRemote struct { backend *Backend state *smtp.ConnectionState public ed25519.PublicKey from string localparts []string } func (s *SessionRemote) Mail(from string, opts smtp.MailOptions) error { _, host, err := utils.ParseAddress(from) if err != nil { return fmt.Errorf("mail.ParseAddress: %w", err) } pks, err := hex.DecodeString(s.state.RemoteAddr.String()) if err != nil { return fmt.Errorf("hex.DecodeString: %w", err) } if remote := base62.EncodeToString(pks); host != remote { return fmt.Errorf("not allowed to send incoming mail as %s", from) } s.from = from return nil } func (s *SessionRemote) Rcpt(to string) error { user, host, err := utils.ParseAddress(to) if err != nil { return fmt.Errorf("mail.ParseAddress: %w", err) } if local := base62.EncodeToString(s.backend.Config.PublicKey); host != local { return fmt.Errorf("not allowed to send mail to %q", host) } s.localparts = append(s.localparts, user) return nil } func (s *SessionRemote) Data(r io.Reader) error { m, err := message.Read(r) if err != nil { return fmt.Errorf("message.Read: %w", err) } m.Header.Add( "Received", fmt.Sprintf("from Yggmail %s; %s", base62.EncodeToString(s.public), time.Now().String(), ), ) var b bytes.Buffer if err := m.WriteTo(&b); err != nil { return fmt.Errorf("m.WriteTo: %w", err) } for _, localpart := range s.localparts { if _, err := s.backend.Storage.MailCreate(localpart, "INBOX", b.Bytes()); err != nil { return fmt.Errorf("s.backend.Storage.StoreMessageFor: %w", err) } s.backend.Log.Printf("Stored new mail for local user %q from %s", localpart, s.from) } return nil } func (s *SessionRemote) Reset() {} func (s *SessionRemote) Logout() error { return nil }