1
0
mirror of https://github.com/containrrr/watchtower.git synced 2024-12-12 09:04:17 +02:00
watchtower/cmd/root.go

208 lines
5.0 KiB
Go
Raw Normal View History

2019-06-22 22:04:36 +02:00
package cmd
import (
"os"
"os/signal"
"strconv"
"syscall"
"time"
2019-07-21 22:22:30 +02:00
"github.com/containrrr/watchtower/internal/actions"
2019-06-22 22:04:36 +02:00
"github.com/containrrr/watchtower/internal/flags"
"github.com/containrrr/watchtower/pkg/api"
2019-07-21 20:15:04 +02:00
"github.com/containrrr/watchtower/pkg/container"
2020-03-13 11:38:33 +02:00
"github.com/containrrr/watchtower/pkg/filters"
2019-07-21 22:22:30 +02:00
"github.com/containrrr/watchtower/pkg/notifications"
t "github.com/containrrr/watchtower/pkg/types"
2019-06-22 22:04:36 +02:00
"github.com/robfig/cron"
log "github.com/sirupsen/logrus"
"github.com/spf13/cobra"
)
var (
client container.Client
scheduleSpec string
cleanup bool
noRestart bool
monitorOnly bool
enableLabel bool
notifier *notifications.Notifier
timeout time.Duration
lifecycleHooks bool
2019-06-22 22:04:36 +02:00
)
var rootCmd = &cobra.Command{
Use: "watchtower",
Short: "Automatically updates running Docker containers",
Long: `
2019-06-22 22:04:36 +02:00
Watchtower automatically updates running Docker containers whenever a new image is released.
More information available at https://github.com/containrrr/watchtower/.
`,
Run: Run,
PreRun: PreRun,
}
func init() {
flags.SetDefaults()
flags.RegisterDockerFlags(rootCmd)
flags.RegisterSystemFlags(rootCmd)
flags.RegisterNotificationFlags(rootCmd)
}
2019-06-23 00:32:50 +02:00
// Execute the root func and exit in case of errors
2019-06-22 22:04:36 +02:00
func Execute() {
if err := rootCmd.Execute(); err != nil {
2019-06-23 00:32:50 +02:00
log.Fatal(err)
2019-06-22 22:04:36 +02:00
}
}
2019-06-23 00:32:50 +02:00
// PreRun is a lifecycle hook that runs before the command is executed.
2019-06-22 22:04:36 +02:00
func PreRun(cmd *cobra.Command, args []string) {
f := cmd.PersistentFlags()
2019-07-22 12:10:57 +02:00
if enabled, _ := f.GetBool("debug"); enabled {
2019-06-22 22:04:36 +02:00
log.SetLevel(log.DebugLevel)
}
pollingSet := f.Changed("interval")
schedule, _ := f.GetString("schedule")
cronLen := len(schedule)
if pollingSet && cronLen > 0 {
2019-06-22 22:04:36 +02:00
log.Fatal("Only schedule or interval can be defined, not both.")
} else if cronLen > 0 {
2019-06-22 22:04:36 +02:00
scheduleSpec, _ = f.GetString("schedule")
} else {
interval, _ := f.GetInt("interval")
scheduleSpec = "@every " + strconv.Itoa(interval) + "s"
}
cleanup, noRestart, monitorOnly, timeout = flags.ReadFlags(cmd)
if timeout < 0 {
log.Fatal("Please specify a positive value for timeout value.")
}
2019-06-22 22:04:36 +02:00
enableLabel, _ = f.GetBool("label-enable")
lifecycleHooks, _ = f.GetBool("enable-lifecycle-hooks")
2019-06-22 22:04:36 +02:00
// configure environment vars for client
err := flags.EnvConfig(cmd)
2019-06-22 22:04:36 +02:00
if err != nil {
log.Fatal(err)
}
noPull, _ := f.GetBool("no-pull")
includeStopped, _ := f.GetBool("include-stopped")
reviveStopped, _ := f.GetBool("revive-stopped")
removeVolumes, _ := f.GetBool("remove-volumes")
2019-06-22 22:04:36 +02:00
client = container.NewClient(
!noPull,
includeStopped,
reviveStopped,
removeVolumes,
2019-06-22 22:04:36 +02:00
)
notifier = notifications.NewNotifier(cmd)
}
2019-06-23 00:32:50 +02:00
// Run is the main execution flow of the command
2019-06-22 22:04:36 +02:00
func Run(c *cobra.Command, names []string) {
filter := filters.BuildFilter(names, enableLabel)
2019-06-22 22:04:36 +02:00
runOnce, _ := c.PersistentFlags().GetBool("run-once")
httpAPI, _ := c.PersistentFlags().GetBool("http-api")
if httpAPI {
apiToken, _ := c.PersistentFlags().GetString("http-api-token")
if err := api.SetupHTTPUpdates(apiToken, func() { runUpdatesWithNotifications(filter) }); err != nil {
log.Fatal(err)
os.Exit(1)
}
api.WaitForHTTPUpdates()
}
2019-06-22 22:04:36 +02:00
if runOnce {
2020-03-26 17:58:57 +02:00
if noStartupMessage, _ := c.PersistentFlags().GetBool("no-startup-message"); !noStartupMessage {
log.Info("Running a one time update.")
}
2019-06-22 22:04:36 +02:00
runUpdatesWithNotifications(filter)
os.Exit(0)
2019-06-22 22:04:36 +02:00
return
}
if err := actions.CheckForMultipleWatchtowerInstances(client, cleanup); err != nil {
log.Fatal(err)
}
2020-03-13 11:38:33 +02:00
if err := runUpgradesOnSchedule(c, filter); err != nil {
2019-07-22 12:10:57 +02:00
log.Error(err)
}
2019-06-22 22:04:36 +02:00
os.Exit(1)
}
2020-03-13 11:38:33 +02:00
func runUpgradesOnSchedule(c *cobra.Command, filter t.Filter) error {
2019-06-22 22:04:36 +02:00
tryLockSem := make(chan bool, 1)
tryLockSem <- true
cron := cron.New()
err := cron.AddFunc(
scheduleSpec,
func() {
select {
case v := <-tryLockSem:
defer func() { tryLockSem <- v }()
runUpdatesWithNotifications(filter)
default:
log.Debug("Skipped another update already running.")
}
nextRuns := cron.Entries()
if len(nextRuns) > 0 {
log.Debug("Scheduled next run: " + nextRuns[0].Next.String())
}
})
if err != nil {
return err
}
2020-03-13 11:38:33 +02:00
if noStartupMessage, _ := c.PersistentFlags().GetBool("no-startup-message"); !noStartupMessage {
log.Info("Starting Watchtower and scheduling first run: " + cron.Entries()[0].Schedule.Next(time.Now()).String())
}
2019-06-22 22:04:36 +02:00
cron.Start()
// Graceful shut-down on SIGINT/SIGTERM
interrupt := make(chan os.Signal, 1)
signal.Notify(interrupt, os.Interrupt)
signal.Notify(interrupt, syscall.SIGTERM)
<-interrupt
cron.Stop()
log.Info("Waiting for running update to be finished...")
<-tryLockSem
return nil
}
func runUpdatesWithNotifications(filter t.Filter) {
2019-06-22 22:04:36 +02:00
notifier.StartNotification()
updateParams := t.UpdateParams{
Filter: filter,
Cleanup: cleanup,
NoRestart: noRestart,
Timeout: timeout,
MonitorOnly: monitorOnly,
LifecycleHooks: lifecycleHooks,
2019-06-22 22:04:36 +02:00
}
err := actions.Update(client, updateParams)
if err != nil {
log.Println(err)
}
notifier.SendNotification()
}