2023-09-01 17:06:04 +00:00
|
|
|
package bridge
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"log/slog"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Handler interface {
|
2023-09-03 14:37:02 +00:00
|
|
|
ProduceNotifications(r *http.Request) ([]Notification, error)
|
2023-09-01 17:06:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type NotificationError struct {
|
|
|
|
Code int `json:"code"`
|
|
|
|
Error string `json:"error"`
|
|
|
|
}
|
|
|
|
|
2023-09-03 14:38:03 +00:00
|
|
|
type NotificationAction struct {
|
|
|
|
Action string
|
|
|
|
Label string
|
|
|
|
Params []string
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewViewAction(label, url string, params ...string) NotificationAction {
|
|
|
|
return NotificationAction{
|
|
|
|
Action: "view",
|
|
|
|
Label: label,
|
|
|
|
Params: append([]string{url}, params...),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n NotificationAction) Format() string {
|
|
|
|
var sb strings.Builder
|
|
|
|
sb.WriteString(n.Action + ", " + n.Label)
|
|
|
|
|
|
|
|
if len(n.Params) > 0 {
|
|
|
|
sb.WriteString(", " + strings.Join(n.Params, ", "))
|
|
|
|
}
|
|
|
|
|
|
|
|
return sb.String()
|
|
|
|
}
|
|
|
|
|
2023-09-01 17:06:04 +00:00
|
|
|
type Notification struct {
|
|
|
|
Title string
|
|
|
|
Body string
|
|
|
|
Priority int
|
|
|
|
Tags []string
|
2023-09-03 14:38:03 +00:00
|
|
|
Actions []NotificationAction
|
2023-09-01 17:06:04 +00:00
|
|
|
IsMarkdown bool
|
|
|
|
|
|
|
|
topic string
|
|
|
|
auth Auth
|
|
|
|
}
|
|
|
|
|
2023-09-03 10:19:34 +00:00
|
|
|
func (n Notification) IsEmpty() bool {
|
|
|
|
return n.Title == "" && n.Body == ""
|
|
|
|
}
|
|
|
|
|
2023-09-01 17:06:04 +00:00
|
|
|
func (n Notification) Send(base string) error {
|
|
|
|
req, err := http.NewRequest("POST", base+"/"+n.topic, strings.NewReader(n.Body))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if n.IsMarkdown {
|
|
|
|
req.Header.Set("Content-Type", "text/markdown")
|
|
|
|
} else {
|
|
|
|
req.Header.Set("Content-Type", "text/plain")
|
|
|
|
}
|
|
|
|
|
|
|
|
if n.Title != "" {
|
|
|
|
req.Header.Set("Title", n.Title)
|
|
|
|
}
|
|
|
|
|
|
|
|
if n.Priority > 0 {
|
|
|
|
req.Header.Set("Priority", strconv.Itoa(n.Priority))
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(n.Tags) > 0 {
|
|
|
|
req.Header.Set("Tags", strings.Join(n.Tags, ","))
|
|
|
|
}
|
|
|
|
|
|
|
|
if n.auth.Username != "" {
|
|
|
|
req.Header.Set("Authorization", n.auth.basic())
|
|
|
|
}
|
|
|
|
|
|
|
|
if n.auth.AccessToken != "" {
|
|
|
|
req.Header.Set("Authorization", n.auth.bearer())
|
|
|
|
}
|
|
|
|
|
2023-09-03 14:38:03 +00:00
|
|
|
if len(n.Actions) > 0 {
|
|
|
|
actions := make([]string, len(n.Actions))
|
|
|
|
for i, act := range n.Actions {
|
|
|
|
actions[i] = act.Format()
|
|
|
|
}
|
|
|
|
|
|
|
|
req.Header.Set("Actions", strings.Join(actions, "; "))
|
|
|
|
}
|
|
|
|
|
2023-09-01 17:06:04 +00:00
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode >= 400 {
|
|
|
|
var e NotificationError
|
|
|
|
dec := json.NewDecoder(resp.Body)
|
|
|
|
dec.Decode(&e)
|
|
|
|
|
|
|
|
return errors.New(e.Error)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type Auth struct {
|
|
|
|
Username string
|
|
|
|
Password string
|
|
|
|
|
|
|
|
AccessToken string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a Auth) IsEmpty() bool {
|
|
|
|
return a.Username == "" && a.AccessToken == ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a Auth) basic() string {
|
|
|
|
return "Basic " + base64.StdEncoding.EncodeToString([]byte(a.Username+":"+a.Password))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a Auth) bearer() string {
|
|
|
|
return "Bearer " + a.AccessToken
|
|
|
|
}
|
|
|
|
|
|
|
|
type Bridge struct {
|
|
|
|
baseURL string
|
|
|
|
topic string
|
|
|
|
h Handler
|
|
|
|
auth Auth
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBridge(baseURL, topic string, handler Handler) Bridge {
|
|
|
|
return Bridge{
|
|
|
|
baseURL: baseURL,
|
|
|
|
topic: topic,
|
|
|
|
h: handler,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bridge) WithAuth(auth Auth) {
|
|
|
|
b.auth = auth
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b Bridge) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.Method != "POST" {
|
|
|
|
w.WriteHeader(http.StatusNotFound)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-09-03 14:37:02 +00:00
|
|
|
nots, err := b.h.ProduceNotifications(r)
|
2023-09-03 12:57:43 +00:00
|
|
|
if err != nil {
|
|
|
|
slog.Error("failed to format notification")
|
|
|
|
w.WriteHeader(http.StatusBadRequest)
|
2023-09-03 10:19:34 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-09-11 16:47:25 +00:00
|
|
|
if len(nots) == 0 {
|
|
|
|
slog.Debug("no notification produced")
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-09-03 14:37:02 +00:00
|
|
|
for _, not := range nots {
|
|
|
|
not.topic = b.topic
|
|
|
|
not.auth = b.auth
|
|
|
|
if err = not.Send(b.baseURL); err != nil {
|
|
|
|
slog.Error("unable to send notification", "error", err)
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
2023-09-01 17:06:04 +00:00
|
|
|
}
|
|
|
|
|
2023-09-03 14:37:02 +00:00
|
|
|
slog.Debug("notifications sent with success", "sent", len(nots))
|
2023-09-01 17:06:04 +00:00
|
|
|
|
2023-09-03 14:36:26 +00:00
|
|
|
w.WriteHeader(http.StatusNoContent)
|
2023-09-01 17:06:04 +00:00
|
|
|
}
|