Skip to content

Commit

Permalink
Abstract out runGhcLibDir into runGhc
Browse files Browse the repository at this point in the history
Now we can just call `cabal exec ghc -- --numeric-version` directly
without the need for guessing where the ghc binary lies.
  • Loading branch information
lukel97 committed Jun 25, 2020
1 parent 2628fe7 commit fdea533
Show file tree
Hide file tree
Showing 6 changed files with 48 additions and 54 deletions.
37 changes: 16 additions & 21 deletions src/HIE/Bios/Cradle.hs
Original file line number Diff line number Diff line change
Expand Up @@ -40,9 +40,7 @@ import System.Environment
import Control.Applicative ((<|>), optional)
import System.IO.Temp
import System.IO.Error (isPermissionError)
import Data.Char
import Data.List
import Data.Foldable
import Data.Ord (Down(..))

import System.PosixCompat.Files
Expand Down Expand Up @@ -217,7 +215,7 @@ defaultCradle cur_dir =
{ actionName = Types.Default
, runCradle = \_ _ ->
return (CradleSuccess (ComponentOptions [] cur_dir []))
, runGhcLibDir = getGhcOnPathLibDir cur_dir
, runGhc = runGhcOnPath cur_dir
}
}

Expand All @@ -231,7 +229,7 @@ noneCradle cur_dir =
, cradleOptsProg = CradleAction
{ actionName = Types.None
, runCradle = \_ _ -> return CradleNone
, runGhcLibDir = pure Nothing
, runGhc = const $ pure Nothing
}
}

Expand All @@ -245,13 +243,13 @@ multiCradle buildCustomCradle cur_dir cs =
, cradleOptsProg = CradleAction
{ actionName = multiActionName
, runCradle = \l fp -> canonicalizePath fp >>= multiAction buildCustomCradle cur_dir cs l
, runGhcLibDir =
, runGhc = \args ->
-- We're being lazy here and just returning the ghc path for the
-- first non-none cradle. This shouldn't matter in practice: all
-- sub cradles should be using the same ghc version!
case filter (not . isNoneCradleConfig) $ map snd cs of
[] -> return Nothing
(cfg:_) -> runGhcLibDir $ cradleOptsProg $
(cfg:_) -> flip runGhc args $ cradleOptsProg $
getCradle buildCustomCradle (cfg, cur_dir)
}
}
Expand Down Expand Up @@ -326,7 +324,7 @@ directCradle wdir args =
{ actionName = Types.Direct
, runCradle = \_ _ ->
return (CradleSuccess (ComponentOptions args wdir []))
, runGhcLibDir = getGhcOnPathLibDir wdir
, runGhc = runGhcOnPath wdir
}
}

Expand All @@ -342,7 +340,7 @@ biosCradle wdir biosCall biosDepsCall =
, cradleOptsProg = CradleAction
{ actionName = Types.Bios
, runCradle = biosAction wdir biosCall biosDepsCall
, runGhcLibDir = getGhcOnPathLibDir wdir
, runGhc = runGhcOnPath wdir
}
}

Expand Down Expand Up @@ -395,15 +393,16 @@ cabalCradle wdir mc =
, cradleOptsProg = CradleAction
{ actionName = Types.Cabal
, runCradle = cabalAction wdir mc
, runGhcLibDir = optional $ fmap trim $ do
, runGhc = \args -> optional $ do
-- Workaround for a cabal-install bug on 3.0.0.0:
-- ./dist-newstyle/tmp/environment.-24811: createDirectory: does not exist (No such file or directory)
-- (It's ok to pass 'dist-newstyle' here, as it can only be changed
-- with the --builddir flag and not cabal.project, which we aren't
-- using in our call to v2-exec)
createDirectoryIfMissing True (wdir </> "dist-newstyle" </> "tmp")
-- Need to pass -v0 otherwise we get "resolving dependencies..."
readProcessWithCwd wdir "cabal" ["v2-exec", "ghc", "-v0", "--", "--print-libdir"] ""
readProcessWithCwd
wdir "cabal" (["v2-exec", "ghc", "-v0", "--"] ++ args) ""
}
}

