{-# LANGUAGE BlockArguments #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE OverloadedRecordDot #-}

module MonadicBang.Options where

import Control.Exception
import Control.Algebra
import Control.Carrier.State.Strict
import Control.Effect.Throw
import Control.Monad
import Data.Bifunctor
import Data.Bool
import Data.List (intercalate, partition)

import GHC
import GHC.Plugins

data Verbosity = DumpTransformed | Quiet

data PreserveErrors = Preserve | Don'tPreserve

data Options = MkOptions {Options -> Verbosity
verbosity :: Verbosity, Options -> PreserveErrors
preserveErrors :: PreserveErrors}

parseOptions :: Has (Throw ErrorCall) sig m => Located HsModule -> [CommandLineOption] -> m Options
parseOptions :: forall (sig :: (* -> *) -> * -> *) (m :: * -> *).
Has (Throw ErrorCall) sig m =>
Located HsModule -> [CommandLineOption] -> m Options
parseOptions Located HsModule
mod' [CommandLineOption]
cmdLineOpts = do
  ([CommandLineOption]
remaining, Options
options) <- [CommandLineOption]
-> StateC [CommandLineOption] m Options
-> m ([CommandLineOption], Options)
forall s (m :: * -> *) a. s -> StateC s m a -> m (s, a)
runState [CommandLineOption]
cmdLineOpts do
    Verbosity
verbosity <- Verbosity -> Verbosity -> Bool -> Verbosity
forall a. a -> a -> Bool -> a
bool Verbosity
Quiet Verbosity
DumpTransformed (Bool -> Verbosity)
-> StateC [CommandLineOption] m Bool
-> StateC [CommandLineOption] m Verbosity
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [CommandLineOption] -> StateC [CommandLineOption] m Bool
forall {a} {t :: * -> *} {m :: * -> *} {sig :: (* -> *) -> * -> *}.
(Eq a, Foldable t, Member (State [a]) sig, Algebra sig m) =>
t a -> m Bool
extractOpts [CommandLineOption]
verboseOpts
    PreserveErrors
preserveErrors <- PreserveErrors -> PreserveErrors -> Bool -> PreserveErrors
forall a. a -> a -> Bool -> a
bool PreserveErrors
Don'tPreserve PreserveErrors
Preserve (Bool -> PreserveErrors)
-> StateC [CommandLineOption] m Bool
-> StateC [CommandLineOption] m PreserveErrors
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [CommandLineOption] -> StateC [CommandLineOption] m Bool
forall {a} {t :: * -> *} {m :: * -> *} {sig :: (* -> *) -> * -> *}.
(Eq a, Foldable t, Member (State [a]) sig, Algebra sig m) =>
t a -> m Bool
extractOpts [CommandLineOption]
preserveErrorsOpts
    Options -> StateC [CommandLineOption] m Options
forall a. a -> StateC [CommandLineOption] m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Options -> StateC [CommandLineOption] m Options)
-> Options -> StateC [CommandLineOption] m Options
forall a b. (a -> b) -> a -> b
$ Verbosity -> PreserveErrors -> Options
MkOptions Verbosity
verbosity PreserveErrors
preserveErrors
  Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless ([CommandLineOption] -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [CommandLineOption]
remaining) (m () -> m ())
-> (CommandLineOption -> m ()) -> CommandLineOption -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ErrorCall -> m ()
forall e (sig :: (* -> *) -> * -> *) (m :: * -> *) a.
Has (Throw e) sig m =>
e -> m a
throwError (ErrorCall -> m ())
-> (CommandLineOption -> ErrorCall) -> CommandLineOption -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. CommandLineOption -> ErrorCall
ErrorCall (CommandLineOption -> m ()) -> CommandLineOption -> m ()
forall a b. (a -> b) -> a -> b
$
    CommandLineOption
"Incorrect command line options for plugin MonadicBang, encountered in " CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++ CommandLineOption
modName CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++ CommandLineOption
modFile CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++
    CommandLineOption
"\n\tOptions that were supplied (via -fplugin-opt) are: " CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++ CommandLineOption -> [CommandLineOption] -> CommandLineOption
forall a. [a] -> [[a]] -> [a]
intercalate CommandLineOption
", " ((CommandLineOption -> CommandLineOption)
-> [CommandLineOption] -> [CommandLineOption]
forall a b. (a -> b) -> [a] -> [b]
map CommandLineOption -> CommandLineOption
forall a. Show a => a -> CommandLineOption
show [CommandLineOption]
cmdLineOpts) CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++
    CommandLineOption
"\n\tUnrecognized options: " CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++ [CommandLineOption] -> CommandLineOption
showOpts [CommandLineOption]
remaining CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++
    CommandLineOption
"\n\n\tUsage: [-ddump] [-preserve-errors]" CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++
    CommandLineOption
"\n" CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++
    CommandLineOption
"\n\t\t-ddump            Print the altered AST" CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++
    CommandLineOption
"\n\t\t-preserve-errors  Keep parse errors about ! outside of 'do' in their original form, rather then a more relevant explanation." CommandLineOption -> CommandLineOption -> CommandLineOption
forall a. [a] -> [a] -> [a]
++
    CommandLineOption
"\n\t\t                  This is mainly useful if another plugin expects those errors."
  Options -> m Options
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure Options
options

  where
    verboseOpts :: [CommandLineOption]
verboseOpts = [CommandLineOption
"-ddump"]
    preserveErrorsOpts :: [CommandLineOption]
