diff --git a/Cabal/Cabal.cabal b/Cabal/Cabal.cabal index 9c31cd60faa..47dd82dde23 100644 --- a/Cabal/Cabal.cabal +++ b/Cabal/Cabal.cabal @@ -95,6 +95,10 @@ extra-source-files: tests/PackageTests/CopyComponent/Lib/Main.hs tests/PackageTests/CopyComponent/Lib/p.cabal tests/PackageTests/CopyComponent/Lib/src/P.hs + tests/PackageTests/CustomPreProcess/Hello.hs + tests/PackageTests/CustomPreProcess/MyCustomPreprocessor.hs + tests/PackageTests/CustomPreProcess/Setup.hs + tests/PackageTests/CustomPreProcess/internal-preprocessor-test.cabal tests/PackageTests/DeterministicAr/Lib.hs tests/PackageTests/DeterministicAr/my.cabal tests/PackageTests/DuplicateModuleName/DuplicateModuleName.cabal diff --git a/Cabal/Distribution/Simple/Configure.hs b/Cabal/Distribution/Simple/Configure.hs index 00a2a9302fa..280485de428 100644 --- a/Cabal/Distribution/Simple/Configure.hs +++ b/Cabal/Distribution/Simple/Configure.hs @@ -127,6 +127,8 @@ import Text.PrettyPrint import Distribution.Compat.Environment ( lookupEnv ) import Distribution.Compat.Exception ( catchExit, catchIO ) +import Data.Graph (graphFromEdges, topSort) + -- | The errors that can be thrown when reading the @setup-config@ file. data ConfigStateFileError = ConfigStateFileNoHeader -- ^ No header found. @@ -1436,7 +1438,7 @@ mkComponentsGraph pkg_descr internalPkgDeps = | c <- pkgEnabledComponents pkg_descr ] in case checkComponentsCyclic graph of Just ccycle -> Left [ cname | (_,cname,_) <- ccycle ] - Nothing -> Right [ (c, cdeps) | (c, _, cdeps) <- graph ] + Nothing -> Right [ (c, cdeps) | (c, _, cdeps) <- topSortFromEdges graph ] where -- The dependencies for the given component componentDeps component = @@ -1620,6 +1622,12 @@ computeCompatPackageKey comp pkg_name pkg_version (SimpleUnitId (ComponentId str | otherwise = str +topSortFromEdges :: Ord key => [(node, key, [key])] + -> [(node, key, [key])] +topSortFromEdges es = + let (graph, vertexToNode, _) = graphFromEdges es + in reverse (map vertexToNode (topSort graph)) + mkComponentsLocalBuildInfo :: ConfigFlags -> Compiler -> InstalledPackageIndex @@ -1635,14 +1643,15 @@ mkComponentsLocalBuildInfo cfg comp installedPackages pkg_descr graph flagAssignment = foldM go [] graph where - go :: [(ComponentLocalBuildInfo, [UnitId])] - -> (Component, [ComponentName]) - -> IO [(ComponentLocalBuildInfo, [UnitId])] - go z (component, _) = do + go z (component, dep_cnames) = do clbi <- componentLocalBuildInfo z component - -- TODO: Maybe just store the internal deps in the clbi? - let dep_uids = map fst (filter (\(_,e) -> e `elem` internalPkgDeps) - (componentPackageDeps clbi)) + -- NB: We want to preserve cdeps because it contains extra + -- information like build-tools ordering + let dep_uids = [ componentUnitId dep_clbi + | cname <- dep_cnames + -- Being in z relies on topsort! + , (dep_clbi, _) <- z + , componentLocalName dep_clbi == cname ] return ((clbi, dep_uids):z) -- The allPkgDeps contains all the package deps for the whole package diff --git a/Cabal/tests/PackageTests/CustomPreProcess/A.pre b/Cabal/tests/PackageTests/CustomPreProcess/A.pre new file mode 100644 index 00000000000..f35c5e15530 --- /dev/null +++ b/Cabal/tests/PackageTests/CustomPreProcess/A.pre @@ -0,0 +1,4 @@ +module A where + +a :: String +a = "hello from A" diff --git a/Cabal/tests/PackageTests/CustomPreProcess/Hello.hs b/Cabal/tests/PackageTests/CustomPreProcess/Hello.hs new file mode 100644 index 00000000000..89a5e5a026d --- /dev/null +++ b/Cabal/tests/PackageTests/CustomPreProcess/Hello.hs @@ -0,0 +1,6 @@ +module Main where + +import A + +main :: IO () +main = putStrLn a diff --git a/Cabal/tests/PackageTests/CustomPreProcess/MyCustomPreprocessor.hs b/Cabal/tests/PackageTests/CustomPreProcess/MyCustomPreprocessor.hs new file mode 100644 index 00000000000..07a4ef33900 --- /dev/null +++ b/Cabal/tests/PackageTests/CustomPreProcess/MyCustomPreprocessor.hs @@ -0,0 +1,9 @@ +module Main where + +import System.Directory +import System.Environment + +main :: IO () +main = do + (source:target:_) <- getArgs + copyFile source target diff --git a/Cabal/tests/PackageTests/CustomPreProcess/Setup.hs b/Cabal/tests/PackageTests/CustomPreProcess/Setup.hs new file mode 100644 index 00000000000..18252a186d9 --- /dev/null +++ b/Cabal/tests/PackageTests/CustomPreProcess/Setup.hs @@ -0,0 +1,36 @@ +{-# OPTIONS_GHC -Wall #-} + +import Distribution.PackageDescription +import Distribution.Simple +import Distribution.Simple.LocalBuildInfo +import Distribution.Simple.PreProcess +import Distribution.Simple.Utils +import System.Exit +import System.FilePath +import System.Process (rawSystem) + +main :: IO () +main = defaultMainWithHooks + simpleUserHooks { hookedPreProcessors = [("pre", myCustomPreprocessor)] } + where + myCustomPreprocessor :: BuildInfo -> LocalBuildInfo -> ComponentLocalBuildInfo -> PreProcessor + myCustomPreprocessor _bi lbi _clbi = + PreProcessor { + platformIndependent = True, + runPreProcessor = mkSimplePreProcessor $ \inFile outFile verbosity -> + do info verbosity ("Preprocessing " ++ inFile ++ " to " ++ outFile) + callProcess progPath [inFile, outFile] + } + where + builddir = buildDir lbi + progName = "my-custom-preprocessor" + progPath = builddir progName progName + + -- Backwards compat with process < 1.2. + callProcess :: FilePath -> [String] -> IO () + callProcess path args = + do exitCode <- rawSystem path args + case exitCode of ExitSuccess -> return () + f@(ExitFailure _) -> fail $ "callProcess " ++ show path + ++ " " ++ show args ++ " failed: " + ++ show f diff --git a/Cabal/tests/PackageTests/CustomPreProcess/internal-preprocessor-test.cabal b/Cabal/tests/PackageTests/CustomPreProcess/internal-preprocessor-test.cabal new file mode 100644 index 00000000000..26735e39fa5 --- /dev/null +++ b/Cabal/tests/PackageTests/CustomPreProcess/internal-preprocessor-test.cabal @@ -0,0 +1,31 @@ +name: internal-preprocessor-test +version: 0.1.0.0 +synopsis: Internal custom preprocessor example. +description: See https://github.com/haskell/cabal/issues/1541#issuecomment-30155513 +license: GPL-3 +author: Mikhail Glushenkov +maintainer: mikhail.glushenkov@gmail.com +category: Testing +build-type: Custom +cabal-version: >=1.10 + +-- Note that exe comes before the library. +-- The reason is backwards compat: old versions of Cabal (< 1.18) +-- don't have a proper component build graph, so components are +-- built in declaration order. +executable my-custom-preprocessor + main-is: MyCustomPreprocessor.hs + build-depends: base, directory + default-language: Haskell2010 + +library + exposed-modules: A + build-depends: base + build-tools: my-custom-preprocessor + -- ^ Note the internal dependency. + default-language: Haskell2010 + +executable hello-world + main-is: Hello.hs + build-depends: base, internal-preprocessor-test + default-language: Haskell2010 diff --git a/Cabal/tests/PackageTests/PackageTester.hs b/Cabal/tests/PackageTests/PackageTester.hs index 3fdae604656..f5f70ab7bd2 100644 --- a/Cabal/tests/PackageTests/PackageTester.hs +++ b/Cabal/tests/PackageTests/PackageTester.hs @@ -453,8 +453,7 @@ rawCompileSetup verbosity suite e path = do r <- rawRun verbosity (Just path) (ghcPath suite) e $ [ "--make"] ++ ghcPackageDBParams (ghcVersion suite) (packageDBStack suite) ++ - [ "-hide-all-packages" - , "-package base" + [ "-hide-package Cabal" #ifdef LOCAL_COMPONENT_ID -- This is best, but we don't necessarily have it -- if we're bootstrapping with old Cabal. diff --git a/Cabal/tests/PackageTests/Tests.hs b/Cabal/tests/PackageTests/Tests.hs index a511912bfc5..f0e79e0270f 100644 --- a/Cabal/tests/PackageTests/Tests.hs +++ b/Cabal/tests/PackageTests/Tests.hs @@ -338,6 +338,12 @@ tests config = do cabal "build" ["myprog"] cabal "copy" ["myprog"] + -- Test internal custom preprocessor + tc "CustomPreProcess" $ do + cabal_build [] + runExe' "hello-world" [] + >>= assertOutputContains "hello from A" + where ghc_pkg_guess bin_name = do cwd <- packageDir