mirror of
https://github.com/prometheus/alertmanager
synced 2025-01-04 04:52:05 +00:00
efc956c7f7
* SMTP config: add global and local password file fields Add config fields (for both global email config and route-specific email config) that specify path to file containing SMTP password. We don't want the password in the config file itself, and reading the password from a k8s-secret-backed file keeps the password itself "encrypted at rest" in etcd, and cleanly separated from the rest of the AM config. I used the same approach as pull request #2534 "Add support to set the Slack URL in the file" <https://github.com/prometheus/alertmanager/pull/2534/files> in the upstream repo. Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * changed *AuthPasswordFile field types to string per review feedback Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * added error to getPassword() retval per review feedback Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * simplified conf.smtp-* files Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * update docs to reflect field type change Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * don't treat username-without-password as invalid Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * test cleanup Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * Apply suggestions from code review Co-authored-by: Simon Pasquier <spasquie@redhat.com> Signed-off-by: Eric R. Rath <4080262+ericrrath@users.noreply.github.com> * Updated per review feedback Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * added sub-test per review feedback Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * added test on Email.getPassword() per feedback Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * only inherit global SMTP passwords if neither local password field is set Signed-off-by: Eric R. Rath <eric.rath@oracle.com> * removed blank line caught by gofumpt Signed-off-by: Eric R. Rath <eric.rath@oracle.com> Signed-off-by: Eric R. Rath <eric.rath@oracle.com> Signed-off-by: Eric R. Rath <4080262+ericrrath@users.noreply.github.com> Co-authored-by: Simon Pasquier <spasquie@redhat.com>
375 lines
10 KiB
Go
375 lines
10 KiB
Go
// Copyright 2019 Prometheus Team
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package email
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"crypto/tls"
|
|
"fmt"
|
|
"math/rand"
|
|
"mime"
|
|
"mime/multipart"
|
|
"mime/quotedprintable"
|
|
"net"
|
|
"net/mail"
|
|
"net/smtp"
|
|
"net/textproto"
|
|
"os"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/go-kit/log"
|
|
"github.com/go-kit/log/level"
|
|
"github.com/pkg/errors"
|
|
commoncfg "github.com/prometheus/common/config"
|
|
|
|
"github.com/prometheus/alertmanager/config"
|
|
"github.com/prometheus/alertmanager/notify"
|
|
"github.com/prometheus/alertmanager/template"
|
|
"github.com/prometheus/alertmanager/types"
|
|
)
|
|
|
|
// Email implements a Notifier for email notifications.
|
|
type Email struct {
|
|
conf *config.EmailConfig
|
|
tmpl *template.Template
|
|
logger log.Logger
|
|
hostname string
|
|
}
|
|
|
|
// New returns a new Email notifier.
|
|
func New(c *config.EmailConfig, t *template.Template, l log.Logger) *Email {
|
|
if _, ok := c.Headers["Subject"]; !ok {
|
|
c.Headers["Subject"] = config.DefaultEmailSubject
|
|
}
|
|
if _, ok := c.Headers["To"]; !ok {
|
|
c.Headers["To"] = c.To
|
|
}
|
|
if _, ok := c.Headers["From"]; !ok {
|
|
c.Headers["From"] = c.From
|
|
}
|
|
|
|
h, err := os.Hostname()
|
|
// If we can't get the hostname, we'll use localhost
|
|
if err != nil {
|
|
h = "localhost.localdomain"
|
|
}
|
|
return &Email{conf: c, tmpl: t, logger: l, hostname: h}
|
|
}
|
|
|
|
// auth resolves a string of authentication mechanisms.
|
|
func (n *Email) auth(mechs string) (smtp.Auth, error) {
|
|
username := n.conf.AuthUsername
|
|
|
|
// If no username is set, keep going without authentication.
|
|
if n.conf.AuthUsername == "" {
|
|
level.Debug(n.logger).Log("msg", "smtp_auth_username is not configured. Attempting to send email without authenticating")
|
|
return nil, nil
|
|
}
|
|
|
|
err := &types.MultiError{}
|
|
for _, mech := range strings.Split(mechs, " ") {
|
|
switch mech {
|
|
case "CRAM-MD5":
|
|
secret := string(n.conf.AuthSecret)
|
|
if secret == "" {
|
|
err.Add(errors.New("missing secret for CRAM-MD5 auth mechanism"))
|
|
continue
|
|
}
|
|
return smtp.CRAMMD5Auth(username, secret), nil
|
|
|
|
case "PLAIN":
|
|
password, passwordErr := n.getPassword()
|
|
if passwordErr != nil {
|
|
err.Add(passwordErr)
|
|
continue
|
|
}
|
|
if password == "" {
|
|
err.Add(errors.New("missing password for PLAIN auth mechanism"))
|
|
continue
|
|
}
|
|
identity := n.conf.AuthIdentity
|
|
|
|
return smtp.PlainAuth(identity, username, password, n.conf.Smarthost.Host), nil
|
|
case "LOGIN":
|
|
password, passwordErr := n.getPassword()
|
|
if passwordErr != nil {
|
|
err.Add(passwordErr)
|
|
continue
|
|
}
|
|
if password == "" {
|
|
err.Add(errors.New("missing password for LOGIN auth mechanism"))
|
|
continue
|
|
}
|
|
return LoginAuth(username, password), nil
|
|
}
|
|
}
|
|
if err.Len() == 0 {
|
|
err.Add(errors.New("unknown auth mechanism: " + mechs))
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
// Notify implements the Notifier interface.
|
|
func (n *Email) Notify(ctx context.Context, as ...*types.Alert) (bool, error) {
|
|
var (
|
|
c *smtp.Client
|
|
conn net.Conn
|
|
err error
|
|
success = false
|
|
)
|
|
if n.conf.Smarthost.Port == "465" {
|
|
tlsConfig, err := commoncfg.NewTLSConfig(&n.conf.TLSConfig)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "parse TLS configuration")
|
|
}
|
|
if tlsConfig.ServerName == "" {
|
|
tlsConfig.ServerName = n.conf.Smarthost.Host
|
|
}
|
|
|
|
conn, err = tls.Dial("tcp", n.conf.Smarthost.String(), tlsConfig)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "establish TLS connection to server")
|
|
}
|
|
} else {
|
|
var (
|
|
d = net.Dialer{}
|
|
err error
|
|
)
|
|
conn, err = d.DialContext(ctx, "tcp", n.conf.Smarthost.String())
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "establish connection to server")
|
|
}
|
|
}
|
|
c, err = smtp.NewClient(conn, n.conf.Smarthost.Host)
|
|
if err != nil {
|
|
conn.Close()
|
|
return true, errors.Wrap(err, "create SMTP client")
|
|
}
|
|
defer func() {
|
|
// Try to clean up after ourselves but don't log anything if something has failed.
|
|
if err := c.Quit(); success && err != nil {
|
|
level.Warn(n.logger).Log("msg", "failed to close SMTP connection", "err", err)
|
|
}
|
|
}()
|
|
|
|
if n.conf.Hello != "" {
|
|
err = c.Hello(n.conf.Hello)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "send EHLO command")
|
|
}
|
|
}
|
|
|
|
// Global Config guarantees RequireTLS is not nil.
|
|
if *n.conf.RequireTLS {
|
|
if ok, _ := c.Extension("STARTTLS"); !ok {
|
|
return true, errors.Errorf("'require_tls' is true (default) but %q does not advertise the STARTTLS extension", n.conf.Smarthost)
|
|
}
|
|
|
|
tlsConf, err := commoncfg.NewTLSConfig(&n.conf.TLSConfig)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "parse TLS configuration")
|
|
}
|
|
if tlsConf.ServerName == "" {
|
|
tlsConf.ServerName = n.conf.Smarthost.Host
|
|
}
|
|
|
|
if err := c.StartTLS(tlsConf); err != nil {
|
|
return true, errors.Wrap(err, "send STARTTLS command")
|
|
}
|
|
}
|
|
|
|
if ok, mech := c.Extension("AUTH"); ok {
|
|
auth, err := n.auth(mech)
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "find auth mechanism")
|
|
}
|
|
if auth != nil {
|
|
if err := c.Auth(auth); err != nil {
|
|
return true, errors.Wrapf(err, "%T auth", auth)
|
|
}
|
|
}
|
|
}
|
|
|
|
var (
|
|
tmplErr error
|
|
data = notify.GetTemplateData(ctx, n.tmpl, as, n.logger)
|
|
tmpl = notify.TmplText(n.tmpl, data, &tmplErr)
|
|
)
|
|
from := tmpl(n.conf.From)
|
|
if tmplErr != nil {
|
|
return false, errors.Wrap(tmplErr, "execute 'from' template")
|
|
}
|
|
to := tmpl(n.conf.To)
|
|
if tmplErr != nil {
|
|
return false, errors.Wrap(tmplErr, "execute 'to' template")
|
|
}
|
|
|
|
addrs, err := mail.ParseAddressList(from)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "parse 'from' addresses")
|
|
}
|
|
if len(addrs) != 1 {
|
|
return false, errors.Errorf("must be exactly one 'from' address (got: %d)", len(addrs))
|
|
}
|
|
if err = c.Mail(addrs[0].Address); err != nil {
|
|
return true, errors.Wrap(err, "send MAIL command")
|
|
}
|
|
addrs, err = mail.ParseAddressList(to)
|
|
if err != nil {
|
|
return false, errors.Wrapf(err, "parse 'to' addresses")
|
|
}
|
|
for _, addr := range addrs {
|
|
if err = c.Rcpt(addr.Address); err != nil {
|
|
return true, errors.Wrapf(err, "send RCPT command")
|
|
}
|
|
}
|
|
|
|
// Send the email headers and body.
|
|
message, err := c.Data()
|
|
if err != nil {
|
|
return true, errors.Wrapf(err, "send DATA command")
|
|
}
|
|
defer message.Close()
|
|
|
|
buffer := &bytes.Buffer{}
|
|
for header, t := range n.conf.Headers {
|
|
value, err := n.tmpl.ExecuteTextString(t, data)
|
|
if err != nil {
|
|
return false, errors.Wrapf(err, "execute %q header template", header)
|
|
}
|
|
fmt.Fprintf(buffer, "%s: %s\r\n", header, mime.QEncoding.Encode("utf-8", value))
|
|
}
|
|
|
|
if _, ok := n.conf.Headers["Message-Id"]; !ok {
|
|
fmt.Fprintf(buffer, "Message-Id: %s\r\n", fmt.Sprintf("<%d.%d@%s>", time.Now().UnixNano(), rand.Uint64(), n.hostname))
|
|
}
|
|
|
|
multipartBuffer := &bytes.Buffer{}
|
|
multipartWriter := multipart.NewWriter(multipartBuffer)
|
|
|
|
fmt.Fprintf(buffer, "Date: %s\r\n", time.Now().Format(time.RFC1123Z))
|
|
fmt.Fprintf(buffer, "Content-Type: multipart/alternative; boundary=%s\r\n", multipartWriter.Boundary())
|
|
fmt.Fprintf(buffer, "MIME-Version: 1.0\r\n\r\n")
|
|
|
|
// TODO: Add some useful headers here, such as URL of the alertmanager
|
|
// and active/resolved.
|
|
_, err = message.Write(buffer.Bytes())
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "write headers")
|
|
}
|
|
|
|
if len(n.conf.Text) > 0 {
|
|
// Text template
|
|
w, err := multipartWriter.CreatePart(textproto.MIMEHeader{
|
|
"Content-Transfer-Encoding": {"quoted-printable"},
|
|
"Content-Type": {"text/plain; charset=UTF-8"},
|
|
})
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "create part for text template")
|
|
}
|
|
body, err := n.tmpl.ExecuteTextString(n.conf.Text, data)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "execute text template")
|
|
}
|
|
qw := quotedprintable.NewWriter(w)
|
|
_, err = qw.Write([]byte(body))
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "write text part")
|
|
}
|
|
err = qw.Close()
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "close text part")
|
|
}
|
|
}
|
|
|
|
if len(n.conf.HTML) > 0 {
|
|
// Html template
|
|
// Preferred alternative placed last per section 5.1.4 of RFC 2046
|
|
// https://www.ietf.org/rfc/rfc2046.txt
|
|
w, err := multipartWriter.CreatePart(textproto.MIMEHeader{
|
|
"Content-Transfer-Encoding": {"quoted-printable"},
|
|
"Content-Type": {"text/html; charset=UTF-8"},
|
|
})
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "create part for html template")
|
|
}
|
|
body, err := n.tmpl.ExecuteHTMLString(n.conf.HTML, data)
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "execute html template")
|
|
}
|
|
qw := quotedprintable.NewWriter(w)
|
|
_, err = qw.Write([]byte(body))
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "write HTML part")
|
|
}
|
|
err = qw.Close()
|
|
if err != nil {
|
|
return true, errors.Wrap(err, "close HTML part")
|
|
}
|
|
}
|
|
|
|
err = multipartWriter.Close()
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "close multipartWriter")
|
|
}
|
|
|
|
_, err = message.Write(multipartBuffer.Bytes())
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "write body buffer")
|
|
}
|
|
|
|
success = true
|
|
return false, nil
|
|
}
|
|
|
|
type loginAuth struct {
|
|
username, password string
|
|
}
|
|
|
|
func LoginAuth(username, password string) smtp.Auth {
|
|
return &loginAuth{username, password}
|
|
}
|
|
|
|
func (a *loginAuth) Start(server *smtp.ServerInfo) (string, []byte, error) {
|
|
return "LOGIN", []byte{}, nil
|
|
}
|
|
|
|
// Used for AUTH LOGIN. (Maybe password should be encrypted)
|
|
func (a *loginAuth) Next(fromServer []byte, more bool) ([]byte, error) {
|
|
if more {
|
|
switch strings.ToLower(string(fromServer)) {
|
|
case "username:":
|
|
return []byte(a.username), nil
|
|
case "password:":
|
|
return []byte(a.password), nil
|
|
default:
|
|
return nil, errors.New("unexpected server challenge")
|
|
}
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
func (n *Email) getPassword() (string, error) {
|
|
if len(n.conf.AuthPasswordFile) > 0 {
|
|
content, err := os.ReadFile(n.conf.AuthPasswordFile)
|
|
if err != nil {
|
|
return "", fmt.Errorf("could not read %s: %w", n.conf.AuthPasswordFile, err)
|
|
}
|
|
return string(content), nil
|
|
}
|
|
return string(n.conf.AuthPassword), nil
|
|
}
|