forked from cottand/leng
-
Notifications
You must be signed in to change notification settings - Fork 0
/
main.go
145 lines (123 loc) · 3.51 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
package main
import (
"context"
"flag"
"github.com/cottand/leng/internal/metric"
"golang.org/x/sys/unix"
"net/http"
"os"
"os/signal"
"runtime"
"time"
)
var (
configPath string
forceUpdate bool
lengActive bool
)
func reloadBlockCache(config *Config,
blockCache *MemoryBlockCache,
questionCache *MemoryQuestionCache,
apiServer *http.Server,
server *Server,
reloadChan chan bool) (*MemoryBlockCache, *http.Server, error) {
logger.Debug("Reloading the blockcache")
blockCache = PerformUpdate(config, true)
server.Stop()
if apiServer != nil {
if err := apiServer.Shutdown(context.Background()); err != nil {
logger.Debugf("error shutting down api server: %v", err)
}
}
server.Run(config, blockCache, questionCache)
apiServer, err := StartAPIServer(config, reloadChan, blockCache, questionCache)
if err != nil {
logger.Fatal(err)
return nil, nil, err
}
return blockCache, apiServer, nil
}
func main() {
flag.Parse()
config, err := LoadConfig(configPath)
if err != nil {
logger.Fatal(err)
}
loggingState, err := loggerInit(config.LogConfig)
if err != nil {
logger.Fatal(err)
}
defer func() {
loggingState.cleanUp()
}()
cancelMetrics := metric.Start(config.Metrics.ResetPeriodMinutes, config.Metrics.HighCardinalityEnabled)
lengActive = true
quitActivation := make(chan bool)
actChannel := make(chan *ActivationHandler)
go startActivation(actChannel, quitActivation)
_ = <-actChannel
close(actChannel)
server := &Server{
host: config.Bind,
rTimeout: 5 * time.Second,
wTimeout: 5 * time.Second,
}
// BlockCache contains all blocked domains
blockCache := &MemoryBlockCache{Backend: make(map[string]bool)}
// QuestionCache contains all queries to the dns server
questionCache := makeQuestionCache(config.QuestionCacheCap)
reloadChan := make(chan bool)
// The server will start with an empty blockcache soe we can dowload the lists if leng is the
// system's dns server.
server.Run(config, blockCache, questionCache)
var apiServer *http.Server
// Load the block cache, restart the server with the new context
blockCache, apiServer, err = reloadBlockCache(config, blockCache, questionCache, apiServer, server, reloadChan)
if err != nil {
logger.Fatalf("Cannot start the API server %s", err)
}
sig := make(chan os.Signal)
signal.Notify(sig, os.Interrupt, unix.SIGHUP, unix.SIGUSR1)
forever:
for {
select {
case s := <-sig:
switch s {
case os.Interrupt:
logger.Error("SIGINT received, stopping\n")
quitActivation <- true
cancelMetrics()
break forever
case unix.SIGHUP:
logger.Error("SIGHUP received: rotating logs\n")
err := loggingState.reopen()
if err != nil {
logger.Error(err)
}
case unix.SIGUSR1:
logger.Info("SIGUSR1 received: reloading config\n")
reloadConfigFromFile(server)
}
case <-reloadChan:
blockCache, apiServer, err = reloadBlockCache(config, blockCache, questionCache, apiServer, server, reloadChan)
if err != nil {
logger.Fatalf("Cannot start the API server %s", err)
}
}
}
// make sure we give the activation goroutine time to exit
<-quitActivation
logger.Debugf("Main goroutine exiting")
}
func init() {
flag.StringVar(&configPath, "config", "leng.toml", "location of the config file")
flag.BoolVar(&forceUpdate, "update", false, "force an update of the blocklist database")
runtime.GOMAXPROCS(runtime.NumCPU())
}
func reloadConfigFromFile(s *Server) {
config, err := LoadConfig(configPath)
if err != nil {
logger.Errorf("Failed to reload config %v", err)
}
s.ReloadConfig(config)
}