diff --git a/Cabal/Distribution/PackageDescription.hs b/Cabal/Distribution/PackageDescription.hs index b6771b8f3e6..fd745299728 100644 --- a/Cabal/Distribution/PackageDescription.hs +++ b/Cabal/Distribution/PackageDescription.hs @@ -64,7 +64,6 @@ module Distribution.PackageDescription ( hasTests, withTest, testModules, - enabledTests, -- * Benchmarks Benchmark(..), @@ -76,7 +75,6 @@ module Distribution.PackageDescription ( hasBenchmarks, withBenchmark, benchmarkModules, - enabledBenchmarks, -- * Build information BuildInfo(..), @@ -176,7 +174,7 @@ data PackageDescription -- PackageDescription we are, the contents of this field are -- either nonsense, or the collected dependencies of *all* the -- components in this package. buildDepends is initialized by - -- 'finalizePackageDescription' and 'flattenPackageDescription'; + -- 'finalizePD' and 'flattenPackageDescription'; -- prior to that, dependency info is stored in the 'CondTree' -- built around a 'GenericPackageDescription'. When this -- resolution is done, dependency info is written to the inner @@ -447,8 +445,12 @@ hasPublicLib p = any f (libraries p) hasLibs :: PackageDescription -> Bool hasLibs p = any (buildable . libBuildInfo) (libraries p) --- |If the package description has a library section, call the given --- function with the library build info as argument. +-- | If the package description has a buildable library section, +-- call the given function with the library build info as argument. +-- You probably want 'withLibLBI' if you have a 'LocalBuildInfo', +-- see the note in +-- "Distribution.Simple.LocalBuildInfo#buildable_vs_enabled_components" +-- for more information. withLib :: PackageDescription -> (Library -> IO ()) -> IO () withLib pkg_descr f = sequence_ [f lib | lib <- libraries pkg_descr, buildable (libBuildInfo lib)] @@ -532,7 +534,10 @@ hasExes :: PackageDescription -> Bool hasExes p = any (buildable . buildInfo) (executables p) -- | Perform the action on each buildable 'Executable' in the package --- description. +-- description. You probably want 'withExeLBI' if you have a +-- 'LocalBuildInfo', see the note in +-- "Distribution.Simple.LocalBuildInfo#buildable_vs_enabled_components" +-- for more information. withExe :: PackageDescription -> (Executable -> IO ()) -> IO () withExe pkg_descr f = sequence_ [f exe | exe <- executables pkg_descr, buildable (buildInfo exe)] @@ -549,13 +554,7 @@ exeModules exe = otherModules (buildInfo exe) data TestSuite = TestSuite { testName :: String, testInterface :: TestSuiteInterface, - testBuildInfo :: BuildInfo, - testEnabled :: Bool - -- TODO: By having a 'testEnabled' field in the PackageDescription, we - -- are mixing build status information (i.e., arguments to 'configure') - -- with static package description information. This is undesirable, but - -- a better solution is waiting on the next overhaul to the - -- GenericPackageDescription -> PackageDescription resolution process. + testBuildInfo :: BuildInfo } deriving (Generic, Show, Read, Eq, Typeable, Data) @@ -593,8 +592,7 @@ instance Monoid TestSuite where mempty = TestSuite { testName = mempty, testInterface = mempty, - testBuildInfo = mempty, - testEnabled = False + testBuildInfo = mempty } mappend = (Semi.<>) @@ -602,8 +600,7 @@ instance Semigroup TestSuite where a <> b = TestSuite { testName = combine' testName, testInterface = combine testInterface, - testBuildInfo = combine testBuildInfo, - testEnabled = testEnabled a || testEnabled b + testBuildInfo = combine testBuildInfo } where combine field = field a `mappend` field b combine' f = case (f a, f b) of @@ -627,14 +624,14 @@ emptyTestSuite = mempty hasTests :: PackageDescription -> Bool hasTests = any (buildable . testBuildInfo) . testSuites --- | Get all the enabled test suites from a package. -enabledTests :: PackageDescription -> [TestSuite] -enabledTests = filter testEnabled . testSuites - -- | Perform an action on each buildable 'TestSuite' in a package. +-- You probably want 'withTestLBI' if you have a 'LocalBuildInfo', see the note in +-- "Distribution.Simple.LocalBuildInfo#buildable_vs_enabled_components" +-- for more information. + withTest :: PackageDescription -> (TestSuite -> IO ()) -> IO () withTest pkg_descr f = - mapM_ f $ filter (buildable . testBuildInfo) $ enabledTests pkg_descr + sequence_ [ f test | test <- testSuites pkg_descr, buildable (testBuildInfo test) ] -- | Get all the module names from a test suite. testModules :: TestSuite -> [ModuleName] @@ -695,9 +692,7 @@ testType test = case testInterface test of data Benchmark = Benchmark { benchmarkName :: String, benchmarkInterface :: BenchmarkInterface, - benchmarkBuildInfo :: BuildInfo, - benchmarkEnabled :: Bool - -- TODO: See TODO for 'testEnabled'. + benchmarkBuildInfo :: BuildInfo } deriving (Generic, Show, Read, Eq, Typeable, Data) @@ -731,8 +726,7 @@ instance Monoid Benchmark where mempty = Benchmark { benchmarkName = mempty, benchmarkInterface = mempty, - benchmarkBuildInfo = mempty, - benchmarkEnabled = False + benchmarkBuildInfo = mempty } mappend = (Semi.<>) @@ -740,8 +734,7 @@ instance Semigroup Benchmark where a <> b = Benchmark { benchmarkName = combine' benchmarkName, benchmarkInterface = combine benchmarkInterface, - benchmarkBuildInfo = combine benchmarkBuildInfo, - benchmarkEnabled = benchmarkEnabled a || benchmarkEnabled b + benchmarkBuildInfo = combine benchmarkBuildInfo } where combine field = field a `mappend` field b combine' f = case (f a, f b) of @@ -765,14 +758,14 @@ emptyBenchmark = mempty hasBenchmarks :: PackageDescription -> Bool hasBenchmarks = any (buildable . benchmarkBuildInfo) . benchmarks --- | Get all the enabled benchmarks from a package. -enabledBenchmarks :: PackageDescription -> [Benchmark] -enabledBenchmarks = filter benchmarkEnabled . benchmarks - -- | Perform an action on each buildable 'Benchmark' in a package. +-- You probably want 'withBenchLBI' if you have a 'LocalBuildInfo', see the note in +-- "Distribution.Simple.LocalBuildInfo#buildable_vs_enabled_components" +-- for more information. + withBenchmark :: PackageDescription -> (Benchmark -> IO ()) -> IO () withBenchmark pkg_descr f = - mapM_ f $ filter (buildable . benchmarkBuildInfo) $ enabledBenchmarks pkg_descr + sequence_ [f bench | bench <- benchmarks pkg_descr, buildable (benchmarkBuildInfo bench)] -- | Get all the module names from a benchmark. benchmarkModules :: Benchmark -> [ModuleName] @@ -935,12 +928,10 @@ allBuildInfo pkg_descr = [ bi | lib <- libraries pkg_descr , buildable bi ] ++ [ bi | tst <- testSuites pkg_descr , let bi = testBuildInfo tst - , buildable bi - , testEnabled tst ] + , buildable bi ] ++ [ bi | tst <- benchmarks pkg_descr , let bi = benchmarkBuildInfo tst - , buildable bi - , benchmarkEnabled tst ] + , buildable bi ] --FIXME: many of the places where this is used, we actually want to look at -- unbuildable bits too, probably need separate functions diff --git a/Cabal/Distribution/PackageDescription/Check.hs b/Cabal/Distribution/PackageDescription/Check.hs index f8eb83afc74..c9e024aae21 100644 --- a/Cabal/Distribution/PackageDescription/Check.hs +++ b/Cabal/Distribution/PackageDescription/Check.hs @@ -43,6 +43,7 @@ import Distribution.Simple.Utils hiding (findPackageDesc, notice) import Distribution.Version import Distribution.Package import Distribution.Text +import Distribution.Simple.LocalBuildInfo hiding (compiler) import Language.Haskell.Extension import Data.Maybe @@ -1282,8 +1283,8 @@ checkPackageVersions pkg = -- pick a single "typical" configuration and check if that has an -- open upper bound. To get a typical configuration we finalise -- using no package index and the current platform. - finalised = finalizePackageDescription - [] (const True) buildPlatform + finalised = finalizePD + [] defaultComponentEnabled (const True) buildPlatform (unknownCompilerInfo (CompilerId buildCompilerFlavor (Version [] [])) NoAbiTag) [] pkg @@ -1294,7 +1295,7 @@ checkPackageVersions pkg = baseDeps = [ vr | Dependency (PackageName "base") vr <- buildDepends pkg' ] - -- Just in case finalizePackageDescription fails for any reason, + -- Just in case finalizePD fails for any reason, -- or if the package doesn't depend on the base package at all, -- then we will just skip the check, since boundedAbove noVersion = True _ -> noVersion diff --git a/Cabal/Distribution/PackageDescription/Configuration.hs b/Cabal/Distribution/PackageDescription/Configuration.hs index 961dae9e201..cd5e9b740f5 100644 --- a/Cabal/Distribution/PackageDescription/Configuration.hs +++ b/Cabal/Distribution/PackageDescription/Configuration.hs @@ -11,12 +11,13 @@ -- Portability : portable -- -- This is about the cabal configurations feature. It exports --- 'finalizePackageDescription' and 'flattenPackageDescription' which are +-- 'finalizePD' and 'flattenPackageDescription' which are -- functions for converting 'GenericPackageDescription's down to -- 'PackageDescription's. It has code for working with the tree of conditions -- and resolving or flattening conditions. module Distribution.PackageDescription.Configuration ( + finalizePD, finalizePackageDescription, flattenPackageDescription, @@ -48,6 +49,7 @@ import Distribution.Text import Distribution.Compat.ReadP as ReadP hiding ( char ) import qualified Distribution.Compat.ReadP as ReadP ( char ) import Distribution.Compat.Semigroup as Semi +import Distribution.Simple.LocalBuildInfo import Control.Arrow (first) import Data.Char ( isAlphaNum ) @@ -211,6 +213,7 @@ instance Semigroup d => Semigroup (DepTestRslt d) where resolveWithFlags :: [(FlagName,[Bool])] -- ^ Domain for each flag name, will be tested in order. + -> ComponentEnabledSpec -> OS -- ^ OS as returned by Distribution.System.buildOS -> Arch -- ^ Arch as returned by Distribution.System.buildArch -> CompilerInfo -- ^ Compiler information @@ -220,7 +223,7 @@ resolveWithFlags :: -> Either [Dependency] (TargetSet PDTagged, FlagAssignment) -- ^ Either the missing dependencies (error case), or a pair of -- (set of build targets with dependencies, chosen flag assignments) -resolveWithFlags dom os arch impl constrs trees checkDeps = +resolveWithFlags dom enabled os arch impl constrs trees checkDeps = either (Left . fromDepMapUnion) Right $ explore (build [] dom) where extraConstrs = toDepMap constrs @@ -245,7 +248,7 @@ resolveWithFlags dom os arch impl constrs trees checkDeps = -- apply additional constraints to all dependencies first (`constrainBy` extraConstrs) . simplifyCondTree (env flags) - deps = overallDependencies targetSet + deps = overallDependencies enabled targetSet in case checkDeps (fromDepMap deps) of DepOk | null ts -> Right (targetSet, flags) | otherwise -> tryAll $ map explore ts @@ -416,15 +419,15 @@ newtype TargetSet a = TargetSet [(DependencyMap, a)] -- | Combine the target-specific dependencies in a TargetSet to give the -- dependencies for the package as a whole. -overallDependencies :: TargetSet PDTagged -> DependencyMap -overallDependencies (TargetSet targets) = mconcat depss +overallDependencies :: ComponentEnabledSpec -> TargetSet PDTagged -> DependencyMap +overallDependencies enabled (TargetSet targets) = mconcat depss where (depss, _) = unzip $ filter (removeDisabledSections . snd) targets removeDisabledSections :: PDTagged -> Bool - removeDisabledSections (Lib _ l) = buildable (libBuildInfo l) - removeDisabledSections (Exe _ e) = buildable (buildInfo e) - removeDisabledSections (Test _ t) = testEnabled t && buildable (testBuildInfo t) - removeDisabledSections (Bench _ b) = benchmarkEnabled b && buildable (benchmarkBuildInfo b) + removeDisabledSections (Lib _ l) = componentEnabled enabled (CLib l) + removeDisabledSections (Exe _ e) = componentEnabled enabled (CExe e) + removeDisabledSections (Test _ t) = componentEnabled enabled (CTest t) + removeDisabledSections (Bench _ b) = componentEnabled enabled (CBench b) removeDisabledSections PDNull = True -- Apply extra constraints to a dependency map. @@ -505,6 +508,10 @@ flattenTaggedTargets (TargetSet targets) = foldr untag ([], [], [], []) targets -- Convert GenericPackageDescription to PackageDescription -- +-- ezyang: Arguably, this should be: +-- data PDTagged = PDComp String Component +-- | PDNull +-- Also, what the heck is the String? The componentName? data PDTagged = Lib String Library | Exe String Executable | Test String TestSuite @@ -547,8 +554,9 @@ instance Semigroup PDTagged where -- On success, it will return the package description and the full flag -- assignment chosen. -- -finalizePackageDescription :: +finalizePD :: FlagAssignment -- ^ Explicitly specified flag assignments + -> ComponentEnabledSpec -> (Dependency -> Bool) -- ^ Is a given dependency satisfiable from the set of -- available packages? If this is unknown then use -- True. @@ -560,7 +568,7 @@ finalizePackageDescription :: (PackageDescription, FlagAssignment) -- ^ Either missing dependencies or the resolved package -- description along with the flag assignments chosen. -finalizePackageDescription userflags satisfyDep +finalizePD userflags enabled satisfyDep (Platform arch os) impl constraints (GenericPackageDescription pkg flags libs0 exes0 tests0 bms0) = case resolveFlags of @@ -569,7 +577,7 @@ finalizePackageDescription userflags satisfyDep , executables = exes' , testSuites = tests' , benchmarks = bms' - , buildDepends = fromDepMap (overallDependencies targetSet) + , buildDepends = fromDepMap (overallDependencies enabled targetSet) } , flagVals ) @@ -582,7 +590,7 @@ finalizePackageDescription userflags satisfyDep ++ map (\(name,tree) -> mapTreeData (Bench name) tree) bms0 resolveFlags = - case resolveWithFlags flagChoices os arch impl constraints condTrees check of + case resolveWithFlags flagChoices enabled os arch impl constraints condTrees check of Right (targetSet, fs) -> let (libs, exes, tests, bms) = flattenTaggedTargets targetSet in Right ( (map (\(n,l) -> (libFillInDefaults l) { libName = n }) libs, @@ -604,6 +612,20 @@ finalizePackageDescription userflags satisfyDep then DepOk else MissingDeps missingDeps +{-# DEPRECATED finalizePackageDescription "This function now always assumes tests and benchmarks are disabled; use finalizePD with ComponentEnabledSpec to specify something more specific." #-} +finalizePackageDescription :: + FlagAssignment -- ^ Explicitly specified flag assignments + -> (Dependency -> Bool) -- ^ Is a given dependency satisfiable from the set of + -- available packages? If this is unknown then use + -- True. + -> Platform -- ^ The 'Arch' and 'OS' + -> CompilerInfo -- ^ Compiler information + -> [Dependency] -- ^ Additional constraints + -> GenericPackageDescription + -> Either [Dependency] + (PackageDescription, FlagAssignment) +finalizePackageDescription flags = finalizePD flags defaultComponentEnabled + {- let tst_p = (CondNode [1::Int] [Distribution.Package.Dependency "a" AnyVersion] []) let tst_p2 = (CondNode [1::Int] [Distribution.Package.Dependency "a" (EarlierVersion (Version [1,0] [])), Distribution.Package.Dependency "a" (LaterVersion (Version [2,0] []))] []) diff --git a/Cabal/Distribution/PackageDescription/PrettyPrint.hs b/Cabal/Distribution/PackageDescription/PrettyPrint.hs index 297219769a1..94e8b887136 100644 --- a/Cabal/Distribution/PackageDescription/PrettyPrint.hs +++ b/Cabal/Distribution/PackageDescription/PrettyPrint.hs @@ -152,10 +152,10 @@ ppTestSuites suites = maybeTestType | testInterface testsuite == mempty = Nothing | otherwise = Just (testType testsuite) - ppTestSuite (TestSuite _ _ buildInfo' _) - (Just (TestSuite _ _ buildInfo2 _)) = - ppDiffFields binfoFieldDescrs buildInfo' buildInfo2 - $+$ ppCustomFields (customFieldsBI buildInfo') + ppTestSuite test' (Just test2) = + ppDiffFields binfoFieldDescrs + (testBuildInfo test') (testBuildInfo test2) + $+$ ppCustomFields (customFieldsBI (testBuildInfo test')) testSuiteMainIs test = case testInterface test of TestSuiteExeV10 _ f -> Just f @@ -182,10 +182,10 @@ ppBenchmarks suites = maybeBenchmarkType | benchmarkInterface benchmark == mempty = Nothing | otherwise = Just (benchmarkType benchmark) - ppBenchmark (Benchmark _ _ buildInfo' _) - (Just (Benchmark _ _ buildInfo2 _)) = - ppDiffFields binfoFieldDescrs buildInfo' buildInfo2 - $+$ ppCustomFields (customFieldsBI buildInfo') + ppBenchmark bench' (Just bench2) = + ppDiffFields binfoFieldDescrs + (benchmarkBuildInfo bench') (benchmarkBuildInfo bench2) + $+$ ppCustomFields (customFieldsBI (benchmarkBuildInfo bench')) benchmarkMainIs benchmark = case benchmarkInterface benchmark of BenchmarkExeV10 _ f -> Just f diff --git a/Cabal/Distribution/Simple/Bench.hs b/Cabal/Distribution/Simple/Bench.hs index f4f255a3146..d30d614bfe2 100644 --- a/Cabal/Distribution/Simple/Bench.hs +++ b/Cabal/Distribution/Simple/Bench.hs @@ -40,9 +40,7 @@ bench args pkg_descr lbi flags = do let verbosity = fromFlag $ benchmarkVerbosity flags benchmarkNames = args pkgBenchmarks = PD.benchmarks pkg_descr - enabledBenchmarks = [ t | t <- pkgBenchmarks - , PD.benchmarkEnabled t - , PD.buildable (PD.benchmarkBuildInfo t) ] + enabledBenchmarks = map fst (LBI.enabledBenchLBIs pkg_descr lbi) -- Run the benchmark doBench :: PD.Benchmark -> IO ExitCode diff --git a/Cabal/Distribution/Simple/Build.hs b/Cabal/Distribution/Simple/Build.hs index c9bd3c1371f..5b9c0f23a8e 100644 --- a/Cabal/Distribution/Simple/Build.hs +++ b/Cabal/Distribution/Simple/Build.hs @@ -81,7 +81,7 @@ build pkg_descr lbi flags suffixes -- a --assume-deps-up-to-date with multiple arguments. Arguably, we should -- error early in this case. targets <- readBuildTargets pkg_descr (buildArgs flags) - (cname, _) <- checkBuildTargets verbosity pkg_descr targets >>= \r -> case r of + (cname, _) <- checkBuildTargets verbosity pkg_descr lbi targets >>= \r -> case r of [] -> die "In --assume-deps-up-to-date mode you must specify a target" [target'] -> return target' _ -> die "In --assume-deps-up-to-date mode you can only build a single target" @@ -106,7 +106,7 @@ build pkg_descr lbi flags suffixes lbi' suffixes comp clbi distPref | otherwise = do targets <- readBuildTargets pkg_descr (buildArgs flags) - targets' <- checkBuildTargets verbosity pkg_descr targets + targets' <- checkBuildTargets verbosity pkg_descr lbi targets let componentsToBuild = componentsInBuildOrder lbi (map fst targets') info verbosity $ "Component build order: " ++ intercalate ", " (map (showComponentName . componentLocalName) componentsToBuild) @@ -145,9 +145,10 @@ repl pkg_descr lbi flags suffixes args = do targets <- readBuildTargets pkg_descr args targets' <- case targets of + -- This seems DEEPLY questionable. [] -> return $ take 1 [ componentName c - | c <- pkgEnabledComponents pkg_descr ] - [target] -> fmap (map fst) (checkBuildTargets verbosity pkg_descr [target]) + | c <- pkgBuildableComponents pkg_descr ] + [target] -> fmap (map fst) (checkBuildTargets verbosity pkg_descr lbi [target]) _ -> die $ "The 'repl' command does not support multiple targets at once." let componentsToBuild = componentsInBuildOrder lbi targets' componentForRepl = last componentsToBuild diff --git a/Cabal/Distribution/Simple/BuildTarget.hs b/Cabal/Distribution/Simple/BuildTarget.hs index aec56e6a4cf..9cc29242f2e 100644 --- a/Cabal/Distribution/Simple/BuildTarget.hs +++ b/Cabal/Distribution/Simple/BuildTarget.hs @@ -947,16 +947,17 @@ caseFold = lowercase -- -- Also swizzle into a more convenient form. -- -checkBuildTargets :: Verbosity -> PackageDescription -> [BuildTarget] +checkBuildTargets :: Verbosity -> PackageDescription -> LocalBuildInfo -> [BuildTarget] -> IO [(ComponentName, Maybe (Either ModuleName FilePath))] -checkBuildTargets _ pkg [] = - return [ (componentName c, Nothing) | c <- pkgEnabledComponents pkg ] +checkBuildTargets _ pkg lbi [] = + return [ (componentName c, Nothing) | c <- enabledComponents pkg lbi ] -checkBuildTargets verbosity pkg targets = do +checkBuildTargets verbosity pkg lbi targets = do let (enabled, disabled) = partitionEithers - [ case componentDisabledReason (getComponent pkg cname) of + [ case componentDisabledReason (componentEnabledSpec lbi) + (getComponent pkg cname) of Nothing -> Left target' Just reason -> Right (cname, reason) | target <- targets diff --git a/Cabal/Distribution/Simple/Configure.hs b/Cabal/Distribution/Simple/Configure.hs index 645bb5697f8..377d139ce07 100644 --- a/Cabal/Distribution/Simple/Configure.hs +++ b/Cabal/Distribution/Simple/Configure.hs @@ -373,6 +373,12 @@ configure (pkg_descr0', pbi) cfg = do let internalPackageSet :: InstalledPackageIndex internalPackageSet = getInternalPackages pkg_descr0 + -- Make a data structure describing what components are enabled. + let enabled :: ComponentEnabledSpec + enabled = ComponentEnabledSpec + { testsEnabled = fromFlag (configTests cfg) + , benchmarksEnabled = fromFlag (configBenchmarks cfg) } + -- allConstraints: The set of all 'Dependency's we have. Used ONLY -- to 'configureFinalizedPackage'. -- requiredDepsMap: A map from 'PackageName' to the specifically @@ -410,7 +416,7 @@ configure (pkg_descr0', pbi) cfg = do -- cleaner to then configure the dependencies afterwards. (pkg_descr :: PackageDescription, flags :: FlagAssignment) - <- configureFinalizedPackage verbosity cfg + <- configureFinalizedPackage verbosity cfg enabled allConstraints (dependencySatisfiable (fromFlagOrDefault False (configExactConfiguration cfg)) @@ -571,7 +577,7 @@ configure (pkg_descr0', pbi) cfg = do -- From there, we build a ComponentLocalBuildInfo for each of the -- components, which lets us actually build each component. buildComponents <- - case mkComponentsGraph pkg_descr internalPkgDeps of + case mkComponentsGraph enabled pkg_descr internalPkgDeps of Left componentCycle -> reportComponentCycle componentCycle Right comps -> mkComponentsLocalBuildInfo cfg comp packageDependsIndex pkg_descr @@ -668,6 +674,7 @@ configure (pkg_descr0', pbi) cfg = do let lbi = LocalBuildInfo { configFlags = cfg', flagAssignment = flags, + componentEnabledSpec = enabled, extraConfigArgs = [], -- Currently configure does not -- take extra args, but if it -- did they would go here. @@ -800,7 +807,7 @@ checkExactConfiguration pkg_descr0 cfg = do -- -- It must be *any libraries that might be* defined rather than the -- actual definitions, because these depend on conditionals in the .cabal --- file, and we haven't resolved them yet. finalizePackageDescription +-- file, and we haven't resolved them yet. finalizePD -- does the resolution of conditionals, and it takes internalPackageSet -- as part of its input. getInternalPackages :: GenericPackageDescription @@ -825,7 +832,7 @@ getInternalPackages pkg_descr0 = -- | Returns true if a dependency is satisfiable. This is to be passed --- to finalizePackageDescription. +-- to finalizePD. dependencySatisfiable :: Bool -> InstalledPackageIndex -- ^ installed set @@ -841,7 +848,7 @@ dependencySatisfiable -- line. Thus we only consult the 'requiredDepsMap'. Note that -- we're not doing the version range check, so if there's some -- dependency that wasn't specified on the command line, - -- 'finalizePackageDescription' will fail. + -- 'finalizePD' will fail. -- -- TODO: mention '--exact-configuration' in the error message -- when this fails? @@ -884,7 +891,7 @@ relaxPackageDeps (RelaxDepsSome allowNewerDeps') gpd = else d -- | Finalize a generic package description. The workhorse is --- 'finalizePackageDescription' but there's a bit of other nattering +-- 'finalizePD' but there's a bit of other nattering -- about necessary. -- -- TODO: what exactly is the business with @flaggedTests@ and @@ -892,6 +899,7 @@ relaxPackageDeps (RelaxDepsSome allowNewerDeps') gpd = configureFinalizedPackage :: Verbosity -> ConfigFlags + -> ComponentEnabledSpec -> [Dependency] -> (Dependency -> Bool) -- ^ tests if a dependency is satisfiable. -- Might say it's satisfiable even when not. @@ -899,27 +907,18 @@ configureFinalizedPackage -> Platform -> GenericPackageDescription -> IO (PackageDescription, FlagAssignment) -configureFinalizedPackage verbosity cfg +configureFinalizedPackage verbosity cfg enabled allConstraints satisfies comp compPlatform pkg_descr0 = do - let enableTest t = t { testEnabled = fromFlag (configTests cfg) } - flaggedTests = map (\(n, t) -> (n, mapTreeData enableTest t)) - (condTestSuites pkg_descr0) - enableBenchmark bm = bm { benchmarkEnabled = - fromFlag (configBenchmarks cfg) } - flaggedBenchmarks = map (\(n, bm) -> - (n, mapTreeData enableBenchmark bm)) - (condBenchmarks pkg_descr0) - pkg_descr0'' = pkg_descr0 { condTestSuites = flaggedTests - , condBenchmarks = flaggedBenchmarks } (pkg_descr0', flags) <- - case finalizePackageDescription + case finalizePD (configConfigurationsFlags cfg) + enabled satisfies compPlatform (compilerInfo comp) allConstraints - pkg_descr0'' + pkg_descr0 of Right r -> return r Left missing -> die $ "Encountered missing dependencies:\n" @@ -1435,13 +1434,15 @@ configCompilerAux = fmap (\(a,_,b) -> (a,b)) . configCompilerAuxEx -- (although it is in the absence of Backpack.) -- -- TODO: tighten up the type of 'internalPkgDeps' -mkComponentsGraph :: PackageDescription +mkComponentsGraph :: ComponentEnabledSpec + -> PackageDescription -> [PackageId] -> Either [ComponentName] [(Component, [ComponentName])] -mkComponentsGraph pkg_descr internalPkgDeps = +mkComponentsGraph enabled pkg_descr internalPkgDeps = let graph = [ (c, componentName c, componentDeps c) - | c <- pkgEnabledComponents pkg_descr ] + | c <- pkgBuildableComponents pkg_descr + , componentEnabled enabled c ] in case checkComponentsCyclic graph of Just ccycle -> Left [ cname | (_,cname,_) <- ccycle ] Nothing -> Right [ (c, cdeps) | (c, _, cdeps) <- topSortFromEdges graph ] diff --git a/Cabal/Distribution/Simple/Install.hs b/Cabal/Distribution/Simple/Install.hs index 0bec0e9fb9d..1773b9d5e8c 100644 --- a/Cabal/Distribution/Simple/Install.hs +++ b/Cabal/Distribution/Simple/Install.hs @@ -61,7 +61,7 @@ install pkg_descr lbi flags | fromFlag (copyAssumeDepsUpToDate flags) = do checkHasLibsOrExes targets <- readBuildTargets pkg_descr (copyArgs flags) - targets' <- checkBuildTargets verbosity pkg_descr targets + targets' <- checkBuildTargets verbosity pkg_descr lbi targets case targets' of _ | null (copyArgs flags) -> copyPackage verbosity pkg_descr lbi distPref copydest @@ -74,7 +74,7 @@ install pkg_descr lbi flags | otherwise = do checkHasLibsOrExes targets <- readBuildTargets pkg_descr (copyArgs flags) - targets' <- checkBuildTargets verbosity pkg_descr targets + targets' <- checkBuildTargets verbosity pkg_descr lbi targets copyPackage verbosity pkg_descr lbi distPref copydest diff --git a/Cabal/Distribution/Simple/LocalBuildInfo.hs b/Cabal/Distribution/Simple/LocalBuildInfo.hs index b241ac71f43..222dd065432 100644 --- a/Cabal/Distribution/Simple/LocalBuildInfo.hs +++ b/Cabal/Distribution/Simple/LocalBuildInfo.hs @@ -38,11 +38,9 @@ module Distribution.Simple.LocalBuildInfo ( foldComponent, componentName, componentBuildInfo, - componentEnabled, - componentDisabledReason, - ComponentDisabledReason(..), + componentBuildable, pkgComponents, - pkgEnabledComponents, + pkgBuildableComponents, lookupComponent, getComponent, maybeGetDefaultLibraryLocalBuildInfo, @@ -58,7 +56,19 @@ module Distribution.Simple.LocalBuildInfo ( withComponentsLBI, withLibLBI, withExeLBI, + withBenchLBI, withTestLBI, + enabledTestLBIs, + enabledBenchLBIs, + enabledComponents, + + -- $buildable_vs_enabled_components + + ComponentEnabledSpec(..), + defaultComponentEnabled, + componentEnabled, + componentDisabledReason, + ComponentDisabledReason(..), -- * Installation directories module Distribution.Simple.InstallDirs, @@ -102,6 +112,8 @@ data LocalBuildInfo = LocalBuildInfo { -- Needed to re-run configuration when .cabal is out of date flagAssignment :: FlagAssignment, -- ^ The final set of flags which were picked for this package + componentEnabledSpec :: ComponentEnabledSpec, + -- ^ What components were enabled during configuration, and why. extraConfigArgs :: [String], -- ^ Extra args on the command line for the configuration step. -- Needed to re-run configuration when .cabal is out of date @@ -152,6 +164,76 @@ data LocalBuildInfo = LocalBuildInfo { instance Binary LocalBuildInfo +-- $buildable_vs_enabled_components +-- #buildable_vs_enabled_components# +-- +-- = Note: Buildable versus enabled components +-- What's the difference between a buildable component (ala +-- 'componentBuildable') versus enabled component (ala +-- 'componentEnabled')? +-- +-- A component is __buildable__ if, after resolving flags and +-- conditionals, there is no @buildable: False@ property in it. +-- This is a /static/ property that arises from the +-- Cabal file and the package description flattening; once we have +-- a 'PackageDescription' buildability is known. +-- +-- A component is __enabled__ if it is buildable, and the user +-- configured (@./Setup configure@) the package to build it, +-- e.g., using @--enable-tests@ or @--enable-benchmarks@. +-- Once we have a 'LocalBuildInfo', whether or not a component +-- is enabled is known. +-- +-- Generally speaking, most Cabal API code cares if a component +-- is enabled, as opposed to buildable. (For example, if you +-- want to run a preprocessor on each component prior to building +-- them, you want to run this on each /enabled/ component.) + +-- | Describes what components are enabled by user-interaction. +-- See also this note in +-- "Distribution.Simple.LocalBuildInfo#buildable_vs_enabled_components". +-- +-- @since 1.26.0.0 +data ComponentEnabledSpec + = ComponentEnabledSpec { + testsEnabled :: Bool, + benchmarksEnabled :: Bool + } + deriving (Generic, Read, Show) +instance Binary ComponentEnabledSpec + +-- | The default set of enabled components. Historically tests and +-- benchmarks are NOT enabled by default. +-- +-- @since 1.26.0.0 +defaultComponentEnabled :: ComponentEnabledSpec +defaultComponentEnabled = ComponentEnabledSpec False False + +-- | Is this component enabled? See also this note in +-- "Distribution.Simple.LocalBuildInfo#buildable_vs_enabled_components". +-- +-- @since 1.26.0.0 +componentEnabled :: ComponentEnabledSpec -> Component -> Bool +componentEnabled enabled = isNothing . componentDisabledReason enabled + +-- | Is this component disabled, and if so, why? +-- +-- @since 1.26.0.0 +componentDisabledReason :: ComponentEnabledSpec -> Component + -> Maybe ComponentDisabledReason +componentDisabledReason enabled (CTest _) + | not (testsEnabled enabled) = Just DisabledAllTests +componentDisabledReason enabled (CBench _) + | not (benchmarksEnabled enabled) = Just DisabledAllBenchmarks +componentDisabledReason _ _ = Nothing + +-- | A reason explaining why a component is disabled. +-- +-- @since 1.26.0.0 +data ComponentDisabledReason = DisabledComponent + | DisabledAllTests + | DisabledAllBenchmarks + -- TODO: Get rid of these functions, as much as possible. They are -- a bit useful in some cases, but you should be very careful! @@ -245,7 +327,7 @@ componentName = (CTestName . testName) (CBenchName . benchmarkName) --- | All the components in the package (libs, exes, or test suites). +-- | All the components in the package. -- pkgComponents :: PackageDescription -> [Component] pkgComponents pkg = @@ -254,32 +336,24 @@ pkgComponents pkg = ++ [ CTest tst | tst <- testSuites pkg ] ++ [ CBench bm | bm <- benchmarks pkg ] --- | All the components in the package that are buildable and enabled. --- Thus this excludes non-buildable components and test suites or benchmarks --- that have been disabled. +-- | A list of all components in the package that are buildable, +-- i.e., were not marked with @buildable: False@. This does NOT +-- indicate if we are actually going to build the component, +-- see 'enabledComponents' instead. -- -pkgEnabledComponents :: PackageDescription -> [Component] -pkgEnabledComponents = filter componentEnabled . pkgComponents - -componentEnabled :: Component -> Bool -componentEnabled = isNothing . componentDisabledReason - -data ComponentDisabledReason = DisabledComponent - | DisabledAllTests - | DisabledAllBenchmarks +-- @since 1.26.0.0 +-- +pkgBuildableComponents :: PackageDescription -> [Component] +pkgBuildableComponents = filter componentBuildable . pkgComponents -componentDisabledReason :: Component -> Maybe ComponentDisabledReason -componentDisabledReason (CLib lib) - | not (buildable (libBuildInfo lib)) = Just DisabledComponent -componentDisabledReason (CExe exe) - | not (buildable (buildInfo exe)) = Just DisabledComponent -componentDisabledReason (CTest tst) - | not (buildable (testBuildInfo tst)) = Just DisabledComponent - | not (testEnabled tst) = Just DisabledAllTests -componentDisabledReason (CBench bm) - | not (buildable (benchmarkBuildInfo bm)) = Just DisabledComponent - | not (benchmarkEnabled bm) = Just DisabledAllBenchmarks -componentDisabledReason _ = Nothing +-- | Is a component buildable (i.e., not marked with @buildable: False@)? +-- See also this note in +-- "Distribution.Simple.LocalBuildInfo#buildable_vs_enabled_components". +-- +-- @since 1.26.0.0 +-- +componentBuildable :: Component -> Bool +componentBuildable = buildable . componentBuildInfo lookupComponent :: PackageDescription -> ComponentName -> Maybe Component lookupComponent pkg (CLibName "") = lookupComponent pkg (defaultLibName (package pkg)) @@ -416,9 +490,8 @@ componentNameToUnitIds lbi cname = | (clbi, _) <- componentsConfigs lbi , componentName (getLocalComponent (localPkgDescr lbi) clbi) == cname ] --- | Perform the action on each buildable 'library' in the package --- description. Extended version of 'withLib' that also gives --- corresponding build info. +-- | Perform the action on each enabled 'library' in the package +-- description with the 'ComponentLocalBuildInfo'. withLibLBI :: PackageDescription -> LocalBuildInfo -> (Library -> ComponentLocalBuildInfo -> IO ()) -> IO () withLibLBI pkg lbi f = @@ -427,7 +500,7 @@ withLibLBI pkg lbi f = | (clbi@LibComponentLocalBuildInfo{}, _) <- componentsConfigs lbi , CLib lib <- [getComponent pkg (componentLocalName clbi)] ] --- | Perform the action on each buildable 'Executable' in the package +-- | Perform the action on each enabled 'Executable' in the package -- description. Extended version of 'withExe' that also gives corresponding -- build info. withExeLBI :: PackageDescription -> LocalBuildInfo @@ -438,14 +511,43 @@ withExeLBI pkg lbi f = | (clbi@ExeComponentLocalBuildInfo{}, _) <- componentsConfigs lbi , CExe exe <- [getComponent pkg (componentLocalName clbi)] ] +-- | Perform the action on each enabled 'Benchmark' in the package +-- description. +withBenchLBI :: PackageDescription -> LocalBuildInfo + -> (Benchmark -> ComponentLocalBuildInfo -> IO ()) -> IO () +withBenchLBI pkg lbi f = + sequence_ [ f test clbi | (test, clbi) <- enabledBenchLBIs pkg lbi ] + withTestLBI :: PackageDescription -> LocalBuildInfo -> (TestSuite -> ComponentLocalBuildInfo -> IO ()) -> IO () withTestLBI pkg lbi f = - sequence_ - [ f test clbi + sequence_ [ f test clbi | (test, clbi) <- enabledTestLBIs pkg lbi ] + +enabledTestLBIs :: PackageDescription -> LocalBuildInfo + -> [(TestSuite, ComponentLocalBuildInfo)] +enabledTestLBIs pkg lbi = + [ (test, clbi) | (clbi@TestComponentLocalBuildInfo{}, _) <- componentsConfigs lbi , CTest test <- [getComponent pkg (componentLocalName clbi)] ] +enabledBenchLBIs :: PackageDescription -> LocalBuildInfo + -> [(Benchmark, ComponentLocalBuildInfo)] +enabledBenchLBIs pkg lbi = + [ (test, clbi) + | (clbi@BenchComponentLocalBuildInfo{}, _) <- componentsConfigs lbi + , CBench test <- [getComponent pkg (componentLocalName clbi)] ] + +-- | Get a list of all enabled 'Component's (both buildable and +-- requested by the user at configure-time). +-- +-- @since 1.26.0.0 +enabledComponents :: PackageDescription -> LocalBuildInfo + -> [Component] +enabledComponents pkg lbi = + [ getComponent pkg (componentLocalName clbi) + | (clbi, _) <- componentsConfigs lbi ] + + {-# DEPRECATED withComponentsLBI "Use withAllComponentsInBuildOrder" #-} withComponentsLBI :: PackageDescription -> LocalBuildInfo -> (Component -> ComponentLocalBuildInfo -> IO ()) diff --git a/Cabal/Distribution/Simple/PreProcess.hs b/Cabal/Distribution/Simple/PreProcess.hs index bc76e38fbf2..df11ade8bb7 100644 --- a/Cabal/Distribution/Simple/PreProcess.hs +++ b/Cabal/Distribution/Simple/PreProcess.hs @@ -47,7 +47,7 @@ import Distribution.Verbosity import Control.Monad import Data.Maybe (fromMaybe) -import Data.List (nub, isSuffixOf) +import Data.List (nub, isSuffixOf, isPrefixOf) import System.Directory (doesFileExist) import System.Info (os, arch) import System.FilePath (splitExtension, dropExtensions, (), (<.>), @@ -650,4 +650,21 @@ preprocessExtras comp lbi = case comp of BenchmarkUnsupported tt -> die $ "No support for preprocessing benchmark " ++ "type " ++ display tt where - pp dir = (map (dir ) . concat) `fmap` forM knownExtrasHandlers ($ dir) + pp dir = (map (dir ) . filter not_sub . concat) `fmap` forM knownExtrasHandlers ($ dir) + -- TODO: This is a terrible hack to work around #3545 while we don't + -- reorganize the directory layout. Basically, for the main + -- library, we might accidentally pick up autogenerated sources for + -- our subcomponents, because they are all stored as subdirectories + -- in dist/build. This is a cheap and cheerful check to prevent + -- this from happening. It is not particularly correct; for example + -- if a user has a test suite named foobar and puts their C file in + -- foobar/foo.c, this test will incorrectly exclude it. But I + -- didn't want to break BC... + not_sub p = and [ not (pre `isPrefixOf` p) | pre <- component_dirs ] + component_dirs = component_names (localPkgDescr lbi) + -- TODO: libify me + component_names pkg_descr = + map libName (libraries pkg_descr) ++ + map exeName (executables pkg_descr) ++ + map testName (testSuites pkg_descr) ++ + map benchmarkName (benchmarks pkg_descr) diff --git a/Cabal/Distribution/Simple/Register.hs b/Cabal/Distribution/Simple/Register.hs index 24d636e539b..b04c2adf3a3 100644 --- a/Cabal/Distribution/Simple/Register.hs +++ b/Cabal/Distribution/Simple/Register.hs @@ -91,7 +91,7 @@ register pkg_descr lbi flags = when (hasPublicLib pkg_descr) doRegister -- packages, we'll have to relax this. doRegister = do targets <- readBuildTargets pkg_descr (regArgs flags) - targets' <- checkBuildTargets verbosity pkg_descr targets + targets' <- checkBuildTargets verbosity pkg_descr lbi targets -- It's important to register in build order, because ghc-pkg -- will complain if a dependency is not registered. diff --git a/Cabal/Distribution/Simple/Test.hs b/Cabal/Distribution/Simple/Test.hs index f7cd437f220..efa2cdc8cfd 100644 --- a/Cabal/Distribution/Simple/Test.hs +++ b/Cabal/Distribution/Simple/Test.hs @@ -49,9 +49,7 @@ test args pkg_descr lbi flags = do testLogDir = distPref "test" testNames = args pkgTests = PD.testSuites pkg_descr - enabledTests = [ t | t <- pkgTests - , PD.testEnabled t - , PD.buildable (PD.testBuildInfo t) ] + enabledTests = (map fst) (LBI.enabledTestLBIs pkg_descr lbi) doTest :: (PD.TestSuite, Maybe TestSuiteLog) -> IO TestSuiteLog doTest (suite, _) = diff --git a/Cabal/changelog b/Cabal/changelog index dca40054d36..a3e90a2e6cd 100644 --- a/Cabal/changelog +++ b/Cabal/changelog @@ -22,6 +22,27 @@ Similarly, 'autogenModulesDir' now requires a 'ComponentLocalBuildInfo' argument as well in order to disambiguate which component the autogenerated files are for. + * Backwards incompatible change to 'Component': 'TestSuite' and + 'Benchmark' no longer have 'testEnabled' and + 'benchmarkEnabled'. If you used + 'enabledTests' or 'enabledBenchmarks', please instead use + 'enabledTestLBIs' and 'enabledBenchLBIs' + (you will need a 'LocalBuildInfo' for these functions.) + Additionally, the semantics of 'withTest' and 'withBench' + have changed: they now iterate over all buildable + such components, regardless of whether or not they have + been enabled; if you only want enabled components, + use 'withTestLBI' and 'withBenchLBI'. + 'finalizePackageDescription' is deprecated: + its replacement 'finalizePD' now takes an extra argument + 'ComponentEnabledSpec' which specifies what components + are to be enabled: use this instead of modifying the + 'Component' in a 'GenericPackageDescription'. (As + it's not possible now, 'finalizePackageDescription' + will assume tests/benchmarks are disabled.) + If you only need to test if a component is buildable + (i.e., it is marked buildable in the Cabal file) + use the new function 'componentBuildable'. * Improved an error message for process output decoding errors (#3408). diff --git a/Cabal/tests/PackageTests/BenchmarkStanza/Check.hs b/Cabal/tests/PackageTests/BenchmarkStanza/Check.hs index f300d93cd38..ffa53d75e64 100644 --- a/Cabal/tests/PackageTests/BenchmarkStanza/Check.hs +++ b/Cabal/tests/PackageTests/BenchmarkStanza/Check.hs @@ -22,7 +22,6 @@ suite = do [ Dependency (PackageName "base") anyVersion ] , hsSourceDirs = ["."] } - , benchmarkEnabled = False } gotBenchmark = head $ benchmarks (localPkgDescr lbi) assertEqual "parsed benchmark stanza does not match anticipated" diff --git a/Cabal/tests/PackageTests/TestStanza/Check.hs b/Cabal/tests/PackageTests/TestStanza/Check.hs index 9f4947c9f6f..cce4addf03f 100644 --- a/Cabal/tests/PackageTests/TestStanza/Check.hs +++ b/Cabal/tests/PackageTests/TestStanza/Check.hs @@ -21,7 +21,6 @@ suite = do [ Dependency (PackageName "base") anyVersion ] , hsSourceDirs = ["."] } - , testEnabled = False } gotTestSuite = head $ testSuites (localPkgDescr lbi) assertEqual "parsed test-suite stanza does not match anticipated" diff --git a/cabal-install/Distribution/Client/Configure.hs b/cabal-install/Distribution/Client/Configure.hs index 6d351c04933..aaf77db405e 100644 --- a/cabal-install/Distribution/Client/Configure.hs +++ b/cabal-install/Distribution/Client/Configure.hs @@ -62,7 +62,7 @@ import qualified Distribution.PackageDescription as PkgDesc import Distribution.PackageDescription.Parse ( readPackageDescription ) import Distribution.PackageDescription.Configuration - ( finalizePackageDescription ) + ( finalizePD ) import Distribution.Version ( anyVersion, thisVersion ) import Distribution.Simple.Utils as Utils @@ -386,8 +386,8 @@ configurePackage verbosity platform comp scriptOptions configFlags configTests = toFlag (TestStanzas `elem` stanzas) } - pkg = case finalizePackageDescription flags + pkg = case finalizePD flags (enableStanzas stanzas) (const True) - platform comp [] (enableStanzas stanzas gpkg) of - Left _ -> error "finalizePackageDescription ReadyPackage failed" + platform comp [] gpkg of + Left _ -> error "finalizePD ReadyPackage failed" Right (desc, _) -> desc diff --git a/cabal-install/Distribution/Client/Dependency.hs b/cabal-install/Distribution/Client/Dependency.hs index 1b5c387a422..02d19898f9d 100644 --- a/cabal-install/Distribution/Client/Dependency.hs +++ b/cabal-install/Distribution/Client/Dependency.hs @@ -74,8 +74,7 @@ import Distribution.Client.SolverInstallPlan (SolverInstallPlan) import qualified Distribution.Client.SolverInstallPlan as SolverInstallPlan import Distribution.Client.Types ( SourcePackageDb(SourcePackageDb) - , UnresolvedPkgLoc, UnresolvedSourcePackage - , enableStanzas ) + , UnresolvedPkgLoc, UnresolvedSourcePackage ) import Distribution.Client.Dependency.Types ( PreSolver(..), Solver(..) , PackagesPreferenceDefault(..) ) @@ -90,7 +89,7 @@ import Distribution.Package import qualified Distribution.PackageDescription as PD import qualified Distribution.PackageDescription.Configuration as PD import Distribution.PackageDescription.Configuration - ( finalizePackageDescription ) + ( finalizePD ) import Distribution.Client.PackageUtils ( externalBuildDepends ) import Distribution.Version @@ -848,12 +847,13 @@ configuredPackageProblems platform cinfo -- of the `nubOn` in `mergeDeps`. requiredDeps :: [Dependency] requiredDeps = - --TODO: use something lower level than finalizePackageDescription - case finalizePackageDescription specifiedFlags + --TODO: use something lower level than finalizePD + case finalizePD specifiedFlags + (enableStanzas stanzas) (const True) platform cinfo [] - (enableStanzas stanzas $ packageDescription pkg) of + (packageDescription pkg) of Right (resolvedPkg, _) -> externalBuildDepends resolvedPkg ++ maybe [] PD.setupDepends (PD.setupBuildInfo resolvedPkg) diff --git a/cabal-install/Distribution/Client/Dependency/TopDown.hs b/cabal-install/Distribution/Client/Dependency/TopDown.hs index aefc67c0d6a..baf47f1bc8a 100644 --- a/cabal-install/Distribution/Client/Dependency/TopDown.hs +++ b/cabal-install/Distribution/Client/Dependency/TopDown.hs @@ -21,7 +21,7 @@ import Distribution.Client.Dependency.TopDown.Constraints ( Satisfiable(..) ) import Distribution.Client.Types ( UnresolvedPkgLoc - , UnresolvedSourcePackage, enableStanzas ) + , UnresolvedSourcePackage ) import qualified Distribution.Simple.PackageIndex as InstalledPackageIndex import Distribution.Simple.PackageIndex (InstalledPackageIndex) @@ -36,7 +36,7 @@ import Distribution.PackageDescription import Distribution.Client.PackageUtils ( externalBuildDepends ) import Distribution.PackageDescription.Configuration - ( finalizePackageDescription, flattenPackageDescription ) + ( finalizePD, flattenPackageDescription ) import Distribution.Version ( Version(..), VersionRange, withinRange, simplifyVersionRange , UpperBound(..), asVersionIntervals ) @@ -54,6 +54,7 @@ import qualified Distribution.Solver.Types.ComponentDeps as CD import Distribution.Solver.Types.DependencyResolver import Distribution.Solver.Types.InstalledPreference import Distribution.Solver.Types.LabeledPackageConstraint +import Distribution.Solver.Types.OptionalStanza import Distribution.Solver.Types.PackageConstraint import qualified Distribution.Solver.Types.PackageIndex as PackageIndex import Distribution.Solver.Types.PackageIndex (PackageIndex) @@ -395,8 +396,8 @@ pruneBottomUp platform comp constraints = | dep <- missing ] configure cs (UnconfiguredPackage (SourcePackage _ pkg _ _) _ flags stanzas) = - finalizePackageDescription flags (dependencySatisfiable cs) - platform comp [] (enableStanzas stanzas pkg) + finalizePD flags (enableStanzas stanzas) (dependencySatisfiable cs) + platform comp [] pkg dependencySatisfiable cs = not . null . PackageIndex.lookupDependency (Constraints.choices cs) @@ -424,9 +425,8 @@ configurePackage platform cinfo available spkg = case spkg of (configure apkg) where configure (UnconfiguredPackage apkg@(SourcePackage _ p _ _) _ flags stanzas) = - case finalizePackageDescription flags dependencySatisfiable - platform cinfo [] - (enableStanzas stanzas p) of + case finalizePD flags (enableStanzas stanzas) dependencySatisfiable + platform cinfo [] p of Left missing -> Left missing Right (pkg, flags') -> Right $ SemiConfiguredPackage apkg flags' stanzas (externalBuildDepends pkg) diff --git a/cabal-install/Distribution/Client/GenBounds.hs b/cabal-install/Distribution/Client/GenBounds.hs index 3f8eed8ef16..cbcb3275d27 100644 --- a/cabal-install/Distribution/Client/GenBounds.hs +++ b/cabal-install/Distribution/Client/GenBounds.hs @@ -30,9 +30,11 @@ import Distribution.Package import Distribution.PackageDescription ( buildDepends ) import Distribution.PackageDescription.Configuration - ( finalizePackageDescription ) + ( finalizePD ) import Distribution.PackageDescription.Parse ( readPackageDescription ) +import Distribution.Simple.LocalBuildInfo + ( defaultComponentEnabled ) import Distribution.Simple.Compiler ( Compiler, PackageDBStack, compilerInfo ) import Distribution.Simple.Program @@ -107,9 +109,12 @@ genBounds verbosity packageDBs repoCtxt comp platform conf mSandboxPkgInfo cwd <- getCurrentDirectory path <- tryFindPackageDesc cwd gpd <- readPackageDescription verbosity path - let epd = finalizePackageDescription [] (const True) platform cinfo [] gpd + -- NB: We don't enable tests or benchmarks, since often they + -- don't really have useful bounds. + let epd = finalizePD [] defaultComponentEnabled + (const True) platform cinfo [] gpd case epd of - Left _ -> putStrLn "finalizePackageDescription failed" + Left _ -> putStrLn "finalizePD failed" Right (pd,_) -> do let needBounds = filter (not . hasUpperBound . depVersion) $ buildDepends pd diff --git a/cabal-install/Distribution/Client/Install.hs b/cabal-install/Distribution/Client/Install.hs index 44ac01eb42e..f1211cac829 100644 --- a/cabal-install/Distribution/Client/Install.hs +++ b/cabal-install/Distribution/Client/Install.hs @@ -159,7 +159,7 @@ import Distribution.PackageDescription ( PackageDescription, GenericPackageDescription(..), Flag(..) , FlagName(..), FlagAssignment ) import Distribution.PackageDescription.Configuration - ( finalizePackageDescription ) + ( finalizePD ) import Distribution.ParseUtils ( showPWarning ) import Distribution.Version @@ -1302,10 +1302,10 @@ installReadyPackage platform cinfo configFlags configTests = toFlag (TestStanzas `elem` stanzas) } source pkg pkgoverride where - pkg = case finalizePackageDescription flags + pkg = case finalizePD flags (enableStanzas stanzas) (const True) - platform cinfo [] (enableStanzas stanzas gpkg) of - Left _ -> error "finalizePackageDescription ReadyPackage failed" + platform cinfo [] gpkg of + Left _ -> error "finalizePD ReadyPackage failed" Right (desc, _) -> desc fetchSourcePackage diff --git a/cabal-install/Distribution/Client/InstallSymlink.hs b/cabal-install/Distribution/Client/InstallSymlink.hs index 875466dc116..692c4e25e49 100644 --- a/cabal-install/Distribution/Client/InstallSymlink.hs +++ b/cabal-install/Distribution/Client/InstallSymlink.hs @@ -38,7 +38,7 @@ symlinkBinary _ _ _ _ = fail "Symlinking feature not available on Windows" #else import Distribution.Client.Types - ( GenericReadyPackage(..), ReadyPackage, enableStanzas + ( GenericReadyPackage(..), ReadyPackage , ConfiguredPackage(..)) import Distribution.Client.Setup ( InstallFlags(installSymlinkBinDir) ) @@ -46,6 +46,7 @@ import qualified Distribution.Client.InstallPlan as InstallPlan import Distribution.Client.InstallPlan (InstallPlan) import Distribution.Solver.Types.SourcePackage +import Distribution.Solver.Types.OptionalStanza import Distribution.Package ( PackageIdentifier, Package(packageId), UnitId(..), installedUnitId ) @@ -55,7 +56,7 @@ import qualified Distribution.PackageDescription as PackageDescription import Distribution.PackageDescription ( PackageDescription ) import Distribution.PackageDescription.Configuration - ( finalizePackageDescription ) + ( finalizePD ) import Distribution.Simple.Setup ( ConfigFlags(..), fromFlag, fromFlagOrDefault, flagToMaybe ) import qualified Distribution.Simple.InstallDirs as InstallDirs @@ -144,10 +145,10 @@ symlinkBinaries platform comp configFlags installFlags plan = pkgDescription (ReadyPackage (ConfiguredPackage _ (SourcePackage _ pkg _ _) flags stanzas _)) = - case finalizePackageDescription flags + case finalizePD flags (enableStanzas stanzas) (const True) - platform cinfo [] (enableStanzas stanzas pkg) of - Left _ -> error "finalizePackageDescription ReadyPackage failed" + platform cinfo [] pkg of + Left _ -> error "finalizePD ReadyPackage failed" Right (desc, _) -> desc -- This is sadly rather complicated. We're kind of re-doing part of the diff --git a/cabal-install/Distribution/Client/ProjectPlanning.hs b/cabal-install/Distribution/Client/ProjectPlanning.hs index fbc3a6ddf51..fd642f8ad6c 100644 --- a/cabal-install/Distribution/Client/ProjectPlanning.hs +++ b/cabal-install/Distribution/Client/ProjectPlanning.hs @@ -1065,8 +1065,8 @@ elaborateInstallPlan platform compiler compilerprogdb -- pkgSourceId = pkgid pkgDescription = let Right (desc, _) = - PD.finalizePackageDescription - flags (const True) + PD.finalizePD + flags enabled (const True) platform (compilerInfo compiler) [] gdesc in desc @@ -1074,6 +1074,7 @@ elaborateInstallPlan platform compiler compilerprogdb pkgFlagDefaults = [ (Cabal.flagName flag, Cabal.flagDefault flag) | flag <- PD.genPackageFlags gdesc ] pkgDependencies = deps + enabled = enableStanzas stanzas pkgStanzasAvailable = Set.fromList stanzas pkgStanzasRequested = -- NB: even if a package stanza is requested, if the package diff --git a/cabal-install/Distribution/Client/Types.hs b/cabal-install/Distribution/Client/Types.hs index 40d9f382875..368acf41eda 100644 --- a/cabal-install/Distribution/Client/Types.hs +++ b/cabal-install/Distribution/Client/Types.hs @@ -24,10 +24,7 @@ import Distribution.Package import Distribution.InstalledPackageInfo ( InstalledPackageInfo ) import Distribution.PackageDescription - ( Benchmark(..), GenericPackageDescription(..), FlagAssignment - , TestSuite(..) ) -import Distribution.PackageDescription.Configuration - ( mapTreeData ) + ( FlagAssignment ) import Distribution.Version ( VersionRange ) @@ -148,20 +145,6 @@ type ReadyPackage = GenericReadyPackage (ConfiguredPackage UnresolvedPkgLoc) -- | Convenience alias for 'SourcePackage UnresolvedPkgLoc'. type UnresolvedSourcePackage = SourcePackage UnresolvedPkgLoc -enableStanzas - :: [OptionalStanza] - -> GenericPackageDescription - -> GenericPackageDescription -enableStanzas stanzas gpkg = gpkg - { condBenchmarks = flagBenchmarks $ condBenchmarks gpkg - , condTestSuites = flagTests $ condTestSuites gpkg - } - where - enableTest t = t { testEnabled = TestStanzas `elem` stanzas } - enableBenchmark bm = bm { benchmarkEnabled = BenchStanzas `elem` stanzas } - flagBenchmarks = map (\(n, bm) -> (n, mapTreeData enableBenchmark bm)) - flagTests = map (\(n, t) -> (n, mapTreeData enableTest t)) - -- ------------------------------------------------------------ -- * Package locations and repositories -- ------------------------------------------------------------ diff --git a/cabal-install/Distribution/Solver/Modular/IndexConversion.hs b/cabal-install/Distribution/Solver/Modular/IndexConversion.hs index 69aeb75fa5b..f7620555f6e 100644 --- a/cabal-install/Distribution/Solver/Modular/IndexConversion.hs +++ b/cabal-install/Distribution/Solver/Modular/IndexConversion.hs @@ -101,7 +101,7 @@ convSP os arch cinfo strfl (SourcePackage (PackageIdentifier pn pv) gpd _ _pl) = let i = I pv InRepo in (pn, i, convGPD os arch cinfo strfl (PI pn i) gpd) --- We do not use 'flattenPackageDescription' or 'finalizePackageDescription' +-- We do not use 'flattenPackageDescription' or 'finalizePD' -- from 'Distribution.PackageDescription.Configuration' here, because we -- want to keep the condition tree, but simplify much of the test. diff --git a/cabal-install/Distribution/Solver/Types/OptionalStanza.hs b/cabal-install/Distribution/Solver/Types/OptionalStanza.hs index 001918ee614..e7c9894178e 100644 --- a/cabal-install/Distribution/Solver/Types/OptionalStanza.hs +++ b/cabal-install/Distribution/Solver/Types/OptionalStanza.hs @@ -1,14 +1,25 @@ {-# LANGUAGE DeriveGeneric #-} module Distribution.Solver.Types.OptionalStanza ( OptionalStanza(..) + , enableStanzas ) where import GHC.Generics (Generic) import Distribution.Compat.Binary (Binary(..)) +import Distribution.Simple.LocalBuildInfo (ComponentEnabledSpec(..), defaultComponentEnabled) +import Data.List (foldl') data OptionalStanza = TestStanzas | BenchStanzas deriving (Eq, Ord, Enum, Bounded, Show, Generic) +-- | Convert a list of 'OptionalStanza' into the corresponding +-- 'ComponentEnabledSpec' which records what components are enabled. +enableStanzas :: [OptionalStanza] -> ComponentEnabledSpec +enableStanzas = foldl' addStanza defaultComponentEnabled + where + addStanza enabled TestStanzas = enabled { testsEnabled = True } + addStanza enabled BenchStanzas = enabled { benchmarksEnabled = True } + instance Binary OptionalStanza diff --git a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/DSL.hs b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/DSL.hs index 4a08f0649c2..1afb1bb79ea 100644 --- a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/DSL.hs +++ b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/DSL.hs @@ -89,7 +89,7 @@ import Distribution.Solver.Types.Variable `GenericPackageDescription` can be turned into a `PackageDescription` in two ways: - a. `finalizePackageDescription` does the proper translation, by taking + a. `finalizePD` does the proper translation, by taking into account the platform, available dependencies, etc. and picks a flag assignment (or gives an error if no flag assignment can be found) b. `flattenPackageDescription` ignores flag assignment and just joins all