2020-02-07 17:27:45 +01:00
|
|
|
// Vikunja is a to-do list application to facilitate your life.
|
2021-02-02 20:19:13 +01:00
|
|
|
// Copyright 2018-2021 Vikunja and contributors. All rights reserved.
|
2018-11-26 21:17:33 +01:00
|
|
|
//
|
2019-12-04 20:39:56 +01:00
|
|
|
// This program is free software: you can redistribute it and/or modify
|
2020-12-23 16:41:52 +01:00
|
|
|
// it under the terms of the GNU Affero General Public Licensee as published by
|
2019-12-04 20:39:56 +01:00
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
2018-11-26 21:17:33 +01:00
|
|
|
//
|
2019-12-04 20:39:56 +01:00
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2020-12-23 16:41:52 +01:00
|
|
|
// GNU Affero General Public Licensee for more details.
|
2018-11-26 21:17:33 +01:00
|
|
|
//
|
2020-12-23 16:41:52 +01:00
|
|
|
// You should have received a copy of the GNU Affero General Public Licensee
|
2019-12-04 20:39:56 +01:00
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2018-11-26 21:17:33 +01:00
|
|
|
|
2018-10-27 11:33:28 +02:00
|
|
|
package mail
|
|
|
|
|
|
|
|
import (
|
2022-07-07 15:54:38 +02:00
|
|
|
"embed"
|
2022-06-19 16:26:17 +02:00
|
|
|
"io"
|
|
|
|
|
2019-07-06 22:12:26 +02:00
|
|
|
"code.vikunja.io/api/pkg/config"
|
2019-07-11 21:10:42 +02:00
|
|
|
"code.vikunja.io/api/pkg/log"
|
2022-06-19 15:57:00 +02:00
|
|
|
"code.vikunja.io/api/pkg/version"
|
|
|
|
|
|
|
|
"github.com/wneessen/go-mail"
|
2018-10-27 11:33:28 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Opts holds infos for a mail
|
|
|
|
type Opts struct {
|
2021-02-07 22:05:09 +01:00
|
|
|
From string
|
2018-10-27 11:33:28 +02:00
|
|
|
To string
|
|
|
|
Subject string
|
|
|
|
Message string
|
|
|
|
HTMLMessage string
|
|
|
|
ContentType ContentType
|
|
|
|
Boundary string
|
|
|
|
Headers []*header
|
2022-06-19 16:21:11 +02:00
|
|
|
Embeds map[string]io.Reader
|
2022-07-07 15:54:38 +02:00
|
|
|
EmbedFS map[string]*embed.FS
|
2018-10-27 11:33:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// ContentType represents mail content types
|
|
|
|
type ContentType int
|
|
|
|
|
|
|
|
// Enumerate all the team rights
|
|
|
|
const (
|
|
|
|
ContentTypePlain ContentType = iota
|
|
|
|
ContentTypeHTML
|
|
|
|
ContentTypeMultipart
|
|
|
|
)
|
|
|
|
|
|
|
|
type header struct {
|
2022-06-19 15:57:00 +02:00
|
|
|
Field mail.Header
|
2018-10-27 11:33:28 +02:00
|
|
|
Content string
|
|
|
|
}
|
|
|
|
|
2022-06-19 15:57:00 +02:00
|
|
|
// SendTestMail sends a test mail to a recipient.
|
2020-05-29 15:10:06 +02:00
|
|
|
// It works without a queue.
|
2021-02-07 22:05:09 +01:00
|
|
|
func SendTestMail(opts *Opts) error {
|
2020-05-29 15:10:06 +02:00
|
|
|
if config.MailerHost.GetString() == "" {
|
|
|
|
log.Warning("Mailer seems to be not configured! Please see the config docs for more details.")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-19 15:57:00 +02:00
|
|
|
c, err := getClient()
|
2020-05-29 15:10:06 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-06-19 15:57:00 +02:00
|
|
|
m := getMessage(opts)
|
2020-05-29 15:10:06 +02:00
|
|
|
|
2022-06-19 15:57:00 +02:00
|
|
|
return c.DialAndSend(m)
|
2020-05-29 15:10:06 +02:00
|
|
|
}
|
|
|
|
|
2022-06-19 15:57:00 +02:00
|
|
|
func getMessage(opts *Opts) *mail.Msg {
|
|
|
|
m := mail.NewMsg()
|
|
|
|
m.SetUserAgent("Vikunja " + version.Version)
|
2021-02-07 22:05:09 +01:00
|
|
|
if opts.From == "" {
|
2021-07-29 17:42:49 +02:00
|
|
|
opts.From = "Vikunja <" + config.MailerFromEmail.GetString() + ">"
|
2021-02-07 22:05:09 +01:00
|
|
|
}
|
2022-06-19 15:57:00 +02:00
|
|
|
_ = m.From(opts.From)
|
|
|
|
_ = m.To(opts.To)
|
|
|
|
m.Subject(opts.Subject)
|
2022-06-19 16:21:11 +02:00
|
|
|
|
2018-10-27 11:33:28 +02:00
|
|
|
for _, h := range opts.Headers {
|
|
|
|
m.SetHeader(h.Field, h.Content)
|
|
|
|
}
|
|
|
|
|
2022-06-19 16:21:11 +02:00
|
|
|
for name, content := range opts.Embeds {
|
|
|
|
m.EmbedReader(name, content)
|
|
|
|
}
|
|
|
|
|
2022-07-07 15:54:38 +02:00
|
|
|
for name, fs := range opts.EmbedFS {
|
|
|
|
err := m.EmbedFromEmbedFS(name, fs)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error embedding %s via embed.FS into mail: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-27 11:33:28 +02:00
|
|
|
switch opts.ContentType {
|
|
|
|
case ContentTypePlain:
|
2022-06-19 15:57:00 +02:00
|
|
|
m.SetBodyString("text/plain", opts.Message)
|
2018-10-27 11:33:28 +02:00
|
|
|
case ContentTypeHTML:
|
2022-06-19 15:57:00 +02:00
|
|
|
m.SetBodyString("text/html", opts.Message)
|
2018-10-27 11:33:28 +02:00
|
|
|
case ContentTypeMultipart:
|
2022-06-19 15:57:00 +02:00
|
|
|
m.SetBodyString("text/plain", opts.Message)
|
|
|
|
m.AddAlternativeString("text/html", opts.HTMLMessage)
|
2018-10-27 11:33:28 +02:00
|
|
|
}
|
2022-06-19 15:57:00 +02:00
|
|
|
|
2021-02-07 22:05:09 +01:00
|
|
|
return m
|
2018-10-27 11:33:28 +02:00
|
|
|
}
|
|
|
|
|
2021-02-07 22:05:09 +01:00
|
|
|
// SendMail puts a mail in the queue
|
|
|
|
func SendMail(opts *Opts) {
|
|
|
|
if isUnderTest {
|
|
|
|
sentMails = append(sentMails, opts)
|
2018-10-27 11:33:28 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-06-19 15:57:00 +02:00
|
|
|
m := getMessage(opts)
|
2021-02-07 22:05:09 +01:00
|
|
|
Queue <- m
|
2018-10-27 11:33:28 +02:00
|
|
|
}
|