2022-11-13 00:17:42 -06:00
|
|
|
package comm
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
|
|
|
|
"github.com/eternal-flame-AD/yoake/internal/comm/email"
|
|
|
|
"github.com/eternal-flame-AD/yoake/internal/comm/gotify"
|
|
|
|
"github.com/eternal-flame-AD/yoake/internal/comm/model"
|
2022-11-19 11:04:03 -06:00
|
|
|
"github.com/eternal-flame-AD/yoake/internal/comm/telegram"
|
|
|
|
"github.com/eternal-flame-AD/yoake/internal/db"
|
2022-11-13 00:17:42 -06:00
|
|
|
"github.com/eternal-flame-AD/yoake/internal/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Communicator struct {
|
|
|
|
commMethods map[string]model.CommMethod
|
|
|
|
fallbackCommunicators []string
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
errMethodNotSupported = errors.New("method not supported")
|
|
|
|
)
|
|
|
|
|
2022-11-19 11:04:03 -06:00
|
|
|
func (c *Communicator) actualSendGenericMessage(tryMethod string, message *model.GenericMessage) error {
|
2022-11-13 00:17:42 -06:00
|
|
|
if comm, ok := c.commMethods[tryMethod]; ok {
|
2022-11-19 11:04:03 -06:00
|
|
|
if convertedMsg, err := ConvertGenericMessage(message, comm.SupportedMIME()); err == nil {
|
|
|
|
err := comm.SendGenericMessage(convertedMsg)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
message.ThreadID = convertedMsg.ThreadID
|
|
|
|
return nil
|
2022-11-13 00:17:42 -06:00
|
|
|
} else {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return errMethodNotSupported
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMethod returns the method with the given name.
|
|
|
|
func (c *Communicator) GetMethod(method string) model.CommMethod {
|
|
|
|
return c.commMethods[method]
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMethodsByMIME returns a list of methods that support the given MIME type as the message type, MIME convertions were considered.
|
|
|
|
func (c *Communicator) GetMethodsByMIME(mime string) []model.CommMethod {
|
|
|
|
var result []model.CommMethod
|
|
|
|
for _, comm := range c.commMethods {
|
|
|
|
if util.Contain(ConvertOutMIMEToSupportedInMIME(comm.SupportedMIME()), mime) {
|
|
|
|
result = append(result, comm)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
type ErrorSentWithFallback struct {
|
|
|
|
OriginalError error
|
|
|
|
OrignalMethod string
|
|
|
|
FallbackMethod string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e ErrorSentWithFallback) Error() string {
|
|
|
|
return fmt.Sprintf("used fallback method %s because original method %s reeported error: %v", e.FallbackMethod, e.OrignalMethod, e.OriginalError)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SendGenericMethods sends a message using the preferred method
|
|
|
|
// if the preferred method failed to send the message, fallback methods will be tried,
|
|
|
|
// and an ErrorSentWithFabback will be returned if any fallback method succeeded.
|
|
|
|
// if fallback methods failed as well the original error will be returned.
|
2022-11-19 11:04:03 -06:00
|
|
|
func (c *Communicator) SendGenericMessage(preferredMethod string, message *model.GenericMessage, force bool) error {
|
2022-11-13 00:17:42 -06:00
|
|
|
if preferredMethod == "" {
|
|
|
|
preferredMethod = c.fallbackCommunicators[0]
|
|
|
|
}
|
|
|
|
if origErr := c.actualSendGenericMessage(preferredMethod, message); origErr != nil {
|
2022-11-18 23:37:30 -06:00
|
|
|
if force {
|
|
|
|
log.Printf("Failed to send message using preferred method %s: %v", preferredMethod, origErr)
|
|
|
|
return origErr
|
|
|
|
}
|
2022-11-13 00:17:42 -06:00
|
|
|
log.Printf("Failed to send message using preferred method %s: %v. trying fallback methods", preferredMethod, origErr)
|
|
|
|
for _, fallback := range c.fallbackCommunicators {
|
|
|
|
if fallback == preferredMethod {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err := c.actualSendGenericMessage(fallback, message); err == nil {
|
|
|
|
log.Printf("Sent message using fallback method %s", fallback)
|
|
|
|
return ErrorSentWithFallback{
|
|
|
|
OriginalError: origErr,
|
|
|
|
OrignalMethod: preferredMethod,
|
|
|
|
FallbackMethod: fallback,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Printf("Failed to send message using fallback method %s: %v", fallback, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return origErr
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-11-19 11:04:03 -06:00
|
|
|
func InitCommunicator(database db.DB) *Communicator {
|
2022-11-13 00:17:42 -06:00
|
|
|
comm := &Communicator{
|
|
|
|
commMethods: make(map[string]model.CommMethod),
|
|
|
|
}
|
|
|
|
if emailHandler, err := email.NewHandler(); err == nil {
|
|
|
|
comm.commMethods["email"] = emailHandler
|
|
|
|
comm.fallbackCommunicators = append(comm.fallbackCommunicators, "email")
|
|
|
|
} else {
|
|
|
|
log.Printf("Failed to initialize email communicator: %v", err)
|
|
|
|
}
|
|
|
|
if gotifyHandler, err := gotify.NewClient(); err == nil {
|
|
|
|
comm.commMethods["gotify"] = gotifyHandler
|
|
|
|
comm.fallbackCommunicators = append(comm.fallbackCommunicators, "gotify")
|
|
|
|
} else {
|
|
|
|
log.Printf("Failed to initialize gotify communicator: %v", err)
|
|
|
|
}
|
2022-11-19 11:04:03 -06:00
|
|
|
if telegramHandler, err := telegram.NewClient(database); err == nil {
|
|
|
|
comm.commMethods["telegram"] = telegramHandler
|
|
|
|
comm.fallbackCommunicators = append(comm.fallbackCommunicators, "telegram")
|
|
|
|
} else {
|
|
|
|
log.Printf("Failed to initialize telegram communicator: %v", err)
|
|
|
|
}
|
2022-11-13 00:17:42 -06:00
|
|
|
|
|
|
|
return comm
|
|
|
|
}
|