diff options
Diffstat (limited to 'xmpp/pylon.lua')
-rw-r--r-- | xmpp/pylon.lua | 187 |
1 files changed, 187 insertions, 0 deletions
diff --git a/xmpp/pylon.lua b/xmpp/pylon.lua new file mode 100644 index 0000000..ba9c252 --- /dev/null +++ b/xmpp/pylon.lua @@ -0,0 +1,187 @@ +local cqueues = require'cqueues' +local cqaux = require'cqueues.auxlib' +local socket = require'cqueues.socket' +local xml = require'xmpp.xml' +local X = xml.X +local xmlify = xml.xmlify +local pprint=require'pprint' +local Queue = require'util.queue' +local base64 = require'xmpp.base64' +local sha1 = require'xmpp.sha1' +local Channel = require'util.channel' + +local Xmpp = {} + +local function make_auth(authz, authn, password) + -- sasl plain (RFC4616) + return base64.encode(authz..'\0'..authn..'\0'..password) +end + +function Xmpp.make(wilson, conf) + local self = { + name = conf.name, + wilson = wilson, + inbox = Queue.make(), + cq = wilson.cq, -- todo + nicks_inuse = {}, -- todo + } + local function conf_var(name) + assert(conf[name] ~= nil, 'missing conf field '..name) + self[name] = conf[name] + end + -- conf_var 'jid' + conf_var 'server' + -- conf_var 'resource' + conf_var 'component' + conf_var 'component_secret' + + setmetatable(self, {__index=Xmpp}) + return self +end + +function Xmpp._connect_c2s(self) + local sock = assert(socket.connect(self.server, 5222)) + self.sock = sock + sock:setmode('bn','bn') + + local start = ([[ +<?xml version='1.0'?><stream:stream from='%s' to='%s' version='1.0' xml:lang='en' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams'>]]):format(self.jid, self.server) + + -- state of the art xml parser + local function check_and_send(test, text) + local x = sock:read('-2048') + assert(x:find(test)) + if text then sock:write(text) end + end + local function ietf_urn(v) return 'urn:ietf:params:xml:ns:xmpp-'..v end + + sock:write(start) + check_and_send('starttls', xmlify(X.starttls{xmlns=ietf_urn"tls"})) + check_and_send('proceed', nil) + sock:starttls() + sock:write(start) + local auth = make_auth('', self.jid:match"(.*)@", self.password) + check_and_send('PLAIN', + xmlify(X.auth{xmlns=ietf_urn"sasl", mechanism='PLAIN', auth})) + check_and_send('success',start) + check_and_send('bind', + xmlify(X.iq{type='set', id='aaaa', + X.bind{xmlns=ietf_urn"bind", X.resource{self.resource}}})) + check_and_send('jid',X.presence{X.show{'chat'}}) + + return sock +end + +-- this sucks! no tls! no security! only use on local connections! +function Xmpp._connect_component(self) + local sock = assert(socket.connect(self.server, 5347)) + self.sock = sock + sock:setmode('bn','bn') + -- yes, our component name goes in the 'to' field. don't ask me why + local start = ([[<stream:stream to='%s' xmlns='jabber:component:accept' xmlns:stream='http://etherx.jabber.org/streams'>]]):format(self.component) + + print(start) + + -- state of the art xml parser + local function check_and_send(test, text) + local x = sock:read('-2048') + assert(x:find(test)) + if text then sock:write(text) end + end + + sock:write(start) + local streamhead = sock:read('-2048') + print('streamhead', streamhead) + assert(streamhead:find'accept') + local streamid = streamhead:match"id='(.-)'" + sock:write(xmlify(X.handshake{sha1.sha1(streamid..self.component_secret)})) + check_and_send('<handshake/>',nil) + + return sock +end + +local THE_MUC = 'd@conference.ubq323.website' + +function Xmpp.run(self) + self:_connect_component() + self.cq:wrap(self.recving, self) + self.cq:wrap(self.sending, self) +end + +function Xmpp.recving(self) + local function getmore() + local function t(...) + pprint(...) + return ... + end + return t(self.sock:read'-2048') + end + for x in xml.stanzae(getmore) do + pprint(x) + print(xmlify(x)) + local body = x'body' and x'body'[1] + if x.label == 'message' then + local fr = x.xarg.from + local t = x.xarg.to + local from_nick = fr:match("/(.*)") + if not self.nicks_inuse[from_nick] and body and fr:match"/" and t == 'wilson@'..self.component then + self.wilson:deliver(Channel(self, THE_MUC), { + body = body, + sender = '[x]'..from_nick + }) + end + end + end +end + +function Xmpp.sending(self) + local users_inuse = {} + local function ensure_joined(muc,user,nick) + if self.nicks_inuse[nick] then return end + user = user:gsub("[^a-zA-Z0-9%.]","."):match("^%.*(.-)%.*$") + while users_inuse[user] do user = user..'-' end + local jid = user..'@'..self.component + local mucjid = muc..'/'..nick + self.nicks_inuse[nick] = true + users_inuse[user] = true + + self.sock:write(xmlify( + X.presence{from=jid, to=mucjid, + X.x{xmlns='http://jabber.org/protocol/muc', + X.history{maxstanzas='0'}}})) + end + ensure_joined(THE_MUC, 'wilson', 'wilson') + for dest_channel, message in self.inbox:iter() do + pprint(dest_channel, message) + ensure_joined(THE_MUC, message.sender, message.sender) + local user = message.sender:gsub("[^a-zA-Z0-9%.]","."):match("^%.*(.-)%.*$") + local jid = user..'@'..self.component + self.sock:write(xmlify( + X.message{to=THE_MUC, type='groupchat', from=jid, + X.body{message.body}})) + end +end + +function Xmpp.post(self, dest_channel, message) + self.inbox:enqueue(dest_channel, message) +end + +return Xmpp + +-- local cq = cqueues.new() +-- local conf = { +-- jid='wilson@ubq323.website', +-- server='ubq323.website', +-- password='gregory<3', +-- resource='cheese', +-- } +-- local dummy_network = { +-- post = function(self, pylonname, channel, message) +-- pprint(pylonname, channel, message) +-- end +-- } +-- local pylon = Xmpp.makepylon('xmpptest',conf, cq, dummy_network) +-- pylon:run() + +-- pprint('peas', cq:loop()) + |