mirror of
https://github.com/bjc/prosody.git
synced 2025-04-03 05:07:42 +03:00
mod_websocket: Merge session close handling changes from mod_c2s (bug fixes)
This should bring some fixes and general robustness that mod_websocket had missed out on. The duplicated code here is not at all ideal. To prevent this happening again, we should figure out how to have the common logic in a single place, while still being able to do the websocket-specific parts that we need. The main known bug that this fixes is that it's possible for a session to get into a non-destroyable state. For example, if we try to session:close() a hibernating session, then session.conn is nil and the function will simply return without doing anything. In the mod_c2s code we already handle this, and just destroy the session. But if a hibernating websocket session is never resumed or becomes non-resumable, it will become immortal! By merging the fix from mod_c2s, the session should now be correctly destroyed.
This commit is contained in:
parent
5b4624137d
commit
8005ac825f
1 changed files with 17 additions and 11 deletions
|
@ -87,6 +87,7 @@ local function session_close(session, reason)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
stream_error = tostring(stream_error);
|
||||||
log("debug", "Disconnecting client, <stream:error> is: %s", stream_error);
|
log("debug", "Disconnecting client, <stream:error> is: %s", stream_error);
|
||||||
session.send(stream_error);
|
session.send(stream_error);
|
||||||
end
|
end
|
||||||
|
@ -94,28 +95,33 @@ local function session_close(session, reason)
|
||||||
session.send(st.stanza("close", { xmlns = xmlns_framing }));
|
session.send(st.stanza("close", { xmlns = xmlns_framing }));
|
||||||
function session.send() return false; end
|
function session.send() return false; end
|
||||||
|
|
||||||
-- luacheck: ignore 422/reason
|
local reason_text = (reason and (reason.name or reason.text or reason.condition)) or reason;
|
||||||
-- FIXME reason should be handled in common place
|
session.log("debug", "c2s stream for %s closed: %s", session.full_jid or session.ip or "<unknown>", reason_text or "session closed");
|
||||||
local reason = (reason and (reason.name or reason.text or reason.condition)) or reason;
|
|
||||||
session.log("debug", "c2s stream for %s closed: %s", session.full_jid or ("<"..session.ip..">"), reason or "session closed");
|
|
||||||
|
|
||||||
-- Authenticated incoming stream may still be sending us stanzas, so wait for </stream:stream> from remote
|
-- Authenticated incoming stream may still be sending us stanzas, so wait for </stream:stream> from remote
|
||||||
local conn = session.conn;
|
local conn = session.conn;
|
||||||
if reason == nil and not session.notopen and session.type == "c2s" then
|
if reason_text == nil and not session.notopen and session.type == "c2s" then
|
||||||
-- Grace time to process data from authenticated cleanly-closed stream
|
-- Grace time to process data from authenticated cleanly-closed stream
|
||||||
add_task(stream_close_timeout, function ()
|
add_task(stream_close_timeout, function ()
|
||||||
if not session.destroyed then
|
if not session.destroyed then
|
||||||
session.log("warn", "Failed to receive a stream close response, closing connection anyway...");
|
session.log("warn", "Failed to receive a stream close response, closing connection anyway...");
|
||||||
sm_destroy_session(session, reason);
|
sm_destroy_session(session, reason_text);
|
||||||
conn:write(build_close(1000, "Stream closed"));
|
if conn then
|
||||||
conn:close();
|
conn:write(build_close(1000, "Stream closed"));
|
||||||
|
conn:close();
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end);
|
end);
|
||||||
else
|
else
|
||||||
sm_destroy_session(session, reason);
|
sm_destroy_session(session, reason_text);
|
||||||
conn:write(build_close(1000, "Stream closed"));
|
if conn then
|
||||||
conn:close();
|
conn:write(build_close(1000, "Stream closed"));
|
||||||
|
conn:close();
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
else
|
||||||
|
local reason_text = (reason and (reason.name or reason.text or reason.condition)) or reason;
|
||||||
|
sm_destroy_session(session, reason_text);
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue