LuaIRC/init.lua

275 lines
5.8 KiB
Lua
Raw Normal View History

2010-02-21 15:59:29 +01:00
local socket = require "socket"
local error = error
local setmetatable = setmetatable
local rawget = rawget
local unpack = unpack
local pairs = pairs
2010-03-10 12:03:36 +01:00
local assert = assert
local require = require
2010-02-21 15:59:29 +01:00
2010-03-10 12:03:36 +01:00
local print = print
2010-02-21 15:59:29 +01:00
2010-03-10 12:03:36 +01:00
module "irc"
2010-02-21 15:59:29 +01:00
local meta = {}
meta.__index = meta
2010-03-10 12:03:36 +01:00
_META = meta
2010-02-21 15:59:29 +01:00
2010-03-10 12:03:36 +01:00
require "irc.util"
require "irc.asyncoperations"
2010-02-21 15:59:29 +01:00
2010-03-10 12:03:36 +01:00
local meta_preconnect = {}
2010-02-21 15:59:29 +01:00
function meta_preconnect.__index(o, k)
2010-03-10 12:03:36 +01:00
local v = rawget(meta_preconnect, k)
if not v and meta[k] then
error("field '"..k.."' is not accessible before connecting", 2)
end
return v
2010-02-21 15:59:29 +01:00
end
2010-03-10 12:03:36 +01:00
2010-02-21 15:59:29 +01:00
function new(user)
2010-03-10 12:03:36 +01:00
local o = {
nick = assert(user.nick, "Field 'nick' is required");
username = user.username or "lua";
realname = user.realname or "Lua owns";
hooks = {};
2010-06-01 01:39:34 +02:00
track_users = true;
2010-03-10 12:03:36 +01:00
}
2010-02-21 15:59:29 +01:00
return setmetatable(o, meta_preconnect)
end
function meta:hook(name, id, f)
f = f or id
self.hooks[name] = self.hooks[name] or {}
self.hooks[name][id] = f
2010-06-01 01:39:34 +02:00
return id or f
2010-02-21 15:59:29 +01:00
end
meta_preconnect.hook = meta.hook
function meta:unhook(name, id)
local hooks = self.hooks[name]
2010-06-01 01:39:34 +02:00
assert(hooks, "no hooks exist for this event")
2010-02-21 15:59:29 +01:00
assert(hooks[id], "hook ID not found")
2010-06-01 01:39:34 +02:00
2010-02-21 15:59:29 +01:00
hooks[id] = nil
end
meta_preconnect.unhook = meta.unhook
function meta:invoke(name, ...)
local hooks = self.hooks[name]
if hooks then
for id,f in pairs(hooks) do
f(...)
end
end
end
2010-03-10 12:03:36 +01:00
function meta_preconnect:connect(server, port, timeout)
2010-02-21 15:59:29 +01:00
port = port or 6667
2010-03-10 12:03:36 +01:00
local s = socket.tcp()
self.socket = s
s:settimeout(timeout or 30)
assert(s:connect(server, port))
2010-02-21 15:59:29 +01:00
setmetatable(self, meta)
self:send("USER %s 0 * :%s", self.username, self.realname)
self:send("NICK %s", self.nick)
2010-03-10 12:03:36 +01:00
2010-06-01 01:39:34 +02:00
self.channels = {}
2010-03-10 12:03:36 +01:00
s:settimeout(0)
repeat
self:think()
until self.authed
2010-02-21 15:59:29 +01:00
end
function meta:disconnect(message)
local message = message or "Bye!"
2010-02-21 19:09:13 +01:00
self:invoke("OnDisconnect", message, false)
2010-02-21 15:59:29 +01:00
self:send("QUIT :%s", message)
self:shutdown()
end
function meta:shutdown()
self.socket:shutdown()
self.socket:close()
setmetatable(self, nil)
end
local function getline(self, errlevel)
line, err = self.socket:receive("*l")
if not line and err ~= "timeout" then
2010-07-16 07:01:05 +02:00
self:invoke("OnDisconnect", err, true)
2010-02-21 15:59:29 +01:00
self:shutdown()
error(err, errlevel)
end
2010-03-10 12:03:36 +01:00
2010-02-21 15:59:29 +01:00
return line
end
function meta:think()
while true do
2010-03-10 12:03:36 +01:00
local line = getline(self, 3)
2010-02-21 15:59:29 +01:00
if line then
self:handle(parse(line))
else
break
end
end
end
local handlers = {}
handlers["PING"] = function(o, prefix, query)
o:send("PONG :%s", query)
end
handlers["001"] = function(o)
2010-03-10 12:03:36 +01:00
o.authed = true
2010-02-21 15:59:29 +01:00
end
handlers["PRIVMSG"] = function(o, prefix, channel, message)
o:invoke("OnChat", parsePrefix(prefix), channel, message)
end
handlers["NOTICE"] = function(o, prefix, channel, message)
o:invoke("OnNotice", parsePrefix(prefix), channel, message)
end
handlers["JOIN"] = function(o, prefix, channel)
2010-06-01 01:39:34 +02:00
local user = parsePrefix(prefix)
if o.track_users then
if user.nick == o.nick then
o.channels[channel] = {users = {}}
else
o.channels[channel].users[user.nick] = user
end
end
o:invoke("OnJoin", user, channel)
2010-02-21 15:59:29 +01:00
end
handlers["PART"] = function(o, prefix, channel, reason)
2010-06-01 01:39:34 +02:00
local user = parsePrefix(prefix)
if o.track_users then
if user.nick == o.nick then
o.channels[channel] = nil
else
o.channels[channel].users[user.nick] = nil
end
end
o:invoke("OnPart", user, channel, reason)
end
handlers["QUIT"] = function(o, prefix, msg)
local user = parsePrefix(prefix)
if o.track_users then
for channel, v in pairs(o.channels) do
v.users[user.nick] = nil
end
end
o:invoke("OnQuit", user, msg)
end
handlers["NICK"] = function(o, prefix, newnick)
local user = parsePrefix(prefix)
if o.track_users then
for channel, v in pairs(o.channels) do
local users = v.users
local oldinfo = users[user.nick]
if oldinfo then
users[newnick] = oldinfo
users[user.nick] = nil
o:invoke("NickChange", user, newnick, channel)
end
end
else
o:invoke("NickChange", user, newnick)
end
end
2010-06-01 01:39:34 +02:00
--NAMES list
handlers["353"] = function(o, prefix, me, chanType, channel, names)
if o.track_users then
o.channels[channel] = o.channels[channel] or {users = {}, type = chanType}
local users = o.channels[channel].users
for nick in names:gmatch("(%S+)") do
2010-06-01 14:18:51 +02:00
local access, name = parseNick(nick)
users[name] = {type = access}
2010-06-01 01:39:34 +02:00
end
end
end
--end of NAMES
handlers["366"] = function(o, prefix, me, channel, msg)
if o.track_users then
o:invoke("NameList", channel, msg)
end
2010-02-21 15:59:29 +01:00
end
2010-07-17 04:00:58 +02:00
handlers["TOPIC"] = function(o, prefix, channel, topic)
o:invoke("TopicChange", channel, topic)
end
2010-02-21 15:59:29 +01:00
handlers["ERROR"] = function(o, prefix, message)
o:invoke("OnDisconnect", message, true)
o:shutdown()
2010-03-10 12:03:36 +01:00
error(message, 3)
2010-02-21 15:59:29 +01:00
end
function meta:handle(prefix, cmd, params)
local handler = handlers[cmd]
if handler then
2010-03-10 12:03:36 +01:00
handler(self, prefix, unpack(params))
2010-02-21 15:59:29 +01:00
end
end
local whoisHandlers = {
["311"] = "userinfo";
["312"] = "node";
["319"] = "channels";
["330"] = "account"; -- Freenode
["307"] = "registered"; -- Unreal
2010-02-21 15:59:29 +01:00
}
function meta:whois(nick)
self:send("WHOIS %s", nick)
local result = {}
while true do
local line = getline(self, 3)
if line then
local prefix, cmd, args = parse(line)
local handler = whoisHandlers[cmd]
if handler then
result[handler] = args
elseif cmd == "318" then
break
else
self:handle(prefix, cmd, args)
end
end
end
if result.account then
result.account = result.account[3]
2010-07-17 04:15:17 +02:00
elseif result.registered then
result.account = result.registered[2]
end
2010-02-21 15:59:29 +01:00
return result
end