module Camfort.Output where
import Camfort.Helpers
import Camfort.Traverse
import qualified Language.Fortran.AST as F
import qualified Language.Fortran.Util.Position as FU
import qualified Language.Fortran.Analysis as FA
import qualified Language.Fortran.Parser as Fortran
import Language.Fortran
import Language.Fortran.Pretty
import Language.Fortran.PreProcess
import Camfort.Analysis.Annotations
import Camfort.Analysis.Syntax
import Camfort.PrettyPrint
import Camfort.Reprint
import Camfort.Transformation.Syntax
import Camfort.Specification.Units.Environment
import System.FilePath
import System.Directory
import qualified Data.ByteString.Char8 as B
import Data.Map.Lazy hiding (map, foldl)
import Data.Functor.Identity
import Data.Generics
import GHC.Generics
import Data.List hiding (zip)
import Data.Generics.Uniplate.Data
import Generics.Deriving.Copoint
import Data.Char
import Data.Generics.Zipper
import Data.Maybe
import Debug.Trace
import Control.Monad.Trans.State.Lazy
import Text.Printf
class Show' s where
show' :: s -> String
instance Show' String where
show' = id
instance (Show' a, Show' b) => Show' (a, b) where
show' (a, b) = "(" ++ show' a ++ "," ++ show' b ++")"
instance (Show a) => Show' a where
show' = show
class OutputFiles t where
mkOutputText :: FileOrDir -> t -> SourceText
outputFile :: t -> Filename
outputFiles :: FileOrDir -> FileOrDir -> [t] -> IO ()
outputFiles inp outp pdata = do
outIsDir <- isDirectory outp
inIsDir <- isDirectory inp
inIsFile <- doesFileExist inp
if outIsDir then do
createDirectoryIfMissing True outp
putStrLn $ "Writing refactored files to directory: " ++ outp ++ "/"
isdir <- isDirectory inp
let inSrc = if isdir then inp else getDir inp
mapM_ (\x -> let f' = changeDir outp inSrc (outputFile x)
in do checkDir f'
putStrLn $ "Writing " ++ f'
B.writeFile f' (mkOutputText outp x)) pdata
else
if inIsDir || length pdata > 1
then error $ "Error: attempting to output multiple files, but the \
\given output destination is a single file. \n\
\Please specify an output directory"
else
if inIsFile
then do
putStrLn $ "Writing refactored file to: " ++ outp
putStrLn $ "Writing " ++ outp
B.writeFile outp (mkOutputText outp (head pdata))
else let outSrc = getDir outp
in do createDirectoryIfMissing True outSrc
putStrLn $ "Writing refactored file to: " ++ outp
putStrLn $ "Writing " ++ outp
B.writeFile outp (mkOutputText outp (head pdata))
instance OutputFiles (Filename, SourceText) where
mkOutputText _ (_, output) = output
outputFile (f, _) = f
data PR a = PR (Program a) deriving Data
instance PrettyPrint (PR Annotation) where
prettyPrint (PR x) = prettyPrint x
instance OutputFiles (Filename, SourceText, Program Annotation) where
mkOutputText f' (f, input, ast') = B.pack $ reprint refactoringLF input f' (PR ast')
where
outputFile (f, _, _) = f
instance OutputFiles (Filename, SourceText, F.ProgramFile Annotation) where
mkOutputText f' (f, input, ast') = B.pack $ reprint refactoringForPar input f' ast'
outputFile (f, _, _) = f
srcSpanToSrcLocs :: FU.SrcSpan -> (SrcLoc, SrcLoc)
srcSpanToSrcLocs (FU.SrcSpan lpos upos) = (toSrcLoc lpos, toSrcLoc upos)
where
toSrcLoc pos = SrcLoc { srcFilename = ""
, srcLine = FU.posLine pos
, srcColumn = FU.posColumn pos }
instance (PrettyPrint (F.ProgramFile Annotation)) where
prettyPrint _ = ""
refactoringForPar :: (Typeable a) => [String] -> SrcLoc -> a -> State Int (String, SrcLoc, Bool)
refactoringForPar inp cursor =
(\_ -> return ("", cursor, False)) `extQ` (outputComments inp cursor)
where
outputComments :: [String] -> SrcLoc -> F.Block Annotation -> State Int (String, SrcLoc, Bool)
outputComments inp cursor e@(F.BlComment ann span comment) = return $
if (pRefactored ann)
then let (lb, ub) = srcSpanToSrcLocs span
lb' = leftOne lb
(p0, _) = takeBounds (cursor, lb') inp
nl = if comment == [] then "" else "\n"
in (p0 ++ comment ++ nl, ub, True)
else ("", cursor, False)
where leftOne (SrcLoc f l c) = SrcLoc f (l1) (c1)
outputComments _ _ _ = return ("", cursor, False)
changeDir newDir oldDir oldFilename = newDir ++ (listDiffL oldDir oldFilename)
where listDiffL [] ys = ys
listDiffL xs [] = []
listDiffL (x:xs) (y:ys) | x==y = listDiffL xs ys
| otherwise = ys
outputAnalysisFiles :: FileOrDir -> [Program Annotation] -> [Filename] -> IO ()
outputAnalysisFiles src asts files = do
isdir <- isDirectory src
let src' = if isdir then src else dropFileName src
putStrLn $ "Writing analysis files to directory: " ++ src'
mapM (\(ast', f) -> writeFile (f ++ ".html") ((concatMap outputHTML) ast')) (zip asts files)
return ()
refactoringLF :: (Typeable a, Monad m) => [String] -> SrcLoc -> a -> StateT Int m (String, SrcLoc, Bool)
refactoringLF inp cursor = ((((\_ -> return ("", cursor, False))
`extQ` (refactorUses inp cursor))
`extQ` (refactorDecl inp cursor))
`extQ` (refactorArgName inp cursor))
`extQ` (refactorFortran inp cursor)
refactorFortran :: Monad m => [String] -> SrcLoc -> Fortran Annotation -> StateT Int m (String, SrcLoc, Bool)
refactorFortran inp cursor e = return $
if (pRefactored $ tag e) then
let (lb, ub) = srcSpan e
(p0, _) = takeBounds (cursor, lb) inp
outE = pprint e
lnl = case e of (NullStmt _ _) -> (if ((p0 /= []) && (Prelude.last p0 /= '\n')) then "\n" else "")
_ -> ""
lnl2 = if ((p0 /= []) && (Prelude.last p0 /= '\n')) then "\n" else ""
textOut = if p0 == "\n" then outE else (p0 ++ lnl2 ++ outE ++ lnl)
in (textOut, ub, True)
else ("", cursor, False)
refactorDecl :: Monad m => [String] -> SrcLoc -> Decl Annotation -> StateT Int m (String, SrcLoc, Bool)
refactorDecl inp cursor d =
if (pRefactored $ tag d) then
let (lb, ub) = srcSpan d
(p0, _) = takeBounds (cursor, lb) inp
textOut = p0 ++ (pprint d)
in do textOut' <-
case d of
(NullDecl _ _) ->
do added <- get
let diff = linesCovered ub lb
let (text, removed) = if added <= diff
then removeNewLines textOut added
else removeNewLines textOut diff
put (added removed)
return text
otherwise -> return textOut
return (textOut', ub, True)
else return ("", cursor, False)
refactorArgName :: Monad m => [String] -> SrcLoc -> ArgName Annotation -> m (String, SrcLoc, Bool)
refactorArgName inp cursor a = return $
case (refactored $ tag a) of
Just lb -> let (p0, _) = takeBounds (cursor, lb) inp
in (p0 ++ pprint a, lb, True)
Nothing -> ("", cursor, False)
refactorUses :: Monad m => [String] -> SrcLoc -> Uses Annotation -> StateT Int m (String, SrcLoc, Bool)
refactorUses inp cursor u =
let ?variant = HTMLPP in
case (refactored $ tag u) of
Just lb -> let (p0, _) = takeBounds (cursor, lb) inp
syntax = printSlave u
in do added <- get
if (newNode $ tag u) then put (added + (countLines syntax))
else return ()
return (p0 ++ syntax, toCol0 lb, True)
Nothing -> return ("", cursor, False)
countLines [] = 0
countLines ('\n':xs) = 1 + countLines xs
countLines (x:xs) = countLines xs
removeNewLines [] n = ([], 0)
removeNewLines xs 0 = (xs, 0)
removeNewLines ('\r':('\n':('\r':('\n':xs)))) n = let (xs', n') = removeNewLines ('\r':'\n':xs) (n 1)
in (xs', n' + 1)
removeNewLines ('\n':('\n':xs)) n = let (xs', n') = removeNewLines ('\n':xs) (n 1)
in (xs', n' + 1)
removeNewLines (x:xs) n = let (xs', n') = removeNewLines xs n
in (x:xs', n)