autobrr/internal/notification/discord.go
Ludvig Lundgren 38addb99e6
feat(notifications): add telegram support (#299)
* feat(notifications): add telegram support

* feat(notifications): change list view

* refactor(notifications): overall setup

* feat(notifications): forms add telegram
2022-06-13 17:01:36 +02:00

220 lines
5 KiB
Go

package notification
import (
"bytes"
"crypto/tls"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
"strings"
"time"
"github.com/autobrr/autobrr/internal/domain"
"github.com/autobrr/autobrr/internal/logger"
)
type DiscordMessage struct {
Content interface{} `json:"content"`
Embeds []DiscordEmbeds `json:"embeds,omitempty"`
Username string `json:"username"`
}
type DiscordEmbeds struct {
Title string `json:"title"`
Description string `json:"description"`
Color int `json:"color"`
Fields []DiscordEmbedsFields `json:"fields,omitempty"`
Timestamp time.Time `json:"timestamp"`
}
type DiscordEmbedsFields struct {
Name string `json:"name"`
Value string `json:"value"`
Inline bool `json:"inline,omitempty"`
}
type EmbedColors int
const (
LIGHT_BLUE EmbedColors = 5814783 // 58b9ff
RED EmbedColors = 15548997 // ed4245
GREEN EmbedColors = 5763719 // 57f287
GRAY EmbedColors = 10070709 // 99aab5
)
type discordSender struct {
log logger.Logger
Settings domain.Notification
}
func NewDiscordSender(log logger.Logger, settings domain.Notification) domain.NotificationSender {
return &discordSender{log: log, Settings: settings}
}
func (a *discordSender) Send(event domain.NotificationEvent, payload domain.NotificationPayload) error {
m := DiscordMessage{
Content: nil,
Embeds: []DiscordEmbeds{a.buildEmbed(event, payload)},
Username: "brr",
}
jsonData, err := json.Marshal(m)
if err != nil {
a.log.Error().Err(err).Msgf("discord client could not marshal data: %v", m)
return err
}
req, err := http.NewRequest(http.MethodPost, a.Settings.Webhook, bytes.NewBuffer(jsonData))
if err != nil {
a.log.Error().Err(err).Msgf("discord client request error: %v", event)
return err
}
req.Header.Set("Content-Type", "application/json")
//req.Header.Set("User-Agent", "autobrr")
t := &http.Transport{
TLSClientConfig: &tls.Config{
InsecureSkipVerify: true,
},
}
client := http.Client{Transport: t, Timeout: 30 * time.Second}
res, err := client.Do(req)
if err != nil {
a.log.Error().Err(err).Msgf("discord client request error: %v", event)
return err
}
body, err := ioutil.ReadAll(res.Body)
if err != nil {
a.log.Error().Err(err).Msgf("discord client request error: %v", event)
return err
}
defer res.Body.Close()
a.log.Trace().Msgf("discord status: %v response: %v", res.StatusCode, string(body))
if res.StatusCode != http.StatusNoContent {
a.log.Error().Err(err).Msgf("discord client request error: %v", string(body))
return fmt.Errorf("err: %v", string(body))
}
a.log.Debug().Msg("notification successfully sent to discord")
return nil
}
func (a *discordSender) CanSend(event domain.NotificationEvent) bool {
if a.isEnabled() && a.isEnabledEvent(event) {
return true
}
return false
}
func (a *discordSender) isEnabled() bool {
if a.Settings.Enabled && a.Settings.Webhook != "" {
return true
}
return false
}
func (a *discordSender) isEnabledEvent(event domain.NotificationEvent) bool {
for _, e := range a.Settings.Events {
if e == string(event) {
return true
}
}
return false
}
func (a *discordSender) buildEmbed(event domain.NotificationEvent, payload domain.NotificationPayload) DiscordEmbeds {
color := LIGHT_BLUE
switch event {
case domain.NotificationEventPushApproved:
color = GREEN
case domain.NotificationEventPushRejected:
color = GRAY
case domain.NotificationEventPushError:
color = RED
case domain.NotificationEventTest:
color = LIGHT_BLUE
}
var fields []DiscordEmbedsFields
if payload.Status != "" {
f := DiscordEmbedsFields{
Name: "Status",
Value: payload.Status.String(),
Inline: true,
}
fields = append(fields, f)
}
if payload.Indexer != "" {
f := DiscordEmbedsFields{
Name: "Indexer",
Value: payload.Indexer,
Inline: true,
}
fields = append(fields, f)
}
if payload.Filter != "" {
f := DiscordEmbedsFields{
Name: "Filter",
Value: payload.Filter,
Inline: true,
}
fields = append(fields, f)
}
if payload.Action != "" {
f := DiscordEmbedsFields{
Name: "Action",
Value: payload.Action,
Inline: true,
}
fields = append(fields, f)
}
if payload.ActionType != "" {
f := DiscordEmbedsFields{
Name: "Action type",
Value: string(payload.ActionType),
Inline: true,
}
fields = append(fields, f)
}
if payload.ActionClient != "" {
f := DiscordEmbedsFields{
Name: "Action client",
Value: payload.ActionClient,
Inline: true,
}
fields = append(fields, f)
}
if len(payload.Rejections) > 0 {
f := DiscordEmbedsFields{
Name: "Reasons",
Value: fmt.Sprintf("```\n%v\n```", strings.Join(payload.Rejections, ", ")),
Inline: false,
}
fields = append(fields, f)
}
embed := DiscordEmbeds{
Title: payload.ReleaseName,
Description: "New release!",
Color: int(color),
Fields: fields,
Timestamp: time.Now(),
}
if payload.Subject != "" && payload.Message != "" {
embed.Title = payload.Subject
embed.Description = payload.Message
}
return embed
}