129
|
1 module.host = "*" -- Global module
|
|
2
|
|
3 local logger = require "util.logger";
|
|
4 local log = logger.init("mod_websocket");
|
|
5 local httpserver = require "net.httpserver";
|
|
6 local lxp = require "lxp";
|
|
7 local init_xmlhandlers = require "core.xmlhandlers";
|
|
8 local st = require "util.stanza";
|
|
9 local sm = require "core.sessionmanager";
|
|
10
|
|
11 local sessions = {};
|
|
12 local default_headers = { };
|
|
13
|
|
14
|
|
15 local stream_callbacks = { default_ns = "jabber:client",
|
|
16 streamopened = sm.streamopened,
|
|
17 streamclosed = sm.streamclosed,
|
|
18 handlestanza = core_process_stanza };
|
|
19 function stream_callbacks.error(session, error, data)
|
|
20 if error == "no-stream" then
|
|
21 session.log("debug", "Invalid opening stream header");
|
|
22 session:close("invalid-namespace");
|
|
23 elseif session.close then
|
|
24 (session.log or log)("debug", "Client XML parse error: %s", tostring(error));
|
|
25 session:close("xml-not-well-formed");
|
|
26 end
|
|
27 end
|
|
28
|
|
29
|
|
30 local function session_reset_stream(session)
|
|
31 local parser = lxp.new(init_xmlhandlers(session, stream_callbacks), "\1");
|
|
32 session.parser = parser;
|
|
33
|
|
34 session.notopen = true;
|
|
35
|
|
36 function session.data(conn, data)
|
|
37 data, _ = data:gsub("[%z\255]", "")
|
|
38 log("debug", "Parsing: %s", data)
|
|
39
|
|
40 local ok, err = parser:parse(data)
|
|
41 if not ok then
|
|
42 log("debug", "Received invalid XML (%s) %d bytes: %s", tostring(err), #data,
|
|
43 data:sub(1, 300):gsub("[\r\n]+", " "):gsub("[%z\1-\31]", "_"));
|
|
44 session:close("xml-not-well-formed");
|
|
45 end
|
|
46 end
|
|
47 end
|
|
48
|
|
49 local stream_xmlns_attr = {xmlns='urn:ietf:params:xml:ns:xmpp-streams'};
|
|
50 local default_stream_attr = { ["xmlns:stream"] = "http://etherx.jabber.org/streams", xmlns = stream_callbacks.default_ns, version = "1.0", id = "" };
|
|
51 local function session_close(session, reason)
|
|
52 local log = session.log or log;
|
|
53 if session.conn then
|
|
54 if session.notopen then
|
|
55 session.send("<?xml version='1.0'?>");
|
|
56 session.send(st.stanza("stream:stream", default_stream_attr):top_tag());
|
|
57 end
|
|
58 if reason then
|
|
59 if type(reason) == "string" then -- assume stream error
|
|
60 log("info", "Disconnecting client, <stream:error> is: %s", reason);
|
|
61 session.send(st.stanza("stream:error"):tag(reason, {xmlns = 'urn:ietf:params:xml:ns:xmpp-streams' }));
|
|
62 elseif type(reason) == "table" then
|
|
63 if reason.condition then
|
|
64 local stanza = st.stanza("stream:error"):tag(reason.condition, stream_xmlns_attr):up();
|
|
65 if reason.text then
|
|
66 stanza:tag("text", stream_xmlns_attr):text(reason.text):up();
|
|
67 end
|
|
68 if reason.extra then
|
|
69 stanza:add_child(reason.extra);
|
|
70 end
|
|
71 log("info", "Disconnecting client, <stream:error> is: %s", tostring(stanza));
|
|
72 session.send(stanza);
|
|
73 elseif reason.name then -- a stanza
|
|
74 log("info", "Disconnecting client, <stream:error> is: %s", tostring(reason));
|
|
75 session.send(reason);
|
|
76 end
|
|
77 end
|
|
78 end
|
|
79 session.send("</stream:stream>");
|
|
80 session.conn:close();
|
|
81 websocket_listener.ondisconnect(session.conn, (reason and (reason.text or reason.condition)) or reason or "session closed");
|
|
82 end
|
|
83 end
|
|
84
|
|
85
|
|
86 local websocket_listener = { default_mode = "*a" };
|
|
87 function websocket_listener.onincoming(conn, data)
|
|
88 local session = sessions[conn];
|
|
89 if not session then
|
|
90 session = { type = "c2s_unauthed",
|
|
91 conn = conn,
|
|
92 reset_stream = session_reset_stream,
|
|
93 close = session_close,
|
|
94 dispatch_stanza = stream_callbacks.handlestanza,
|
|
95 log = logger.init("websocket"),
|
|
96 secure = conn.ssl };
|
|
97
|
|
98 function session.send(s)
|
|
99 conn:write("\00" .. tostring(s) .. "\255");
|
|
100 end
|
|
101
|
|
102 sessions[conn] = session;
|
|
103 end
|
|
104
|
|
105 session_reset_stream(session);
|
|
106
|
|
107 if data then
|
|
108 session.data(conn, data);
|
|
109 end
|
|
110 end
|
|
111
|
|
112 function websocket_listener.ondisconnect(conn, err)
|
|
113 local session = sessions[conn];
|
|
114 if session then
|
|
115 (session.log or log)("info", "Client disconnected: %s", err);
|
|
116 sm.destroy_session(session, err);
|
|
117 sessions[conn] = nil;
|
|
118 session = nil;
|
|
119 end
|
|
120 end
|
|
121
|
|
122
|
|
123 function handle_request(method, body, request)
|
|
124 if request.method ~= "GET" or request.headers["upgrade"] ~= "WebSocket" or request.headers["connection"] ~= "Upgrade" then
|
|
125 if request.method == "OPTIONS" then
|
|
126 return { headers = default_headers, body = "" };
|
|
127 else
|
|
128 return "<html><body>You really don't look like a Websocket client to me... what do you want?</body></html>";
|
|
129 end
|
|
130 end
|
|
131
|
|
132 local subprotocol = request.headers["Websocket-Protocol"];
|
|
133 if subprotocol ~= nil and subprotocol ~= "XMPP" then
|
|
134 return "<html><body>You really don't look like an XMPP Websocket client to me... what do you want?</body></html>";
|
|
135 end
|
|
136
|
|
137 if not method then
|
|
138 log("debug", "Request %s suffered error %s", tostring(request.id), body);
|
|
139 return;
|
|
140 end
|
|
141
|
|
142 request.conn:setlistener(websocket_listener);
|
|
143 request.write("HTTP/1.1 101 Web Socket Protocol Handshake\r\n");
|
|
144 request.write("Upgrade: WebSocket\r\n");
|
|
145 request.write("Connection: Upgrade\r\n");
|
|
146 request.write("WebSocket-Origin: file://\r\n"); -- FIXME
|
|
147 request.write("WebSocket-Location: ws://localhost:5281/xmpp-websocket\r\n"); -- FIXME
|
|
148 request.write("WebSocket-Protocol: XMPP\r\n");
|
|
149 request.write("\r\n");
|
|
150
|
|
151 return true;
|
|
152 end
|
|
153
|
|
154 local function setup()
|
|
155 local ports = module:get_option("websocket_ports") or { 5281 };
|
|
156 httpserver.new_from_config(ports, handle_request, { base = "xmpp-websocket" });
|
|
157 end
|
|
158 if prosody.start_time then -- already started
|
|
159 setup();
|
|
160 else
|
|
161 prosody.events.add_handler("server-started", setup);
|
|
162 end
|