Expand Down Expand Up @@ -521,8 +520,9 @@ stackCradle wdir mc =
, cradleOptsProg = CradleAction
{ actionName = Types.Stack
, runCradle = stackAction wdir mc
, runGhcLibDir = optional $ fmap trim $
readProcessWithCwd wdir "stack" ["exec", "--silent", "ghc", "--", "--print-libdir"] ""
, runGhc = \args -> optional $
readProcessWithCwd
wdir "stack" (["exec", "--silent", "ghc", "--"] <> args) ""
}
}

Expand Down Expand Up @@ -709,8 +709,8 @@ readProcessWithOutputFile l work_dir cp = do

where
withHieBiosOutput :: [(String,String)] -> (FilePath -> IO a) -> IO a
withHieBiosOutput env action = do
let mbHieBiosOut = lookup hieBiosOutput env
withHieBiosOutput env' action = do
let mbHieBiosOut = lookup hieBiosOutput env'
case mbHieBiosOut of
Just file@(_:_) -> action file
_ -> withSystemTempFile "hie-bios" $
Expand All @@ -726,14 +726,9 @@ makeCradleResult (ex, err, componentDir, gopts) deps =
let compOpts = ComponentOptions gopts componentDir deps
in CradleSuccess compOpts

-- Used for clipping the trailing newlines on some commands
trim :: String -> String
trim = dropWhileEnd isSpace

-- | Calls @ghc --print-libdir@, with just whatever's on the PATH.
getGhcOnPathLibDir :: FilePath -> IO (Maybe FilePath)
getGhcOnPathLibDir wdir = optional $
fmap trim $ readProcessWithCwd wdir "ghc" ["--print-libdir"] ""
runGhcOnPath :: FilePath -> [String] -> IO (Maybe String)
runGhcOnPath wdir args = optional $ readProcessWithCwd wdir "ghc" args ""

-- | Wrapper around 'readCreateProcess' that sets the working directory
readProcessWithCwd :: FilePath -> FilePath -> [String] -> String -> IO String
Expand Down
27 changes: 10 additions & 17 deletions src/HIE/Bios/Environment.hs
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,12 @@ import DynFlags
import qualified GHC.Paths as Paths

import Control.Applicative
import Control.Monad (msum, void)
import Control.Monad (void)
import Control.Monad.Trans.Maybe

import System.Directory
import System.FilePath
import System.Environment (lookupEnv)
import System.Process

import qualified Crypto.Hash.SHA1 as H
import qualified Data.ByteString.Char8 as B
Expand Down Expand Up @@ -78,7 +77,8 @@ getRuntimeGhcLibDir cradle avoidHardcoding =
runMaybeT $ fromNix <|> fromCradle <|> fromGhcPaths
where
fromNix = MaybeT $ lookupEnv "NIX_GHC_LIBDIR"
fromCradle = MaybeT $ runGhcLibDir $ cradleOptsProg cradle
fromCradle = MaybeT $ fmap (fmap trim) $
runGhc (cradleOptsProg cradle) ["--print-libdir"]
fromGhcPaths = MaybeT $ pure $
if avoidHardcoding then Just Paths.libdir else Nothing

