{-# LANGUAGE CPP #-}
module Diagrams.Util
(
with
, applyAll
, (#)
, (##)
, iterateN
, tau
, findHsFile
, findSandbox
, globalPackage
, foldB
) where
import Control.Applicative
import Control.Lens hiding (( # ))
import Control.Monad
import Control.Monad.Catch
import Control.Monad.Trans
import Control.Monad.Trans.Maybe
import Data.Default.Class
import Data.List
import Data.Maybe
import Data.Monoid
import System.Directory
import System.Environment
import System.FilePath
import System.FilePath.Lens
import System.Process
with :: Default d => d
with :: forall d. Default d => d
with = forall d. Default d => d
def
applyAll :: [a -> a] -> a -> a
applyAll :: forall a. [a -> a] -> a -> a
applyAll = forall a. Endo a -> a -> a
appEndo forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. Monoid a => [a] -> a
mconcat forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a b. (a -> b) -> [a] -> [b]
map forall a. (a -> a) -> Endo a
Endo
infixl 8 #
(#) :: a -> (a -> b) -> b
# :: forall a b. a -> (a -> b) -> b
(#) = forall a b c. (a -> b -> c) -> b -> a -> c
flip forall a b. (a -> b) -> a -> b
($)
(##) :: AReview t b -> b -> t
## :: forall t b. AReview t b -> b -> t
(##) = forall b (m :: * -> *) t. MonadReader b m => AReview t b -> m t
review
{-# INLINE (##) #-}
infixr 8 ##
iterateN :: Int -> (a -> a) -> a -> [a]
iterateN :: forall a. Int -> (a -> a) -> a -> [a]
iterateN Int
n a -> a
f = forall a. Int -> [a] -> [a]
take Int
n forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall a. (a -> a) -> a -> [a]
iterate a -> a
f
tau :: Floating a => a
tau :: forall a. Floating a => a
tau = a
2forall a. Num a => a -> a -> a
*forall a. Floating a => a
pi
foldB :: (a -> a -> a) -> a -> [a] -> a
foldB :: forall a. (a -> a -> a) -> a -> [a] -> a
foldB a -> a -> a
_ a
z [] = a
z
foldB a -> a -> a
f a
_ [a]
as = [a] -> a
foldB' [a]
as
where foldB' :: [a] -> a
foldB' [a
x] = a
x
foldB' [a]
xs = [a] -> a
foldB' ([a] -> [a]
go [a]
xs)
go :: [a] -> [a]
go [] = []
go [a
x] = [a
x]
go (a
x1:a
x2:[a]
xs) = a -> a -> a
f a
x1 a
x2 forall a. a -> [a] -> [a]
: [a] -> [a]
go [a]
xs
findHsFile :: FilePath -> IO (Maybe FilePath)
findHsFile :: String -> IO (Maybe String)
findHsFile String
file = forall (m :: * -> *) a. MaybeT m a -> m (Maybe a)
runMaybeT forall a b. (a -> b) -> a -> b
$ MaybeT IO String
hs forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> MaybeT IO String
lhs
where
hs :: MaybeT IO String
hs = forall {t :: (* -> *) -> * -> *}.
(Monad (t IO), MonadTrans t, Alternative (t IO)) =>
String -> t IO String
check (String -> String -> String
addExtension String
file String
"hs")
lhs :: MaybeT IO String
lhs = forall {t :: (* -> *) -> * -> *}.
(Monad (t IO), MonadTrans t, Alternative (t IO)) =>
String -> t IO String
check (String -> String -> String
addExtension String
file String
"lhs")
check :: String -> t IO String
check String
f = do
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (String -> IO Bool
doesFileExist String
f) forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= forall (f :: * -> *). Alternative f => Bool -> f ()
guard
forall (f :: * -> *) a. Applicative f => a -> f a
pure String
f
parseConfig :: FilePath -> MaybeT IO FilePath
parseConfig :: String -> MaybeT IO String
parseConfig String
file = do
String
config <- forall (m :: * -> *) a.
(MonadCatch m, MonadIO m) =>
IO a -> MaybeT m a
maybeIO forall a b. (a -> b) -> a -> b
$ String -> IO String
readFile String
file
forall (m :: * -> *) a. Monad m => Maybe a -> MaybeT m a
hoistMaybe forall a b. (a -> b) -> a -> b
$ String
config forall s a. s -> Getting (First a) s a -> Maybe a
^? forall (f :: * -> *).
Applicative f =>
IndexedLensLike' Int f String String
lined forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall t. Prefixed t => t -> Prism' t t
prefixed String
"package-db: "
configSearch :: FilePath -> MaybeT IO FilePath
configSearch :: String -> MaybeT IO String
configSearch String
p0 = do
String
p0' <- forall (m :: * -> *) a.
(MonadCatch m, MonadIO m) =>
IO a -> MaybeT m a
maybeIO forall a b. (a -> b) -> a -> b
$ String -> IO String
canonicalizePath String
p0
let mkPaths :: String -> [String]
mkPaths String
p
| forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all Char -> Bool
isPathSeparator String
p Bool -> Bool -> Bool
|| String
p forall a. Eq a => a -> a -> Bool
== String
"."
= []
| Bool
otherwise = (String
p String -> String -> String
</> String
"cabal.sandbox.config")
forall a. a -> [a] -> [a]
: String -> [String]
mkPaths (String
p forall s a. s -> Getting a s a -> a
^. Lens' String String
directory)
forall (m :: * -> *) a b.
Monad m =>
(a -> MaybeT m b) -> [a] -> MaybeT m b
foldMaybeT String -> MaybeT IO String
parseConfig (String -> [String]
mkPaths String
p0')
isDB :: FilePath -> MaybeT IO FilePath
isDB :: String -> MaybeT IO String
isDB String
path =
if String -> Bool
isConf String
path
then forall (m :: * -> *) a. Monad m => a -> m a
return String
path
else forall (m :: * -> *) a.
(MonadCatch m, MonadIO m) =>
IO a -> MaybeT m a
maybeIO (String -> IO [String]
getDirectoryContents String
path) forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= forall (m :: * -> *) a. Monad m => Maybe a -> MaybeT m a
hoistMaybe forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Maybe a
find String -> Bool
isConf
where
isConf :: String -> Bool
isConf = forall a. Eq a => [a] -> [a] -> Bool
isSuffixOf String
".conf.d"
findSandbox :: [FilePath] -> IO (Maybe FilePath)
findSandbox :: [String] -> IO (Maybe String)
findSandbox [String]
paths = forall (m :: * -> *) a. MaybeT m a -> m (Maybe a)
runMaybeT forall a b. (a -> b) -> a -> b
$ MaybeT IO String
pathsTest forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> MaybeT IO String
diaSB forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> MaybeT IO String
envDB forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> MaybeT IO String
wdConfig
where
lookEnv :: String -> MaybeT IO String
lookEnv = forall (m :: * -> *) a. m (Maybe a) -> MaybeT m a
MaybeT forall b c a. (b -> c) -> (a -> b) -> a -> c
. (forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap) (forall a. [a] -> a
head forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> [String]
splitSearchPath) forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> IO (Maybe String)
lookupEnv
envDB :: MaybeT IO String
envDB = forall (m :: * -> *) a b.
Monad m =>
(a -> MaybeT m b) -> [a] -> MaybeT m b
foldMaybeT String -> MaybeT IO String
lookEnv [String
"GHC_PACKAGE_PATH", String
"HSENV", String
"PACKAGE_DB_FOR_GHC"]
test :: String -> MaybeT IO String
test String
x = String -> MaybeT IO String
isDB String
x forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> String -> MaybeT IO String
configSearch String
x
pathsTest :: MaybeT IO String
pathsTest = forall (m :: * -> *) a b.
Monad m =>
(a -> MaybeT m b) -> [a] -> MaybeT m b
foldMaybeT String -> MaybeT IO String
test [String]
paths
diaSB :: MaybeT IO String
diaSB = String -> MaybeT IO String
lookEnv String
"DIAGRAMS_SANDBOX" forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= String -> MaybeT IO String
test
wdConfig :: MaybeT IO String
wdConfig = forall (m :: * -> *) a.
(MonadCatch m, MonadIO m) =>
IO a -> MaybeT m a
maybeIO IO String
getCurrentDirectory forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= String -> MaybeT IO String
configSearch
globalPackage :: IO FilePath
globalPackage :: IO String
globalPackage = do
[(String, String)]
info <- forall a. Read a => String -> a
read forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> String -> [String] -> String -> IO String
readProcess String
"ghc" [String
"--info"] String
""
forall (m :: * -> *) a. Monad m => a -> m a
return forall a b. (a -> b) -> a -> b
$ forall a. a -> Maybe a -> a
fromMaybe (forall a. HasCallStack => String -> a
error String
"Unable to parse ghc --info.")
(forall a b. Eq a => a -> [(a, b)] -> Maybe b
lookup String
"Global Package DB" [(String, String)]
info)
maybeIO :: (MonadCatch m, MonadIO m) => IO a -> MaybeT m a
maybeIO :: forall (m :: * -> *) a.
(MonadCatch m, MonadIO m) =>
IO a -> MaybeT m a
maybeIO IO a
io = forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO a
io forall (m :: * -> *) a.
MonadCatch m =>
m a -> (SomeException -> m a) -> m a
`catchAll` forall a b. a -> b -> a
const forall (m :: * -> *) a. MonadPlus m => m a
mzero
#if MIN_VERSION_transformers(0,6,0)
#else
hoistMaybe :: Monad m => Maybe a -> MaybeT m a
hoistMaybe :: forall (m :: * -> *) a. Monad m => Maybe a -> MaybeT m a
hoistMaybe = forall (m :: * -> *) a. m (Maybe a) -> MaybeT m a
MaybeT forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (m :: * -> *) a. Monad m => a -> m a
return
#endif
foldMaybeT :: Monad m => (a -> MaybeT m b) -> [a] -> MaybeT m b
foldMaybeT :: forall (m :: * -> *) a b.
Monad m =>
(a -> MaybeT m b) -> [a] -> MaybeT m b
foldMaybeT a -> MaybeT m b
_ [] = forall (m :: * -> *) a. MonadPlus m => m a
mzero
foldMaybeT a -> MaybeT m b
f (a
a:[a]
as) = forall (m :: * -> *) a. m (Maybe a) -> MaybeT m a
MaybeT forall a b. (a -> b) -> a -> b
$ do
Maybe b
x <- forall (m :: * -> *) a. MaybeT m a -> m (Maybe a)
runMaybeT (a -> MaybeT m b
f a
a)
if forall a. Maybe a -> Bool
isJust Maybe b
x
then forall (m :: * -> *) a. Monad m => a -> m a
return Maybe b
x
else forall (m :: * -> *) a. MaybeT m a -> m (Maybe a)
runMaybeT (forall (m :: * -> *) a b.
Monad m =>
(a -> MaybeT m b) -> [a] -> MaybeT m b
foldMaybeT a -> MaybeT m b
f [a]
as)