Software /
code /
prosody-modules
Diff
mod_tcpproxy/mod_tcpproxy.lua @ 147:4db80a46b064
mod_tcpproxy: Initial commit. The moment you didn't know you've been waiting for is here... the dawn of TCP over XMPP.
author | Matthew Wild <mwild1@gmail.com> |
---|---|
date | Tue, 13 Apr 2010 04:38:07 +0100 |
child | 153:31e24026e4fd |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/mod_tcpproxy/mod_tcpproxy.lua Tue Apr 13 04:38:07 2010 +0100 @@ -0,0 +1,100 @@ +local st = require "util.stanza"; + +local xmlns_ibb = "http://jabber.org/protocol/ibb"; +local xmlns_tcp = "http://prosody.im/protocol/tcpproxy"; + +local host_attr, port_attr = xmlns_tcp.."\1host", xmlns_tcp.."\1port"; + +local base64 = require "util.encodings".base64; +local b64, unb64 = base64.encode, base64.decode; + +local host = module.host; + +local open_connections = {}; + +local function new_session(jid, sid, conn) + if not open_connections[jid] then + open_connections[jid] = {}; + end + open_connections[jid][sid] = conn; +end +local function close_session(jid, sid) + if open_connections[jid] then + open_connections[jid][sid] = nil; + if next(open_connections[jid]) == nil then + open_connections[jid] = nil; + end + return true; + end +end + +function proxy_component(origin, stanza) + local ibb_tag = stanza.tags[1]; + if (not (stanza.name == "iq" and stanza.attr.type == "set") + and stanza.name ~= "message") + or + (not (ibb_tag) + or ibb_tag.attr.xmlns ~= xmlns_ibb) then + if stanza.attr.type ~= "error" then + origin.send(st.error_reply(stanza, "cancel", "service-unavailable")); + end + return; + end + + if ibb_tag.name == "open" then + -- Starting a new stream + local to_host, to_port = ibb_tag.attr[host_attr], ibb_tag.attr[port_attr]; + local jid, sid = stanza.attr.from, ibb_tag.attr.sid; + if not (to_host and to_port) then + return origin.send(st.error_reply(stanza, "modify", "bad-request", "No host/port specified")); + elseif not sid or sid == "" then + return origin.send(st.error_reply(stanza, "modify", "bad-request", "No sid specified")); + elseif ibb_tag.attr.stanza ~= "message" then + return origin.send(st.error_reply(stanza, "modify", "bad-request", "Only 'message' stanza transport is supported")); + end + local conn, err = socket.tcp(); + if not conn then + return origin.send(st.error_reply(stanza, "wait", "resource-constraint", err)); + end + conn:settimeout(0); + + local success, err = conn:connect(to_host, to_port); + if not success and err ~= "timeout" then + return origin.send(st.error_reply(stanza, "wait", "remote-server-not-found", err)); + end + + local listener,seq = {}, 0; + function listener.onconnect(conn) + origin.send(st.reply(stanza)); + end + function listener.onincoming(conn, data) + origin.send(st.message({to=jid,from=host}) + :tag("data", {xmlns=xmlns_ibb,seq=seq,sid=sid}) + :text(b64(data))); + seq = seq + 1; + end + function listener.ondisconnect(conn, err) + origin.send(st.message({to=jid,from=host}) + :tag("close", {xmlns=xmlns_ibb,sid=sid})); + close_session(jid, sid); + end + + conn = server.wrapclient(conn, to_host, to_port, listener, "*a" ); + new_session(jid, sid, conn); + elseif ibb_tag.name == "data" then + local conn = open_connections[stanza.attr.from][ibb_tag.attr.sid]; + if conn then + conn:write(unb64(ibb_tag:get_text())); + else + return origin.send(st.error_reply(stanza, "cancel", "item-not-found")); + end + elseif ibb_tag.name == "close" then + if close_session(stanza.attr.from, ibb_tag.attr.sid) then + origin.send(st.reply(stanza)); + else + return origin.send(st.error_reply(stanza, "cancel", "item-not-found")); + end + end +end + +require "core.componentmanager".register_component(host, proxy_component);