preserveErrorsOpts = [CommandLineOption
"-preserve-errors"]
    extractOpts :: t a -> m Bool
extractOpts t a
opt = do
      (Bool
isOpt, [a]
opts') <- ([a] -> (Bool, [a])) -> m (Bool, [a])
forall s (sig :: (* -> *) -> * -> *) (m :: * -> *) a.
Has (State s) sig m =>
(s -> a) -> m a
gets (([a] -> (Bool, [a])) -> m (Bool, [a]))
-> ([a] -> (Bool, [a])) -> m (Bool, [a])
forall a b. (a -> b) -> a -> b
$ ([a] -> Bool) -> ([a], [a]) -> (Bool, [a])
forall a b c. (a -> b) -> (a, c) -> (b, c)
forall (p :: * -> * -> *) a b c.
Bifunctor p =>
(a -> b) -> p a c -> p b c
first (Bool -> Bool
not (Bool -> Bool) -> ([a] -> Bool) -> [a] -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [a] -> Bool
forall a. [a] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null) (([a], [a]) -> (Bool, [a]))
-> ([a] -> ([a], [a])) -> [a] -> (Bool, [a])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (a -> Bool) -> [a] -> ([a], [a])
forall a. (a -> Bool) -> [a] -> ([a], [a])
partition (a -> t a -> Bool
forall a. Eq a => a -> t a -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` t a
opt)
      [a] -> m ()
forall s (sig :: (* -> *) -> * -> *) (m :: * -> *).
Has (State s) sig m =>
s -> m ()
put [a]
opts'
      Bool -> m Bool
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure Bool
isOpt

    showOpts :: [CommandLineOption] -> CommandLineOption
showOpts = CommandLineOption -> [CommandLineOption] -> CommandLineOption
forall a. [a] -> [[a]] -> [a]
intercalate CommandLineOption
", " ([CommandLineOption] -> CommandLineOption)
-> ([CommandLineOption] -> [CommandLineOption])
-> [CommandLineOption]
-> CommandLineOption
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (CommandLineOption -> CommandLineOption)
-> [CommandLineOption] -> [CommandLineOption]
forall a b. (a -> b) -> [a] -> [b]
map CommandLineOption -> CommandLineOption
forall a. Show a => a -> CommandLineOption
show

    modFile :: CommandLineOption
modFile = CommandLineOption
-> (RealSrcSpan -> CommandLineOption)
-> Maybe RealSrcSpan
-> CommandLineOption
forall b a. b -> (a -> b) -> Maybe a -> b
maybe CommandLineOption
"" ((CommandLineOption
" in file " ++) (CommandLineOption -> CommandLineOption)
-> (RealSrcSpan -> CommandLineOption)
-> RealSrcSpan
-> CommandLineOption
forall b c a. (b -> c) -> (a -> b) -> a -> c
. FastString -> CommandLineOption
unpackFS (FastString -> CommandLineOption)
-> (RealSrcSpan -> FastString) -> RealSrcSpan -> CommandLineOption
forall b c a. (b -> c) -> (a -> b) -> a -> c
. RealSrcSpan -> FastString
srcSpanFile) (Maybe RealSrcSpan -> CommandLineOption)
-> Maybe RealSrcSpan -> CommandLineOption
forall a b. (a -> b) -> a -> b
$ SrcSpan -> Maybe RealSrcSpan
toRealSrcSpan (Located HsModule -> SrcSpan
forall l e. GenLocated l e -> l
getLoc Located HsModule
mod')
    modName :: CommandLineOption
modName = CommandLineOption
-> (GenLocated SrcSpanAnnA ModuleName -> CommandLineOption)
-> Maybe (GenLocated SrcSpanAnnA ModuleName)
-> CommandLineOption
forall b a. b -> (a -> b) -> Maybe a -> b
maybe CommandLineOption
"an unnamed module" ((CommandLineOption
"module " ++) (CommandLineOption -> CommandLineOption)
-> (GenLocated SrcSpanAnnA ModuleName -> CommandLineOption)
-> GenLocated SrcSpanAnnA ModuleName
-> CommandLineOption
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ModuleName -> CommandLineOption
moduleNameString (ModuleName -> CommandLineOption)
-> (GenLocated SrcSpanAnnA ModuleName -> ModuleName)
-> GenLocated SrcSpanAnnA ModuleName
-> CommandLineOption
forall b c a. (b -> c) -> (a -> b) -> a -> c
. GenLocated SrcSpanAnnA ModuleName -> ModuleName
forall l e. GenLocated l e -> e
unLoc) (Maybe (GenLocated SrcSpanAnnA ModuleName) -> CommandLineOption)
-> Maybe (GenLocated SrcSpanAnnA ModuleName) -> CommandLineOption
forall a b. (a -> b) -> a -> b
$ (Located HsModule -> HsModule
forall l e. GenLocated l e -> e
unLoc Located HsModule
mod').hsmodName
    toRealSrcSpan :: SrcSpan -> Maybe RealSrcSpan
toRealSrcSpan = \cases
      (RealSrcSpan RealSrcSpan
rss Maybe BufSpan
_) -> RealSrcSpan -> Maybe RealSrcSpan
forall a. a -> Maybe a
Just RealSrcSpan
rss
      (UnhelpfulSpan UnhelpfulSpanReason
_) -> Maybe RealSrcSpan
forall a. Maybe a
Nothing