3 changed files with 63 additions and 9 deletions
@ -0,0 +1,54 @@ |
|||
{-# LANGUAGE OverloadedStrings #-} |
|||
module IRCParser where |
|||
|
|||
import Data.ByteString (pack) |
|||
import Data.Attoparsec.ByteString |
|||
import Control.Applicative |
|||
|
|||
-- :nisstyre!wes@oftn/oswg-member/Nisstyre PRIVMSG #thisisatestwhatever :yay\r\n |
|||
-- :nisstyre!wes@oftn/oswg-member/Nisstyre PART #thisisatestwhatever :\"WeeChat 2.2\"\r\n |
|||
-- :nisstyre!wes@oftn/oswg-member/Nisstyre JOIN #thisisatestwhatever\r\n |
|||
-- :nisstyre!wes@oftn/oswg-member/Nisstyre QUIT :Quit: WeeChat 2.2\r\n |
|||
-- PING :asimov.freenode.net\r\n |
|||
|
|||
space = word8 32 |
|||
notSpace = notWord8 32 |
|||
|
|||
-- 13 = \r, 10 = \n |
|||
crlf c = c == 13 || c == 10 |
|||
|
|||
-- : |
|||
colon = word8 58 |
|||
|
|||
command = choice $ map string ["PRIVMSG", "PART", "JOIN", "QUIT", "PING"] |
|||
|
|||
argument = takeWhile1 (\c -> c /= 58 && c /= 32) |
|||
|
|||
parseFinalArg = do |
|||
_ <- colon |
|||
command <- takeWhile1 (not . crlf) |
|||
return command |
|||
|
|||
parseArgs = do |
|||
args <- (argument `sepBy` space) <|> alwaysMatch [] |
|||
_ <- space <|> (alwaysMatch 0) |
|||
finalArg <- parseFinalArg <|> (alwaysMatch "") |
|||
return (args, finalArg) |
|||
|
|||
source = do |
|||
_ <- colon |
|||
source <- manyTill notSpace space |
|||
return source |
|||
|
|||
-- Matches any character and does not consume input |
|||
-- Returns value passed in |
|||
alwaysMatch x = (maybe x (const x)) <$> peekWord8 |
|||
|
|||
parseMessage = do |
|||
sourceName <- source <|> (alwaysMatch []) |
|||
commandName <- command |
|||
_ <- space |
|||
arguments <- parseArgs |
|||
return (pack sourceName, commandName, arguments) |
|||
|
|||
runParseMessage = parseOnly parseMessage |
Loading…
Reference in new issue