{-# LANGUAGE Trustworthy #-}
{-# OPTIONS_GHC -Wno-partial-fields #-}
{-# OPTIONS_HADDOCK hide #-}
module Text.Gigaparsec.Internal.Token.Lexer (
    Lexer, mkLexer, mkLexerWithErrorConfig,
    Lexeme, lexeme, nonlexeme, fully, space,
    apply, sym, symbol, names, -- more go here, no numeric and no text
    -- Numeric
    integer, natural,
    -- Text
    stringLiteral, rawStringLiteral, multiStringLiteral, rawMultiStringLiteral,
    charLiteral,
    -- Space
    Space, skipComments, whiteSpace, alter, initSpace,
  ) where

import Text.Gigaparsec (Parsec, eof, void, empty, (<|>), atomic, unit)
import Text.Gigaparsec.Char (satisfy, string, item, endOfLine)
import Text.Gigaparsec.Combinator (skipMany, skipManyTill)
import Text.Gigaparsec.State (set, get, setDuring, rollback)
import Text.Gigaparsec.Errors.Combinator (hide)

import Text.Gigaparsec.Token.Descriptions qualified as Desc
import Text.Gigaparsec.Token.Errors (
    ErrorConfig (labelSpaceEndOfLineComment, labelSpaceEndOfMultiComment),
    defaultErrorConfig
  )
import Text.Gigaparsec.Internal.Token.Errors (annotate)
import Text.Gigaparsec.Internal.Token.Generic (mkGeneric)
import Text.Gigaparsec.Internal.Token.Symbol (Symbol, mkSym, mkSymbol)
import Text.Gigaparsec.Internal.Token.Symbol qualified as Symbol (lexeme)
import Text.Gigaparsec.Internal.Token.Names (Names, mkNames)
import Text.Gigaparsec.Internal.Token.Names qualified as Names (lexeme)
import Text.Gigaparsec.Internal.Token.Numeric (
    IntegerParsers, mkSigned, mkUnsigned,
    --FloatingParsers, mkSignedFloating, mkUnsignedFloating,
    --CombinedParsers, mkSignedCombined, mkUnsignedCombined,
  )
import Text.Gigaparsec.Internal.Token.BitBounds (CanHoldSigned, CanHoldUnsigned)
import Text.Gigaparsec.Internal.Token.Numeric qualified as Numeric (lexemeInteger, {-lexemeFloating, lexemeCombined-})
import Text.Gigaparsec.Internal.Token.Text (
    TextParsers,
    mkStringParsers, mkCharacterParsers, mkEscape, mkEscapeChar, StringChar(RawChar)
  )
import Text.Gigaparsec.Internal.Token.Text qualified as Text (lexeme)

import Text.Gigaparsec.Internal.Require (require)

import Data.List (isPrefixOf)
import Data.IORef (newIORef)
import Data.Ref (fromIORef)
import Control.Exception (Exception, throw)
import Control.Monad (join, guard)
import System.IO.Unsafe (unsafePerformIO)

type Lexer :: *
data Lexer = Lexer { Lexer -> Lexeme
lexeme :: !Lexeme
                   , Lexer -> Lexeme
nonlexeme :: !Lexeme
                   , Lexer -> forall a. Parsec a -> Parsec a
fully :: !(forall a. Parsec a -> Parsec a)
                   , Lexer -> Space
space :: !Space
                   }

mkLexer :: Desc.LexicalDesc -> Lexer
mkLexer :: LexicalDesc -> Lexer
mkLexer !LexicalDesc
desc = LexicalDesc -> ErrorConfig -> Lexer
mkLexerWithErrorConfig LexicalDesc
desc ErrorConfig
defaultErrorConfig

mkLexerWithErrorConfig :: Desc.LexicalDesc -> ErrorConfig -> Lexer
mkLexerWithErrorConfig :: LexicalDesc -> ErrorConfig -> Lexer
mkLexerWithErrorConfig Desc.LexicalDesc{SpaceDesc
TextDesc
NumericDesc
SymbolDesc
NameDesc
nameDesc :: NameDesc
symbolDesc :: SymbolDesc
numericDesc :: NumericDesc
textDesc :: TextDesc
spaceDesc :: SpaceDesc
nameDesc :: LexicalDesc -> NameDesc
symbolDesc :: LexicalDesc -> SymbolDesc
numericDesc :: LexicalDesc -> NumericDesc
textDesc :: LexicalDesc -> TextDesc
spaceDesc :: LexicalDesc -> SpaceDesc
..} !ErrorConfig
errConfig = Lexer {Space
Lexeme
Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
lexeme :: Lexeme
nonlexeme :: Lexeme
fully :: forall a. Parsec a -> Parsec a
space :: Space
lexeme :: Lexeme
nonlexeme :: Lexeme
fully :: forall a. Parsec a -> Parsec a
space :: Space
..}
  where apply :: Parsec a -> Parsec a
apply Parsec a
p = Parsec a
p Parsec a -> Parsec () -> Parsec a
forall a b. Parsec a -> Parsec b -> Parsec a
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a
<* Space -> Parsec ()
whiteSpace Space
space
        gen :: GenericNumeric
gen = ErrorConfig -> GenericNumeric
mkGeneric ErrorConfig
errConfig
        -- DO NOT HAVE MUTUALLY RECURSIVE FIELDS
        lexeme :: Lexeme
lexeme = Lexeme { apply :: forall a. Parsec a -> Parsec a
apply = Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply
                        , sym :: String -> Parsec ()
sym = Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a
apply (Parsec () -> Parsec ())
-> (String -> Parsec ()) -> String -> Parsec ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Lexeme -> String -> Parsec ()
sym Lexeme
nonlexeme
                        , symbol :: Symbol
symbol = (forall a. Parsec a -> Parsec a) -> Symbol -> Symbol
Symbol.lexeme Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply Symbol
symbolNonLexeme
                        , names :: Names
names = (forall a. Parsec a -> Parsec a) -> Names -> Names
Names.lexeme Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply (Lexeme -> Names
names Lexeme
nonlexeme)
                        , natural :: IntegerParsers CanHoldUnsigned
natural = (forall a. Parsec a -> Parsec a)
-> IntegerParsers CanHoldUnsigned -> IntegerParsers CanHoldUnsigned
forall (c :: Bits -> * -> Constraint).
(forall a. Parsec a -> Parsec a)
-> IntegerParsers c -> IntegerParsers c
Numeric.lexemeInteger Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply IntegerParsers CanHoldUnsigned
naturalNonLexeme
                        , integer :: IntegerParsers CanHoldSigned
integer = (forall a. Parsec a -> Parsec a)
-> IntegerParsers CanHoldSigned -> IntegerParsers CanHoldSigned
forall (c :: Bits -> * -> Constraint).
(forall a. Parsec a -> Parsec a)
-> IntegerParsers c -> IntegerParsers c
Numeric.lexemeInteger Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply (Lexeme -> IntegerParsers CanHoldSigned
integer Lexeme
nonlexeme)
                        {-, floating = Numeric.lexemeFloating apply (floating nonlexeme)
                        , unsignedCombined =
                            Numeric.lexemeCombined apply (unsignedCombined nonlexeme)
                        , signedCombined =
                            Numeric.lexemeCombined apply (signedCombined nonlexeme)-}
                        , stringLiteral :: TextParsers String
stringLiteral = (forall a. Parsec a -> Parsec a)
-> TextParsers String -> TextParsers String
forall t.
(forall a. Parsec a -> Parsec a) -> TextParsers t -> TextParsers t
Text.lexeme Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply (Lexeme -> TextParsers String
stringLiteral Lexeme
nonlexeme)
                        , rawStringLiteral :: TextParsers String
rawStringLiteral = (forall a. Parsec a -> Parsec a)
-> TextParsers String -> TextParsers String
forall t.
(forall a. Parsec a -> Parsec a) -> TextParsers t -> TextParsers t
Text.lexeme Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply (Lexeme -> TextParsers String
rawStringLiteral Lexeme
nonlexeme)
                        , multiStringLiteral :: TextParsers String
multiStringLiteral = (forall a. Parsec a -> Parsec a)
-> TextParsers String -> TextParsers String
forall t.
(forall a. Parsec a -> Parsec a) -> TextParsers t -> TextParsers t
Text.lexeme Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply (Lexeme -> TextParsers String
multiStringLiteral Lexeme
nonlexeme)
                        , rawMultiStringLiteral :: TextParsers String
rawMultiStringLiteral = (forall a. Parsec a -> Parsec a)
-> TextParsers String -> TextParsers String
forall t.
(forall a. Parsec a -> Parsec a) -> TextParsers t -> TextParsers t
Text.lexeme Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply (Lexeme -> TextParsers String
rawMultiStringLiteral Lexeme
nonlexeme)
                        , charLiteral :: TextParsers Char
charLiteral = (forall a. Parsec a -> Parsec a)
-> TextParsers Char -> TextParsers Char
forall t.
(forall a. Parsec a -> Parsec a) -> TextParsers t -> TextParsers t
Text.lexeme Parsec a -> Parsec a
forall a. Parsec a -> Parsec a
apply (Lexeme -> TextParsers Char
charLiteral Lexeme
nonlexeme)
                        }
        nonlexeme :: Lexeme
nonlexeme = NonLexeme { sym :: String -> Parsec ()
sym = SymbolDesc -> Symbol -> ErrorConfig -> String -> Parsec ()
mkSym SymbolDesc
symbolDesc Symbol
symbolNonLexeme ErrorConfig
errConfig
                              , symbol :: Symbol
symbol = Symbol
symbolNonLexeme
                              , names :: Names
names = NameDesc -> SymbolDesc -> ErrorConfig -> Names
mkNames NameDesc
nameDesc SymbolDesc
symbolDesc ErrorConfig
errConfig
                              , natural :: IntegerParsers CanHoldUnsigned
natural = IntegerParsers CanHoldUnsigned
naturalNonLexeme
                              , integer :: IntegerParsers CanHoldSigned
integer = NumericDesc
-> IntegerParsers CanHoldUnsigned
-> ErrorConfig
-> IntegerParsers CanHoldSigned
forall (c :: Bits -> * -> Constraint).
NumericDesc
-> IntegerParsers c -> ErrorConfig -> IntegerParsers CanHoldSigned
mkSigned NumericDesc
numericDesc IntegerParsers CanHoldUnsigned
naturalNonLexeme ErrorConfig
errConfig
                              {-, floating = mkSignedFloating numericDesc positiveFloating
                              , unsignedCombined = mkUnsignedCombined numericDesc naturalNonLexeme positiveFloating
                              , signedCombined = mkSignedCombined numericDesc (unsignedCombined nonlexeme)-}
                              , stringLiteral :: TextParsers String
stringLiteral = Set (String, String)
-> StringChar
-> CharPredicate
-> Bool
-> ErrorConfig
-> TextParsers String
mkStringParsers Set (String, String)
stringEnds StringChar
escapeChar CharPredicate
graphicCharacter Bool
False ErrorConfig
errConfig
                              , rawStringLiteral :: TextParsers String
rawStringLiteral = Set (String, String)
-> StringChar
-> CharPredicate
-> Bool
-> ErrorConfig
-> TextParsers String
mkStringParsers Set (String, String)
stringEnds StringChar
rawChar CharPredicate
graphicCharacter Bool
False ErrorConfig
errConfig
                              , multiStringLiteral :: TextParsers String
multiStringLiteral = Set (String, String)
-> StringChar
-> CharPredicate
-> Bool
-> ErrorConfig
-> TextParsers String
mkStringParsers Set (String, String)
multiStringEnds StringChar
escapeChar CharPredicate
graphicCharacter Bool
True ErrorConfig
errConfig
                              , rawMultiStringLiteral :: TextParsers String
rawMultiStringLiteral = Set (String, String)
-> StringChar
-> CharPredicate
-> Bool
-> ErrorConfig
-> TextParsers String
mkStringParsers Set (String, String)
multiStringEnds StringChar
rawChar CharPredicate
graphicCharacter Bool
True ErrorConfig
errConfig
                              , charLiteral :: TextParsers Char
charLiteral = TextDesc -> Escape -> ErrorConfig -> TextParsers Char
mkCharacterParsers TextDesc
textDesc Escape
escape ErrorConfig
errConfig
                              }
        !symbolNonLexeme :: Symbol
symbolNonLexeme = SymbolDesc -> NameDesc -> ErrorConfig -> Symbol
mkSymbol SymbolDesc
symbolDesc NameDesc
nameDesc ErrorConfig
errConfig
        !naturalNonLexeme :: IntegerParsers CanHoldUnsigned
naturalNonLexeme = NumericDesc
-> GenericNumeric -> ErrorConfig -> IntegerParsers CanHoldUnsigned
mkUnsigned NumericDesc
numericDesc GenericNumeric
gen ErrorConfig
errConfig
        --positiveFloating = mkUnsignedFloating numericDesc naturalNonLexeme gen
        !escape :: Escape
escape = EscapeDesc -> GenericNumeric -> ErrorConfig -> Escape
mkEscape (TextDesc -> EscapeDesc
Desc.escapeSequences TextDesc
textDesc) GenericNumeric
gen ErrorConfig
errConfig
        graphicCharacter :: CharPredicate
graphicCharacter = TextDesc -> CharPredicate
Desc.graphicCharacter TextDesc
textDesc
        stringEnds :: Set (String, String)
stringEnds = TextDesc -> Set (String, String)
Desc.stringEnds TextDesc
textDesc
        multiStringEnds :: Set (String, String)
multiStringEnds = TextDesc -> Set (String, String)
Desc.multiStringEnds TextDesc
textDesc
        rawChar :: StringChar
rawChar = StringChar
RawChar
        escapeChar :: StringChar
escapeChar = EscapeDesc -> Escape -> Parsec () -> ErrorConfig -> StringChar
mkEscapeChar (TextDesc -> EscapeDesc
Desc.escapeSequences TextDesc
textDesc) Escape
escape (Space -> Parsec ()
whiteSpace Space
space) ErrorConfig
errConfig
        fully' :: Parsec a -> Parsec a
fully' Parsec a
p = Space -> Parsec ()
whiteSpace Space
space Parsec () -> Parsec a -> Parsec a
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parsec a
p Parsec a -> Parsec () -> Parsec a
forall a b. Parsec a -> Parsec b -> Parsec a
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a
<* Parsec ()
eof
        fully :: Parsec b -> Parsec b
fully Parsec b
p
          | SpaceDesc -> Bool
Desc.whitespaceIsContextDependent SpaceDesc
spaceDesc = Space -> Parsec ()
initSpace Space
space Parsec () -> Parsec b -> Parsec b
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parsec b -> Parsec b
forall a. Parsec a -> Parsec a
fully' Parsec b
p
          | Bool
otherwise                                   = Parsec b -> Parsec b
forall a. Parsec a -> Parsec a
fully' Parsec b
p
        space :: Space
space = SpaceDesc -> ErrorConfig -> Space
mkSpace SpaceDesc
spaceDesc ErrorConfig
errConfig

--TODO: better name for this, I guess?
type Lexeme :: *
data Lexeme = Lexeme
                { Lexeme -> forall a. Parsec a -> Parsec a
apply :: !(forall a. Parsec a -> Parsec a) -- this is tricky...
                , Lexeme -> String -> Parsec ()
sym :: !(String -> Parsec ())
                , Lexeme -> Symbol
symbol :: !Symbol
                , Lexeme -> Names
names :: !Names
                , Lexeme -> IntegerParsers CanHoldUnsigned
natural :: !(IntegerParsers CanHoldUnsigned)
                , Lexeme -> IntegerParsers CanHoldSigned
integer :: !(IntegerParsers CanHoldSigned)
                -- desperate times, desperate measures
                --, floating :: !FloatingParsers
                --, unsignedCombined :: !CombinedParsers
                --, signedCombined :: !CombinedParsers
                , Lexeme -> TextParsers String
stringLiteral :: !(TextParsers String)
                , Lexeme -> TextParsers String
rawStringLiteral :: !(TextParsers String)
                , Lexeme -> TextParsers String
multiStringLiteral :: !(TextParsers String)
                , Lexeme -> TextParsers String
rawMultiStringLiteral :: !(TextParsers String)
                , Lexeme -> TextParsers Char
charLiteral :: !(TextParsers Char)
                }
            | NonLexeme
                { sym :: !(String -> Parsec ())
                , symbol :: !Symbol
                , names :: !Names
                , natural :: !(IntegerParsers CanHoldUnsigned)
                , integer :: !(IntegerParsers CanHoldSigned)
                -- desperate times, desperate measures
                --, floating :: !FloatingParsers
                --, unsignedCombined :: !CombinedParsers
                --, signedCombined :: !CombinedParsers
                , stringLiteral :: !(TextParsers String)
                , rawStringLiteral :: !(TextParsers String)
                , multiStringLiteral :: !(TextParsers String)
                , rawMultiStringLiteral :: !(TextParsers String)
                , charLiteral :: !(TextParsers Char)
                }

type Space :: *
data Space = Space { Space -> Parsec ()
whiteSpace :: !(Parsec ())
                   , Space -> Parsec ()
skipComments :: !(Parsec ())
                   , Space -> forall a. CharPredicate -> Parsec a -> Parsec a
alter :: forall a. Desc.CharPredicate -> Parsec a -> Parsec a
                   , Space -> Parsec ()
initSpace :: Parsec ()
                   }

mkSpace :: Desc.SpaceDesc -> ErrorConfig -> Space
mkSpace :: SpaceDesc -> ErrorConfig -> Space
mkSpace desc :: SpaceDesc
desc@Desc.SpaceDesc{Bool
String
CharPredicate
whitespaceIsContextDependent :: SpaceDesc -> Bool
lineCommentStart :: String
lineCommentAllowsEOF :: Bool
multiLineCommentStart :: String
multiLineCommentEnd :: String
multiLineNestedComments :: Bool
space :: CharPredicate
whitespaceIsContextDependent :: Bool
lineCommentStart :: SpaceDesc -> String
lineCommentAllowsEOF :: SpaceDesc -> Bool
multiLineCommentStart :: SpaceDesc -> String
multiLineCommentEnd :: SpaceDesc -> String
multiLineNestedComments :: SpaceDesc -> Bool
space :: SpaceDesc -> CharPredicate
..} !ErrorConfig
errConfig = Space {Parsec ()
CharPredicate -> Parsec a -> Parsec a
forall a. CharPredicate -> Parsec a -> Parsec a
skipComments :: Parsec ()
whiteSpace :: Parsec ()
alter :: forall a. CharPredicate -> Parsec a -> Parsec a
initSpace :: Parsec ()
whiteSpace :: Parsec ()
skipComments :: Parsec ()
alter :: forall a. CharPredicate -> Parsec a -> Parsec a
initSpace :: Parsec ()
..}
  where -- don't think we can trust doing initialisation here, it'll happen in some random order
        {-# NOINLINE wsImpl #-}
        !wsImpl :: Ref r a
wsImpl = IORef a -> Ref r a
forall a r. IORef a -> Ref r a
fromIORef (IO (IORef a) -> IORef a
forall a. IO a -> a
unsafePerformIO (a -> IO (IORef a)
forall a. a -> IO (IORef a)
newIORef (String -> a
forall a. HasCallStack => String -> a
error String
"uninitialised space")))
        comment :: ErrorConfig -> Parsec ()
comment = SpaceDesc -> ErrorConfig -> Parsec ()
commentParser SpaceDesc
desc -- do not make this strict
        implOf :: CharPredicate -> Parsec ()
implOf
          | SpaceDesc -> Bool
supportsComments SpaceDesc
desc = Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a
hide (Parsec () -> Parsec ())
-> (CharPredicate -> Parsec ()) -> CharPredicate -> Parsec ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Parsec ()
-> ((Char -> Bool) -> Parsec ()) -> CharPredicate -> Parsec ()
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Parsec ()
skipComments (Parsec () -> Parsec ()
forall a. Parsec a -> Parsec ()
skipMany (Parsec () -> Parsec ())
-> ((Char -> Bool) -> Parsec ()) -> (Char -> Bool) -> Parsec ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Parsec () -> Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a -> Parsec a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> ErrorConfig -> Parsec ()
comment ErrorConfig
errConfig) (Parsec () -> Parsec ())
-> ((Char -> Bool) -> Parsec ()) -> (Char -> Bool) -> Parsec ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Parsec Char -> Parsec ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (Parsec Char -> Parsec ())
-> ((Char -> Bool) -> Parsec Char) -> (Char -> Bool) -> Parsec ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Char -> Bool) -> Parsec Char
satisfy)
          | Bool
otherwise             = Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a
hide (Parsec () -> Parsec ())
-> (CharPredicate -> Parsec ()) -> CharPredicate -> Parsec ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Parsec ()
-> ((Char -> Bool) -> Parsec ()) -> CharPredicate -> Parsec ()
forall b a. b -> (a -> b) -> Maybe a -> b
maybe Parsec ()
forall a. Parsec a
forall (f :: * -> *) a. Alternative f => f a
empty (Parsec Char -> Parsec ()
forall a. Parsec a -> Parsec ()
skipMany (Parsec Char -> Parsec ())
-> ((Char -> Bool) -> Parsec Char) -> (Char -> Bool) -> Parsec ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Char -> Bool) -> Parsec Char
satisfy)
        !configuredWhitespace :: Parsec ()
configuredWhitespace = CharPredicate -> Parsec ()
implOf CharPredicate
space
        !whiteSpace :: Parsec ()
whiteSpace
          | Bool
whitespaceIsContextDependent = Parsec (Parsec ()) -> Parsec ()
forall (m :: * -> *) a. Monad m => m (m a) -> m a
join (Ref Any (Parsec ()) -> Parsec (Parsec ())
forall r a. Ref r a -> Parsec a
get Ref Any (Parsec ())
forall {r} {a}. Ref r a
wsImpl)
          | Bool
otherwise                    = Parsec ()
configuredWhitespace
        !skipComments :: Parsec ()
skipComments = Parsec () -> Parsec ()
forall a. Parsec a -> Parsec ()
skipMany (ErrorConfig -> Parsec ()
comment ErrorConfig
errConfig)
        alter :: CharPredicate -> Parsec b -> Parsec b
alter CharPredicate
p
          | Bool
whitespaceIsContextDependent = Ref Any Any -> Parsec b -> Parsec b
forall r a b. Ref r a -> Parsec b -> Parsec b
rollback Ref Any Any
forall {r} {a}. Ref r a
wsImpl (Parsec b -> Parsec b)
-> (Parsec b -> Parsec b) -> Parsec b -> Parsec b
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Ref Any (Parsec ()) -> Parsec () -> Parsec b -> Parsec b
forall r a b. Ref r a -> a -> Parsec b -> Parsec b
setDuring Ref Any (Parsec ())
forall {r} {a}. Ref r a
wsImpl (CharPredicate -> Parsec ()
implOf CharPredicate
p)
          | Bool
otherwise                    = UnsupportedOperation -> Parsec b -> Parsec b
forall a e. Exception e => e -> a
throw (String -> UnsupportedOperation
UnsupportedOperation String
badAlter)
        initSpace :: Parsec ()
initSpace
          | Bool
whitespaceIsContextDependent = Ref Any (Parsec ()) -> Parsec () -> Parsec ()
forall r a. Ref r a -> a -> Parsec ()
set Ref Any (Parsec ())
forall {r} {a}. Ref r a
wsImpl Parsec ()
configuredWhitespace
          | Bool
otherwise                    = UnsupportedOperation -> Parsec ()
forall a e. Exception e => e -> a
throw (String -> UnsupportedOperation
UnsupportedOperation String
badInit)
        badInit :: String
badInit = String
"whitespace cannot be initialised unless `spaceDesc.whitespaceIsContextDependent` is True"
        badAlter :: String
badAlter = String
"whitespace cannot be altered unless `spaceDesc.whitespaceIsContextDependent` is True"

{-
We have the following invariances to be checked up front:
  * at least one kind of comment must be enabled
  * the starts of line and multiline must not overlap

-- TODO: needs error messages put in (is the hide correct)
-- TODO: remove guard, configure properly
-}
commentParser :: Desc.SpaceDesc -> ErrorConfig -> Parsec ()
commentParser :: SpaceDesc -> ErrorConfig -> Parsec ()
commentParser Desc.SpaceDesc{Bool
String
CharPredicate
whitespaceIsContextDependent :: SpaceDesc -> Bool
lineCommentStart :: SpaceDesc -> String
lineCommentAllowsEOF :: SpaceDesc -> Bool
multiLineCommentStart :: SpaceDesc -> String
multiLineCommentEnd :: SpaceDesc -> String
multiLineNestedComments :: SpaceDesc -> Bool
space :: SpaceDesc -> CharPredicate
lineCommentStart :: String
lineCommentAllowsEOF :: Bool
multiLineCommentStart :: String
multiLineCommentEnd :: String
multiLineNestedComments :: Bool
space :: CharPredicate
whitespaceIsContextDependent :: Bool
..} !ErrorConfig
errConfig =
  Bool -> String -> String -> Parsec () -> Parsec ()
forall a. Bool -> String -> String -> a -> a
require (Bool
multiEnabled Bool -> Bool -> Bool
|| Bool
singleEnabled) String
"skipComments" String
noComments (Parsec () -> Parsec ()) -> Parsec () -> Parsec ()
forall a b. (a -> b) -> a -> b
$
    Bool -> String -> String -> Parsec () -> Parsec ()
forall a. Bool -> String -> String -> a -> a
require (Bool -> Bool
not (Bool
multiEnabled Bool -> Bool -> Bool
&& String -> String -> Bool
forall a. Eq a => [a] -> [a] -> Bool
isPrefixOf String
multiLineCommentStart String
lineCommentStart)) String
"skipComments" String
noOverlap (Parsec () -> Parsec ()) -> Parsec () -> Parsec ()
forall a b. (a -> b) -> a -> b
$
      Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a
hide (Parsec ()
multiLine Parsec () -> Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a -> Parsec a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Parsec ()
singleLine)
  where
    -- can't make these strict until guard is gone
    openComment :: Parsec String
openComment = Parsec String -> Parsec String
forall a. Parsec a -> Parsec a
atomic (String -> Parsec String
string String
multiLineCommentStart)
    closeComment :: Parsec String
closeComment = LabelWithExplainConfig -> Parsec String -> Parsec String
forall config a. Annotate config => config -> Parsec a -> Parsec a
forall a. LabelWithExplainConfig -> Parsec a -> Parsec a
annotate (ErrorConfig -> LabelWithExplainConfig
labelSpaceEndOfMultiComment ErrorConfig
errConfig) (Parsec String -> Parsec String
forall a. Parsec a -> Parsec a
atomic (String -> Parsec String
string String
multiLineCommentEnd))
    multiLine :: Parsec ()
multiLine = Bool -> Parsec ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard Bool
multiEnabled Parsec () -> Parsec String -> Parsec String
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parsec String
openComment Parsec String -> Parsec () -> Parsec ()
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Int -> Parsec ()
wellNested Int
1
    wellNested :: Int -> Parsec ()
    wellNested :: Int -> Parsec ()
wellNested Int
0 = Parsec ()
unit
    wellNested Int
n = Parsec String
closeComment Parsec String -> Parsec () -> Parsec ()
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Int -> Parsec ()
wellNested (Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1)
               Parsec () -> Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a -> Parsec a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Bool -> Parsec ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard Bool
multiLineNestedComments Parsec () -> Parsec String -> Parsec String
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parsec String
openComment Parsec String -> Parsec () -> Parsec ()
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Int -> Parsec ()
wellNested (Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
               Parsec () -> Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a -> Parsec a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Parsec Char
item Parsec Char -> Parsec () -> Parsec ()
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Int -> Parsec ()
wellNested Int
n
    singleLine :: Parsec ()
singleLine = Bool -> Parsec ()
forall (f :: * -> *). Alternative f => Bool -> f ()
guard Bool
singleEnabled
              Parsec () -> Parsec String -> Parsec String
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parsec String -> Parsec String
forall a. Parsec a -> Parsec a
atomic (String -> Parsec String
string String
lineCommentStart)
              Parsec String -> Parsec () -> Parsec ()
forall a b. Parsec a -> Parsec b -> Parsec b
forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b
*> Parsec Char -> Parsec () -> Parsec ()
forall a end. Parsec a -> Parsec end -> Parsec ()
skipManyTill Parsec Char
item (LabelWithExplainConfig -> Parsec () -> Parsec ()
forall config a. Annotate config => config -> Parsec a -> Parsec a
forall a. LabelWithExplainConfig -> Parsec a -> Parsec a
annotate (ErrorConfig -> LabelWithExplainConfig
labelSpaceEndOfLineComment ErrorConfig
errConfig) Parsec ()
endOfLineComment)

    endOfLineComment :: Parsec ()
endOfLineComment
      | Bool
lineCommentAllowsEOF = Parsec Char -> Parsec ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void Parsec Char
endOfLine Parsec () -> Parsec () -> Parsec ()
forall a. Parsec a -> Parsec a -> Parsec a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> Parsec ()
eof
      | Bool
otherwise            = Parsec Char -> Parsec ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void Parsec Char
endOfLine

    multiEnabled :: Bool
multiEnabled = Bool -> Bool
not (String -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null String
multiLineCommentStart Bool -> Bool -> Bool
|| String -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null String
multiLineCommentEnd)
    singleEnabled :: Bool
singleEnabled = Bool -> Bool
not (String -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null String
lineCommentStart)
    noComments :: String
noComments = String
"one of single- or multi-line comments must be enabled"
    noOverlap :: String
noOverlap = String
"single-line comments must not overlap with multi-line comments"

supportsComments :: Desc.SpaceDesc -> Bool
supportsComments :: SpaceDesc -> Bool
supportsComments Desc.SpaceDesc{Bool
String
CharPredicate
whitespaceIsContextDependent :: SpaceDesc -> Bool
lineCommentStart :: SpaceDesc -> String
lineCommentAllowsEOF :: SpaceDesc -> Bool
multiLineCommentStart :: SpaceDesc -> String
multiLineCommentEnd :: SpaceDesc -> String
multiLineNestedComments :: SpaceDesc -> Bool
space :: SpaceDesc -> CharPredicate
lineCommentStart :: String
lineCommentAllowsEOF :: Bool
multiLineCommentStart :: String
multiLineCommentEnd :: String
multiLineNestedComments :: Bool
space :: CharPredicate
whitespaceIsContextDependent :: Bool
..} = Bool -> Bool
not (String -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null String
lineCommentStart Bool -> Bool -> Bool
&& String -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null String
multiLineCommentStart)

type UnsupportedOperation :: *
newtype UnsupportedOperation = UnsupportedOperation String deriving stock UnsupportedOperation -> UnsupportedOperation -> Bool
(UnsupportedOperation -> UnsupportedOperation -> Bool)
-> (UnsupportedOperation -> UnsupportedOperation -> Bool)
-> Eq UnsupportedOperation
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: UnsupportedOperation -> UnsupportedOperation -> Bool
== :: UnsupportedOperation -> UnsupportedOperation -> Bool
$c/= :: UnsupportedOperation -> UnsupportedOperation -> Bool
/= :: UnsupportedOperation -> UnsupportedOperation -> Bool
Eq
instance Show UnsupportedOperation where
  show :: UnsupportedOperation -> String
show (UnsupportedOperation String
msg) = String
"unsupported operation: " String -> ShowS
forall a. [a] -> [a] -> [a]
++ String
msg
instance Exception UnsupportedOperation