2022-04-26 20:50:09 +03:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
2023-06-13 13:41:13 +03:00
|
|
|
"io/fs"
|
2022-04-26 20:50:09 +03:00
|
|
|
"os"
|
2022-10-10 14:05:24 +03:00
|
|
|
"time"
|
2022-04-26 20:50:09 +03:00
|
|
|
|
|
|
|
"github.com/AdguardTeam/AdGuardHome/internal/aghos"
|
2022-10-04 16:02:55 +03:00
|
|
|
"github.com/AdguardTeam/AdGuardHome/internal/next/agh"
|
2022-04-26 20:50:09 +03:00
|
|
|
"github.com/AdguardTeam/golibs/log"
|
|
|
|
)
|
|
|
|
|
|
|
|
// signalHandler processes incoming signals and shuts services down.
|
|
|
|
type signalHandler struct {
|
2022-10-10 14:05:24 +03:00
|
|
|
// signal is the channel to which OS signals are sent.
|
2022-04-26 20:50:09 +03:00
|
|
|
signal chan os.Signal
|
|
|
|
|
2022-10-10 14:05:24 +03:00
|
|
|
// confFile is the path to the configuration file.
|
|
|
|
confFile string
|
|
|
|
|
2023-06-13 13:41:13 +03:00
|
|
|
// frontend is the filesystem with the frontend and other statically
|
|
|
|
// compiled files.
|
|
|
|
frontend fs.FS
|
|
|
|
|
2022-10-10 14:05:24 +03:00
|
|
|
// start is the time at which AdGuard Home has been started.
|
|
|
|
start time.Time
|
|
|
|
|
|
|
|
// services are the services that are shut down before application exiting.
|
2022-04-26 20:50:09 +03:00
|
|
|
services []agh.Service
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle processes OS signals.
|
|
|
|
func (h *signalHandler) handle() {
|
2022-04-29 14:39:02 +03:00
|
|
|
defer log.OnPanic("signalHandler.handle")
|
2022-04-26 20:50:09 +03:00
|
|
|
|
|
|
|
for sig := range h.signal {
|
2022-04-29 14:39:02 +03:00
|
|
|
log.Info("sighdlr: received signal %q", sig)
|
2022-04-26 20:50:09 +03:00
|
|
|
|
2022-10-10 14:05:24 +03:00
|
|
|
if aghos.IsReconfigureSignal(sig) {
|
|
|
|
h.reconfigure()
|
|
|
|
} else if aghos.IsShutdownSignal(sig) {
|
|
|
|
status := h.shutdown()
|
|
|
|
log.Info("sighdlr: exiting with status %d", status)
|
|
|
|
|
|
|
|
os.Exit(status)
|
2022-04-26 20:50:09 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-10 14:05:24 +03:00
|
|
|
// reconfigure rereads the configuration file and updates and restarts services.
|
|
|
|
func (h *signalHandler) reconfigure() {
|
|
|
|
log.Info("sighdlr: reconfiguring adguard home")
|
|
|
|
|
|
|
|
status := h.shutdown()
|
|
|
|
if status != statusSuccess {
|
2023-06-19 12:21:32 +03:00
|
|
|
log.Info("sighdlr: reconfiguring: exiting with status %d", status)
|
2022-10-10 14:05:24 +03:00
|
|
|
|
|
|
|
os.Exit(status)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(a.garipov): This is a very rough way to do it. Some services can be
|
|
|
|
// reconfigured without the full shutdown, and the error handling is
|
|
|
|
// currently not the best.
|
|
|
|
|
2023-06-29 14:34:06 +03:00
|
|
|
confMgr, err := newConfigMgr(h.confFile, h.frontend, h.start)
|
2023-06-13 13:41:13 +03:00
|
|
|
check(err)
|
2022-10-10 14:05:24 +03:00
|
|
|
|
|
|
|
web := confMgr.Web()
|
|
|
|
err = web.Start()
|
2023-06-13 13:41:13 +03:00
|
|
|
check(err)
|
2022-10-10 14:05:24 +03:00
|
|
|
|
|
|
|
dns := confMgr.DNS()
|
|
|
|
err = dns.Start()
|
2023-06-13 13:41:13 +03:00
|
|
|
check(err)
|
2022-10-10 14:05:24 +03:00
|
|
|
|
|
|
|
h.services = []agh.Service{
|
|
|
|
dns,
|
|
|
|
web,
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Info("sighdlr: successfully reconfigured adguard home")
|
|
|
|
}
|
|
|
|
|
2022-04-26 20:50:09 +03:00
|
|
|
// Exit status constants.
|
|
|
|
const (
|
|
|
|
statusSuccess = 0
|
|
|
|
statusError = 1
|
|
|
|
)
|
|
|
|
|
|
|
|
// shutdown gracefully shuts down all services.
|
2022-10-10 14:05:24 +03:00
|
|
|
func (h *signalHandler) shutdown() (status int) {
|
2022-04-26 20:50:09 +03:00
|
|
|
ctx, cancel := ctxWithDefaultTimeout()
|
|
|
|
defer cancel()
|
|
|
|
|
2022-10-10 14:05:24 +03:00
|
|
|
status = statusSuccess
|
2022-04-26 20:50:09 +03:00
|
|
|
|
2022-04-29 14:39:02 +03:00
|
|
|
log.Info("sighdlr: shutting down services")
|
2022-04-26 20:50:09 +03:00
|
|
|
for i, service := range h.services {
|
|
|
|
err := service.Shutdown(ctx)
|
|
|
|
if err != nil {
|
2022-04-29 14:39:02 +03:00
|
|
|
log.Error("sighdlr: shutting down service at index %d: %s", i, err)
|
2022-04-26 20:50:09 +03:00
|
|
|
status = statusError
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-10 14:05:24 +03:00
|
|
|
return status
|
2022-04-26 20:50:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// newSignalHandler returns a new signalHandler that shuts down svcs.
|
2023-06-13 13:41:13 +03:00
|
|
|
func newSignalHandler(
|
|
|
|
confFile string,
|
|
|
|
frontend fs.FS,
|
|
|
|
start time.Time,
|
|
|
|
svcs ...agh.Service,
|
|
|
|
) (h *signalHandler) {
|
2022-04-26 20:50:09 +03:00
|
|
|
h = &signalHandler{
|
|
|
|
signal: make(chan os.Signal, 1),
|
2022-10-10 14:05:24 +03:00
|
|
|
confFile: confFile,
|
2023-06-13 13:41:13 +03:00
|
|
|
frontend: frontend,
|
2022-10-10 14:05:24 +03:00
|
|
|
start: start,
|
2022-04-26 20:50:09 +03:00
|
|
|
services: svcs,
|
|
|
|
}
|
|
|
|
|
|
|
|
aghos.NotifyShutdownSignal(h.signal)
|
2022-10-10 14:05:24 +03:00
|
|
|
aghos.NotifyReconfigureSignal(h.signal)
|
2022-04-26 20:50:09 +03:00
|
|
|
|
|
|
|
return h
|
|
|
|
}
|