{-# LANGUAGE BangPatterns, CPP, FlexibleInstances, TypeSynonymInstances #-}
#if __GLASGOW_HASKELL__ >= 702
{-# LANGUAGE Trustworthy #-}
#endif
{-# OPTIONS_GHC -fno-warn-warnings-deprecations #-}
module Data.Attoparsec.Text
(
Parser
, Result
, T.IResult(..)
, I.compareResults
, parse
, feed
, I.parseOnly
, parseWith
, parseTest
, maybeResult
, eitherResult
, I.char
, I.anyChar
, I.notChar
, I.satisfy
, I.satisfyWith
, I.skip
, I.peekChar
, I.peekChar'
, digit
, letter
, space
, I.inClass
, I.notInClass
, I.string
, I.stringCI
, I.asciiCI
, skipSpace
, I.skipWhile
, I.scan
, I.runScanner
, I.take
, I.takeWhile
, I.takeWhile1
, I.takeTill
, (.*>)
, (<*.)
, I.takeText
, I.takeLazyText
, I.endOfLine
, isEndOfLine
, isHorizontalSpace
, decimal
, hexadecimal
, signed
, double
, Number(..)
, number
, rational
, scientific
, try
, (<?>)
, choice
, count
, option
, many'
, many1
, many1'
, manyTill
, manyTill'
, sepBy
, sepBy'
, sepBy1
, sepBy1'
, skipMany
, skipMany1
, eitherP
, I.match
, I.endOfInput
, I.atEnd
) where
#if !MIN_VERSION_base(4,8,0)
import Control.Applicative (pure, (*>), (<*), (<$>))
import Data.Word (Word)
#endif
import Control.Applicative ((<|>))
import Data.Attoparsec.Combinator
import Data.Attoparsec.Number (Number(..))
import Data.Scientific (Scientific)
import qualified Data.Scientific as Sci
import Data.Attoparsec.Text.Internal (Parser, Result, parse, takeWhile1)
import Data.Bits (Bits, (.|.), shiftL)
import Data.Char (isAlpha, isDigit, isSpace, ord)
import Data.Int (Int8, Int16, Int32, Int64)
import Data.List (intercalate)
import Data.Text (Text)
import Data.Word (Word8, Word16, Word32, Word64)
import qualified Data.Attoparsec.Internal as I
import qualified Data.Attoparsec.Internal.Types as T
import qualified Data.Attoparsec.Text.Internal as I
import qualified Data.Text as T
parseTest :: (Show a) => I.Parser a -> Text -> IO ()
parseTest :: forall a. Show a => Parser a -> Text -> IO ()
parseTest Parser a
p Text
s = forall a. Show a => a -> IO ()
print (forall a. Parser a -> Text -> Result a
parse Parser a
p Text
s)
parseWith :: Monad m =>
(m Text)
-> I.Parser a
-> Text
-> m (Result a)
parseWith :: forall (m :: * -> *) a.
Monad m =>
m Text -> Parser a -> Text -> m (Result a)
parseWith m Text
refill Parser a
p Text
s = forall {r}. IResult Text r -> m (IResult Text r)
step forall a b. (a -> b) -> a -> b
$ forall a. Parser a -> Text -> Result a
parse Parser a
p Text
s
where step :: IResult Text r -> m (IResult Text r)
step (T.Partial Text -> IResult Text r
k) = (IResult Text r -> m (IResult Text r)
step forall b c a. (b -> c) -> (a -> b) -> a -> c
. Text -> IResult Text r
k) forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< m Text
refill
step IResult Text r
r = forall (m :: * -> *) a. Monad m => a -> m a
return IResult Text r
r
{-# INLINE parseWith #-}
maybeResult :: Result r -> Maybe r
maybeResult :: forall r. Result r -> Maybe r
maybeResult (T.Done Text
_ r
r) = forall a. a -> Maybe a
Just r
r
maybeResult IResult Text r
_ = forall a. Maybe a
Nothing
eitherResult :: Result r -> Either String r
eitherResult :: forall r. Result r -> Either String r
eitherResult (T.Done Text
_ r
r) = forall a b. b -> Either a b
Right r
r
eitherResult (T.Fail Text
_ [] String
msg) = forall a b. a -> Either a b
Left String
msg
eitherResult (T.Fail Text
_ [String]
ctxs String
msg) = forall a b. a -> Either a b
Left (forall a. [a] -> [[a]] -> [a]
intercalate String
" > " [String]
ctxs forall a. [a] -> [a] -> [a]
++ String
": " forall a. [a] -> [a] -> [a]
++ String
msg)
eitherResult IResult Text r
_ = forall a b. a -> Either a b
Left String
"Result: incomplete input"
isEndOfLine :: Char -> Bool
isEndOfLine :: Char -> Bool
isEndOfLine Char
c = Char
c forall a. Eq a => a -> a -> Bool
== Char
'\n' Bool -> Bool -> Bool
|| Char
c forall a. Eq a => a -> a -> Bool
== Char
'\r'
{-# INLINE isEndOfLine #-}
isHorizontalSpace :: Char -> Bool
isHorizontalSpace :: Char -> Bool
isHorizontalSpace Char
c = Char
c forall a. Eq a => a -> a -> Bool
== Char
' ' Bool -> Bool -> Bool
|| Char
c forall a. Eq a => a -> a -> Bool
== Char
'\t'
{-# INLINE isHorizontalSpace #-}
hexadecimal :: (Integral a, Bits a) => Parser a
hexadecimal :: forall a. (Integral a, Bits a) => Parser a
hexadecimal = forall a. (a -> Char -> a) -> a -> Text -> a
T.foldl' forall {a}. (Bits a, Num a) => a -> Char -> a
step a
0 forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
`fmap` (Char -> Bool) -> Parser Text
takeWhile1 Char -> Bool
isHexDigit
where
isHexDigit :: Char -> Bool
isHexDigit Char
c = (Char
c forall a. Ord a => a -> a -> Bool
>= Char
'0' Bool -> Bool -> Bool
&& Char
c forall a. Ord a => a -> a -> Bool
<= Char
'9') Bool -> Bool -> Bool
||
(Char
c forall a. Ord a => a -> a -> Bool
>= Char
'a' Bool -> Bool -> Bool
&& Char
c forall a. Ord a => a -> a -> Bool
<= Char
'f') Bool -> Bool -> Bool
||
(Char
c forall a. Ord a => a -> a -> Bool
>= Char
'A' Bool -> Bool -> Bool
&& Char
c forall a. Ord a => a -> a -> Bool
<= Char
'F')
step :: a -> Char -> a
step a
a Char
c | Int
w forall a. Ord a => a -> a -> Bool
>= Int
48 Bool -> Bool -> Bool
&& Int
w forall a. Ord a => a -> a -> Bool
<= Int
57 = (a
a forall a. Bits a => a -> Int -> a
`shiftL` Int
4) forall a. Bits a => a -> a -> a
.|. forall a b. (Integral a, Num b) => a -> b
fromIntegral (Int
w forall a. Num a => a -> a -> a
- Int
48)
| Int
w forall a. Ord a => a -> a -> Bool
>= Int
97 = (a
a forall a. Bits a => a -> Int -> a
`shiftL` Int
4) forall a. Bits a => a -> a -> a
.|. forall a b. (Integral a, Num b) => a -> b
fromIntegral (Int
w forall a. Num a => a -> a -> a
- Int
87)
| Bool
otherwise = (a
a forall a. Bits a => a -> Int -> a
`shiftL` Int
4) forall a. Bits a => a -> a -> a
.|. forall a b. (Integral a, Num b) => a -> b
fromIntegral (Int
w forall a. Num a => a -> a -> a
- Int
55)
where w :: Int
w = Char -> Int
ord Char
c
{-# SPECIALISE hexadecimal :: Parser Int #-}
{-# SPECIALISE hexadecimal :: Parser Int8 #-}
{-# SPECIALISE hexadecimal :: Parser Int16 #-}
{-# SPECIALISE hexadecimal :: Parser Int32 #-}
{-# SPECIALISE hexadecimal :: Parser Int64 #-}
{-# SPECIALISE hexadecimal :: Parser Integer #-}
{-# SPECIALISE hexadecimal :: Parser Word #-}
{-# SPECIALISE hexadecimal :: Parser Word8 #-}
{-# SPECIALISE hexadecimal :: Parser Word16 #-}
{-# SPECIALISE hexadecimal :: Parser Word32 #-}
{-# SPECIALISE hexadecimal :: Parser Word64 #-}
decimal :: Integral a => Parser a
decimal :: forall a. Integral a => Parser a
decimal = forall a. (a -> Char -> a) -> a -> Text -> a
T.foldl' forall {a}. Num a => a -> Char -> a
step a
0 forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
`fmap` (Char -> Bool) -> Parser Text
takeWhile1 Char -> Bool
isDecimal
where step :: a -> Char -> a
step a
a Char
c = a
a forall a. Num a => a -> a -> a
* a
10 forall a. Num a => a -> a -> a
+ forall a b. (Integral a, Num b) => a -> b
fromIntegral (Char -> Int
ord Char
c forall a. Num a => a -> a -> a
- Int
48)
{-# SPECIALISE decimal :: Parser Int #-}
{-# SPECIALISE decimal :: Parser Int8 #-}
{-# SPECIALISE decimal :: Parser Int16 #-}
{-# SPECIALISE decimal :: Parser Int32 #-}
{-# SPECIALISE decimal :: Parser Int64 #-}
{-# SPECIALISE decimal :: Parser Integer #-}
{-# SPECIALISE decimal :: Parser Word #-}
{-# SPECIALISE decimal :: Parser Word8 #-}
{-# SPECIALISE decimal :: Parser Word16 #-}
{-# SPECIALISE decimal :: Parser Word32 #-}
{-# SPECIALISE decimal :: Parser Word64 #-}
isDecimal :: Char -> Bool
isDecimal :: Char -> Bool
isDecimal Char
c = Char
c forall a. Ord a => a -> a -> Bool
>= Char
'0' Bool -> Bool -> Bool
&& Char
c forall a. Ord a => a -> a -> Bool
<= Char
'9'
{-# INLINE isDecimal #-}
signed :: Num a => Parser a -> Parser a
{-# SPECIALISE signed :: Parser Int -> Parser Int #-}
{-# SPECIALISE signed :: Parser Int8 -> Parser Int8 #-}
{-# SPECIALISE signed :: Parser Int16 -> Parser Int16 #-}
{-# SPECIALISE signed :: Parser Int32 -> Parser Int32 #-}
{-# SPECIALISE signed :: Parser Int64 -> Parser Int64 #-}
{-# SPECIALISE signed :: Parser Integer -> Parser Integer #-}
signed :: forall a. Num a => Parser a -> Parser a
signed Parser a
p = (forall a. Num a => a -> a
negate forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (Char -> Parser Char
I.char Char
'-' forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parser a
p))
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> (Char -> Parser Char
I.char Char
'+' forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parser a
p)
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Parser a
p
rational :: Fractional a => Parser a
{-# SPECIALIZE rational :: Parser Double #-}
{-# SPECIALIZE rational :: Parser Float #-}
{-# SPECIALIZE rational :: Parser Rational #-}
{-# SPECIALIZE rational :: Parser Scientific #-}
rational :: forall a. Fractional a => Parser a
rational = forall a. (Scientific -> a) -> Parser a
scientifically forall a b. (Real a, Fractional b) => a -> b
realToFrac
double :: Parser Double
double :: Parser Double
double = forall a. (Scientific -> a) -> Parser a
scientifically forall a. RealFloat a => Scientific -> a
Sci.toRealFloat
number :: Parser Number
number :: Parser Number
number = forall a. (Scientific -> a) -> Parser a
scientifically forall a b. (a -> b) -> a -> b
$ \Scientific
s ->
let e :: Int
e = Scientific -> Int
Sci.base10Exponent Scientific
s
c :: Integer
c = Scientific -> Integer
Sci.coefficient Scientific
s
in if Int
e forall a. Ord a => a -> a -> Bool
>= Int
0
then Integer -> Number
I (Integer
c forall a. Num a => a -> a -> a
* Integer
10 forall a b. (Num a, Integral b) => a -> b -> a
^ Int
e)
else Double -> Number
D (forall a. RealFloat a => Scientific -> a
Sci.toRealFloat Scientific
s)
{-# DEPRECATED number "Use 'scientific' instead." #-}
scientific :: Parser Scientific
scientific :: Parser Scientific
scientific = forall a. (Scientific -> a) -> Parser a
scientifically forall a. a -> a
id
data SP = SP !Integer {-# UNPACK #-}!Int
{-# INLINE scientifically #-}
scientifically :: (Scientific -> a) -> Parser a
scientifically :: forall a. (Scientific -> a) -> Parser a
scientifically Scientific -> a
h = do
!Bool
positive <- ((forall a. Eq a => a -> a -> Bool
== Char
'+') forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (Char -> Bool) -> Parser Char
I.satisfy (\Char
c -> Char
c forall a. Eq a => a -> a -> Bool
== Char
'-' Bool -> Bool -> Bool
|| Char
c forall a. Eq a => a -> a -> Bool
== Char
'+')) forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|>
forall (f :: * -> *) a. Applicative f => a -> f a
pure Bool
True
Integer
n <- forall a. Integral a => Parser a
decimal
let f :: Text -> SP
f Text
fracDigits = Integer -> Int -> SP
SP (forall a. (a -> Char -> a) -> a -> Text -> a
T.foldl' forall {a}. Num a => a -> Char -> a
step Integer
n Text
fracDigits)
(forall a. Num a => a -> a
negate forall a b. (a -> b) -> a -> b
$ Text -> Int
T.length Text
fracDigits)
step :: a -> Char -> a
step a
a Char
c = a
a forall a. Num a => a -> a -> a
* a
10 forall a. Num a => a -> a -> a
+ forall a b. (Integral a, Num b) => a -> b
fromIntegral (Char -> Int
ord Char
c forall a. Num a => a -> a -> a
- Int
48)
SP Integer
c Int
e <- ((Char -> Bool) -> Parser Char
I.satisfy (forall a. Eq a => a -> a -> Bool
==Char
'.') forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> (Text -> SP
f forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> (Char -> Bool) -> Parser Text
I.takeWhile Char -> Bool
isDigit)) forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|>
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Integer -> Int -> SP
SP Integer
n Int
0)
let !signedCoeff :: Integer
signedCoeff | Bool
positive = Integer
c
| Bool
otherwise = -Integer
c
((Char -> Bool) -> Parser Char
I.satisfy (\Char
w -> Char
w forall a. Eq a => a -> a -> Bool
== Char
'e' Bool -> Bool -> Bool
|| Char
w forall a. Eq a => a -> a -> Bool
== Char
'E') forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*>
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Scientific -> a
h forall b c a. (b -> c) -> (a -> b) -> a -> c
. Integer -> Int -> Scientific
Sci.scientific Integer
signedCoeff forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Int
e forall a. Num a => a -> a -> a
+)) (forall a. Num a => Parser a -> Parser a
signed forall a. Integral a => Parser a
decimal)) forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|>
forall (m :: * -> *) a. Monad m => a -> m a
return (Scientific -> a
h forall a b. (a -> b) -> a -> b
$ Integer -> Int -> Scientific
Sci.scientific Integer
signedCoeff Int
e)
digit :: Parser Char
digit :: Parser Char
digit = (Char -> Bool) -> Parser Char
I.satisfy Char -> Bool
isDigit forall i a. Parser i a -> String -> Parser i a
<?> String
"digit"
{-# INLINE digit #-}
letter :: Parser Char
letter :: Parser Char
letter = (Char -> Bool) -> Parser Char
I.satisfy Char -> Bool
isAlpha forall i a. Parser i a -> String -> Parser i a
<?> String
"letter"
{-# INLINE letter #-}
space :: Parser Char
space :: Parser Char
space = (Char -> Bool) -> Parser Char
I.satisfy Char -> Bool
isSpace forall i a. Parser i a -> String -> Parser i a
<?> String
"space"
{-# INLINE space #-}
skipSpace :: Parser ()
skipSpace :: Parser ()
skipSpace = (Char -> Bool) -> Parser ()
I.skipWhile Char -> Bool
isSpace
{-# INLINE skipSpace #-}
(.*>) :: Text -> Parser a -> Parser a
Text
s .*> :: forall a. Text -> Parser a -> Parser a
.*> Parser a
f = Text -> Parser Text
I.string Text
s forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parser a
f
{-# DEPRECATED (.*>) "This is no longer necessary, and will be removed. Use '*>' instead." #-}
(<*.) :: Parser a -> Text -> Parser a
Parser a
f <*. :: forall a. Parser a -> Text -> Parser a
<*. Text
s = Parser a
f forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a
<* Text -> Parser Text
I.string Text
s
{-# DEPRECATED (<*.) "This is no longer necessary, and will be removed. Use '<*' instead." #-}