* Add criticals to email summary * chore(report/email): add Critical keys Co-authored-by: MaineK00n <mainek00n.1229@gmail.com>
216 lines
5.5 KiB
Go
216 lines
5.5 KiB
Go
package reporter
|
|
|
|
import (
|
|
"crypto/tls"
|
|
"fmt"
|
|
"net"
|
|
"net/mail"
|
|
"strings"
|
|
"time"
|
|
|
|
sasl "github.com/emersion/go-sasl"
|
|
smtp "github.com/emersion/go-smtp"
|
|
"github.com/future-architect/vuls/config"
|
|
"github.com/future-architect/vuls/models"
|
|
"golang.org/x/xerrors"
|
|
)
|
|
|
|
// EMailWriter send mail
|
|
type EMailWriter struct {
|
|
FormatOneEMail bool
|
|
FormatOneLineText bool
|
|
FormatList bool
|
|
Cnf config.SMTPConf
|
|
}
|
|
|
|
func (w EMailWriter) Write(rs ...models.ScanResult) (err error) {
|
|
var message string
|
|
sender := NewEMailSender(w.Cnf)
|
|
m := map[string]int{}
|
|
for _, r := range rs {
|
|
if w.FormatOneEMail {
|
|
message += formatFullPlainText(r) + "\r\n\r\n"
|
|
mm := r.ScannedCves.CountGroupBySeverity()
|
|
keys := []string{"Critical", "High", "Medium", "Low", "Unknown"}
|
|
for _, k := range keys {
|
|
m[k] += mm[k]
|
|
}
|
|
} else {
|
|
var subject string
|
|
if len(r.Errors) != 0 {
|
|
subject = fmt.Sprintf("%s%s An error occurred while scanning",
|
|
w.Cnf.SubjectPrefix, r.ServerInfo())
|
|
} else {
|
|
subject = fmt.Sprintf("%s%s %s",
|
|
w.Cnf.SubjectPrefix,
|
|
r.ServerInfo(),
|
|
r.ScannedCves.FormatCveSummary())
|
|
}
|
|
if w.FormatList {
|
|
message = formatList(r)
|
|
} else {
|
|
message = formatFullPlainText(r)
|
|
}
|
|
if w.FormatOneLineText {
|
|
message = fmt.Sprintf("One Line Summary\r\n================\r\n%s", formatOneLineSummary(r))
|
|
}
|
|
if err := sender.Send(subject, message); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
summary := fmt.Sprintf("Total: %d (Critical:%d High:%d Medium:%d Low:%d ?:%d)",
|
|
m["Critical"]+m["High"]+m["Medium"]+m["Low"]+m["Unknown"],
|
|
m["Critical"], m["High"], m["Medium"], m["Low"], m["Unknown"])
|
|
|
|
origmessage := message
|
|
if w.FormatOneEMail {
|
|
message = fmt.Sprintf("One Line Summary\r\n================\r\n%s", formatOneLineSummary(rs...))
|
|
if !w.FormatOneLineText {
|
|
message += fmt.Sprintf("\r\n\r\n%s", origmessage)
|
|
}
|
|
|
|
subject := fmt.Sprintf("%s %s",
|
|
w.Cnf.SubjectPrefix, summary)
|
|
return sender.Send(subject, message)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// EMailSender is interface of sending e-mail
|
|
type EMailSender interface {
|
|
Send(subject, body string) error
|
|
}
|
|
|
|
type emailSender struct {
|
|
conf config.SMTPConf
|
|
}
|
|
|
|
func (e *emailSender) sendMail(smtpServerAddr, message string) (err error) {
|
|
var c *smtp.Client
|
|
var auth sasl.Client
|
|
emailConf := e.conf
|
|
//TLS Config
|
|
tlsConfig := &tls.Config{
|
|
ServerName: emailConf.SMTPAddr,
|
|
}
|
|
switch emailConf.SMTPPort {
|
|
case "465":
|
|
//New TLS connection
|
|
c, err = smtp.DialTLS(smtpServerAddr, tlsConfig)
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to create TLS connection to SMTP server: %w", err)
|
|
}
|
|
default:
|
|
c, err = smtp.Dial(smtpServerAddr)
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to create connection to SMTP server: %w", err)
|
|
}
|
|
}
|
|
defer c.Close()
|
|
|
|
if err = c.Hello("localhost"); err != nil {
|
|
return xerrors.Errorf("Failed to send Hello command: %w", err)
|
|
}
|
|
|
|
if ok, _ := c.Extension("STARTTLS"); ok {
|
|
if err := c.StartTLS(tlsConfig); err != nil {
|
|
return xerrors.Errorf("Failed to STARTTLS: %w", err)
|
|
}
|
|
}
|
|
|
|
if ok, param := c.Extension("AUTH"); ok {
|
|
authList := strings.Split(param, " ")
|
|
auth = e.newSaslClient(authList)
|
|
if err = c.Auth(auth); err != nil {
|
|
return xerrors.Errorf("Failed to authenticate: %w", err)
|
|
}
|
|
}
|
|
|
|
if err = c.Mail(emailConf.From, nil); err != nil {
|
|
return xerrors.Errorf("Failed to send Mail command: %w", err)
|
|
}
|
|
for _, to := range emailConf.To {
|
|
if err = c.Rcpt(to); err != nil {
|
|
return xerrors.Errorf("Failed to send Rcpt command: %w", err)
|
|
}
|
|
}
|
|
|
|
w, err := c.Data()
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to send Data command: %w", err)
|
|
}
|
|
_, err = w.Write([]byte(message))
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to write EMail message: %w", err)
|
|
}
|
|
err = w.Close()
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to close Writer: %w", err)
|
|
}
|
|
err = c.Quit()
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to close connection: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (e *emailSender) Send(subject, body string) (err error) {
|
|
emailConf := e.conf
|
|
to := strings.Join(emailConf.To[:], ", ")
|
|
cc := strings.Join(emailConf.Cc[:], ", ")
|
|
mailAddresses := append(emailConf.To, emailConf.Cc...)
|
|
if _, err := mail.ParseAddressList(strings.Join(mailAddresses[:], ", ")); err != nil {
|
|
return xerrors.Errorf("Failed to parse email addresses: %w", err)
|
|
}
|
|
|
|
headers := make(map[string]string)
|
|
headers["From"] = emailConf.From
|
|
headers["To"] = to
|
|
headers["Cc"] = cc
|
|
headers["Subject"] = subject
|
|
headers["Date"] = time.Now().Format(time.RFC1123Z)
|
|
headers["Content-Type"] = "text/plain; charset=utf-8"
|
|
|
|
var header string
|
|
for k, v := range headers {
|
|
header += fmt.Sprintf("%s: %s\r\n", k, v)
|
|
}
|
|
message := fmt.Sprintf("%s\r\n%s", header, body)
|
|
|
|
smtpServer := net.JoinHostPort(emailConf.SMTPAddr, emailConf.SMTPPort)
|
|
|
|
if emailConf.User != "" && emailConf.Password != "" {
|
|
err = e.sendMail(smtpServer, message)
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to send emails: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
err = e.sendMail(smtpServer, message)
|
|
if err != nil {
|
|
return xerrors.Errorf("Failed to send emails: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// NewEMailSender creates emailSender
|
|
func NewEMailSender(cnf config.SMTPConf) EMailSender {
|
|
return &emailSender{cnf}
|
|
}
|
|
|
|
func (e *emailSender) newSaslClient(authList []string) sasl.Client {
|
|
for _, v := range authList {
|
|
switch v {
|
|
case "PLAIN":
|
|
auth := sasl.NewPlainClient("", e.conf.User, e.conf.Password)
|
|
return auth
|
|
case "LOGIN":
|
|
auth := sasl.NewLoginClient(e.conf.User, e.conf.Password)
|
|
return auth
|
|
}
|
|
}
|
|
return nil
|
|
}
|