-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathserver.lua
executable file
·151 lines (139 loc) · 3.68 KB
/
server.lua
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
#!/usr/bin/env lua5.3
xpcall(function()
local ignore_newindex = { lfs = true }
setmetatable(_ENV or getfenv(), { __index = function()
error("__index on env", 2)
end, __newindex = function(_, key)
if ignore_newindex[key] then
return
end
error("__newindex on env", 2)
end })
local dump_stack_cond
if true then
-- Load cqueues early.
local cqueues = require("cqueues")
local real_poll = cqueues.poll
local util_named_traceback
local function pack(...)
return select("#", ...), { ... }
end
cqueues.poll = function(...)
if not ... then
return
end
local ready
while true do
local nready
nready, ready = pack(real_poll(dump_stack_cond, ...))
if nready >= 2 and not ready[1] then
assert(ready[1], ready[2])
end
for i = 1, #ready do
if ready[i] == dump_stack_cond then
print(util_named_traceback("dump stack requested"))
table.remove(ready, i)
break
end
end
if #ready > 0 then
break
end
end
return table.unpack(ready)
end
dump_stack_cond = require("cqueues.condition").new()
util_named_traceback = require("tptmp.server.util").named_traceback
end
local lfs = require("lfs")
local cqueues = require("cqueues")
local config = require("tptmp.server.config")
local log = require("tptmp.server.log")
local util = require("tptmp.server.util")
local remote_console = require("tptmp.server.remote_console")
local authenticator = require("tptmp.server.authenticator")
local server = require("tptmp.server.server")
local dynamic_config = require("tptmp.server.dynamic_config")
local plugin_host = require("tptmp.server.plugin_host")
local room = require("tptmp.server.room")
local client = require("tptmp.server.client")
util.periodic_tracebacks(config.periodic_tracebacks)
math.randomseed(os.time())
local plugins = {}
for file in lfs.dir("tptmp/server/plugins") do
local name = file:match("^(.+)%.lua$")
if name then
plugins[name] = require("tptmp.server.plugins." .. name)
log.inf("[plugin] loaded " .. name)
end
end
local phost = plugin_host.new({
plugins = plugins,
mtidx = {
room = room.room_i,
client = client.client_i,
server = server.server_i,
},
})
print = log.dump
local queue = cqueues.new()
local rcon, auth, serv
local function stop()
log.inf("stopping")
rcon:stop()
serv:stop()
end
util.cqueues_wrap(queue, function()
local dconf = dynamic_config.new({
name = "dconf",
})
if config.auth then
auth = authenticator.new({
name = "auth",
})
end
serv = server.new({
auth = auth,
version = config.version,
name = "server",
dconf = dconf,
phost = phost,
})
serv:start()
rcon = remote_console.new({
server = serv,
name = "rcon",
phost = phost,
})
rcon:start()
end, "rcon")
if dump_stack_cond then
local signal = require("cqueues.signal")
signal.discard(signal.SIGUSR1)
local listener = signal.listen(signal.SIGUSR1)
util.cqueues_wrap(queue, function()
while true do
local ready = util.cqueues_poll(listener)
if ready[listener] then
print("SIGUSR1, requesting stacks to be dumped")
dump_stack_cond:signal()
end
end
end)
end
-- * util.cqueues_wrap shouldn't throw errors.
local ok, err = pcall(function()
assert(queue:loop())
end)
if type(err) == "string" and err:find("interrupted!$") then
log.inf("interrupted")
ok = true
end
assert(ok or err == util.CQUEUES_WRAP_RETHROW, "sanity check failure")
end, function(err)
local function rip(str)
io.stderr:write(str:gsub("\n", "\n[rip] ") .. "\n")
end
rip("[rip] top-level error: " .. tostring(err))
rip("[rip] " .. debug.traceback())
end)