module HsImport.Parse
( parseFile
, lastImportSrcLine
) where
import qualified Data.Text.IO as TIO
import qualified Data.Text as T
import Data.Maybe (fromMaybe)
import Data.List (isPrefixOf)
import qualified Language.Haskell.Exts as HS
import Control.Applicative ((<$>))
import Control.Exception (catch, SomeException)
type Error = String
parseFile :: FilePath -> IO (Either Error (HS.ParseResult HS.Module))
parseFile file = do
srcFile <- unlines. replaceCPPByComment . lines . T.unpack <$> TIO.readFile file
catch (do let result = parseFileContents srcFile
case result of
HS.ParseOk _ -> return $ Right result
HS.ParseFailed srcLoc _ -> do
srcResult <- parseInvalidSource (lines srcFile) (HS.srcLine srcLoc) 0 (HS.srcLine srcLoc)
return $ Right $ fromMaybe result srcResult)
(\(e :: SomeException) -> do
let srcLines = lines srcFile
srcResult <- parseInvalidSource srcLines (length srcLines) 0 (length srcLines)
return $ maybe (Left $ show e) Right srcResult)
where
replaceCPPByComment = map $ \line ->
if "#" `isPrefixOf` line
then "-- fake hsimport comment"
else line
type SrcLine = Int
lastImportSrcLine :: [String] -> Maybe SrcLine
lastImportSrcLine srcLines
| null srcLines
= Nothing
| "import" `isPrefixOf` head srcLines
= parseImport 1
| otherwise
= Nothing
where
parseImport lastLine
| lastLine <= numSrcLines
= case parseFileContents source of
HS.ParseOk module_
| oneImportDeclWithoutSymbols module_ && startsWithImportDeclSymbols (lastLine + 1)
-> parseImport (lastLine + 1)
| otherwise
-> Just lastLine
HS.ParseFailed _ _ -> parseImport (lastLine + 1)
| otherwise
= Nothing
where
source = unlines $ take lastLine srcLines
numSrcLines = length srcLines
oneImportDeclWithoutSymbols (HS.Module _ _ _ _ _ [HS.ImportDecl {HS.importSpecs = Nothing}] _) = True
oneImportDeclWithoutSymbols _ = False
startsWithImportDeclSymbols lineNum
| line : _ <- drop (lineNum 1) srcLines
, ' ' : _ <- line
, '(' : _ <- dropWhile (== ' ') line
= True
| otherwise
= False
parseInvalidSource :: [String] -> Int -> Int -> Int -> IO (Maybe (HS.ParseResult HS.Module))
parseInvalidSource srcLines firstInvalidLine lastValidLine currLastLine
| null srcLines || lastValidLine >= currLastLine = return Nothing
| otherwise =
catch (case parseFileContents source of
result@(HS.ParseOk _)
| (nextLine + 1) == currLastLine ->
return $ Just result
| otherwise ->
parseInvalidSource srcLines firstInvalidLine nextLine currLastLine
HS.ParseFailed srcLoc _
| HS.srcLine srcLoc == firstInvalidLine ->
parseInvalidSource srcLines firstInvalidLine lastValidLine nextLine
| otherwise ->
parseInvalidSource srcLines firstInvalidLine nextLine currLastLine)
(\(_ :: SomeException) -> parseInvalidSource srcLines firstInvalidLine lastValidLine nextLine)
where
source = unlines $ take (nextLine + 1) srcLines
nextLine = lastValidLine + (floor ((realToFrac (currLastLine lastValidLine) / 2) :: Double) :: Int)
parseFileContents :: String -> HS.ParseResult HS.Module
parseFileContents = HS.parseFileContentsWithMode parseMode
where
parseMode = HS.defaultParseMode { HS.fixities = Just [] }