1
0
mirror of https://github.com/ShadowNinja/LuaIRC.git synced 2025-01-25 01:10:20 +01:00
LuaIRC/messages.lua

204 lines
4.3 KiB
Lua
Raw Normal View History

2014-06-03 17:13:12 +02:00
-- Module table
local m = {}
local msg_meta = {}
msg_meta.__index = msg_meta
2014-06-03 17:13:12 +02:00
local function Message(opts)
2014-05-31 06:52:54 +02:00
opts = opts or {}
setmetatable(opts, msg_meta)
if opts.raw then
opts:fromRFC1459(opts.raw)
end
return opts
end
2014-06-03 17:13:12 +02:00
m.Message = Message
2014-05-31 04:46:37 +02:00
local tag_escapes = {
[";"] = "\\:",
[" "] = "\\s",
["\0"] = "\\0",
["\\"] = "\\\\",
["\r"] = "\\r",
["\n"] = "\\n",
}
local tag_unescapes = {}
for x, y in pairs(tag_escapes) do tag_unescapes[y] = x end
function msg_meta:toRFC1459()
2014-06-03 17:13:12 +02:00
local s = ""
2014-03-18 09:36:02 +01:00
if self.tags then
s = s.."@"
for key, value in pairs(self.tags) do
s = s..key
if value ~= true then
2014-05-31 04:46:37 +02:00
value = value:gsub("[; %z\\\r\n]", tag_escapes)
2014-03-18 09:36:02 +01:00
s = s.."="..value
end
s = s..";"
end
-- Strip trailing semicolon
s = s:sub(1, -2)
s = s.." "
end
s = s..self.command
2014-06-03 17:13:12 +02:00
local argnum = #self.args
for i = 1, argnum do
local arg = self.args[i]
local startsWithColon = (arg:sub(1, 1) == ":")
local hasSpace = arg:find(" ")
if i == argnum and (hasSpace or startsWithColon) then
s = s.." :"
else
assert(not hasSpace and not startsWithColon,
"Message arguments can not be "
.."serialized to RFC1459 format")
s = s.." "
end
s = s..arg
end
2014-03-18 09:36:02 +01:00
return s
end
2014-05-31 04:46:37 +02:00
local function parsePrefix(prefix)
local user = {}
user.nick, user.username, user.host = prefix:match("^(.+)!(.+)@(.+)$")
if not user.nick and prefix:find(".", 1, true) then
user.server = prefix
end
return user
end
function msg_meta:fromRFC1459(line)
-- IRCv3 tags
if line:sub(1, 1) == "@" then
self.tags = {}
local space = line:find(" ", 1, true)
-- For each semicolon-delimited section from after
-- the @ character to before the space character.
for tag in line:sub(2, space - 1):gmatch("([^;]+)") do
local eq = tag:find("=", 1, true)
if eq then
self.tags[tag:sub(1, eq - 1)] =
tag:sub(eq + 1):gsub("\\([:s0\\rn])", tag_unescapes)
else
self.tags[tag] = true
end
end
line = line:sub(space + 1)
end
if line:sub(1, 1) == ":" then
local space = line:find(" ", 1, true)
self.prefix = line:sub(2, space - 1)
self.user = parsePrefix(self.prefix)
line = line:sub(space + 1)
end
local pos
self.command, pos = line:match("(%S+)()")
line = line:sub(pos)
2014-05-31 06:52:54 +02:00
self.args = self.args or {}
2014-05-31 04:46:37 +02:00
for pos, param in line:gmatch("()(%S+)") do
if param:sub(1, 1) == ":" then
param = line:sub(pos + 1)
2014-06-03 17:13:12 +02:00
table.insert(self.args, param)
2014-05-31 04:46:37 +02:00
break
end
2014-06-03 17:13:12 +02:00
table.insert(self.args, param)
2014-05-31 04:46:37 +02:00
end
end
2014-06-03 17:13:12 +02:00
function m.privmsg(to, text)
2014-05-31 06:52:54 +02:00
return Message({command="PRIVMSG", args={to, text}})
end
2014-06-03 17:13:12 +02:00
function m.notice(to, text)
2014-05-31 06:52:54 +02:00
return Message({command="NOTICE", args={to, text}})
end
2014-06-03 17:13:12 +02:00
function m.action(to, text)
2014-05-31 06:52:54 +02:00
return Message({command="PRIVMSG", args={to, ("\x01ACTION %s\x01"):format(text)}})
end
2014-06-03 17:13:12 +02:00
function m.ctcp(command, to, args)
s = "\x01"..command
if args then
s = ' '..args
end
s = s..'\x01'
2014-05-31 06:52:54 +02:00
return Message({command="PRIVMSG", args={to, s}})
end
2014-06-03 17:13:12 +02:00
function m.kick(channel, target, reason)
2014-05-31 06:52:54 +02:00
return Message({command="KICK", args={channel, target, reason}})
end
2014-06-03 17:13:12 +02:00
function m.join(channel, key)
2014-05-31 06:52:54 +02:00
return Message({command="JOIN", args={channel, key}})
end
2014-06-03 17:13:12 +02:00
function m.part(channel, reason)
2014-05-31 06:52:54 +02:00
return Message({command="PART", args={channel, reason}})
end
2014-06-03 17:13:12 +02:00
function m.quit(reason)
2014-05-31 06:52:54 +02:00
return Message({command="QUIT", args={reason}})
end
2014-06-03 17:13:12 +02:00
function m.kill(target, reason)
2014-05-31 06:52:54 +02:00
return Message({command="KILL", args={target, reason}})
end
2014-06-03 17:13:12 +02:00
function m.kline(time, mask, reason, operreason)
local args = nil
if time then
args = {time, mask, reason..'|'..operreason}
else
args = {mask, reason..'|'..operreason}
end
2014-05-31 06:52:54 +02:00
return Message({command="KLINE", args=args})
end
2014-06-03 17:13:12 +02:00
function m.whois(nick, server)
local args = nil
if server then
args = {server, nick}
else
args = {nick}
end
2014-05-31 06:52:54 +02:00
return Message({command="WHOIS", args=args})
end
2014-06-03 17:13:12 +02:00
function m.topic(channel, text)
2014-05-31 06:52:54 +02:00
return Message({command="TOPIC", args={channel, text}})
end
2014-06-03 17:13:12 +02:00
function m.invite(channel, target)
2014-05-31 06:52:54 +02:00
return Message({command="INVITE", args={channel, target}})
end
2014-06-03 17:13:12 +02:00
function m.nick(nick)
2014-05-31 06:52:54 +02:00
return Message({command="NICK", args={nick}})
end
2014-06-03 17:13:12 +02:00
function m.mode(target, modes)
-- We have to split the modes parameter because the mode string and
-- each parameter are seperate arguments (The first command is incorrect)
2014-05-31 04:46:37 +02:00
-- MODE foo :+ov Nick1 Nick2
-- MODE foo +ov Nick1 Nick2
2014-06-03 17:13:12 +02:00
local mt = util.split(modes)
2014-05-31 06:52:54 +02:00
return Message({command="MODE", args={target, unpack(mt)}})
end
2014-06-03 17:13:12 +02:00
return m