diff --git a/cabal-install/solver-dsl/UnitTests/Distribution/Solver/Modular/DSL.hs b/cabal-install/solver-dsl/UnitTests/Distribution/Solver/Modular/DSL.hs index 4d356615826..765dfea4e3e 100644 --- a/cabal-install/solver-dsl/UnitTests/Distribution/Solver/Modular/DSL.hs +++ b/cabal-install/solver-dsl/UnitTests/Distribution/Solver/Modular/DSL.hs @@ -6,6 +6,7 @@ module UnitTests.Distribution.Solver.Modular.DSL ( ExampleDependency(..) , Dependencies(..) + , ExSubLib(..) , ExTest(..) , ExExe(..) , ExConstraint(..) @@ -21,13 +22,21 @@ module UnitTests.Distribution.Solver.Modular.DSL ( , ExampleQualifier(..) , ExampleVar(..) , EnableAllTests(..) + , dependencies + , publicDependencies + , unbuildableDependencies , exAv , exAvNoLibrary , exInst + , exSubLib + , exTest + , exExe , exFlagged , exResolve , extractInstallPlan , declareFlags + , withSubLibrary + , withSubLibraries , withSetupDeps , withTest , withTests @@ -58,6 +67,7 @@ import qualified Distribution.Package as C import qualified Distribution.Types.ExeDependency as C import qualified Distribution.Types.ForeignLib as C import qualified Distribution.Types.LegacyExeDependency as C +import qualified Distribution.Types.LibraryVisibility as C import qualified Distribution.Types.PkgconfigDependency as C import qualified Distribution.Types.PkgconfigVersion as C import qualified Distribution.Types.PkgconfigVersionRange as C @@ -140,12 +150,40 @@ type ExamplePkgName = String type ExamplePkgVersion = Int type ExamplePkgHash = String -- for example "installed" packages type ExampleFlagName = String +type ExampleSubLibName = String type ExampleTestName = String type ExampleExeName = String type ExampleVersionRange = C.VersionRange -data Dependencies = NotBuildable | Buildable [ExampleDependency] - deriving Show +data Dependencies = Dependencies { + depsVisibility :: C.LibraryVisibility + , depsIsBuildable :: Bool + , depsExampleDependencies :: [ExampleDependency] + } deriving Show + +instance Semigroup Dependencies where + deps1 <> deps2 = Dependencies { + depsVisibility = depsVisibility deps1 <> depsVisibility deps2 + , depsIsBuildable = depsIsBuildable deps1 && depsIsBuildable deps2 + , depsExampleDependencies = depsExampleDependencies deps1 ++ depsExampleDependencies deps2 + } + +instance Monoid Dependencies where + mempty = Dependencies { + depsVisibility = mempty + , depsIsBuildable = True + , depsExampleDependencies = [] + } + mappend = (<>) + +dependencies :: [ExampleDependency] -> Dependencies +dependencies deps = mempty { depsExampleDependencies = deps } + +publicDependencies :: Dependencies +publicDependencies = mempty { depsVisibility = C.LibraryVisibilityPublic } + +unbuildableDependencies :: Dependencies +unbuildableDependencies = mempty { depsIsBuildable = False } data ExampleDependency = -- | Simple dependency on any version @@ -158,6 +196,12 @@ data ExampleDependency = -- and an exclusive upper bound. | ExRange ExamplePkgName ExamplePkgVersion ExamplePkgVersion + -- | Sub-library dependency + | ExSubLibAny ExamplePkgName ExampleSubLibName + + -- | Sub-library dependency on a fixed version + | ExSubLibFix ExamplePkgName ExampleSubLibName ExamplePkgVersion + -- | Build-tool-depends dependency | ExBuildToolAny ExamplePkgName ExampleExeName @@ -191,13 +235,24 @@ data ExFlag = ExFlag { , exFlagType :: FlagType } deriving Show -data ExTest = ExTest ExampleTestName [ExampleDependency] +data ExSubLib = ExSubLib ExampleSubLibName Dependencies + +data ExTest = ExTest ExampleTestName Dependencies + +data ExExe = ExExe ExampleExeName Dependencies -data ExExe = ExExe ExampleExeName [ExampleDependency] +exSubLib :: ExampleSubLibName -> [ExampleDependency] -> ExSubLib +exSubLib name deps = ExSubLib name (dependencies deps) + +exTest :: ExampleTestName -> [ExampleDependency] -> ExTest +exTest name deps = ExTest name (dependencies deps) + +exExe :: ExampleExeName -> [ExampleDependency] -> ExExe +exExe name deps = ExExe name (dependencies deps) exFlagged :: ExampleFlagName -> [ExampleDependency] -> [ExampleDependency] -> ExampleDependency -exFlagged n t e = ExFlagged n (Buildable t) (Buildable e) +exFlagged n t e = ExFlagged n (dependencies t) (dependencies e) data ExConstraint = ExVersionConstraint ConstraintScope ExampleVersionRange @@ -213,7 +268,7 @@ data ExPreference = data ExampleAvailable = ExAv { exAvName :: ExamplePkgName , exAvVersion :: ExamplePkgVersion - , exAvDeps :: ComponentDeps [ExampleDependency] + , exAvDeps :: ComponentDeps Dependencies -- Setting flags here is only necessary to override the default values of -- the fields in C.Flag. @@ -253,7 +308,7 @@ newtype EnableAllTests = EnableAllTests Bool -- exAv :: ExamplePkgName -> ExamplePkgVersion -> [ExampleDependency] -> ExampleAvailable -exAv n v ds = (exAvNoLibrary n v) { exAvDeps = CD.fromLibraryDeps ds } +exAv n v ds = (exAvNoLibrary n v) { exAvDeps = CD.fromLibraryDeps (dependencies ds) } -- | Constructs an 'ExampleAvailable' package without a default library -- component. @@ -270,9 +325,18 @@ declareFlags flags ex = ex { exAvFlags = flags } +withSubLibrary :: ExampleAvailable -> ExSubLib -> ExampleAvailable +withSubLibrary ex lib = withSubLibraries ex [lib] + +withSubLibraries :: ExampleAvailable -> [ExSubLib] -> ExampleAvailable +withSubLibraries ex libs = + let subLibCDs = CD.fromList [(CD.ComponentSubLib $ C.mkUnqualComponentName name, deps) + | ExSubLib name deps <- libs] + in ex { exAvDeps = exAvDeps ex <> subLibCDs } + withSetupDeps :: ExampleAvailable -> [ExampleDependency] -> ExampleAvailable withSetupDeps ex setupDeps = ex { - exAvDeps = exAvDeps ex <> CD.fromSetupDeps setupDeps + exAvDeps = exAvDeps ex <> CD.fromSetupDeps (dependencies setupDeps) } withTest :: ExampleAvailable -> ExTest -> ExampleAvailable @@ -342,7 +406,7 @@ exAvSrcPkg ex = usedFlags :: Map ExampleFlagName C.Flag usedFlags = Map.fromList [(fn, mkDefaultFlag fn) | fn <- names] where - names = concatMap extractFlags $ CD.flatDeps (exAvDeps ex) + names = extractFlags $ CD.flatDeps (exAvDeps ex) in -- 'declaredFlags' overrides 'usedFlags' to give flags non-default settings: Map.elems $ declaredFlags `Map.union` usedFlags @@ -351,7 +415,7 @@ exAvSrcPkg ex = testSuites = [(name, deps) | (CD.ComponentTest name, deps) <- CD.toList (exAvDeps ex)] benchmarks = [(name, deps) | (CD.ComponentBench name, deps) <- CD.toList (exAvDeps ex)] executables = [(name, deps) | (CD.ComponentExe name, deps) <- CD.toList (exAvDeps ex)] - setup = case CD.setupDeps (exAvDeps ex) of + setup = case depsExampleDependencies $ CD.setupDeps (exAvDeps ex) of [] -> Nothing deps -> Just C.SetupBuildInfo { C.setupDepends = mkSetupDeps deps, @@ -379,30 +443,30 @@ exAvSrcPkg ex = } , C.genPackageFlags = flags , C.condLibrary = - let mkLib bi = mempty { C.libBuildInfo = bi } + let mkLib v bi = mempty { C.libVisibility = v, C.libBuildInfo = bi } -- Avoid using the Monoid instance for [a] when getting -- the library dependencies, to allow for the possibility -- that the package doesn't have a library: libDeps = lookup CD.ComponentLib (CD.toList (exAvDeps ex)) - in mkCondTree defaultLib mkLib . mkBuildInfoTree . Buildable <$> libDeps + in mkTopLevelCondTree defaultLib mkLib <$> libDeps , C.condSubLibraries = - let mkTree = mkCondTree defaultLib mkLib . mkBuildInfoTree . Buildable - mkLib bi = mempty { C.libBuildInfo = bi } + let mkTree = mkTopLevelCondTree defaultSubLib mkLib + mkLib v bi = mempty { C.libVisibility = v, C.libBuildInfo = bi } in map (second mkTree) subLibraries , C.condForeignLibs = - let mkTree = mkCondTree mempty mkLib . mkBuildInfoTree . Buildable + let mkTree = mkTopLevelCondTree (mkLib defaultTopLevelBuildInfo) (const mkLib) mkLib bi = mempty { C.foreignLibBuildInfo = bi } in map (second mkTree) foreignLibraries , C.condExecutables = - let mkTree = mkCondTree defaultExe mkExe . mkBuildInfoTree . Buildable + let mkTree = mkTopLevelCondTree defaultExe (const mkExe) mkExe bi = mempty { C.buildInfo = bi } in map (second mkTree) executables , C.condTestSuites = - let mkTree = mkCondTree defaultTest mkTest . mkBuildInfoTree . Buildable + let mkTree = mkTopLevelCondTree defaultTest (const mkTest) mkTest bi = mempty { C.testBuildInfo = bi } in map (second mkTree) testSuites , C.condBenchmarks = - let mkTree = mkCondTree defaultBenchmark mkBench . mkBuildInfoTree . Buildable + let mkTree = mkTopLevelCondTree defaultBenchmark (const mkBench) mkBench bi = mempty { C.benchmarkBuildInfo = bi } in map (second mkTree) benchmarks } @@ -423,19 +487,34 @@ exAvSrcPkg ex = defaultTopLevelBuildInfo = mempty { C.defaultLanguage = Just Haskell98 } defaultLib :: C.Library - defaultLib = mempty { C.exposedModules = [Module.fromString "Module"] } + defaultLib = mempty { + C.libBuildInfo = defaultTopLevelBuildInfo + , C.exposedModules = [Module.fromString "Module"] + , C.libVisibility = C.LibraryVisibilityPublic + } + + defaultSubLib :: C.Library + defaultSubLib = mempty { + C.libBuildInfo = defaultTopLevelBuildInfo + , C.exposedModules = [Module.fromString "Module"] + } defaultExe :: C.Executable - defaultExe = mempty { C.modulePath = "Main.hs" } + defaultExe = mempty { + C.buildInfo = defaultTopLevelBuildInfo + , C.modulePath = "Main.hs" + } defaultTest :: C.TestSuite defaultTest = mempty { - C.testInterface = C.TestSuiteExeV10 (C.mkVersion [1,0]) "Test.hs" + C.testBuildInfo = defaultTopLevelBuildInfo + , C.testInterface = C.TestSuiteExeV10 (C.mkVersion [1,0]) "Test.hs" } defaultBenchmark :: C.Benchmark defaultBenchmark = mempty { - C.benchmarkInterface = C.BenchmarkExeV10 (C.mkVersion [1,0]) "Benchmark.hs" + C.benchmarkBuildInfo = defaultTopLevelBuildInfo + , C.benchmarkInterface = C.BenchmarkExeV10 (C.mkVersion [1,0]) "Benchmark.hs" } -- Split the set of dependencies into the set of dependencies of the library, @@ -477,57 +556,43 @@ exAvSrcPkg ex = in (dep:other, exts, lang, pcpkgs, exes, legacyExes) -- Extract the total set of flags used - extractFlags :: ExampleDependency -> [ExampleFlagName] - extractFlags (ExAny _) = [] - extractFlags (ExFix _ _) = [] - extractFlags (ExRange _ _ _) = [] - extractFlags (ExBuildToolAny _ _) = [] - extractFlags (ExBuildToolFix _ _ _) = [] - extractFlags (ExLegacyBuildToolAny _) = [] - extractFlags (ExLegacyBuildToolFix _ _) = [] - extractFlags (ExFlagged f a b) = - f : concatMap extractFlags (deps a ++ deps b) - where - deps :: Dependencies -> [ExampleDependency] - deps NotBuildable = [] - deps (Buildable ds) = ds - extractFlags (ExExt _) = [] - extractFlags (ExLang _) = [] - extractFlags (ExPkg _) = [] - - -- Convert a tree of BuildInfos into a tree of a specific component type. - -- 'defaultTopLevel' contains the default values for the component, and - -- 'mkComponent' creates a component from a 'BuildInfo'. - mkCondTree :: forall a. Semigroup a => - a -> (C.BuildInfo -> a) - -> DependencyTree C.BuildInfo - -> DependencyTree a - mkCondTree defaultTopLevel mkComponent (C.CondNode topData topConstraints topComps) = - C.CondNode { - C.condTreeData = - defaultTopLevel <> mkComponent (defaultTopLevelBuildInfo <> topData) - , C.condTreeConstraints = topConstraints - , C.condTreeComponents = goComponents topComps - } + extractFlags :: Dependencies -> [ExampleFlagName] + extractFlags deps = concatMap go (depsExampleDependencies deps) where - go :: DependencyTree C.BuildInfo -> DependencyTree a - go (C.CondNode ctData constraints comps) = - C.CondNode (mkComponent ctData) constraints (goComponents comps) - - goComponents :: [DependencyComponent C.BuildInfo] - -> [DependencyComponent a] - goComponents comps = [C.CondBranch cond (go t) (go <$> me) | C.CondBranch cond t me <- comps] - - mkBuildInfoTree :: Dependencies -> DependencyTree C.BuildInfo - mkBuildInfoTree NotBuildable = - C.CondNode { - C.condTreeData = mempty { C.buildable = False } - , C.condTreeConstraints = [] - , C.condTreeComponents = [] - } - mkBuildInfoTree (Buildable deps) = - let (libraryDeps, exts, mlang, pcpkgs, buildTools, legacyBuildTools) = splitTopLevel deps + go :: ExampleDependency -> [ExampleFlagName] + go (ExAny _) = [] + go (ExFix _ _) = [] + go (ExRange _ _ _) = [] + go (ExSubLibAny _ _) = [] + go (ExSubLibFix _ _ _) = [] + go (ExBuildToolAny _ _) = [] + go (ExBuildToolFix _ _ _) = [] + go (ExLegacyBuildToolAny _) = [] + go (ExLegacyBuildToolFix _ _) = [] + go (ExFlagged f a b) = f : extractFlags a ++ extractFlags b + go (ExExt _) = [] + go (ExLang _) = [] + go (ExPkg _) = [] + + -- Convert 'Dependencies' into a tree of a specific component type, using + -- the given top level component and function for creating a component at + -- any level. + mkTopLevelCondTree :: forall a. Semigroup a => + a + -> (C.LibraryVisibility -> C.BuildInfo -> a) + -> Dependencies + -> DependencyTree a + mkTopLevelCondTree defaultTopLevel mkComponent deps = + let condNode = mkCondTree mkComponent deps + in condNode { C.condTreeData = defaultTopLevel <> C.condTreeData condNode } + + -- Convert 'Dependencies' into a tree of a specific component type, using + -- the given function to generate each component. + mkCondTree :: (C.LibraryVisibility -> C.BuildInfo -> a) -> Dependencies -> DependencyTree a + mkCondTree mkComponent deps = + let (libraryDeps, exts, mlang, pcpkgs, buildTools, legacyBuildTools) = splitTopLevel (depsExampleDependencies deps) (directDeps, flaggedDeps) = splitDeps libraryDeps + component = mkComponent (depsVisibility deps) bi bi = mempty { C.otherExtensions = exts , C.defaultLanguage = mlang @@ -539,25 +604,27 @@ exAvSrcPkg ex = | (n,v) <- pcpkgs , let n' = C.mkPkgconfigName n , let v' = C.PcThisVersion (mkSimplePkgconfigVersion v) ] + , C.buildable = depsIsBuildable deps } in C.CondNode { - C.condTreeData = bi -- Necessary for language extensions + C.condTreeData = component -- TODO: Arguably, build-tools dependencies should also -- effect constraints on conditional tree. But no way to -- distinguish between them , C.condTreeConstraints = map mkDirect directDeps - , C.condTreeComponents = map mkFlagged flaggedDeps + , C.condTreeComponents = map (mkFlagged mkComponent) flaggedDeps } - mkDirect :: (ExamplePkgName, C.VersionRange) -> C.Dependency - mkDirect (dep, vr) = C.Dependency (C.mkPackageName dep) vr (Set.singleton C.LMainLibName) + mkDirect :: (ExamplePkgName, C.LibraryName, C.VersionRange) -> C.Dependency + mkDirect (dep, name, vr) = C.Dependency (C.mkPackageName dep) vr (Set.singleton name) - mkFlagged :: (ExampleFlagName, Dependencies, Dependencies) - -> DependencyComponent C.BuildInfo - mkFlagged (f, a, b) = + mkFlagged :: (C.LibraryVisibility -> C.BuildInfo -> a) + -> (ExampleFlagName, Dependencies, Dependencies) + -> DependencyComponent a + mkFlagged mkComponent (f, a, b) = C.CondBranch (C.Var (C.Flag (C.mkFlagName f))) - (mkBuildInfoTree a) - (Just (mkBuildInfoTree b)) + (mkCondTree mkComponent a) + (Just (mkCondTree mkComponent b)) -- Split a set of dependencies into direct dependencies and flagged -- dependencies. A direct dependency is a tuple of the name of package and @@ -565,20 +632,26 @@ exAvSrcPkg ex = -- 'mkDirect' for example. A flagged dependency is the set of dependencies -- guarded by a flag. splitDeps :: [ExampleDependency] - -> ( [(ExamplePkgName, C.VersionRange)] + -> ( [(ExamplePkgName, C.LibraryName, C.VersionRange)] , [(ExampleFlagName, Dependencies, Dependencies)] ) splitDeps [] = ([], []) splitDeps (ExAny p:deps) = let (directDeps, flaggedDeps) = splitDeps deps - in ((p, C.anyVersion):directDeps, flaggedDeps) + in ((p, C.LMainLibName, C.anyVersion):directDeps, flaggedDeps) splitDeps (ExFix p v:deps) = let (directDeps, flaggedDeps) = splitDeps deps - in ((p, C.thisVersion $ mkSimpleVersion v):directDeps, flaggedDeps) + in ((p, C.LMainLibName, C.thisVersion $ mkSimpleVersion v):directDeps, flaggedDeps) splitDeps (ExRange p v1 v2:deps) = let (directDeps, flaggedDeps) = splitDeps deps - in ((p, mkVersionRange v1 v2):directDeps, flaggedDeps) + in ((p, C.LMainLibName, mkVersionRange v1 v2):directDeps, flaggedDeps) + splitDeps (ExSubLibAny p lib:deps) = + let (directDeps, flaggedDeps) = splitDeps deps + in ((p, C.LSubLibName (C.mkUnqualComponentName lib), C.anyVersion):directDeps, flaggedDeps) + splitDeps (ExSubLibFix p lib v:deps) = + let (directDeps, flaggedDeps) = splitDeps deps + in ((p, C.LSubLibName (C.mkUnqualComponentName lib), C.thisVersion $ mkSimpleVersion v):directDeps, flaggedDeps) splitDeps (ExFlagged f a b:deps) = let (directDeps, flaggedDeps) = splitDeps deps in (directDeps, (f, a, b):flaggedDeps) diff --git a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/MemoryUsage.hs b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/MemoryUsage.hs index 93f7c5a5d20..463e56bd7e6 100644 --- a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/MemoryUsage.hs +++ b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/MemoryUsage.hs @@ -150,7 +150,7 @@ duplicateDependencies name = pkgs :: ExampleDb pkgs = [ Right $ exAv "A" 1 (dependencyTree 1) - , Right $ exAv "B" 1 [] `withExe` ExExe "exe" [] + , Right $ exAv "B" 1 [] `withExe` exExe "exe" [] ] dependencyTree :: Int -> [ExampleDependency] @@ -178,7 +178,7 @@ duplicateFlaggedDependencies name = pkgs :: ExampleDb pkgs = [ Right $ exAv "A" 1 (dependencyTree 1) - , Right $ exAv "B" 1 [] `withExe` ExExe "exe" [] + , Right $ exAv "B" 1 [] `withExe` exExe "exe" [] ] dependencyTree :: Int -> [ExampleDependency] diff --git a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/QuickCheck.hs b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/QuickCheck.hs index a1a6412d014..46b2fdfaf3c 100644 --- a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/QuickCheck.hs +++ b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/QuickCheck.hs @@ -26,6 +26,7 @@ import Distribution.Utils.ShortText (ShortText) import Distribution.Client.Setup (defaultMaxBackjumps) +import Distribution.Types.LibraryVisibility import Distribution.Types.PackageName import Distribution.Types.UnqualComponentName @@ -310,8 +311,8 @@ arbitraryExInst pn v pkgs = do deps <- randomSubset numDeps pkgs return $ ExInst (unPN pn) (unPV v) pkgHash (map exInstHash deps) -arbitraryComponentDeps :: PN -> TestDb -> Gen (ComponentDeps [ExampleDependency]) -arbitraryComponentDeps _ (TestDb []) = return $ CD.fromLibraryDeps [] +arbitraryComponentDeps :: PN -> TestDb -> Gen (ComponentDeps Dependencies) +arbitraryComponentDeps _ (TestDb []) = return $ CD.fromLibraryDeps (dependencies []) arbitraryComponentDeps pn db = do -- dedupComponentNames removes components with duplicate names, for example, -- 'ComponentExe x' and 'ComponentTest x', and then CD.fromList combines @@ -321,7 +322,7 @@ arbitraryComponentDeps pn db = do return $ if isCompleteComponentDeps cds then cds else -- Add a library if the ComponentDeps isn't complete. - CD.fromLibraryDeps [] <> cds + CD.fromLibraryDeps (dependencies []) <> cds where isValid :: Component -> Bool isValid (ComponentSubLib name) = name /= mkUnqualComponentName (unPN pn) @@ -352,13 +353,20 @@ isCompleteComponentDeps = any (completesPkg . fst) . CD.toList completesPkg (ComponentFLib _) = False completesPkg ComponentSetup = False -arbitraryComponentDep :: TestDb -> Gen (ComponentDep [ExampleDependency]) +arbitraryComponentDep :: TestDb -> Gen (ComponentDep Dependencies) arbitraryComponentDep db = do comp <- arbitrary deps <- case comp of ComponentSetup -> smallListOf (arbitraryExDep db SetupDep) _ -> boundedListOf 5 (arbitraryExDep db NonSetupDep) - return (comp, deps) + return ( comp + , Dependencies { + depsExampleDependencies = deps + + -- TODO: Test different values for visibility and buildability. + , depsVisibility = LibraryVisibilityPublic + , depsIsBuildable = True + } ) -- | Location of an 'ExampleDependency'. It determines which values are valid. data ExDepLocation = SetupDep | NonSetupDep @@ -387,8 +395,8 @@ arbitraryExDep db@(TestDb pkgs) level = arbitraryDeps :: TestDb -> Gen Dependencies arbitraryDeps db = frequency - [ (1, return NotBuildable) - , (20, Buildable <$> smallListOf (arbitraryExDep db NonSetupDep)) + [ (1, return unbuildableDependencies) + , (20, dependencies <$> smallListOf (arbitraryExDep db NonSetupDep)) ] arbitraryFlagName :: Gen String @@ -432,6 +440,12 @@ instance Arbitrary IndependentGoals where shrink (IndependentGoals indep) = [IndependentGoals False | indep] +instance Arbitrary LibraryVisibility where + arbitrary = elements [LibraryVisibilityPrivate, LibraryVisibilityPublic] + + shrink LibraryVisibilityPublic = [LibraryVisibilityPrivate] + shrink LibraryVisibilityPrivate = [] + instance Arbitrary UnqualComponentName where -- The "component-" prefix prevents component names and build-depends -- dependency names from overlapping. @@ -476,19 +490,18 @@ instance Arbitrary ExampleDependency where shrink (ExFix "base" _) = [] -- preserve bounds on base shrink (ExFix pn _) = [ExAny pn] shrink (ExFlagged flag th el) = - deps th ++ deps el + depsExampleDependencies th ++ depsExampleDependencies el ++ [ExFlagged flag th' el | th' <- shrink th] ++ [ExFlagged flag th el' | el' <- shrink el] - where - deps NotBuildable = [] - deps (Buildable ds) = ds shrink dep = error $ "Dependency not handled: " ++ show dep instance Arbitrary Dependencies where arbitrary = error "arbitrary not implemented: Dependencies" - shrink NotBuildable = [Buildable []] - shrink (Buildable deps) = map Buildable (shrink deps) + shrink deps = + [ deps { depsVisibility = v } | v <- shrink $ depsVisibility deps ] + ++ [ deps { depsIsBuildable = b } | b <- shrink $ depsIsBuildable deps ] + ++ [ deps { depsExampleDependencies = ds } | ds <- shrink $ depsExampleDependencies deps ] instance Arbitrary ExConstraint where arbitrary = error "arbitrary not implemented: ExConstraint" diff --git a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/Solver.hs b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/Solver.hs index a7edbeaacf4..65a7d6114bf 100644 --- a/cabal-install/tests/UnitTests/Distribution/Solver/Modular/Solver.hs +++ b/cabal-install/tests/UnitTests/Distribution/Solver/Modular/Solver.hs @@ -282,16 +282,16 @@ tests = [ , runTest $ indep $ mkTest dbBJ8 "bj8" ["A", "B"] (solverSuccess [("A", 1), ("B", 1), ("C", 1)]) ] , testGroup "library dependencies" [ - let db = [Right $ exAvNoLibrary "A" 1 `withExe` ExExe "exe" []] + let db = [Right $ exAvNoLibrary "A" 1 `withExe` exExe "exe" []] in runTest $ mkTest db "install build target without a library" ["A"] $ solverSuccess [("A", 1)] , let db = [ Right $ exAv "A" 1 [ExAny "B"] - , Right $ exAvNoLibrary "B" 1 `withExe` ExExe "exe" [] ] + , Right $ exAvNoLibrary "B" 1 `withExe` exExe "exe" [] ] in runTest $ mkTest db "reject build-depends dependency with no library" ["A"] $ solverFailure (isInfixOf "rejecting: B-1.0.0 (does not contain library, which is required by A)") - , let exe = ExExe "exe" [] + , let exe = exExe "exe" [] db = [ Right $ exAv "A" 1 [ExAny "B"] , Right $ exAvNoLibrary "B" 2 `withExe` exe , Right $ exAv "B" 1 [] `withExe` exe ] @@ -370,19 +370,19 @@ tests = [ , testGroup "Components that are unbuildable in the current environment" $ let flagConstraint = ExFlagConstraint . ScopeAnyQualifier in [ - let db = [ Right $ exAv "A" 1 [ExFlagged "build-lib" (Buildable []) NotBuildable] ] + let db = [ Right $ exAv "A" 1 [ExFlagged "build-lib" (dependencies []) unbuildableDependencies] ] in runTest $ constraints [flagConstraint "A" "build-lib" False] $ mkTest db "install unbuildable library" ["A"] $ solverSuccess [("A", 1)] , let db = [ Right $ exAvNoLibrary "A" 1 - `withExe` ExExe "exe" [ExFlagged "build-exe" (Buildable []) NotBuildable] ] + `withExe` exExe "exe" [ExFlagged "build-exe" (dependencies []) unbuildableDependencies] ] in runTest $ constraints [flagConstraint "A" "build-exe" False] $ mkTest db "install unbuildable exe" ["A"] $ solverSuccess [("A", 1)] , let db = [ Right $ exAv "A" 1 [ExAny "B"] - , Right $ exAv "B" 1 [ExFlagged "build-lib" (Buildable []) NotBuildable] ] + , Right $ exAv "B" 1 [ExFlagged "build-lib" (dependencies []) unbuildableDependencies] ] in runTest $ constraints [flagConstraint "B" "build-lib" False] $ mkTest db "reject library dependency with unbuildable library" ["A"] $ solverFailure $ isInfixOf $ @@ -390,15 +390,15 @@ tests = [ ++ "current environment, but it is required by A)" , let db = [ Right $ exAv "A" 1 [ExBuildToolAny "B" "bt"] - , Right $ exAv "B" 1 [ExFlagged "build-lib" (Buildable []) NotBuildable] - `withExe` ExExe "bt" [] ] + , Right $ exAv "B" 1 [ExFlagged "build-lib" (dependencies []) unbuildableDependencies] + `withExe` exExe "bt" [] ] in runTest $ constraints [flagConstraint "B" "build-lib" False] $ mkTest db "allow build-tool dependency with unbuildable library" ["A"] $ solverSuccess [("A", 1), ("B", 1)] , let db = [ Right $ exAv "A" 1 [ExBuildToolAny "B" "bt"] , Right $ exAv "B" 1 [] - `withExe` ExExe "bt" [ExFlagged "build-exe" (Buildable []) NotBuildable] ] + `withExe` exExe "bt" [ExFlagged "build-exe" (dependencies []) unbuildableDependencies] ] in runTest $ constraints [flagConstraint "B" "build-exe" False] $ mkTest db "reject build-tool dependency with unbuildable exe" ["A"] $ solverFailure $ isInfixOf $ @@ -874,11 +874,11 @@ db5 = [ Right $ exAv "A" 1 [] , Right $ exAv "A" 2 [] , Right $ exAv "B" 1 [] - , Right $ exAv "C" 1 [] `withTest` ExTest "testC" [ExAny "A"] - , Right $ exAv "D" 1 [] `withTest` ExTest "testD" [ExFix "B" 2] - , Right $ exAv "E" 1 [ExFix "A" 1] `withTest` ExTest "testE" [ExAny "A"] - , Right $ exAv "F" 1 [ExFix "A" 1] `withTest` ExTest "testF" [ExFix "A" 2] - , Right $ exAv "G" 1 [ExFix "A" 2] `withTest` ExTest "testG" [ExAny "A"] + , Right $ exAv "C" 1 [] `withTest` exTest "testC" [ExAny "A"] + , Right $ exAv "D" 1 [] `withTest` exTest "testD" [ExFix "B" 2] + , Right $ exAv "E" 1 [ExFix "A" 1] `withTest` exTest "testE" [ExAny "A"] + , Right $ exAv "F" 1 [ExFix "A" 1] `withTest` exTest "testF" [ExFix "A" 2] + , Right $ exAv "G" 1 [ExFix "A" 2] `withTest` exTest "testG" [ExAny "A"] ] -- Now the _dependencies_ have test suites @@ -893,7 +893,7 @@ db6 :: ExampleDb db6 = [ Right $ exAv "A" 1 [] , Right $ exAv "A" 2 [] - , Right $ exAv "B" 1 [] `withTest` ExTest "testA" [ExAny "A"] + , Right $ exAv "B" 1 [] `withTest` exTest "testA" [ExAny "A"] , Right $ exAv "C" 1 [ExFix "A" 1, ExAny "B"] , Right $ exAv "D" 1 [ExAny "B"] ] @@ -915,7 +915,7 @@ testTestSuiteWithFlag name = db = [ Right $ exAv "A" 1 [] `withTest` - ExTest "test" [exFlagged "flag" [ExFix "B" 2] []] + exTest "test" [exFlagged "flag" [ExFix "B" 2] []] , Right $ exAv "B" 1 [] ] @@ -1086,13 +1086,13 @@ dbConstraints = dbStanzaPreferences1 :: ExampleDb dbStanzaPreferences1 = [ - Right $ exAv "pkg" 1 [] `withTest` ExTest "test" [ExAny "test-dep"] + Right $ exAv "pkg" 1 [] `withTest` exTest "test" [ExAny "test-dep"] , Right $ exAv "test-dep" 1 [] ] dbStanzaPreferences2 :: ExampleDb dbStanzaPreferences2 = [ - Right $ exAv "pkg" 1 [] `withTest` ExTest "test" [ExAny "unknown"] + Right $ exAv "pkg" 1 [] `withTest` exTest "test" [ExAny "unknown"] ] -- | This is a test case for a bug in stanza preferences (#3930). The solver @@ -1108,7 +1108,7 @@ testStanzaPreference name = [] [ExAny "unknown-pkg1"]] `withTest` - ExTest "test" [exFlagged "flag" + exTest "test" [exFlagged "flag" [ExAny "unknown-pkg2"] []] goals = [ @@ -1278,8 +1278,8 @@ testIndepGoals2 name = where db :: ExampleDb db = [ - Right $ exAv "A" 1 [ExAny "C"] `withTest` ExTest "test" [ExFix "D" 1] - , Right $ exAv "B" 1 [ExAny "C"] `withTest` ExTest "test" [ExFix "D" 1] + Right $ exAv "A" 1 [ExAny "C"] `withTest` exTest "test" [ExFix "D" 1] + , Right $ exAv "B" 1 [ExAny "C"] `withTest` exTest "test" [ExFix "D" 1] , Right $ exAv "C" 1 [ExAny "D"] , Right $ exAv "D" 1 [] , Right $ exAv "D" 2 [] @@ -1460,7 +1460,7 @@ testIndepGoals4 name = db = [ Right $ exAv "A" 1 [ExFix "E" 2] , Right $ exAv "B" 1 [ExAny "D"] - , Right $ exAv "C" 1 [ExAny "D"] `withTest` ExTest "test" [ExFix "E" 1] + , Right $ exAv "C" 1 [ExAny "D"] `withTest` exTest "test" [ExFix "E" 1] , Right $ exAv "D" 1 [ExAny "E"] , Right $ exAv "E" 1 [] , Right $ exAv "E" 2 [] @@ -1617,8 +1617,8 @@ testBuildable testName unavailableDep = [ExAny "true-dep"] [ExAny "false-dep"]] `withExe` - ExExe "exe" [ unavailableDep - , ExFlagged "enable-exe" (Buildable []) NotBuildable ] + exExe "exe" [ unavailableDep + , ExFlagged "enable-exe" (dependencies []) unbuildableDependencies ] , Right $ exAv "true-dep" 1 [] , Right $ exAv "false-dep" 1 [] ] @@ -1631,15 +1631,15 @@ dbBuildable1 = [ [ exFlagged "flag1" [ExAny "flag1-true"] [ExAny "flag1-false"] , exFlagged "flag2" [ExAny "flag2-true"] [ExAny "flag2-false"]] `withExes` - [ ExExe "exe1" + [ exExe "exe1" [ ExAny "unknown" - , ExFlagged "flag1" (Buildable []) NotBuildable - , ExFlagged "flag2" (Buildable []) NotBuildable] - , ExExe "exe2" + , ExFlagged "flag1" (dependencies []) unbuildableDependencies + , ExFlagged "flag2" (dependencies []) unbuildableDependencies] + , exExe "exe2" [ ExAny "unknown" , ExFlagged "flag1" - (Buildable []) - (Buildable [ExFlagged "flag2" NotBuildable (Buildable [])])] + (dependencies []) + (dependencies [ExFlagged "flag2" unbuildableDependencies (dependencies [])])] ] , Right $ exAv "flag1-true" 1 [] , Right $ exAv "flag1-false" 1 [] @@ -1654,9 +1654,9 @@ dbBuildable2 = [ , Right $ exAv "B" 1 [ExAny "unknown"] , Right $ exAv "B" 2 [] `withExe` - ExExe "exe" + exExe "exe" [ ExAny "unknown" - , ExFlagged "disable-exe" NotBuildable (Buildable []) + , ExFlagged "disable-exe" unbuildableDependencies (dependencies []) ] , Right $ exAv "B" 3 [ExAny "unknown"] ] @@ -1870,17 +1870,17 @@ dbBuildTools = [ Right $ exAv "A" 1 [ExBuildToolAny "bt-pkg" "exe1"] , Right $ exAv "B" 1 [exFlagged "flagB" [ExAny "unknown"] [ExBuildToolAny "bt-pkg" "exe1"]] - , Right $ exAv "C" 1 [] `withTest` ExTest "testC" [ExBuildToolAny "bt-pkg" "exe1"] + , Right $ exAv "C" 1 [] `withTest` exTest "testC" [ExBuildToolAny "bt-pkg" "exe1"] , Right $ exAv "D" 1 [ExBuildToolAny "bt-pkg" "unknown-exe"] , Right $ exAv "E" 1 [ExBuildToolAny "unknown-pkg" "exe1"] , Right $ exAv "F" 1 [exFlagged "flagF" [ExBuildToolAny "bt-pkg" "unknown-exe"] [ExAny "unknown"]] - , Right $ exAv "G" 1 [] `withTest` ExTest "testG" [ExBuildToolAny "bt-pkg" "unknown-exe"] + , Right $ exAv "G" 1 [] `withTest` exTest "testG" [ExBuildToolAny "bt-pkg" "unknown-exe"] , Right $ exAv "H" 1 [ExBuildToolFix "bt-pkg" "exe1" 3] , Right $ exAv "bt-pkg" 4 [] - , Right $ exAv "bt-pkg" 3 [] `withExe` ExExe "exe2" [] - , Right $ exAv "bt-pkg" 2 [] `withExe` ExExe "exe1" [] + , Right $ exAv "bt-pkg" 3 [] `withExe` exExe "exe2" [] + , Right $ exAv "bt-pkg" 2 [] `withExe` exExe "exe1" [] , Right $ exAv "bt-pkg" 1 [] ] @@ -1924,7 +1924,7 @@ chooseExeAfterBuildToolsPackage shouldSucceed name = [ExAny "unknown"]] , Right $ exAv "B" 1 [] `withExes` - [ExExe exe [] | exe <- if shouldSucceed then ["exe1", "exe2"] else ["exe1"]] + [exExe exe [] | exe <- if shouldSucceed then ["exe1", "exe2"] else ["exe1"]] ] goals :: [ExampleVar] @@ -1952,7 +1952,7 @@ requireConsistentBuildToolVersions name = , Right $ exAv "B" 1 [] `withExes` exes ] - exes = [ExExe "exe1" [], ExExe "exe2" []] + exes = [exExe "exe1" [], exExe "exe2" []] -- | This test is similar to the failure case for -- chooseExeAfterBuildToolsPackage, except that the build tool is unbuildable @@ -1972,8 +1972,8 @@ chooseUnbuildableExeAfterBuildToolsPackage name = [ExAny "unknown"]] , Right $ exAvNoLibrary "B" 1 `withExes` - [ ExExe "bt1" [] - , ExExe "bt2" [ExFlagged "build-bt2" (Buildable []) NotBuildable] + [ exExe "bt1" [] + , exExe "bt2" [ExFlagged "build-bt2" (dependencies []) unbuildableDependencies] ] ] @@ -1989,7 +1989,7 @@ chooseUnbuildableExeAfterBuildToolsPackage name = -------------------------------------------------------------------------------} dbLegacyBuildTools1 :: ExampleDb dbLegacyBuildTools1 = [ - Right $ exAv "alex" 1 [] `withExe` ExExe "alex" [], + Right $ exAv "alex" 1 [] `withExe` exExe "alex" [], Right $ exAv "A" 1 [ExLegacyBuildToolAny "alex"] ] @@ -1997,8 +1997,8 @@ dbLegacyBuildTools1 = [ -- package and the executable. This db has no solution. dbLegacyBuildTools2 :: ExampleDb dbLegacyBuildTools2 = [ - Right $ exAv "alex" 1 [] `withExe` ExExe "other-exe" [], - Right $ exAv "other-package" 1 [] `withExe` ExExe "alex" [], + Right $ exAv "alex" 1 [] `withExe` exExe "other-exe" [], + Right $ exAv "other-package" 1 [] `withExe` exExe "alex" [], Right $ exAv "A" 1 [ExLegacyBuildToolAny "alex"] ] @@ -2012,8 +2012,8 @@ dbLegacyBuildTools3 = [ -- Test that we can solve for different versions of executables dbLegacyBuildTools4 :: ExampleDb dbLegacyBuildTools4 = [ - Right $ exAv "alex" 1 [] `withExe` ExExe "alex" [], - Right $ exAv "alex" 2 [] `withExe` ExExe "alex" [], + Right $ exAv "alex" 1 [] `withExe` exExe "alex" [], + Right $ exAv "alex" 2 [] `withExe` exExe "alex" [], Right $ exAv "A" 1 [ExLegacyBuildToolFix "alex" 1], Right $ exAv "B" 1 [ExLegacyBuildToolFix "alex" 2], Right $ exAv "C" 1 [ExAny "A", ExAny "B"] @@ -2022,7 +2022,7 @@ dbLegacyBuildTools4 = [ -- Test that exe is not related to library choices dbLegacyBuildTools5 :: ExampleDb dbLegacyBuildTools5 = [ - Right $ exAv "alex" 1 [ExFix "A" 1] `withExe` ExExe "alex" [], + Right $ exAv "alex" 1 [ExFix "A" 1] `withExe` exExe "alex" [], Right $ exAv "A" 1 [], Right $ exAv "A" 2 [], Right $ exAv "B" 1 [ExLegacyBuildToolFix "alex" 1, ExFix "A" 2] @@ -2031,8 +2031,8 @@ dbLegacyBuildTools5 = [ -- Test that build-tools on build-tools works dbLegacyBuildTools6 :: ExampleDb dbLegacyBuildTools6 = [ - Right $ exAv "alex" 1 [] `withExe` ExExe "alex" [], - Right $ exAv "happy" 1 [ExLegacyBuildToolAny "alex"] `withExe` ExExe "happy" [], + Right $ exAv "alex" 1 [] `withExe` exExe "alex" [], + Right $ exAv "happy" 1 [ExLegacyBuildToolAny "alex"] `withExe` exExe "happy" [], Right $ exAv "A" 1 [ExLegacyBuildToolAny "happy"] ] @@ -2043,7 +2043,7 @@ dbIssue3775 = [ Right $ exAv "warp" 1 [], -- NB: the warp build-depends refers to the package, not the internal -- executable! - Right $ exAv "A" 2 [ExFix "warp" 1] `withExe` ExExe "warp" [ExAny "A"], + Right $ exAv "A" 2 [ExFix "warp" 1] `withExe` exExe "warp" [ExAny "A"], Right $ exAv "B" 2 [ExAny "A", ExAny "warp"] ]