-- README -- Squish verse into this dir, then squish them into one, which you move -- and rename to mod_ircd.lua in your prosody modules/plugins dir. -- -- IRC spec: -- http://tools.ietf.org/html/rfc2812 local _module = module module = _G.module local module = _module -- local component_jid, component_secret, muc_server = module.host, nil, module:get_option("conference_server"); package.loaded["util.sha1"] = require "util.encodings"; local verse = require "verse" require "verse.component" require "socket" c = verse.new();--verse.logger()) c:add_plugin("groupchat"); local function verse2prosody(e) return c:event("stanza", e.stanza) or true; end module:hook("message/bare", verse2prosody); module:hook("message/full", verse2prosody); module:hook("presence/bare", verse2prosody); module:hook("presence/full", verse2prosody); c.type = "component"; c.send = core_post_stanza; -- This plugin is actually a verse based component, but that mode is currently commented out -- Add some hooks for debugging --c:hook("opened", function () print("Stream opened!") end); --c:hook("closed", function () print("Stream closed!") end); --c:hook("stanza", function (stanza) print("Stanza:", stanza) end); -- This one prints all received data --c:hook("incoming-raw", print, 1000); --c:hook("stanza", print, 1000); --c:hook("outgoing-raw", print, 1000); -- Print a message after authentication --c:hook("authentication-success", function () print("Logged in!"); end); --c:hook("authentication-failure", function (err) print("Failed to log in! Error: "..tostring(err.condition)); end); -- Print a message and exit when disconnected --c:hook("disconnected", function () print("Disconnected!"); os.exit(); end); -- Now, actually start the connection: --c.connect_host = "127.0.0.1" --c:connect_component(component_jid, component_secret); local jid = require "util.jid"; local nodeprep = require "util.encodings".stringprep.nodeprep; local function parse_line(line) local ret = {}; if line:sub(1,1) == ":" then ret.from, line = line:match("^:(%w+)%s+(.*)$"); end for part in line:gmatch("%S+") do if part:sub(1,1) == ":" then ret[#ret+1] = line:match(":(.*)$"); break end ret[#ret+1]=part; end return ret; end local function build_line(parts) if #parts > 1 then parts[#parts] = ":" .. parts[#parts]; end return (parts.from and ":"..parts.from.." " or "")..table.concat(parts, " "); end local function irc2muc(channel, nick) local room = channel and nodeprep(channel:match("^#(%w+)")) or nil; return jid.join(room, muc_server, nick) end local function muc2irc(room) local channel, _, nick = jid.split(room); return "#"..channel, nick; end local rolemap = { moderator = "@", participant = "+", } local modemap = { moderator = "o", participant = "v", } local irc_listener = { default_port = 6667, default_mode = "*l" }; local sessions = {}; local jids = {}; local commands = {}; local nicks = {}; local st = require "util.stanza"; local conference_server = muc_server; local function irc_close_session(session) session.conn:close(); end function irc_listener.onincoming(conn, data) local session = sessions[conn]; if not session then session = { conn = conn, host = component_jid, reset_stream = function () end, close = irc_close_session, log = logger.init("irc"..(conn.id or "1")), rooms = {}, roster = {} }; sessions[conn] = session; function session.data(data) local parts = parse_line(data); module:log("debug", require"util.serialization".serialize(parts)); local command = table.remove(parts, 1); if not command then return; end command = command:upper(); if not session.nick then if not (command == "USER" or command == "NICK") then module:log("debug", "Client tried to send command %s before registering", command); return session.send{from=muc_server, 451, command, "You have not registered"} end end if commands[command] then local ret = commands[command](session, parts); if ret then return session.send(ret); end else session.send{from=muc_server, 421, session.nick, command, "Unknown command"}; return module:log("debug", "Unknown command: %s", command); end end function session.send(data) if type(data) == "string" then return conn:write(data.."\r\n"); elseif type(data) == "table" then local line = build_line(data); module:log("debug", line); conn:write(line.."\r\n"); end end end if data then session.data(data); end end function irc_listener.ondisconnect(conn, error) local session = sessions[conn]; if session then for _, room in pairs(session.rooms) do room:leave("Disconnected"); end if session.nick then nicks[session.nick] = nil; end if session.full_jid then jids[session.full_jid] = nil; end end sessions[conn] = nil; end function commands.NICK(session, args) if session.nick then session.send(":"..muc_server.." 484 * "..nick.." :I'm afraid I can't let you do that, "..session.nick); --TODO Loop throug all rooms and change nick, with help from Verse. return; end local nick = args[1]; nick = nick:gsub("[^%w_]",""); if nicks[nick] then session.send{from=muc_server, 433, nick, "The nickname "..nick.." is already in use"}; return; end local full_jid = jid.join(nick, component_jid, "ircd"); jids[full_jid] = session; nicks[nick] = session; session.nick = nick; session.full_jid = full_jid; session.type = "c2s"; session.send{from = muc_server, 001, nick, "Welcome to XMPP via the "..session.host.." gateway "..session.nick}; session.send{from=nick, "MODE", nick, "+i"}; -- why end function commands.USER(session, params) -- FIXME -- Empty command for now end function commands.JOIN(session, args) local channel = args[1]; local room_jid = irc2muc(channel); print(session.full_jid); local room, err = c:join_room(room_jid, session.nick, { source = session.full_jid } ); if not room then return ":"..session.host.." ERR :Could not join room: "..err end session.rooms[channel] = room; room.channel = channel; room.session = session; session.send{from=session.nick, "JOIN", channel}; session.send{from=muc_server, 332, session.nick, channel ,"Connection in progress..."}; room:hook("message", function(event) if not event.body then return end local nick, body = event.nick, event.body; if nick ~= session.nick then if body:sub(1,4) == "/me " then body = "\1ACTION ".. body:sub(5) .. "\1" end local type = event.stanza.attr.type; session.send{from=nick, "PRIVMSG", type == "groupchat" and channel or nick, body}; --FIXME PM's probably won't work end end); end c:hook("groupchat/joined", function(room) local session = room.session or jids[room.opts.source]; local channel = room.channel; session.send{from=session.nick.."!"..session.nick, "JOIN", channel}; session.send((":%s!%s JOIN %s :"):format(session.nick, session.nick, channel)); if room.topic then session.send{from=muc_server, 332, room.topic}; end commands.NAMES(session, channel) if session.nick.role then session.send{from=muc_server, "MODE", channel, session.nick, modemap[session.nick.role], session.nick} end room:hook("occupant-joined", function(nick) session.send{from=nick.nick.."!"..nick.nick, "JOIN", channel}; if nick.role and modemap[nick.role] then session.send{from=nick.nick.."!"..nick.nick, "MODE", channel, modemap[nick.role], nick.nick}; end end); room:hook("occupant-left", function(nick) session.send{from=nick.nick.."!"..nick.nick, "PART", room.channel}; end); end); function commands.NAMES(session, channel) local nicks = { }; local room = session.rooms[channel]; if not room then return end -- TODO Break this out into commands.NAMES for nick, n in pairs(room.occupants) do if n.role and rolemap[n.role] then nick = rolemap[n.role] .. nick; end table.insert(nicks, nick); end nicks = table.concat(nicks, " "); session.send((":%s 353 %s = %s :%s"):format(session.host, session.nick, channel, nicks)); session.send((":%s 366 %s %s :End of /NAMES list."):format(session.host, session.nick, channel)); session.send(":"..session.host.." 353 "..session.nick.." = "..channel.." :"..nicks); end function commands.PART(session, args) local channel, part_message = unpack(args); channel = channel:match("^([%S]*)"); session.rooms[channel]:leave(part_message); session.send(":"..session.nick.." PART :"..channel); end function commands.PRIVMSG(session, args) local channel, message = unpack(args); if message and #message > 0 then if message:sub(1,8) == "\1ACTION " then message = "/me ".. message:sub(9,-2) end -- TODO clean out invalid chars if channel:sub(1,1) == "#" then if session.rooms[channel] then module:log("debug", "%s sending PRIVMSG \"%s\" to %s", session.nick, message, channel); session.rooms[channel]:send_message(message); end else -- private message local nick = channel; module:log("debug", "PM to %s", nick); for channel, room in pairs(session.rooms) do module:log("debug", "looking for %s in %s", nick, channel); if room.occupants[nick] then module:log("debug", "found %s in %s", nick, channel); local who = room.occupants[nick]; -- FIXME PMs in verse --room:send_private_message(nick, message); local pm = st.message({type="chat",to=who.jid}, message); module:log("debug", "sending PM to %s: %s", nick, tostring(pm)); room:send(pm) break end end end end end function commands.PING(session, args) session.send{from=muc_server, "PONG", args[1]}; end function commands.WHO(session, args) local channel = args[1]; if session.rooms[channel] then local room = session.rooms[channel] for nick in pairs(room.occupants) do --n=MattJ 91.85.191.50 irc.freenode.net MattJ H :0 Matthew Wild session.send{from=muc_server, 352, session.nick, channel, nick, nick, muc_server, nick, "H", "0 "..nick} end session.send{from=muc_server, 315, session.nick, channel, "End of /WHO list"}; end end function commands._MODE(session, args) -- FIXME local channel, target = unpack(args); if target then -- do stuff? --room:set_affiliation(...) else -- What's 324? And +J ? session.send{from=muc_server, 324, session.nick, channel, "+J"} end end function commands.QUIT(session, args) session.send{"ERROR", "Closing Link: "..session.nick}; for _, room in pairs(session.rooms) do room:leave(args[1]); end jids[session.full_jid] = nil; nicks[session.nick] = nil; sessions[session.conn] = nil; session:close(); end function commands.RAW(session, data) --c:send(data) end local function desetup() require "net.connlisteners".deregister("irc"); end --c:hook("ready", function () require "net.connlisteners".register("irc", irc_listener); require "net.connlisteners".start("irc"); --end); module:hook("module-unloaded", desetup) --print("Starting loop...") --verse.loop()