199 lines
3.7 KiB
Go
199 lines
3.7 KiB
Go
|
package main
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"flag"
|
||
|
daemon "github.com/sevlyar/go-daemon/oldapi"
|
||
|
"log"
|
||
|
"os"
|
||
|
"syscall"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
pidFileName = "dmn.pid"
|
||
|
logFileName = "dmn.log"
|
||
|
|
||
|
fileMask = 0600
|
||
|
)
|
||
|
const (
|
||
|
ret_OK = iota
|
||
|
ret_ALREADYRUN
|
||
|
ret_PIDFERROR
|
||
|
ret_REBORNERROR
|
||
|
ret_CONFERROR
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
status = flag.Bool("status", false,
|
||
|
`Check status of the daemon. The program immediately exits after these
|
||
|
checks with either a return code of 0 (Daemon Stopped) or return code
|
||
|
not equal to 0 (Daemon Running)`)
|
||
|
|
||
|
silent = flag.Bool("silent", false, "Don't write in stdout")
|
||
|
|
||
|
test = flag.Bool("t", false,
|
||
|
`Run syntax tests for configuration files only. The program
|
||
|
immediately exits after these syntax parsing tests with either
|
||
|
a return code of 0 (Syntax OK) or return code not equal to 0
|
||
|
(Syntax Error)`)
|
||
|
|
||
|
configFileName = flag.String("f", "dmn.conf",
|
||
|
`Specifies the name of the configuration file. The default is dmn.conf.
|
||
|
Daemon refuses to start if there is no configuration file.`)
|
||
|
)
|
||
|
|
||
|
var confProv = make(chan Config, 8)
|
||
|
|
||
|
func main() {
|
||
|
flag.Parse()
|
||
|
|
||
|
setupLogging()
|
||
|
|
||
|
conf, err := loadConfig(*configFileName)
|
||
|
if err != nil {
|
||
|
log.Println("Config error:", err)
|
||
|
os.Exit(ret_CONFERROR)
|
||
|
}
|
||
|
if *test {
|
||
|
os.Exit(ret_OK)
|
||
|
}
|
||
|
|
||
|
pidf := lockPidFile()
|
||
|
err = daemon.Reborn(027, "./")
|
||
|
if err != nil {
|
||
|
log.Println("Reborn error:", err)
|
||
|
os.Exit(ret_REBORNERROR)
|
||
|
}
|
||
|
|
||
|
confProv <- conf
|
||
|
go watchdog(confProv)
|
||
|
|
||
|
serveSignals()
|
||
|
|
||
|
pidf.Unlock()
|
||
|
}
|
||
|
|
||
|
func setupLogging() {
|
||
|
if daemon.WasReborn() {
|
||
|
file, _ := os.OpenFile(logFileName, os.O_WRONLY|os.O_CREATE|os.O_APPEND, fileMask)
|
||
|
daemon.RedirectStream(os.Stdout, file)
|
||
|
daemon.RedirectStream(os.Stderr, file)
|
||
|
file.Close()
|
||
|
log.Println("--- log ---")
|
||
|
} else {
|
||
|
log.SetFlags(0)
|
||
|
if *silent {
|
||
|
file, _ := os.OpenFile(os.DevNull, os.O_WRONLY, fileMask)
|
||
|
daemon.RedirectStream(os.Stdout, file)
|
||
|
daemon.RedirectStream(os.Stderr, file)
|
||
|
file.Close()
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type Config []string
|
||
|
|
||
|
func loadConfig(path string) (config Config, err error) {
|
||
|
var file *os.File
|
||
|
file, err = os.OpenFile(path, os.O_RDONLY, 0700)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
defer file.Close()
|
||
|
|
||
|
config = make([]string, 0)
|
||
|
err = json.NewDecoder(file).Decode(&config)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
for _, path = range config {
|
||
|
if _, err = os.Stat(path); os.IsNotExist(err) {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func lockPidFile() *daemon.PidFile {
|
||
|
pidf, err := daemon.LockPidFile(pidFileName, fileMask)
|
||
|
if err != nil {
|
||
|
if err == daemon.ErrWouldBlock {
|
||
|
log.Println("daemon copy is already running")
|
||
|
os.Exit(ret_ALREADYRUN)
|
||
|
} else {
|
||
|
log.Println("pid file creation error:", err)
|
||
|
os.Exit(ret_PIDFERROR)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if !daemon.WasReborn() {
|
||
|
pidf.Unlock()
|
||
|
}
|
||
|
|
||
|
if *status {
|
||
|
os.Exit(ret_OK)
|
||
|
}
|
||
|
|
||
|
return pidf
|
||
|
}
|
||
|
|
||
|
func watchdog(confProv <-chan Config) {
|
||
|
states := make(map[string]time.Time)
|
||
|
conf := <-confProv
|
||
|
for {
|
||
|
select {
|
||
|
case conf = <-confProv:
|
||
|
default:
|
||
|
}
|
||
|
|
||
|
for _, path := range conf {
|
||
|
fi, err := os.Stat(path)
|
||
|
if err != nil {
|
||
|
log.Println(err)
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
cur := fi.ModTime()
|
||
|
if pre, exists := states[path]; exists {
|
||
|
if pre != cur {
|
||
|
log.Printf("file %s modified at %s", path, cur)
|
||
|
}
|
||
|
}
|
||
|
states[path] = cur
|
||
|
}
|
||
|
time.Sleep(time.Second)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func serveSignals() {
|
||
|
daemon.SetHandler(termHandler, syscall.SIGTERM, syscall.SIGKILL)
|
||
|
daemon.SetHandler(hupHandler, syscall.SIGHUP)
|
||
|
|
||
|
err := daemon.ServeSignals()
|
||
|
if err != nil {
|
||
|
log.Println("Error:", err)
|
||
|
}
|
||
|
|
||
|
log.Println("--- end ---")
|
||
|
}
|
||
|
|
||
|
func termHandler(sig os.Signal) error {
|
||
|
log.Println("SIGTERM:", sig)
|
||
|
return daemon.ErrStop
|
||
|
}
|
||
|
|
||
|
func hupHandler(sig os.Signal) error {
|
||
|
log.Println("SIGHUP:", sig)
|
||
|
|
||
|
conf, err := loadConfig(*configFileName)
|
||
|
if err != nil {
|
||
|
log.Println("Config error:", err)
|
||
|
} else {
|
||
|
confProv <- conf
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|