{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE OverloadedStrings #-}
module Data.Ini
(
readIniFile
,parseIni
,lookupValue
,lookupArray
,readValue
,readArray
,parseValue
,sections
,keys
,printIni
,writeIniFile
,KeySeparator(..)
,WriteIniSettings(..)
,defaultWriteIniSettings
,printIniWith
,writeIniFileWith
,Ini(..)
,unIni
,iniParser
,sectionParser
,keyValueParser
)
where
import Control.Applicative
import Control.Monad
import Data.Attoparsec.Combinator
import Data.Attoparsec.Text
import Data.Char
import Data.HashMap.Strict (HashMap)
import qualified Data.HashMap.Strict as M
import Data.Maybe
import Data.Semigroup
import Data.Text (Text)
import qualified Data.Text as T
import qualified Data.Text.IO as T
import Prelude hiding (takeWhile)
data Ini =
Ini
{ Ini -> HashMap Text [(Text, Text)]
iniSections :: HashMap Text [(Text, Text)]
, Ini -> [(Text, Text)]
iniGlobals :: [(Text, Text)]
}
deriving (Int -> Ini -> ShowS
[Ini] -> ShowS
Ini -> String
(Int -> Ini -> ShowS)
-> (Ini -> String) -> ([Ini] -> ShowS) -> Show Ini
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Ini] -> ShowS
$cshowList :: [Ini] -> ShowS
show :: Ini -> String
$cshow :: Ini -> String
showsPrec :: Int -> Ini -> ShowS
$cshowsPrec :: Int -> Ini -> ShowS
Show, Ini -> Ini -> Bool
(Ini -> Ini -> Bool) -> (Ini -> Ini -> Bool) -> Eq Ini
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: Ini -> Ini -> Bool
$c/= :: Ini -> Ini -> Bool
== :: Ini -> Ini -> Bool
$c== :: Ini -> Ini -> Bool
Eq)
instance Semigroup Ini where
Ini
x <> :: Ini -> Ini -> Ini
<> Ini
y = Ini :: HashMap Text [(Text, Text)] -> [(Text, Text)] -> Ini
Ini {iniGlobals :: [(Text, Text)]
iniGlobals = [(Text, Text)]
forall a. Monoid a => a
mempty, iniSections :: HashMap Text [(Text, Text)]
iniSections = Ini -> HashMap Text [(Text, Text)]
iniSections Ini
x HashMap Text [(Text, Text)]
-> HashMap Text [(Text, Text)] -> HashMap Text [(Text, Text)]
forall a. Semigroup a => a -> a -> a
<> Ini -> HashMap Text [(Text, Text)]
iniSections Ini
y}
instance Monoid Ini where
mempty :: Ini
mempty = Ini :: HashMap Text [(Text, Text)] -> [(Text, Text)] -> Ini
Ini {iniGlobals :: [(Text, Text)]
iniGlobals = [(Text, Text)]
forall a. Monoid a => a
mempty, iniSections :: HashMap Text [(Text, Text)]
iniSections = HashMap Text [(Text, Text)]
forall a. Monoid a => a
mempty}
mappend :: Ini -> Ini -> Ini
mappend = Ini -> Ini -> Ini
forall a. Semigroup a => a -> a -> a
(<>)
{-# DEPRECATED #-}
unIni :: Ini -> HashMap Text (HashMap Text Text)
unIni :: Ini -> HashMap Text (HashMap Text Text)
unIni = ([(Text, Text)] -> HashMap Text Text)
-> HashMap Text [(Text, Text)] -> HashMap Text (HashMap Text Text)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap [(Text, Text)] -> HashMap Text Text
forall k v. (Eq k, Hashable k) => [(k, v)] -> HashMap k v
M.fromList (HashMap Text [(Text, Text)] -> HashMap Text (HashMap Text Text))
-> (Ini -> HashMap Text [(Text, Text)])
-> Ini
-> HashMap Text (HashMap Text Text)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Ini -> HashMap Text [(Text, Text)]
iniSections
readIniFile :: FilePath -> IO (Either String Ini)
readIniFile :: String -> IO (Either String Ini)
readIniFile = (Text -> Either String Ini) -> IO Text -> IO (Either String Ini)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap Text -> Either String Ini
parseIni (IO Text -> IO (Either String Ini))
-> (String -> IO Text) -> String -> IO (Either String Ini)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> IO Text
T.readFile
parseIni :: Text -> Either String Ini
parseIni :: Text -> Either String Ini
parseIni = Parser Ini -> Text -> Either String Ini
forall a. Parser a -> Text -> Either String a
parseOnly Parser Ini
iniParser
lookupValue :: Text
-> Text
-> Ini -> Either String Text
lookupValue :: Text -> Text -> Ini -> Either String Text
lookupValue Text
name Text
key (Ini {iniSections :: Ini -> HashMap Text [(Text, Text)]
iniSections=HashMap Text [(Text, Text)]
secs}) =
case Text -> HashMap Text [(Text, Text)] -> Maybe [(Text, Text)]
forall k v. (Eq k, Hashable k) => k -> HashMap k v -> Maybe v
M.lookup Text
name HashMap Text [(Text, Text)]
secs of
Maybe [(Text, Text)]
Nothing -> String -> Either String Text
forall a b. a -> Either a b
Left (String
"Couldn't find section: " String -> ShowS
forall a. [a] -> [a] -> [a]
++ Text -> String
T.unpack Text
name)
Just [(Text, Text)]
section ->
case Text -> [(Text, Text)] -> Maybe Text
forall a b. Eq a => a -> [(a, b)] -> Maybe b
lookup Text
key [(Text, Text)]
section of
Maybe Text
Nothing -> String -> Either String Text
forall a b. a -> Either a b
Left (String
"Couldn't find key: " String -> ShowS
forall a. [a] -> [a] -> [a]
++ Text -> String
T.unpack Text
key)
Just Text
value -> Text -> Either String Text
forall (m :: * -> *) a. Monad m => a -> m a
return Text
value
lookupArray :: Text
-> Text
-> Ini -> Either String [Text]
lookupArray :: Text -> Text -> Ini -> Either String [Text]
lookupArray Text
name Text
key (Ini {iniSections :: Ini -> HashMap Text [(Text, Text)]
iniSections = HashMap Text [(Text, Text)]
secs}) =
case Text -> HashMap Text [(Text, Text)] -> Maybe [(Text, Text)]
forall k v. (Eq k, Hashable k) => k -> HashMap k v -> Maybe v
M.lookup Text
name HashMap Text [(Text, Text)]
secs of
Maybe [(Text, Text)]
Nothing -> String -> Either String [Text]
forall a b. a -> Either a b
Left (String
"Couldn't find section: " String -> ShowS
forall a. [a] -> [a] -> [a]
++ Text -> String
T.unpack Text
name)
Just [(Text, Text)]
section ->
case ((Text, Text) -> Maybe Text) -> [(Text, Text)] -> [Text]
forall a b. (a -> Maybe b) -> [a] -> [b]
mapMaybe
(\(Text
k, Text
v) ->
if Text
k Text -> Text -> Bool
forall a. Eq a => a -> a -> Bool
== Text
key
then Text -> Maybe Text
forall a. a -> Maybe a
Just Text
v
else Maybe Text
forall a. Maybe a
Nothing)
[(Text, Text)]
section of
[] -> String -> Either String [Text]
forall a b. a -> Either a b
Left (String
"Couldn't find key: " String -> ShowS
forall a. [a] -> [a] -> [a]
++ Text -> String
T.unpack Text
key)
[Text]
values -> [Text] -> Either String [Text]
forall (m :: * -> *) a. Monad m => a -> m a
return [Text]
values
sections :: Ini -> [Text]
sections :: Ini -> [Text]
sections = HashMap Text [(Text, Text)] -> [Text]
forall k v. HashMap k v -> [k]
M.keys (HashMap Text [(Text, Text)] -> [Text])
-> (Ini -> HashMap Text [(Text, Text)]) -> Ini -> [Text]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Ini -> HashMap Text [(Text, Text)]
iniSections
keys :: Text
-> Ini -> Either String [Text]
keys :: Text -> Ini -> Either String [Text]
keys Text
name Ini
i =
case Text -> HashMap Text [(Text, Text)] -> Maybe [(Text, Text)]
forall k v. (Eq k, Hashable k) => k -> HashMap k v -> Maybe v
M.lookup Text
name (Ini -> HashMap Text [(Text, Text)]
iniSections Ini
i) of
Maybe [(Text, Text)]
Nothing -> String -> Either String [Text]
forall a b. a -> Either a b
Left (String
"Couldn't find section: " String -> ShowS
forall a. [a] -> [a] -> [a]
++ Text -> String
T.unpack Text
name)
Just [(Text, Text)]
section -> [Text] -> Either String [Text]
forall a b. b -> Either a b
Right (((Text, Text) -> Text) -> [(Text, Text)] -> [Text]
forall a b. (a -> b) -> [a] -> [b]
map (Text, Text) -> Text
forall a b. (a, b) -> a
fst [(Text, Text)]
section)
readValue :: Text
-> Text
-> (Text -> Either String (a, Text))
-> Ini
-> Either String a
readValue :: Text
-> Text
-> (Text -> Either String (a, Text))
-> Ini
-> Either String a
readValue Text
section Text
key Text -> Either String (a, Text)
f Ini
ini =
Text -> Text -> Ini -> Either String Text
lookupValue Text
section Text
key Ini
ini Either String Text
-> (Text -> Either String (a, Text)) -> Either String (a, Text)
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= Text -> Either String (a, Text)
f Either String (a, Text)
-> ((a, Text) -> Either String a) -> Either String a
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= a -> Either String a
forall (m :: * -> *) a. Monad m => a -> m a
return (a -> Either String a)
-> ((a, Text) -> a) -> (a, Text) -> Either String a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (a, Text) -> a
forall a b. (a, b) -> a
fst
readArray :: Text
-> Text
-> (Text -> Either String (a, Text))
-> Ini
-> Either String [a]
readArray :: Text
-> Text
-> (Text -> Either String (a, Text))
-> Ini
-> Either String [a]
readArray Text
section Text
key Text -> Either String (a, Text)
f Ini
ini =
([(a, Text)] -> [a])
-> Either String [(a, Text)] -> Either String [a]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (((a, Text) -> a) -> [(a, Text)] -> [a]
forall a b. (a -> b) -> [a] -> [b]
map (a, Text) -> a
forall a b. (a, b) -> a
fst) (Text -> Text -> Ini -> Either String [Text]
lookupArray Text
section Text
key Ini
ini Either String [Text]
-> ([Text] -> Either String [(a, Text)])
-> Either String [(a, Text)]
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= (Text -> Either String (a, Text))
-> [Text] -> Either String [(a, Text)]
forall (t :: * -> *) (m :: * -> *) a b.
(Traversable t, Monad m) =>
(a -> m b) -> t a -> m (t b)
mapM Text -> Either String (a, Text)
f)
parseValue :: Text
-> Text
-> Parser a
-> Ini
-> Either String a
parseValue :: Text -> Text -> Parser a -> Ini -> Either String a
parseValue Text
section Text
key Parser a
f Ini
ini =
Text -> Text -> Ini -> Either String Text
lookupValue Text
section Text
key Ini
ini Either String Text -> (Text -> Either String a) -> Either String a
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= Parser a -> Text -> Either String a
forall a. Parser a -> Text -> Either String a
parseOnly (Parser a
f Parser a -> Parser Text () -> Parser a
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a
<* (Parser Text ()
skipSpace Parser Text () -> Parser Text () -> Parser Text ()
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> Parser Text ()
forall t. Chunk t => Parser t ()
endOfInput))
writeIniFile :: FilePath -> Ini -> IO ()
writeIniFile :: String -> Ini -> IO ()
writeIniFile = WriteIniSettings -> String -> Ini -> IO ()
writeIniFileWith WriteIniSettings
defaultWriteIniSettings
printIni :: Ini -> Text
printIni :: Ini -> Text
printIni = WriteIniSettings -> Ini -> Text
printIniWith WriteIniSettings
defaultWriteIniSettings
data KeySeparator
= ColonKeySeparator
| EqualsKeySeparator
deriving (KeySeparator -> KeySeparator -> Bool
(KeySeparator -> KeySeparator -> Bool)
-> (KeySeparator -> KeySeparator -> Bool) -> Eq KeySeparator
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: KeySeparator -> KeySeparator -> Bool
$c/= :: KeySeparator -> KeySeparator -> Bool
== :: KeySeparator -> KeySeparator -> Bool
$c== :: KeySeparator -> KeySeparator -> Bool
Eq, Int -> KeySeparator -> ShowS
[KeySeparator] -> ShowS
KeySeparator -> String
(Int -> KeySeparator -> ShowS)
-> (KeySeparator -> String)
-> ([KeySeparator] -> ShowS)
-> Show KeySeparator
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [KeySeparator] -> ShowS
$cshowList :: [KeySeparator] -> ShowS
show :: KeySeparator -> String
$cshow :: KeySeparator -> String
showsPrec :: Int -> KeySeparator -> ShowS
$cshowsPrec :: Int -> KeySeparator -> ShowS
Show)
data WriteIniSettings = WriteIniSettings
{ WriteIniSettings -> KeySeparator
writeIniKeySeparator :: KeySeparator
} deriving (Int -> WriteIniSettings -> ShowS
[WriteIniSettings] -> ShowS
WriteIniSettings -> String
(Int -> WriteIniSettings -> ShowS)
-> (WriteIniSettings -> String)
-> ([WriteIniSettings] -> ShowS)
-> Show WriteIniSettings
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [WriteIniSettings] -> ShowS
$cshowList :: [WriteIniSettings] -> ShowS
show :: WriteIniSettings -> String
$cshow :: WriteIniSettings -> String
showsPrec :: Int -> WriteIniSettings -> ShowS
$cshowsPrec :: Int -> WriteIniSettings -> ShowS
Show)
defaultWriteIniSettings :: WriteIniSettings
defaultWriteIniSettings :: WriteIniSettings
defaultWriteIniSettings = WriteIniSettings :: KeySeparator -> WriteIniSettings
WriteIniSettings
{ writeIniKeySeparator :: KeySeparator
writeIniKeySeparator = KeySeparator
ColonKeySeparator
}
writeIniFileWith :: WriteIniSettings -> FilePath -> Ini -> IO ()
writeIniFileWith :: WriteIniSettings -> String -> Ini -> IO ()
writeIniFileWith WriteIniSettings
wis String
fp = String -> Text -> IO ()
T.writeFile String
fp (Text -> IO ()) -> (Ini -> Text) -> Ini -> IO ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. WriteIniSettings -> Ini -> Text
printIniWith WriteIniSettings
wis
printIniWith :: WriteIniSettings -> Ini -> Text
printIniWith :: WriteIniSettings -> Ini -> Text
printIniWith WriteIniSettings
wis Ini
i =
[Text] -> Text
T.concat ([Text] -> Text) -> [Text] -> Text
forall a b. (a -> b) -> a -> b
$ (((Text, Text) -> Text) -> [(Text, Text)] -> [Text]
forall a b. (a -> b) -> [a] -> [b]
map (Text, Text) -> Text
buildPair (Ini -> [(Text, Text)]
iniGlobals Ini
i)) [Text] -> [Text] -> [Text]
forall a. [a] -> [a] -> [a]
++
(((Text, [(Text, Text)]) -> Text)
-> [(Text, [(Text, Text)])] -> [Text]
forall a b. (a -> b) -> [a] -> [b]
map (Text, [(Text, Text)]) -> Text
buildSection (HashMap Text [(Text, Text)] -> [(Text, [(Text, Text)])]
forall k v. HashMap k v -> [(k, v)]
M.toList (Ini -> HashMap Text [(Text, Text)]
iniSections Ini
i)))
where buildSection :: (Text, [(Text, Text)]) -> Text
buildSection (Text
name,[(Text, Text)]
pairs) =
Text
"[" Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
name Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
"]\n" Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<>
[Text] -> Text
T.concat (((Text, Text) -> Text) -> [(Text, Text)] -> [Text]
forall a b. (a -> b) -> [a] -> [b]
map (Text, Text) -> Text
buildPair [(Text, Text)]
pairs)
buildPair :: (Text, Text) -> Text
buildPair (Text
name,Text
value) =
Text
name Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
separator Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
value Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<> Text
"\n"
separator :: Text
separator = case WriteIniSettings -> KeySeparator
writeIniKeySeparator WriteIniSettings
wis of
KeySeparator
ColonKeySeparator -> Text
": "
KeySeparator
EqualsKeySeparator -> Text
"="
iniParser :: Parser Ini
iniParser :: Parser Ini
iniParser =
(\[(Text, Text)]
kv [(Text, [(Text, Text)])]
secs -> Ini :: HashMap Text [(Text, Text)] -> [(Text, Text)] -> Ini
Ini {iniSections :: HashMap Text [(Text, Text)]
iniSections = [(Text, [(Text, Text)])] -> HashMap Text [(Text, Text)]
forall k v. (Eq k, Hashable k) => [(k, v)] -> HashMap k v
M.fromList [(Text, [(Text, Text)])]
secs, iniGlobals :: [(Text, Text)]
iniGlobals = [(Text, Text)]
kv}) ([(Text, Text)] -> [(Text, [(Text, Text)])] -> Ini)
-> Parser Text [(Text, Text)]
-> Parser Text ([(Text, [(Text, Text)])] -> Ini)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$>
Parser Text (Text, Text) -> Parser Text [(Text, Text)]
forall (f :: * -> *) a. Alternative f => f a -> f [a]
many Parser Text (Text, Text)
keyValueParser Parser Text ([(Text, [(Text, Text)])] -> Ini)
-> Parser Text [(Text, [(Text, Text)])] -> Parser Ini
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*>
Parser Text (Text, [(Text, Text)])
-> Parser Text [(Text, [(Text, Text)])]
forall (f :: * -> *) a. Alternative f => f a -> f [a]
many Parser Text (Text, [(Text, Text)])
sectionParser Parser Ini -> Parser Text () -> Parser Ini
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a
<*
(Parser Text ()
forall t. Chunk t => Parser t ()
endOfInput Parser Text () -> Parser Text () -> Parser Text ()
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> (String -> Parser Text ()
forall (m :: * -> *) a. MonadFail m => String -> m a
fail (String -> Parser Text ())
-> (Text -> String) -> Text -> Parser Text ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Text -> String
T.unpack (Text -> Parser Text ()) -> Parser Text Text -> Parser Text ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< (Char -> Bool) -> Parser Text Text
takeWhile (Bool -> Bool
not (Bool -> Bool) -> (Char -> Bool) -> Char -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Char -> Bool
isControl)))
sectionParser :: Parser (Text,[(Text, Text)])
sectionParser :: Parser Text (Text, [(Text, Text)])
sectionParser =
do Parser Text ()
skipEndOfLine
Parser Text ()
skipComments
Parser Text ()
skipEndOfLine
Char
_ <- Char -> Parser Char
char Char
'['
Text
name <- (Char -> Bool) -> Parser Text Text
takeWhile (\Char
c -> Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/=Char
']' Bool -> Bool -> Bool
&& Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/= Char
'[')
Char
_ <- Char -> Parser Char
char Char
']'
Parser Text ()
skipEndOfLine
[(Text, Text)]
values <- Parser Text (Text, Text) -> Parser Text [(Text, Text)]
forall (f :: * -> *) a. Alternative f => f a -> f [a]
many Parser Text (Text, Text)
keyValueParser
(Text, [(Text, Text)]) -> Parser Text (Text, [(Text, Text)])
forall (m :: * -> *) a. Monad m => a -> m a
return (Text -> Text
T.strip Text
name, [(Text, Text)]
values)
keyValueParser :: Parser (Text,Text)
keyValueParser :: Parser Text (Text, Text)
keyValueParser =
do Parser Text ()
skipEndOfLine
Parser Text ()
skipComments
Parser Text ()
skipEndOfLine
Text
key <- (Char -> Bool) -> Parser Text Text
takeWhile1 (\Char
c -> Bool -> Bool
not (Char -> Bool
isDelim Char
c Bool -> Bool -> Bool
|| Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
== Char
'[' Bool -> Bool -> Bool
|| Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
== Char
']'))
Char
delim <- (Char -> Bool) -> Parser Char
satisfy Char -> Bool
isDelim
Text
value <- (Text -> Text) -> Parser Text Text -> Parser Text Text
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Char -> Text -> Text
clean Char
delim) ((Char -> Bool) -> Parser Text Text
takeWhile (Bool -> Bool
not (Bool -> Bool) -> (Char -> Bool) -> Char -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Char -> Bool
isEndOfLine))
Parser Text ()
skipEndOfLine
(Text, Text) -> Parser Text (Text, Text)
forall (m :: * -> *) a. Monad m => a -> m a
return (Text -> Text
T.strip Text
key, Text -> Text
T.strip Text
value)
where clean :: Char -> Text -> Text
clean Char
':' = Int -> Text -> Text
T.drop Int
1
clean Char
_ = Text -> Text
forall a. a -> a
id
isDelim :: Char -> Bool
isDelim :: Char -> Bool
isDelim Char
x = Char
x Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
== Char
'=' Bool -> Bool -> Bool
|| Char
x Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
== Char
':'
skipEndOfLine :: Parser ()
skipEndOfLine :: Parser Text ()
skipEndOfLine = (Char -> Bool) -> Parser Text ()
skipWhile Char -> Bool
isSpace
skipComments :: Parser ()
=
Parser Text () -> Parser Text ()
forall (f :: * -> *) a. Alternative f => f a -> f ()
skipMany (do Char
_ <- (Char -> Bool) -> Parser Char
satisfy (\Char
c -> Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
== Char
';' Bool -> Bool -> Bool
|| Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
== Char
'#')
(Char -> Bool) -> Parser Text ()
skipWhile (Bool -> Bool
not (Bool -> Bool) -> (Char -> Bool) -> Char -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Char -> Bool
isEndOfLine)
Parser Text ()
skipEndOfLine)