{-# LANGUAGE CPP #-}
{- |
  Module      :  Language.Haskell.Meta.Parse
  Copyright   :  (c) Matt Morrow 2008
  License     :  BSD3
  Maintainer  :  Matt Morrow <mjm2002@gmail.com>
  Stability   :  experimental
  Portability :  portable (template-haskell)
-}

module Language.Haskell.Meta.Parse (
  parsePat,
  parseExp,
  parseType,
  parseDecs,
  parseDecsWithMode,
  myDefaultParseMode,
  myDefaultExtensions,
  parseResultToEither,
  parseHsModule,
  parseHsDecls,
  parseHsDeclsWithMode,
  parseHsType,
  parseHsExp,
  parseHsPat,
  pprHsModule,
  moduleDecls,
  noSrcSpanInfo,
  emptyHsModule
 ) where

import           Language.Haskell.Exts.Extension
import           Language.Haskell.Exts.Parser           hiding
  (parseExp, parsePat, parseType)
import           Language.Haskell.Exts.Pretty
import qualified Language.Haskell.Exts.SrcLoc           as Hs
import qualified Language.Haskell.Exts.Syntax           as Hs
import           Language.Haskell.Meta.Syntax.Translate
import           Language.Haskell.TH.Syntax             hiding (Extension (..))

-----------------------------------------------------------------------------

-- * template-haskell

parsePat :: String -> Either String Pat
parsePat :: String -> Either String Pat
parsePat = forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a b. a -> Either a b
Left (forall a b. b -> Either a b
Right forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. ToPat a => a -> Pat
toPat) forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> Either String (Pat SrcSpanInfo)
parseHsPat

parseExp :: String -> Either String Exp
parseExp :: String -> Either String Exp
parseExp = forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a b. a -> Either a b
Left (forall a b. b -> Either a b
Right forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. ToExp a => a -> Exp
toExp) forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> Either String (Exp SrcSpanInfo)
parseHsExp

parseType :: String -> Either String Type
parseType :: String -> Either String Type
parseType = forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a b. a -> Either a b
Left (forall a b. b -> Either a b
Right forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. ToType a => a -> Type
toType) forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> Either String (Type SrcSpanInfo)
parseHsType

parseDecs :: String -> Either String [Dec]
parseDecs :: String -> Either String [Dec]
parseDecs  = forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a b. a -> Either a b
Left (forall a b. b -> Either a b
Right forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. ToDecs a => a -> [Dec]
toDecs) forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> Either String [Decl SrcSpanInfo]
parseHsDecls

-- | @since 0.8.2
parseDecsWithMode :: ParseMode -> String -> Either String [Dec]
parseDecsWithMode :: ParseMode -> String -> Either String [Dec]
parseDecsWithMode ParseMode
parseMode = forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a b. a -> Either a b
Left (forall a b. b -> Either a b
Right forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. ToDecs a => a -> [Dec]
toDecs)
  forall b c a. (b -> c) -> (a -> b) -> a -> c
. ParseMode -> String -> Either String [Decl SrcSpanInfo]
parseHsDeclsWithMode ParseMode
parseMode

-----------------------------------------------------------------------------

{-# DEPRECATED myDefaultParseMode, myDefaultExtensions
  "The provided ParseModes aren't very meaningful, use your own instead" #-}
myDefaultParseMode :: ParseMode
myDefaultParseMode :: ParseMode
myDefaultParseMode = ParseMode
defaultParseMode
  {parseFilename :: String
parseFilename = []
  ,baseLanguage :: Language
baseLanguage = Language
Haskell2010
  ,extensions :: [Extension]
extensions = forall a b. (a -> b) -> [a] -> [b]
map KnownExtension -> Extension
EnableExtension [KnownExtension]
myDefaultExtensions
  }

myDefaultExtensions :: [KnownExtension]
myDefaultExtensions :: [KnownExtension]
myDefaultExtensions = [KnownExtension
PostfixOperators
                      ,KnownExtension
QuasiQuotes
                      ,KnownExtension
UnicodeSyntax
                      ,KnownExtension
PatternSignatures
                      ,KnownExtension
MagicHash
                      ,KnownExtension
ForeignFunctionInterface
                      ,KnownExtension
TemplateHaskell
                      ,KnownExtension
RankNTypes
                      ,KnownExtension
MultiParamTypeClasses
                      ,KnownExtension
RecursiveDo
                      ,KnownExtension
TypeApplications]

parseResultToEither :: ParseResult a -> Either String a
parseResultToEither :: forall a. ParseResult a -> Either String a
parseResultToEither (ParseOk a
a) = forall a b. b -> Either a b
Right a
a
parseResultToEither (ParseFailed SrcLoc
loc String
e)
  = let line :: Int
line = SrcLoc -> Int
Hs.srcLine SrcLoc
loc forall a. Num a => a -> a -> a
- Int
1
    in forall a b. a -> Either a b
Left ([String] -> String
unlines [forall a. Show a => a -> String
show Int
line,forall a. Show a => a -> String
show SrcLoc
loc,String
e])

parseHsModule :: String -> Either String (Hs.Module Hs.SrcSpanInfo)
parseHsModule :: String -> Either String (Module SrcSpanInfo)
parseHsModule = forall a. ParseResult a -> Either String a
parseResultToEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. ParseMode -> String -> ParseResult (Module SrcSpanInfo)
parseModuleWithMode ParseMode
myDefaultParseMode

parseHsDecls :: String -> Either String [Hs.Decl Hs.SrcSpanInfo]
parseHsDecls :: String -> Either String [Decl SrcSpanInfo]
parseHsDecls = forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a b. a -> Either a b
Left (forall a b. b -> Either a b
Right forall b c a. (b -> c) -> (a -> b) -> a -> c
. Module SrcSpanInfo -> [Decl SrcSpanInfo]
moduleDecls)
  forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. ParseResult a -> Either String a
parseResultToEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. ParseMode -> String -> ParseResult (Module SrcSpanInfo)
parseModuleWithMode ParseMode
myDefaultParseMode

-- | @since 0.8.2
parseHsDeclsWithMode :: ParseMode -> String -> Either String [Hs.Decl Hs.SrcSpanInfo]
parseHsDeclsWithMode :: ParseMode -> String -> Either String [Decl SrcSpanInfo]
parseHsDeclsWithMode ParseMode
parseMode = forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either forall a b. a -> Either a b
Left (forall a b. b -> Either a b
Right forall b c a. (b -> c) -> (a -> b) -> a -> c
. Module SrcSpanInfo -> [Decl SrcSpanInfo]
moduleDecls)
  forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. ParseResult a -> Either String a
parseResultToEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. ParseMode -> String -> ParseResult (Module SrcSpanInfo)
parseModuleWithMode ParseMode
parseMode


parseHsType :: String -> Either String (Hs.Type Hs.SrcSpanInfo)
parseHsType :: String -> Either String (Type SrcSpanInfo)
parseHsType = forall a. ParseResult a -> Either String a
parseResultToEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. ParseMode -> String -> ParseResult (Type SrcSpanInfo)
parseTypeWithMode ParseMode
myDefaultParseMode


parseHsExp :: String -> Either String (Hs.Exp Hs.SrcSpanInfo)
parseHsExp :: String -> Either String (Exp SrcSpanInfo)
parseHsExp = forall a. ParseResult a -> Either String a
parseResultToEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. ParseMode -> String -> ParseResult (Exp SrcSpanInfo)
parseExpWithMode ParseMode
myDefaultParseMode

parseHsPat :: String -> Either String (Hs.Pat Hs.SrcSpanInfo)
parseHsPat :: String -> Either String (Pat SrcSpanInfo)
parseHsPat = forall a. ParseResult a -> Either String a
parseResultToEither forall b c a. (b -> c) -> (a -> b) -> a -> c
. ParseMode -> String -> ParseResult (Pat SrcSpanInfo)
parsePatWithMode ParseMode
myDefaultParseMode

pprHsModule :: Hs.Module Hs.SrcSpanInfo -> String
pprHsModule :: Module SrcSpanInfo -> String
pprHsModule = forall a. Pretty a => a -> String
prettyPrint


moduleDecls :: Hs.Module Hs.SrcSpanInfo -> [Hs.Decl Hs.SrcSpanInfo]
moduleDecls :: Module SrcSpanInfo -> [Decl SrcSpanInfo]
moduleDecls (Hs.Module SrcSpanInfo
_ Maybe (ModuleHead SrcSpanInfo)
_ [ModulePragma SrcSpanInfo]
_ [ImportDecl SrcSpanInfo]
_ [Decl SrcSpanInfo]
x) = [Decl SrcSpanInfo]
x
moduleDecls Module SrcSpanInfo
m                     = forall (f :: * -> *) e a.
(Functor f, Show (f ())) =>
String -> f e -> a
todo String
"" Module SrcSpanInfo
m
-- TODO
--             (Hs.XmlPage _ _ _ _ _ _ _)
--          (Hs.XmlHybrid _ _ _ _ _ _ _ _ _)

-- mkModule :: String -> Hs.Module
-- mkModule s = Hs.Module undefined (Hs.ModuleName s) Nothing [] []

emptyHsModule :: String -> Hs.Module Hs.SrcSpanInfo
emptyHsModule :: String -> Module SrcSpanInfo
emptyHsModule String
n =
    (forall l.
l
-> Maybe (ModuleHead l)
-> [ModulePragma l]
-> [ImportDecl l]
-> [Decl l]
-> Module l
Hs.Module
        SrcSpanInfo
noSrcSpanInfo
        (forall a. a -> Maybe a
Just (forall l.
l
-> ModuleName l
-> Maybe (WarningText l)
-> Maybe (ExportSpecList l)
-> ModuleHead l
Hs.ModuleHead SrcSpanInfo
noSrcSpanInfo (forall l. l -> String -> ModuleName l
Hs.ModuleName SrcSpanInfo
noSrcSpanInfo String
n) forall a. Maybe a
Nothing forall a. Maybe a
Nothing))
        []
        []
        [])

noSrcSpanInfo :: Hs.SrcSpanInfo
noSrcSpanInfo :: SrcSpanInfo
noSrcSpanInfo = SrcSpan -> SrcSpanInfo
Hs.noInfoSpan (SrcLoc -> SrcLoc -> SrcSpan
Hs.mkSrcSpan SrcLoc
Hs.noLoc SrcLoc
Hs.noLoc)

{-
ghci> :i Module
data Module
  = Module SrcLoc
           ModuleName
           [OptionPragma]
           (Maybe WarningText)
           (Maybe [ExportSpec])
           [ImportDecl]
           [Decl]
        -- Defined in Language.Haskell.Exts.Syntax
instance Show Module -- Defined in Language.Haskell.Exts.Syntax
-}

-----------------------------------------------------------------------------