Merge pull request #15 from clean-cloogle/no-results-response
[cloogle-irc.git] / IRC.icl
1 implementation module IRC
2
3 import StdList, StdTuple, StdOverloaded, StdFunc, StdString, StdChar, StdBool
4 import _SystemArray
5
6 import GenPrint
7 import GenIRC
8
9 import Control.Applicative
10 import Control.Monad
11 import Data.Either
12 import Data.Maybe
13 import Data.Tuple
14 import Text.Parsers.Simple.Chars
15 import Text.Parsers.Simple.Core
16
17 import StdDebug
18
19 from Data.Functor import <$>
20 from Data.Func import $
21 from StdMisc import undef, abort
22 from Text import class Text(lpad,trim,rtrim,split,indexOf,concat),
23 instance Text String
24 import qualified Text
25
26 jon :== 'Text'.join
27
28 derive gPrint IRCErrors, IRCReplies, Maybe, Either, IRCUser, IRCNumReply
29
30 Start = (map (fmap toString) msgs, msgs)
31 where
32 msgs =
33 [ parseIRCMessage ":clooglebot!~cloogle@dhcp-077-249-221-037.chello.nl QUIT\r\n"
34 , parseIRCMessage ":clooglebot!~cloogle QUIT\r\n"
35 , parseIRCMessage ":frobnicator!~frobnicat@92.110.128.124 QUIT\r\n"
36 , parseIRCMessage ":frobnicator!~frobnicat@92.110.128.124 AWAY test\r\n"
37 , parseIRCMessage ":frobnicator!~frobnicat@92.110.128.124 AWAY :test with spaces\r\n"
38 , parseIRCMessage ":cherryh.freenode.net NOTICE * :*** Found your hostname\r\n"
39 , parseIRCMessage ":cherryh.freenode.net QUIT :hoi hoi\r\n"
40 , parseIRCMessage ":cherryh.freenode.net JOIN #cha,#ch-b #twilight\r\n"
41 , parseIRCMessage ":cherryh.freenode.net ISON a b c d e f :g h\r\n"
42 , parseIRCMessage ":wilhelm.freenode.net 001 clooglebot :Welcome to the freenode Internet Relay Chat Network clooglebot\r\n"
43 , parseIRCMessage "PING :orwell.freenode.net\r\n"
44 , parseIRCMessage ":ChanServ!ChanServ@services. MODE #cloogle +o frobnicator\r\n"
45 ]
46
47 parseIRCMessage :: String -> Either [Error] IRCMessage
48 parseIRCMessage s = case runParser parsePrefix (fromString s) of
49 // Prefix is parsed
50 ([(prefix, rest):_], _)
51 //Try parsing a numeric reply
52 = case parse parseReply rest of
53 //Try a normal command
54 Left e = case parseCmd rest of
55 Left e2 = Left [e2:e]
56 Right cmd = Right {IRCMessage | irc_prefix=prefix, irc_command=Right cmd}
57 Right repl = Right {IRCMessage | irc_prefix=prefix, irc_command=Left repl}
58 // Error parsing prefix
59 (_, es) = Left ["Error parsing prefix"]
60
61 //Prefix
62 parsePrefix :: Parser Char (Maybe (Either IRCUser String))
63 parsePrefix
64 = optional (pToken ':' >>| parseEither parseUser parseHost <* pToken ' ')
65 where
66 parseEither :: (Parser a b) (Parser a c) -> Parser a (Either b c)
67 parseEither p q = Left <$> p <|> Right <$> q
68
69 parseUser :: Parser Char IRCUser
70 parseUser = parseNick
71 >>= \nick->optional (pToken '!' >>| parseUsr)
72 >>= \muser->optional (pToken '@' >>| parseHost)
73 >>= \mhost->pure {IRCUser
74 | irc_nick=nick, irc_user=muser, irc_host=mhost}
75
76 parseUsr :: Parser Char String
77 parseUsr = toString <$> pSome (pNoneOf [' ', '@':illegal])
78
79 parseNick :: Parser Char String
80 parseNick = pAlpha
81 >>= \c ->pMany (pAlpha <|> pDigit <|> pOneOf (fromString "-[]\\`^{}"))
82 >>= \cs->pure (toString [c:cs])
83
84 parseHost :: Parser Char String
85 parseHost = jon "." <$> (pSepBy parseName (pToken '.'))
86 >>= \s->optional (pToken '.') >>= pure o maybe s (\p->s+++toString s)
87 where
88 parseName :: Parser Char String
89 parseName = toString <$> pSome (pAlpha <|> pDigit <|> pOneOf ['-', '/'])
90
91 //Parse Cmd
92 parseCmd :: [Char] -> Either Error IRCCommand
93 parseCmd cs = fst $ gIRCParse{|*|} $ argfun $ split " " $ toString cs
94 where
95 argfun :: [String] -> [String]
96 argfun [] = []
97 argfun [x:xs]
98 # x = trim x
99 | x.[0] == ':' = [jon " " $ [x % (1, size x):map rtrim xs]]
100 | otherwise = [x:argfun xs]
101
102 //Reply
103 parseReply :: Parser Char IRCNumReply
104 parseReply = spaceParser
105 >>| (pMany (pToken '0') >>| pSome pDigit <* spaceParser)
106 >>= \rep->(toString <$> pSome (pNoneOf [' ':illegal]) <* spaceParser)
107 >>= \rec->(toString <$> pSome (pNoneOf illegal))
108 >>= \msg->pure {IRCNumReply
109 | irc_reply = fromInt $ toInt $ toString rep
110 , irc_recipient = rec
111 , irc_message = msg % (if (msg.[0] == ':') 1 0, size msg)
112 }
113 <* pToken '\r' <* pToken '\n'
114 where
115 spaceParser :: Parser Char [Char]
116 spaceParser = pMany $ pToken ' '
117
118 //Common parsers
119 pNoneOf :: [a] -> Parser a a | Eq a
120 pNoneOf l = pSatisfy (not o flip isMember l)
121
122 illegal :: [Char]
123 illegal = ['\x00','\r','\n']
124
125 instance toString IRCNumReply where
126 toString m = lpad (toString $ toInt m.irc_reply) 3 '0' <+ " " <+
127 m.irc_recipient <+ " " <+ concat (gIRCPrint{|*|} m.irc_message)
128 instance toString IRCMessage where
129 toString m = maybe "" (\s->either ((<+) ":") id s <+ " ") m.irc_prefix
130 <+ either toString toString m.irc_command
131 instance toString IRCUser where
132 toString m = m.irc_nick <+ maybe "" ((<+) "!") m.irc_user
133 <+ maybe "" ((<+) "@") m.irc_host
134 instance toString IRCCommand where
135 toString m = jon " " (gIRCPrint{|*|} m) +++ "\r\n"
136 instance toString IRCReplies where toString r = printToString r
137 instance toString IRCErrors where toString r = printToString r
138
139 (<+) infixr 5 :: a b -> String | toString a & toString b
140 (<+) a b = toString a +++ toString b
141
142 instance fromInt IRCReplies where
143 fromInt r = case r of
144 1 = RPL_WELCOME
145 2 = RPL_YOURHOST
146 3 = RPL_CREATED
147 4 = RPL_MYINFO
148 5 = RPL_BOUNCE
149 200 = RPL_TRACELINK
150 201 = RPL_TRACECONNECTING
151 202 = RPL_TRACEHANDSHAKE
152 203 = RPL_TRACEUNKNOWN
153 204 = RPL_TRACEOPERATOR
154 205 = RPL_TRACEUSER
155 206 = RPL_TRACESERVER
156 207 = RPL_TRACESERVICE
157 208 = RPL_TRACENEWTYPE
158 209 = RPL_TRACECLASS
159 210 = RPL_TRACERECONNECT
160 211 = RPL_STATSLINKINFO
161 212 = RPL_STATSCOMMANDS
162 219 = RPL_ENDOFSTATS
163 221 = RPL_UMODEIS
164 234 = RPL_SERVLIST
165 235 = RPL_SERVLISTEND
166 242 = RPL_STATSUPTIME
167 243 = RPL_STATSOLINE
168 251 = RPL_LUSERCLIENT
169 252 = RPL_LUSEROP
170 253 = RPL_LUSERUNKNOWN
171 254 = RPL_LUSERCHANNELS
172 255 = RPL_LUSERME
173 256 = RPL_ADMINME
174 257 = RPL_ADMINLOC1
175 258 = RPL_ADMINLOC2
176 259 = RPL_ADMINEMAIL
177 261 = RPL_TRACELOG
178 262 = RPL_TRACEEND
179 263 = RPL_TRYAGAIN
180 301 = RPL_AWAY
181 302 = RPL_USERHOST
182 303 = RPL_ISON
183 304 = RPL_UNAWAY
184 305 = RPL_NOWAWAY
185 311 = RPL_WHOISUSER
186 312 = RPL_WHOISSERVER
187 313 = RPL_WHOISOPERATOR
188 314 = RPL_WHOWASUSER
189 315 = RPL_ENDOFWHO
190 317 = RPL_WHOISIDLE
191 318 = RPL_ENDOFWHOIS
192 319 = RPL_WHOISCHANNELS
193 321 = RPL_LISTSTART
194 322 = RPL_LIST
195 323 = RPL_LISTEND
196 324 = RPL_CHANNELMODEIS
197 325 = RPL_UNIQOPIS
198 331 = RPL_NOTOPIC
199 332 = RPL_TOPIC
200 341 = RPL_INVITING
201 342 = RPL_SUMMONING
202 346 = RPL_INVITELIST
203 347 = RPL_ENDOFINVITELIST
204 348 = RPL_EXCEPTLIST
205 349 = RPL_ENDOFEXCEPTLIST
206 351 = RPL_VERSION
207 352 = RPL_WHOREPLY
208 353 = RPL_NAMREPLY
209 364 = RPL_LINKS
210 365 = RPL_ENDOFLINKS
211 366 = RPL_ENDOFNAMES
212 367 = RPL_BANLIST
213 368 = RPL_ENDOFBANLIST
214 369 = RPL_ENDOFWHOWAS
215 371 = RPL_INFO
216 372 = RPL_MOTD
217 374 = RPL_ENDOFINFO
218 375 = RPL_MOTDSTART
219 376 = RPL_ENDOFMOTD
220 381 = RPL_YOUREOPER
221 382 = RPL_REHASHING
222 383 = RPL_YOURESERVICE
223 391 = RPL_TIME
224 392 = RPL_USERSSTART
225 393 = RPL_USERS
226 394 = RPL_ENDOFUSERS
227 395 = RPL_NOUSERS
228 _ = RPL_UNKNOWN
229
230 instance toInt IRCReplies where
231 toInt r = case r of
232 RPL_WELCOME = 1
233 RPL_YOURHOST = 2
234 RPL_CREATED = 3
235 RPL_MYINFO = 4
236 RPL_BOUNCE = 5
237 RPL_TRACELINK = 200
238 RPL_TRACECONNECTING = 201
239 RPL_TRACEHANDSHAKE = 202
240 RPL_TRACEUNKNOWN = 203
241 RPL_TRACEOPERATOR = 204
242 RPL_TRACEUSER = 205
243 RPL_TRACESERVER = 206
244 RPL_TRACESERVICE = 207
245 RPL_TRACENEWTYPE = 208
246 RPL_TRACECLASS = 209
247 RPL_TRACERECONNECT = 210
248 RPL_STATSLINKINFO = 211
249 RPL_STATSCOMMANDS = 212
250 RPL_ENDOFSTATS = 219
251 RPL_UMODEIS = 221
252 RPL_SERVLIST = 234
253 RPL_SERVLISTEND = 234
254 RPL_STATSUPTIME = 242
255 RPL_STATSOLINE = 243
256 RPL_LUSERCLIENT = 251
257 RPL_LUSEROP = 252
258 RPL_LUSERUNKNOWN = 253
259 RPL_LUSERCHANNELS = 254
260 RPL_LUSERME = 255
261 RPL_ADMINME = 256
262 RPL_ADMINLOC1 = 257
263 RPL_ADMINLOC2 = 258
264 RPL_ADMINEMAIL = 259
265 RPL_TRACELOG = 261
266 RPL_TRACEEND = 262
267 RPL_TRYAGAIN = 263
268 RPL_AWAY = 301
269 RPL_USERHOST = 302
270 RPL_ISON = 303
271 RPL_UNAWAY = 304
272 RPL_NOWAWAY = 305
273 RPL_WHOISUSER = 311
274 RPL_WHOISSERVER = 312
275 RPL_WHOISOPERATOR = 313
276 RPL_WHOWASUSER = 314
277 RPL_ENDOFWHO = 315
278 RPL_WHOISIDLE = 317
279 RPL_ENDOFWHOIS = 318
280 RPL_WHOISCHANNELS = 319
281 RPL_LISTSTART = 321
282 RPL_LIST = 322
283 RPL_LISTEND = 323
284 RPL_CHANNELMODEIS = 324
285 RPL_UNIQOPIS = 325
286 RPL_NOTOPIC = 331
287 RPL_TOPIC = 332
288 RPL_INVITING = 341
289 RPL_SUMMONING = 342
290 RPL_INVITELIST = 346
291 RPL_ENDOFINVITELIST = 347
292 RPL_EXCEPTLIST = 348
293 RPL_ENDOFEXCEPTLIST = 349
294 RPL_VERSION = 351
295 RPL_WHOREPLY = 352
296 RPL_NAMREPLY = 353
297 RPL_LINKS = 364
298 RPL_ENDOFLINKS = 365
299 RPL_ENDOFNAMES = 366
300 RPL_BANLIST = 367
301 RPL_ENDOFBANLIST = 367
302 RPL_ENDOFWHOWAS = 369
303 RPL_INFO = 371
304 RPL_MOTD = 372
305 RPL_ENDOFINFO = 374
306 RPL_MOTDSTART = 375
307 RPL_ENDOFMOTD = 376
308 RPL_YOUREOPER = 381
309 RPL_REHASHING = 382
310 RPL_YOURESERVICE = 383
311 RPL_TIME = 391
312 RPL_USERSSTART = 392
313 RPL_USERS = 393
314 RPL_ENDOFUSERS = 394
315 RPL_NOUSERS = 395
316 RPL_UNKNOWN = 998
317
318 instance fromInt IRCErrors where
319 fromInt r = case r of
320 401 = ERR_NOSUCHNICK
321 402 = ERR_NOSUCHSERVER
322 403 = ERR_NOSUCHCHANNEL
323 404 = ERR_CANNOTSENDTOCHAN
324 405 = ERR_TOOMANYCHANNELS
325 406 = ERR_WASNOSUCHNICK
326 407 = ERR_TOOMANYTARGETS
327 408 = ERR_NOSUCHSERVICE
328 409 = ERR_NOORIGIN
329 411 = ERR_NORECIPIENT
330 412 = ERR_NOTEXTTOSEND
331 413 = ERR_NOTOPLEVEL
332 414 = ERR_WILDTOPLEVEL
333 415 = ERR_BADMASK
334 421 = ERR_UNKNOWNCOMMAND
335 422 = ERR_NOMOTD
336 423 = ERR_NOADMININFO
337 424 = ERR_FILEERROR
338 431 = ERR_NONICKNAMEGIVEN
339 432 = ERR_ERRONEUSNICKNAME
340 433 = ERR_NICKNAMEINUSE
341 436 = ERR_NICKCOLLISION
342 437 = ERR_UNAVAILRESOURCE
343 441 = ERR_USERNOTINCHANNEL
344 442 = ERR_NOTONCHANNEL
345 443 = ERR_USERONCHANNEL
346 444 = ERR_NOLOGIN
347 445 = ERR_SUMMONDISABLED
348 446 = ERR_USERSDISABLED
349 451 = ERR_NOTREGISTERED
350 461 = ERR_NEEDMOREPARAMS
351 462 = ERR_ALREADYREGISTRED
352 463 = ERR_NOPERMFORHOST
353 464 = ERR_PASSWDMISMATCH
354 465 = ERR_YOUREBANNEDCREEP
355 466 = ERR_YOUWILLBEBANNED
356 467 = ERR_KEYSET
357 471 = ERR_CHANNELISFULL
358 472 = ERR_UNKNOWNMODE
359 473 = ERR_INVITEONLYCHAN
360 474 = ERR_BANNEDFROMCHAN
361 475 = ERR_BADCHANNELKEY
362 476 = ERR_BADCHANMASK
363 477 = ERR_NOCHANMODES
364 478 = ERR_BANLISTFULL
365 481 = ERR_NOPRIVILEGES
366 482 = ERR_CHANOPRIVSNEEDED
367 483 = ERR_CANTKILLSERVER
368 484 = ERR_RESTRICTED
369 485 = ERR_UNIQOPPRIVSNEEDED
370 491 = ERR_NOOPERHOST
371 501 = ERR_UMODEUNKNOWNFLAG
372 502 = ERR_USERSDONTMATCH
373 _ = ERR_UNKNOWN
374
375 instance toInt IRCErrors where
376 toInt r = case r of
377 ERR_NOSUCHNICK = 401
378 ERR_NOSUCHSERVER = 402
379 ERR_NOSUCHCHANNEL = 403
380 ERR_CANNOTSENDTOCHAN = 404
381 ERR_TOOMANYCHANNELS = 405
382 ERR_WASNOSUCHNICK = 406
383 ERR_TOOMANYTARGETS = 407
384 ERR_NOSUCHSERVICE = 408
385 ERR_NOORIGIN = 409
386 ERR_NORECIPIENT = 411
387 ERR_NOTEXTTOSEND = 412
388 ERR_NOTOPLEVEL = 413
389 ERR_WILDTOPLEVEL = 414
390 ERR_BADMASK = 415
391 ERR_UNKNOWNCOMMAND = 421
392 ERR_NOMOTD = 422
393 ERR_NOADMININFO = 423
394 ERR_FILEERROR = 424
395 ERR_NONICKNAMEGIVEN = 431
396 ERR_ERRONEUSNICKNAME = 432
397 ERR_NICKNAMEINUSE = 433
398 ERR_NICKCOLLISION = 436
399 ERR_UNAVAILRESOURCE = 437
400 ERR_USERNOTINCHANNEL = 441
401 ERR_NOTONCHANNEL = 442
402 ERR_USERONCHANNEL = 443
403 ERR_NOLOGIN = 444
404 ERR_SUMMONDISABLED = 445
405 ERR_USERSDISABLED = 446
406 ERR_NOTREGISTERED = 451
407 ERR_NEEDMOREPARAMS = 461
408 ERR_ALREADYREGISTRED = 462
409 ERR_NOPERMFORHOST = 463
410 ERR_PASSWDMISMATCH = 464
411 ERR_YOUREBANNEDCREEP = 465
412 ERR_YOUWILLBEBANNED = 466
413 ERR_KEYSET = 467
414 ERR_CHANNELISFULL = 471
415 ERR_UNKNOWNMODE = 472
416 ERR_INVITEONLYCHAN = 473
417 ERR_BANNEDFROMCHAN = 474
418 ERR_BADCHANNELKEY = 475
419 ERR_BADCHANMASK = 476
420 ERR_NOCHANMODES = 477
421 ERR_BANLISTFULL = 478
422 ERR_NOPRIVILEGES = 481
423 ERR_CHANOPRIVSNEEDED = 482
424 ERR_CANTKILLSERVER = 483
425 ERR_RESTRICTED = 484
426 ERR_UNIQOPPRIVSNEEDED = 485
427 ERR_NOOPERHOST = 491
428 ERR_UMODEUNKNOWNFLAG = 501
429 ERR_USERSDONTMATCH = 502
430 ERR_UNKNOWN = 999