mirror of
https://github.com/containrrr/watchtower.git
synced 2024-12-12 09:04:17 +02:00
Make sure all shoutrrr notifications are sent (#564)
This commit is contained in:
parent
b8408269bb
commit
fdecd40189
@ -138,6 +138,7 @@ func Run(c *cobra.Command, names []string) {
|
|||||||
log.Info("Running a one time update.")
|
log.Info("Running a one time update.")
|
||||||
}
|
}
|
||||||
runUpdatesWithNotifications(filter)
|
runUpdatesWithNotifications(filter)
|
||||||
|
notifier.Close()
|
||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -153,3 +153,5 @@ func (e *emailTypeNotifier) Fire(entry *log.Entry) error {
|
|||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e *emailTypeNotifier) Close() {}
|
@ -59,6 +59,8 @@ func (n *gotifyTypeNotifier) StartNotification() {}
|
|||||||
|
|
||||||
func (n *gotifyTypeNotifier) SendNotification() {}
|
func (n *gotifyTypeNotifier) SendNotification() {}
|
||||||
|
|
||||||
|
func (n *gotifyTypeNotifier) Close() {}
|
||||||
|
|
||||||
func (n *gotifyTypeNotifier) Levels() []log.Level {
|
func (n *gotifyTypeNotifier) Levels() []log.Level {
|
||||||
return n.logLevels
|
return n.logLevels
|
||||||
}
|
}
|
||||||
|
@ -47,6 +47,8 @@ func (n *msTeamsTypeNotifier) StartNotification() {}
|
|||||||
|
|
||||||
func (n *msTeamsTypeNotifier) SendNotification() {}
|
func (n *msTeamsTypeNotifier) SendNotification() {}
|
||||||
|
|
||||||
|
func (n *msTeamsTypeNotifier) Close() {}
|
||||||
|
|
||||||
func (n *msTeamsTypeNotifier) Levels() []log.Level {
|
func (n *msTeamsTypeNotifier) Levels() []log.Level {
|
||||||
return n.levels
|
return n.levels
|
||||||
}
|
}
|
||||||
|
@ -66,3 +66,10 @@ func (n *Notifier) SendNotification() {
|
|||||||
t.SendNotification()
|
t.SendNotification()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Close closes all notifiers.
|
||||||
|
func (n *Notifier) Close() {
|
||||||
|
for _, t := range n.types {
|
||||||
|
t.Close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -3,11 +3,11 @@ package notifications
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/containrrr/shoutrrr/pkg/types"
|
||||||
"text/template"
|
"text/template"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/containrrr/shoutrrr"
|
"github.com/containrrr/shoutrrr"
|
||||||
"github.com/containrrr/shoutrrr/pkg/router"
|
|
||||||
t "github.com/containrrr/watchtower/pkg/types"
|
t "github.com/containrrr/watchtower/pkg/types"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
@ -18,13 +18,19 @@ const (
|
|||||||
shoutrrrType = "shoutrrr"
|
shoutrrrType = "shoutrrr"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type router interface {
|
||||||
|
Send(message string, params *types.Params) []error
|
||||||
|
}
|
||||||
|
|
||||||
// Implements Notifier, logrus.Hook
|
// Implements Notifier, logrus.Hook
|
||||||
type shoutrrrTypeNotifier struct {
|
type shoutrrrTypeNotifier struct {
|
||||||
Urls []string
|
Urls []string
|
||||||
Router *router.ServiceRouter
|
Router router
|
||||||
entries []*log.Entry
|
entries []*log.Entry
|
||||||
logLevels []log.Level
|
logLevels []log.Level
|
||||||
template *template.Template
|
template *template.Template
|
||||||
|
messages chan string
|
||||||
|
done chan bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func newShoutrrrNotifier(c *cobra.Command, acceptedLogLevels []log.Level) t.Notifier {
|
func newShoutrrrNotifier(c *cobra.Command, acceptedLogLevels []log.Level) t.Notifier {
|
||||||
@ -41,13 +47,33 @@ func newShoutrrrNotifier(c *cobra.Command, acceptedLogLevels []log.Level) t.Noti
|
|||||||
Router: r,
|
Router: r,
|
||||||
logLevels: acceptedLogLevels,
|
logLevels: acceptedLogLevels,
|
||||||
template: getShoutrrrTemplate(c),
|
template: getShoutrrrTemplate(c),
|
||||||
|
messages: make(chan string, 1),
|
||||||
|
done: make(chan bool),
|
||||||
}
|
}
|
||||||
|
|
||||||
log.AddHook(n)
|
log.AddHook(n)
|
||||||
|
|
||||||
|
// Do the sending in a separate goroutine so we don't block the main process.
|
||||||
|
go sendNotifications(n)
|
||||||
|
|
||||||
return n
|
return n
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func sendNotifications(n *shoutrrrTypeNotifier) {
|
||||||
|
for msg := range n.messages {
|
||||||
|
errs := n.Router.Send(msg, nil)
|
||||||
|
|
||||||
|
for i, err := range errs {
|
||||||
|
if err != nil {
|
||||||
|
// Use fmt so it doesn't trigger another notification.
|
||||||
|
fmt.Println("Failed to send notification via shoutrrr (url="+n.Urls[i]+"): ", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
n.done <- true
|
||||||
|
}
|
||||||
|
|
||||||
func (e *shoutrrrTypeNotifier) buildMessage(entries []*log.Entry) string {
|
func (e *shoutrrrTypeNotifier) buildMessage(entries []*log.Entry) string {
|
||||||
var body bytes.Buffer
|
var body bytes.Buffer
|
||||||
if err := e.template.Execute(&body, entries); err != nil {
|
if err := e.template.Execute(&body, entries); err != nil {
|
||||||
@ -58,20 +84,8 @@ func (e *shoutrrrTypeNotifier) buildMessage(entries []*log.Entry) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (e *shoutrrrTypeNotifier) sendEntries(entries []*log.Entry) {
|
func (e *shoutrrrTypeNotifier) sendEntries(entries []*log.Entry) {
|
||||||
|
|
||||||
msg := e.buildMessage(entries)
|
msg := e.buildMessage(entries)
|
||||||
|
e.messages <- msg
|
||||||
// Do the sending in a separate goroutine so we don't block the main process.
|
|
||||||
go func() {
|
|
||||||
errs := e.Router.Send(msg, nil)
|
|
||||||
|
|
||||||
for i, err := range errs {
|
|
||||||
if err != nil {
|
|
||||||
// Use fmt so it doesn't trigger another notification.
|
|
||||||
fmt.Println("Failed to send notification via shoutrrr (url="+e.Urls[i]+"): ", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *shoutrrrTypeNotifier) StartNotification() {
|
func (e *shoutrrrTypeNotifier) StartNotification() {
|
||||||
@ -89,6 +103,15 @@ func (e *shoutrrrTypeNotifier) SendNotification() {
|
|||||||
e.entries = nil
|
e.entries = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e *shoutrrrTypeNotifier) Close() {
|
||||||
|
close(e.messages)
|
||||||
|
|
||||||
|
// Use fmt so it doesn't trigger another notification.
|
||||||
|
fmt.Println("Waiting for the notification goroutine to finish")
|
||||||
|
|
||||||
|
_ = <-e.done
|
||||||
|
}
|
||||||
|
|
||||||
func (e *shoutrrrTypeNotifier) Levels() []log.Level {
|
func (e *shoutrrrTypeNotifier) Levels() []log.Level {
|
||||||
return e.logLevels
|
return e.logLevels
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package notifications
|
package notifications
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/containrrr/shoutrrr/pkg/types"
|
||||||
"testing"
|
"testing"
|
||||||
"text/template"
|
"text/template"
|
||||||
|
|
||||||
@ -102,3 +103,69 @@ func TestShoutrrrInvalidTemplateUsesTemplate(t *testing.T) {
|
|||||||
|
|
||||||
require.Equal(t, sd, s)
|
require.Equal(t, sd, s)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type blockingRouter struct {
|
||||||
|
unlock chan bool
|
||||||
|
sent chan bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b blockingRouter) Send(message string, params *types.Params) []error {
|
||||||
|
_ = <-b.unlock
|
||||||
|
b.sent <- true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSlowNotificationNotSent(t *testing.T) {
|
||||||
|
_, blockingRouter := sendNotificationsWithBlockingRouter()
|
||||||
|
|
||||||
|
notifSent := false
|
||||||
|
select {
|
||||||
|
case notifSent = <-blockingRouter.sent:
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
require.Equal(t, false, notifSent)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSlowNotificationSent(t *testing.T) {
|
||||||
|
shoutrrr, blockingRouter := sendNotificationsWithBlockingRouter()
|
||||||
|
|
||||||
|
blockingRouter.unlock <- true
|
||||||
|
shoutrrr.Close()
|
||||||
|
|
||||||
|
notifSent := false
|
||||||
|
select {
|
||||||
|
case notifSent = <-blockingRouter.sent:
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
require.Equal(t, true, notifSent)
|
||||||
|
}
|
||||||
|
|
||||||
|
func sendNotificationsWithBlockingRouter() (*shoutrrrTypeNotifier, *blockingRouter) {
|
||||||
|
cmd := new(cobra.Command)
|
||||||
|
|
||||||
|
router := &blockingRouter{
|
||||||
|
unlock: make(chan bool, 1),
|
||||||
|
sent: make(chan bool, 1),
|
||||||
|
}
|
||||||
|
|
||||||
|
shoutrrr := &shoutrrrTypeNotifier{
|
||||||
|
template: getShoutrrrTemplate(cmd),
|
||||||
|
messages: make(chan string, 1),
|
||||||
|
done: make(chan bool),
|
||||||
|
Router: router,
|
||||||
|
}
|
||||||
|
|
||||||
|
entry := &log.Entry{
|
||||||
|
Message: "foo bar",
|
||||||
|
}
|
||||||
|
|
||||||
|
go sendNotifications(shoutrrr)
|
||||||
|
|
||||||
|
shoutrrr.StartNotification()
|
||||||
|
shoutrrr.Fire(entry)
|
||||||
|
|
||||||
|
shoutrrr.SendNotification()
|
||||||
|
|
||||||
|
return shoutrrr, router
|
||||||
|
}
|
||||||
|
@ -42,3 +42,5 @@ func newSlackNotifier(c *cobra.Command, acceptedLogLevels []log.Level) t.Notifie
|
|||||||
func (s *slackTypeNotifier) StartNotification() {}
|
func (s *slackTypeNotifier) StartNotification() {}
|
||||||
|
|
||||||
func (s *slackTypeNotifier) SendNotification() {}
|
func (s *slackTypeNotifier) SendNotification() {}
|
||||||
|
|
||||||
|
func (s *slackTypeNotifier) Close() {}
|
||||||
|
@ -4,4 +4,5 @@ package types
|
|||||||
type Notifier interface {
|
type Notifier interface {
|
||||||
StartNotification()
|
StartNotification()
|
||||||
SendNotification()
|
SendNotification()
|
||||||
|
Close()
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user