Software /
code /
prosody
Comparison
net/server_epoll.lua @ 13490:6f840763fc73
net.server_epoll: Add support for systemd socket activation
Allows creating listening sockets and accepting client connections
before Prosody starts.
This is unlike normal Prosody dynamic resource management, where ports
may added and removed at any time, and the ports defined by the config.
Weird things happen if these are closed (e.g. due to reload) so here we
prevent closing and ensure sockets are reused when opened again.
author | Kim Alvefur <zash@zash.se> |
---|---|
date | Tue, 14 May 2024 17:07:47 +0200 |
parent | 13450:f36e1e1b84fe |
child | 13552:f0b2c026e542 |
comparison
equal
deleted
inserted
replaced
13489:ae65f199f408 | 13490:6f840763fc73 |
---|---|
33 | 33 |
34 local poller = require "prosody.util.poll" | 34 local poller = require "prosody.util.poll" |
35 local EEXIST = poller.EEXIST; | 35 local EEXIST = poller.EEXIST; |
36 local ENOENT = poller.ENOENT; | 36 local ENOENT = poller.ENOENT; |
37 | 37 |
38 -- systemd socket activation | |
39 local SD_LISTEN_FDS_START = 3; | |
40 local SD_LISTEN_FDS = tonumber(os.getenv("LISTEN_FDS")) or 0; | |
41 | |
42 local inherited_sockets = setmetatable({}, { | |
43 __index = function(t, k) | |
44 local serv_mt = debug.getregistry()["tcp{server}"]; | |
45 for i = 1, SD_LISTEN_FDS do | |
46 local serv = socket.tcp(); | |
47 if serv:getfd() ~= _SOCKETINVALID then | |
48 -- If LuaSocket allocated a FD for then we can't really close it and it would leak. | |
49 log("error", "LuaSocket not compatible with socket activation. Upgrade LuaSocket or disable socket activation."); | |
50 setmetatable(t, nil); | |
51 break | |
52 end | |
53 serv:setfd(SD_LISTEN_FDS_START + i - 1); | |
54 debug.setmetatable(serv, serv_mt); | |
55 serv:settimeout(0); | |
56 local ip, port = serv:getsockname(); | |
57 t[ip .. ":" .. port] = serv; | |
58 if ip == "0.0.0.0" then | |
59 -- LuaSocket treats '*' as an alias for '0.0.0.0' | |
60 t["*:" .. port] = serv; | |
61 end | |
62 end | |
63 | |
64 -- Disable lazy-loading mechanism once performed | |
65 setmetatable(t, nil); | |
66 return t[k]; | |
67 end; | |
68 }); | |
69 | |
38 local poll = assert(poller.new()); | 70 local poll = assert(poller.new()); |
39 | 71 |
40 local _ENV = nil; | 72 local _ENV = nil; |
41 -- luacheck: std none | 73 -- luacheck: std none |
42 | 74 |
942 server:add(true, false); | 974 server:add(true, false); |
943 return server; | 975 return server; |
944 end | 976 end |
945 | 977 |
946 local function listen(addr, port, listeners, config) | 978 local function listen(addr, port, listeners, config) |
979 local inherited = inherited_sockets[addr .. ":" .. port]; | |
980 if inherited then | |
981 local conn = wrapserver(inherited, addr, port, listeners, config); | |
982 -- sockets created by systemd must not be :close() since we may not have | |
983 -- privileges to create them | |
984 conn.destroy = interface.del; | |
985 return conn; | |
986 end | |
947 local conn, err = socket.bind(addr, port, cfg.tcp_backlog); | 987 local conn, err = socket.bind(addr, port, cfg.tcp_backlog); |
948 if not conn then return conn, err; end | 988 if not conn then return conn, err; end |
949 conn:settimeout(0); | 989 conn:settimeout(0); |
950 return wrapserver(conn, addr, port, listeners, config); | 990 return wrapserver(conn, addr, port, listeners, config); |
951 end | 991 end |