-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathmain.go
281 lines (232 loc) · 8.61 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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
package main
import (
"io"
"maps"
"os"
"os/signal"
"slices"
"syscall"
"time"
"github.com/synacktiv/bbs/logger"
)
var gChainsConf chainsConf
var gRoutingConf routingConf
var gServerConf serverConf
var gHosts hostMap
var gMetaLogger *logger.MetaLogger
func main() {
// Parse the command line arguments
parseArgs()
// ***** BEGIN Logs setup *****
var auditFile *os.File = nil
var logFile *os.File = nil
if gArgAuditPath != "" {
var err error
auditFile, err = os.OpenFile(gArgAuditPath, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0755)
if err != nil {
panic(err)
}
defer auditFile.Close()
}
if gArgLogPath != "" {
var err error
logFile, err = os.OpenFile(gArgLogPath, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0755)
if err != nil {
panic(err)
}
defer logFile.Close()
}
var logWriter io.Writer = os.Stdout
var auditWriter io.Writer = os.Stdout
if auditFile != nil {
if gArgAuditBoth {
auditWriter = io.MultiWriter(os.Stdout, auditFile)
} else {
auditWriter = auditFile
}
}
if logFile != nil {
if gArgLogBoth {
logWriter = io.MultiWriter(os.Stdout, logFile)
} else {
logWriter = logFile
}
}
gMetaLogger = logger.NewMetaLogger(logWriter, auditWriter)
if gArgQuietBool {
gMetaLogger.SetLogLevel(logger.LogLevelQuiet)
} else if gArgVerboseBool {
gMetaLogger.SetLogLevel(logger.LogLevelVerbose)
} else {
gMetaLogger.SetLogLevel(logger.LogLevelNormal)
}
if gArgNoAuditBool {
gMetaLogger.SetAuditLevel(logger.AuditLevelNo)
} else {
gMetaLogger.SetAuditLevel(logger.AuditLevelYes)
}
// ***** END Logs setup *****
// ***** BEGIN Configuration files loading *****
// Output PID needed to hot reload configuration files
gMetaLogger.Infof("bbs PID: %v. Use the following to reload configuration:", os.Getpid())
gMetaLogger.Infof("kill -HUP %v", os.Getpid())
// Setup a notification channel listening on SIGHUP, used to hot reload configuration files
signalCh := make(chan os.Signal, 1)
signal.Notify(signalCh, syscall.SIGHUP)
// Send a SIGHUP to trigger initial configuration loading
signalCh <- syscall.SIGHUP
// Wait for data on the previously created channel to reload configuration files
for {
sig := <-signalCh
gMetaLogger.Infof("Signal %v received, reloading configurations", sig)
gMetaLogger.Debug("Describing gServerConf.servers : ")
describeServers(gServerConf.servers)
// Load main config from the unified config file (proxies, chains, routes, servers and hosts)
config, err := parseMainConfig(gArgConfigPath)
if err != nil {
gMetaLogger.Errorf("error parsing main config : %v", err)
continue
}
gMetaLogger.Info("JSON configuration file parsed. Checking for errors.")
gMetaLogger.Debugf("Parsed main config : %v", config)
// Create the implicit single proxy chains associated with each declared proxy
duplicateName := false
definedChains := slices.Collect(maps.Keys(config.Chains))
for proxyName, _ := range config.Proxies {
if slices.Contains(definedChains, proxyName) {
gMetaLogger.Errorf("chain %v cannot be named as proxy %v", proxyName, proxyName)
duplicateName = true
break
}
var implicitChain proxyChainDesc
implicitChain.ProxyDns = true
implicitChain.TcpConnectTimeout = 1000
implicitChain.TcpReadTimeout = 2000
implicitChain.Proxies = []string{proxyName}
config.Chains[proxyName] = implicitChain
}
if duplicateName {
continue
}
// Check that all proxies used in all chains of chains section correspond to an existing proxy in the proxies section
allExist := true
definedProxies := slices.Collect(maps.Keys(config.Proxies))
for chainName, chainDesc := range config.Chains {
for index, proxyName := range chainDesc.Proxies {
if !slices.Contains(definedProxies, proxyName) {
gMetaLogger.Errorf("proxy %v used at index %v of chain %v is not part of the defined proxies in proxies section (%v)", proxyName, index, chainName, definedProxies)
allExist = false
}
}
}
if !allExist {
continue
}
// If -pac is not defined, perform consistency checks on routing configuration
if gArgPACPath == "" {
// Check that all routes defined in routes section correspond to an existing chain in the chains section
allExist = true
definedChains := slices.Collect(maps.Keys(config.Chains))
for routingTableName, routingTable := range config.Routes {
for index, ruleBlock := range routingTable {
if ruleBlock.Route != "drop" && !slices.Contains(definedChains, ruleBlock.Route) {
gMetaLogger.Errorf("route %v defined in ruleBlock number %v of routingTable %v is not part of the defined chains in the chains section (%v)", ruleBlock.Route, index, routingTableName, definedChains)
allExist = false
}
}
}
if !allExist {
continue
}
// Check that all routing tables used in all servers of the servers sections correspond to an existing routing table in the routes section
allExist = true
definedRoutingTables := slices.Collect(maps.Keys(config.Routes))
for index, server := range config.Servers {
if !slices.Contains(definedRoutingTables, server.table) {
gMetaLogger.Errorf("table %v used by server number %v is not part of the defined routing tables in section routes (%v)", server.table, index, definedRoutingTables)
allExist = false
}
}
if !allExist {
continue
}
} else { // Otherwise, load PAC file and do not perform consistency checks
err := reloadPACConf(gArgPACPath)
if err != nil {
gMetaLogger.Errorf("error reloading pac file: %v", err)
continue
}
gMetaLogger.Info("Global PAC configuration updated")
}
// At this point, the defined configuration should be consistent, so we can update the globals
gMetaLogger.Info("No errors detected. Updating global configurations.")
// Build a proxyChain object from the proxyChainDesc parsed in JSON file
proxychains := make(map[string]proxyChain)
for chainName, chainDesc := range config.Chains {
var proxychain proxyChain
proxychain.proxyDns = chainDesc.ProxyDns
proxychain.tcpConnectTimeout = chainDesc.TcpConnectTimeout
proxychain.tcpReadTimeout = chainDesc.TcpReadTimeout
for _, proxyName := range chainDesc.Proxies {
proxychain.proxies = append(proxychain.proxies, config.Proxies[proxyName])
}
proxychains[chainName] = proxychain
}
gChainsConf.mu.Lock()
gChainsConf.proxychains = proxychains
gChainsConf.valid = true
gChainsConf.mu.Unlock()
gMetaLogger.Info("Global chains configuration updated")
gMetaLogger.Debugf("-> %v", gChainsConf.proxychains)
gHosts = config.Hosts
gMetaLogger.Info("Global hosts configuration updated")
gMetaLogger.Debugf("-> %v", gHosts)
if gArgPACPath == "" {
gRoutingConf.mu.Lock()
gRoutingConf.routing = config.Routes
gRoutingConf.valid = true
gRoutingConf.mu.Unlock()
gMetaLogger.Info("Global routing configuration updated")
gMetaLogger.Debugf("-> %v", gRoutingConf.routing)
}
// Update global servers variable, stop old ones and start new ones
// Stoping running servers that are not defined in the new configuration
gMetaLogger.Debug("Describing servers : ")
describeServers(config.Servers)
gServerConf.mu.Lock()
j := 0
for i := range gServerConf.servers {
i_fixed := i - j
stillExists := slices.ContainsFunc(config.Servers, func(s server) bool { return compare(s, gServerConf.servers[i_fixed]) })
if stillExists {
gMetaLogger.Debugf("Server %v still exists in new loaded servers, keeping it", gServerConf.servers[i_fixed])
} else {
gMetaLogger.Debugf("Server %v does not exists anymore, stopping it", gServerConf.servers[i_fixed])
gServerConf.servers[i_fixed].stop()
gServerConf.servers = slices.Delete(gServerConf.servers, i_fixed, i_fixed+1)
j = j + 1
}
}
for i := range config.Servers {
alreadyExists := slices.ContainsFunc(gServerConf.servers, func(s server) bool { return compare(s, config.Servers[i]) })
if !alreadyExists {
gServerConf.servers = append(gServerConf.servers, config.Servers[i])
}
}
gServerConf.mu.Unlock()
gMetaLogger.Debugf("gServerConf.servers : %v", gServerConf.servers)
gMetaLogger.Debug("Describing gServerConf.servers : ")
describeServers(gServerConf.servers)
// Start all servers that are not running
for i := 0; i < len(gServerConf.servers); i++ {
if !gServerConf.servers[i].running {
gMetaLogger.Debugf("myServer %v(%p) is not running, running it", gServerConf.servers[i], &gServerConf.servers[i])
time.Sleep(1 * time.Second)
go (gServerConf.servers[i]).run()
gMetaLogger.Debugf("myServer %v(%p) is running", gServerConf.servers[i], &gServerConf.servers[i])
}
}
gMetaLogger.Debug("Describing gServerConf.servers : ")
describeServers(gServerConf.servers)
}
}