248 lines
8.8 KiB
Haskell
248 lines
8.8 KiB
Haskell
{-# LANGUAGE OverloadedStrings #-}
|
|
{-# LANGUAGE FlexibleContexts #-}
|
|
|
|
{-|
|
|
Module : GHCup.Utils.File.Windows
|
|
Description : File and windows APIs
|
|
Copyright : (c) Julian Ospald, 2020
|
|
License : LGPL-3.0
|
|
Maintainer : hasufell@hasufell.de
|
|
Stability : experimental
|
|
Portability : Windows
|
|
|
|
This module handles file and executable handling.
|
|
Some of these functions use sophisticated logging.
|
|
-}
|
|
module GHCup.Utils.File.Windows where
|
|
|
|
import {-# SOURCE #-} GHCup.Utils ( getLinkTarget, pathIsLink )
|
|
import GHCup.Utils.Dirs
|
|
import GHCup.Utils.File.Common
|
|
import GHCup.Types
|
|
|
|
import Control.Concurrent
|
|
import Control.DeepSeq
|
|
import Control.Exception.Safe
|
|
import Control.Monad
|
|
import Control.Monad.Reader
|
|
import Data.List
|
|
import Foreign.C.Error
|
|
import GHC.IO.Exception
|
|
import GHC.IO.Handle
|
|
import System.Directory
|
|
import System.Environment
|
|
import System.FilePath
|
|
import System.IO
|
|
import System.Process
|
|
|
|
import qualified Control.Exception as EX
|
|
import qualified Data.ByteString as BS
|
|
import qualified Data.ByteString.Lazy as BL
|
|
import qualified Data.Map.Strict as Map
|
|
|
|
|
|
|
|
toProcessError :: FilePath
|
|
-> [FilePath]
|
|
-> ExitCode
|
|
-> Either ProcessError ()
|
|
toProcessError exe args exitcode = case exitcode of
|
|
(ExitFailure xi) -> Left $ NonZeroExit xi exe args
|
|
ExitSuccess -> Right ()
|
|
|
|
|
|
-- | @readCreateProcessWithExitCode@ works exactly like 'readProcessWithExitCode' except that it
|
|
-- lets you pass 'CreateProcess' giving better flexibility.
|
|
--
|
|
-- Note that @Handle@s provided for @std_in@, @std_out@, or @std_err@ via the CreateProcess
|
|
-- record will be ignored.
|
|
--
|
|
-- @since 1.2.3.0
|
|
readCreateProcessWithExitCodeBS
|
|
:: CreateProcess
|
|
-> BL.ByteString
|
|
-> IO (ExitCode, BL.ByteString, BL.ByteString) -- ^ exitcode, stdout, stderr
|
|
readCreateProcessWithExitCodeBS cp input = do
|
|
let cp_opts = cp {
|
|
std_in = CreatePipe,
|
|
std_out = CreatePipe,
|
|
std_err = CreatePipe
|
|
}
|
|
withCreateProcess_ "readCreateProcessWithExitCodeBS" cp_opts $
|
|
\mb_inh mb_outh mb_errh ph ->
|
|
case (mb_inh, mb_outh, mb_errh) of
|
|
(Just inh, Just outh, Just errh) -> do
|
|
|
|
out <- BS.hGetContents outh
|
|
err <- BS.hGetContents errh
|
|
|
|
-- fork off threads to start consuming stdout & stderr
|
|
withForkWait (EX.evaluate $ rnf out) $ \waitOut ->
|
|
withForkWait (EX.evaluate $ rnf err) $ \waitErr -> do
|
|
|
|
-- now write any input
|
|
unless (BL.null input) $
|
|
ignoreSigPipe $ BL.hPut inh input
|
|
-- hClose performs implicit hFlush, and thus may trigger a SIGPIPE
|
|
ignoreSigPipe $ hClose inh
|
|
|
|
-- wait on the output
|
|
waitOut
|
|
waitErr
|
|
|
|
hClose outh
|
|
hClose errh
|
|
|
|
-- wait on the process
|
|
ex <- waitForProcess ph
|
|
return (ex, BL.fromStrict out, BL.fromStrict err)
|
|
|
|
(Nothing,_,_) -> error "readCreateProcessWithExitCodeBS: Failed to get a stdin handle."
|
|
(_,Nothing,_) -> error "readCreateProcessWithExitCodeBS: Failed to get a stdout handle."
|
|
(_,_,Nothing) -> error "readCreateProcessWithExitCodeBS: Failed to get a stderr handle."
|
|
where
|
|
ignoreSigPipe :: IO () -> IO ()
|
|
ignoreSigPipe = EX.handle $ \e -> case e of
|
|
IOError { ioe_type = ResourceVanished
|
|
, ioe_errno = Just ioe }
|
|
| Errno ioe == ePIPE -> return ()
|
|
_ -> throwIO e
|
|
-- wrapper so we can get exceptions with the appropriate function name.
|
|
withCreateProcess_
|
|
:: String
|
|
-> CreateProcess
|
|
-> (Maybe Handle -> Maybe Handle -> Maybe Handle -> ProcessHandle -> IO a)
|
|
-> IO a
|
|
withCreateProcess_ fun c action =
|
|
EX.bracketOnError (createProcess_ fun c) cleanupProcess
|
|
(\(m_in, m_out, m_err, ph) -> action m_in m_out m_err ph)
|
|
|
|
-- | Fork a thread while doing something else, but kill it if there's an
|
|
-- exception.
|
|
--
|
|
-- This is important in the cases above because we want to kill the thread
|
|
-- that is holding the Handle lock, because when we clean up the process we
|
|
-- try to close that handle, which could otherwise deadlock.
|
|
--
|
|
withForkWait :: IO () -> (IO () -> IO a) -> IO a
|
|
withForkWait async' body = do
|
|
waitVar <- newEmptyMVar :: IO (MVar (Either SomeException ()))
|
|
mask $ \restore -> do
|
|
tid <- forkIO $ try (restore async') >>= putMVar waitVar
|
|
let wait' = takeMVar waitVar >>= either throwIO return
|
|
restore (body wait') `EX.onException` killThread tid
|
|
|
|
|
|
-- | Execute the given command and collect the stdout, stderr and the exit code.
|
|
-- The command is run in a subprocess.
|
|
executeOut :: MonadIO m
|
|
=> FilePath -- ^ command as filename, e.g. 'ls'
|
|
-> [String] -- ^ arguments to the command
|
|
-> Maybe FilePath -- ^ chdir to this path
|
|
-> m CapturedProcess
|
|
executeOut path args chdir = do
|
|
cp <- createProcessWithMingwPath ((proc path args){ cwd = chdir })
|
|
(exit, out, err) <- liftIO $ readCreateProcessWithExitCodeBS cp ""
|
|
pure $ CapturedProcess exit out err
|
|
|
|
|
|
execLogged :: (MonadReader AppState m, MonadIO m, MonadThrow m)
|
|
=> FilePath -- ^ thing to execute
|
|
-> [String] -- ^ args for the thing
|
|
-> Maybe FilePath -- ^ optionally chdir into this
|
|
-> FilePath -- ^ log filename (opened in append mode)
|
|
-> Maybe [(String, String)] -- ^ optional environment
|
|
-> m (Either ProcessError ())
|
|
execLogged exe args chdir lfile env = do
|
|
AppState { dirs = Dirs {..} } <- ask
|
|
let stdoutLogfile = logsDir </> lfile <> ".stdout.log"
|
|
stderrLogfile = logsDir </> lfile <> ".stderr.log"
|
|
cp <- createProcessWithMingwPath ((proc exe args)
|
|
{ cwd = chdir
|
|
, env = env
|
|
, std_in = CreatePipe
|
|
, std_out = CreatePipe
|
|
, std_err = CreatePipe
|
|
})
|
|
fmap (toProcessError exe args)
|
|
$ liftIO
|
|
$ withCreateProcess cp
|
|
$ \_ mout merr ph ->
|
|
case (mout, merr) of
|
|
(Just cStdout, Just cStderr) -> do
|
|
withForkWait (tee stdoutLogfile cStdout) $ \waitOut ->
|
|
withForkWait (tee stderrLogfile cStderr) $ \waitErr -> do
|
|
waitOut
|
|
waitErr
|
|
waitForProcess ph
|
|
_ -> fail "Could not acquire out/err handle"
|
|
|
|
where
|
|
tee :: FilePath -> Handle -> IO ()
|
|
tee logFile handle' = go
|
|
where
|
|
go = do
|
|
some <- BS.hGetSome handle' 512
|
|
if BS.null some
|
|
then pure ()
|
|
else do
|
|
void $ BS.appendFile logFile some
|
|
void $ BS.hPut stdout some
|
|
go
|
|
|
|
|
|
-- | Thin wrapper around `executeFile`.
|
|
exec :: MonadIO m
|
|
=> FilePath -- ^ thing to execute
|
|
-> [FilePath] -- ^ args for the thing
|
|
-> Maybe FilePath -- ^ optionally chdir into this
|
|
-> Maybe [(String, String)] -- ^ optional environment
|
|
-> m (Either ProcessError ())
|
|
exec exe args chdir env = do
|
|
cp <- createProcessWithMingwPath ((proc exe args) { cwd = chdir, env = env })
|
|
exit_code <- liftIO $ withCreateProcess cp $ \_ _ _ p -> waitForProcess p
|
|
pure $ toProcessError exe args exit_code
|
|
|
|
|
|
chmod_755 :: MonadIO m => FilePath -> m ()
|
|
chmod_755 fp =
|
|
let perm = setOwnerWritable True emptyPermissions
|
|
in liftIO $ setPermissions fp perm
|
|
|
|
|
|
createProcessWithMingwPath :: MonadIO m
|
|
=> CreateProcess
|
|
-> m CreateProcess
|
|
createProcessWithMingwPath cp = do
|
|
msys2Dir <- liftIO ghcupMsys2Dir
|
|
cEnv <- Map.fromList <$> maybe (liftIO getEnvironment) pure (env cp)
|
|
let mingWPaths = [msys2Dir </> "usr" </> "bin"
|
|
,msys2Dir </> "mingw64" </> "bin"]
|
|
paths = ["PATH", "Path"]
|
|
curPaths = (\x -> maybe [] splitSearchPath (Map.lookup x cEnv)) =<< paths
|
|
newPath = intercalate [searchPathSeparator] (mingWPaths ++ curPaths)
|
|
envWithoutPath = foldr (\x y -> Map.delete x y) cEnv paths
|
|
envWithNewPath = Map.insert "Path" newPath envWithoutPath
|
|
liftIO $ setEnv "Path" newPath
|
|
pure $ cp { env = Just $ Map.toList envWithNewPath }
|
|
|
|
ghcupMsys2Dir :: IO FilePath
|
|
ghcupMsys2Dir =
|
|
lookupEnv "GHCUP_MSYS2" >>= \case
|
|
Just fp -> pure fp
|
|
Nothing -> do
|
|
baseDir <- liftIO ghcupBaseDir
|
|
pure (baseDir </> "msys64")
|
|
|
|
-- | Checks whether the binary is a broken link.
|
|
isBrokenSymlink :: FilePath -> IO Bool
|
|
isBrokenSymlink fp = do
|
|
b <- pathIsLink fp
|
|
if b
|
|
then do
|
|
tfp <- getLinkTarget fp
|
|
not <$> doesPathExist
|
|
-- this drops 'symDir' if 'tfp' is absolute
|
|
(takeDirectory fp </> tfp)
|
|
else pure False
|