Expand All @@ -87,20 +87,9 @@ getRuntimeGhcLibDir cradle avoidHardcoding =
-- fall back to the version of ghc used to compile hie-bios.
getRuntimeGhcVersion :: Cradle a
-> IO String
getRuntimeGhcVersion cradle = fmap (fromMaybe VERSION_ghc) $ runMaybeT $ do
libDir <- MaybeT $ getRuntimeGhcLibDir cradle True
let possibleExes = guessExecutablePathFromLibdir libDir
MaybeT $ msum (fmap getGhcVersion possibleExes)
where
getGhcVersion :: FilePath -> IO (Maybe String)
getGhcVersion ghc = (Just <$> readProcess ghc ["--numeric-version"] "")
<|> (pure Nothing)
-- Taken from ghc-check GHC.Check.Executable
guessExecutablePathFromLibdir :: FilePath -> [FilePath]
guessExecutablePathFromLibdir fp =
[ fp </> "bin" </> "ghc" -- Linux
, fp </> ".." </> "bin" </> "ghc.exe" -- Windows
]
getRuntimeGhcVersion cradle =
fmap (fromMaybe VERSION_ghc) $ fmap (fmap trim) $
runGhc (cradleOptsProg cradle) ["--numeric-version"]

----------------------------------------------------------------

Expand Down Expand Up @@ -325,3 +314,7 @@ value = many1 (satisfy (not . isSpace))

anyToken :: ReadP Char
anyToken = satisfy $ const True

-- Used for clipping the trailing newlines on GHC output
trim :: String -> String
trim = dropWhileEnd isSpace
1 change: 0 additions & 1 deletion src/HIE/Bios/Ghc/Check.hs
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,6 @@ import qualified HIE.Bios.Ghc.Gap as Gap

import qualified DynFlags as G
import qualified GHC as G
import HIE.Bios.Environment

----------------------------------------------------------------

Expand Down
2 changes: 2 additions & 0 deletions src/HIE/Bios/Internal/Debug.hs
Original file line number Diff line number Diff line change
Expand Up @@ -33,13 +33,15 @@ debugInfo fp cradle = unlines <$> do
conf <- findConfig canonFp
crdl <- findCradle' canonFp
ghcLibDir <- getRuntimeGhcLibDir cradle False
ghcVer <- getRuntimeGhcVersion cradle
case res of
CradleSuccess (ComponentOptions gopts croot deps) -> do
return [
"Root directory: " ++ rootDir
, "Component directory: " ++ croot
, "GHC options: " ++ unwords (map quoteIfNeeded gopts)
, "GHC library directory: " ++ show ghcLibDir
, "GHC version: " ++ show ghcVer
, "Config Location: " ++ conf
, "Cradle: " ++ crdl
, "Dependencies: " ++ unwords deps
Expand Down
13 changes: 7 additions & 6 deletions src/HIE/Bios/Types.hs
Original file line number Diff line number Diff line change
Expand Up @@ -53,14 +53,15 @@ data ActionName a
deriving (Show, Eq, Ord, Functor)

data CradleAction a = CradleAction {
actionName :: ActionName a
actionName :: ActionName a
-- ^ Name of the action.
, runCradle :: LoggingFunction -> FilePath -> IO (CradleLoadResult ComponentOptions)
, runCradle :: LoggingFunction -> FilePath -> IO (CradleLoadResult ComponentOptions)
-- ^ Options to compile the given file with.
, runGhcLibDir :: IO (Maybe FilePath)
-- ^ Try to find the path to /the libdir/ of the GHC that
-- this cradle uses to compile stuff normally. You don't
-- want to call this directly, use 'getGhcLibDir' instead.
-- , runGhcLibDir :: IO (Maybe FilePath)
-- -- ^ Try to find the path to /the libdir/ of the GHC that
-- -- this cradle uses to compile stuff normally. You don't
-- -- want to call this directly, use 'getGhcLibDir' instead.
, runGhc :: [String] -> IO (Maybe String)
}
deriving (Functor)

Expand Down
22 changes: 13 additions & 9 deletions tests/BiosTests.hs
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,7 @@ module Main where
import Test.Tasty
import Test.Tasty.HUnit
import Test.Hspec.Expectations
#if __GLASGOW_HASKELL__ < 810
import Test.Tasty.ExpectedFailure
#endif
import qualified GHC as G
import HIE.Bios
import HIE.Bios.Ghc.Api
Expand Down Expand Up @@ -66,7 +64,6 @@ main = do
>> testDirectory isCabalCradle "./tests/projects/multi-cabal/src/Lib.hs"
]
-- TODO: Remove once there's a stackage snapshot for ghc 8.10
#if __GLASGOW_HASKELL__ < 810
++ [ expectFailBecause "stack repl does not fail on an invalid cabal file" $
testCaseSteps "failing-stack" $ testDirectoryFail isStackCradle "./tests/projects/failing-stack/src/Lib.hs"
(\CradleError {..} -> do
Expand All @@ -83,16 +80,13 @@ main = do
$ testDirectory isStackCradle "./tests/projects/space stack/A.hs"
>> testDirectory isStackCradle "./tests/projects/space stack/B.hs"
]
#endif
, testGroup "Implicit cradle tests" $
[ testCaseSteps "implicit-cabal" $ testImplicitCradle "./tests/projects/implicit-cabal/Main.hs" Cabal
-- TODO: Remove once there's a stackage snapshot for ghc 8.10
#if __GLASGOW_HASKELL__ < 810
, testCaseSteps "implicit-stack" $ testImplicitCradle "./tests/projects/implicit-stack/Main.hs" Stack
, testCaseSteps "implicit-stack-multi"
$ testImplicitCradle "./tests/projects/implicit-stack-multi/Main.hs" Stack
>> testImplicitCradle "./tests/projects/implicit-stack-multi/other-package/Main.hs" Stack
#endif
]
]

Expand All @@ -104,16 +98,26 @@ testDirectory :: (Cradle Void -> Bool) -> FilePath -> (String -> IO ()) -> IO ()
testDirectory cradlePred fp step = do
a_fp <- canonicalizePath fp
crd <- initialiseCradle cradlePred a_fp step
step "Get GHC library directory"
step "Get runtime GHC library directory"
testGetGhcLibDir crd
step "Get runtime GHC version"
testGetGhcVersion crd
step "Initialise Flags"
testLoadFile crd a_fp step

-- Here we are testing that the cradle's method of obtaining the ghcLibDir
-- always works.
testGetGhcLibDir :: Cradle a -> IO ()
testGetGhcLibDir crd =
runGhcLibDir (cradleOptsProg crd) `shouldNotReturn` Nothing
getRuntimeGhcLibDir crd True `shouldNotReturn` Nothing

-- Here we are testing that the cradle's method of getting the runtime ghc
-- version is correct - which while testing, should be the version that we have
-- built the tests with. This will fail if you compiled the tests with a ghc
-- that doesn't equal the ghc on your path though :(
testGetGhcVersion :: Cradle a -> IO ()
testGetGhcVersion crd =
getRuntimeGhcVersion crd `shouldReturn` VERSION_ghc

testDirectoryFail :: (Cradle Void -> Bool) -> FilePath -> (CradleError -> Expectation) -> (String -> IO ()) -> IO ()
testDirectoryFail cradlePred fp cradleFailPred step = do
Expand Down Expand Up @@ -203,7 +207,7 @@ stackYaml resolver = unlines ["resolver: " ++ resolver, "packages:", "- ."]
stackYamlResolver :: String
stackYamlResolver =
#if (defined(MIN_VERSION_GLASGOW_HASKELL) && (MIN_VERSION_GLASGOW_HASKELL(8,10,1,0)))
"TODO"
"nightly-2020-06-25"
#elif (defined(MIN_VERSION_GLASGOW_HASKELL) && (MIN_VERSION_GLASGOW_HASKELL(8,8,1,0)))
"lts-15.10"
#elif (defined(MIN_VERSION_GLASGOW_HASKELL) && (MIN_VERSION_GLASGOW_HASKELL(8,6,5,0)))
Expand Down

0 comments on commit fdea533

Please sign in to comment.