diff --git a/cabal-install/src/Distribution/Client/CmdInstall.hs b/cabal-install/src/Distribution/Client/CmdInstall.hs index 0adeca99446..36fc4d10bfa 100644 --- a/cabal-install/src/Distribution/Client/CmdInstall.hs +++ b/cabal-install/src/Distribution/Client/CmdInstall.hs @@ -2,6 +2,7 @@ {-# LANGUAGE NamedFieldPuns #-} {-# LANGUAGE RecordWildCards #-} {-# LANGUAGE ScopedTypeVariables #-} +{-# LANGUAGE TupleSections #-} -- | cabal-install CLI command: build module Distribution.Client.CmdInstall @@ -104,6 +105,7 @@ import Distribution.Client.Types , PackageSpecifier (..) , SourcePackageDb (..) , UnresolvedSourcePackage + , pkgSpecifierTarget ) import Distribution.Client.Types.OverwritePolicy ( OverwritePolicy (..) @@ -371,7 +373,7 @@ installAction flags@NixStyleFlags{extraFlags = clientInstallFlags', ..} targetSt -- First, we need to learn about what's available to be installed. localBaseCtx <- - establishProjectBaseContext reducedVerbosity cliConfig InstallCommand + establishProjectBaseContext reducedVerbosity baseCliConfig InstallCommand let localDistDirLayout = distDirLayout localBaseCtx pkgDb <- projectConfigWithBuilderRepoContext @@ -432,7 +434,7 @@ installAction flags@NixStyleFlags{extraFlags = clientInstallFlags', ..} targetSt withoutProject globalConfig = do tss <- traverse (parseWithoutProjectTargetSelector verbosity) targetStrings' let - projectConfig = globalConfig <> cliConfig + projectConfig = globalConfig <> baseCliConfig ProjectConfigBuildOnly { projectConfigLogsDir @@ -478,10 +480,17 @@ installAction flags@NixStyleFlags{extraFlags = clientInstallFlags', ..} targetSt return (packageSpecifiers, uris, packageTargets, projectConfig) - (specs, uris, targetSelectors, config) <- + (specs, uris, targetSelectors, baseConfig) <- withProjectOrGlobalConfig verbosity ignoreProject globalConfigFlag withProject withoutProject + -- We compute the base context again to determine packages available in the + -- project to be installed, so we can list the available package names when + -- the "all:..." variants of the target selectors are used. + localPkgs <- localPackages <$> establishProjectBaseContext verbosity baseConfig InstallCommand + let + config = addLocalConfigToPkgs baseConfig (map pkgSpecifierTarget specs ++ concatMap (targetPkgNames localPkgs) targetSelectors) + ProjectConfig { projectConfigBuildOnly = ProjectConfigBuildOnly @@ -631,8 +640,7 @@ installAction flags@NixStyleFlags{extraFlags = clientInstallFlags', ..} targetSt globalFlags flags{configFlags = configFlags'} clientInstallFlags' - cliConfig = addLocalConfigToTargets baseCliConfig targetStrings - globalConfigFlag = projectConfigConfigFile (projectConfigShared cliConfig) + globalConfigFlag = projectConfigConfigFile (projectConfigShared baseCliConfig) -- Do the install action for each executable in the install configuration. traverseInstall :: InstallAction -> InstallCfg -> IO () @@ -641,9 +649,9 @@ installAction flags@NixStyleFlags{extraFlags = clientInstallFlags', ..} targetSt actionOnExe <- action v overwritePolicy <$> prepareExeInstall cfg traverse_ actionOnExe . Map.toList $ targetsMap buildCtx --- | Treat all direct targets of install command as local packages: #8637 -addLocalConfigToTargets :: ProjectConfig -> [String] -> ProjectConfig -addLocalConfigToTargets config targetStrings = +-- | Treat all direct targets of install command as local packages: #8637 and later #7297, #8909, #7236. +addLocalConfigToPkgs :: ProjectConfig -> [PackageName] -> ProjectConfig +addLocalConfigToPkgs config pkgs = config { projectConfigSpecificPackage = projectConfigSpecificPackage config @@ -651,7 +659,19 @@ addLocalConfigToTargets config targetStrings = } where localConfig = projectConfigLocalPackages config - targetPackageConfigs = map (\x -> (mkPackageName x, localConfig)) targetStrings + targetPackageConfigs = map (,localConfig) pkgs + +targetPkgNames + :: [PackageSpecifier UnresolvedSourcePackage] + -- ^ The local packages, to resolve 'TargetAllPackages' selectors + -> TargetSelector + -> [PackageName] +targetPkgNames localPkgs = \case + TargetPackage _ pkgIds _ -> map pkgName pkgIds + TargetPackageNamed name _ -> [name] + TargetAllPackages _ -> map pkgSpecifierTarget localPkgs + TargetComponent pkgId _ -> [pkgName pkgId] + TargetComponentUnknown name _ -> [name] -- | Verify that invalid config options were not passed to the install command. -- diff --git a/cabal-testsuite/PackageTests/Install/T7297-8909-7236/Main.hs b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/Main.hs new file mode 100644 index 00000000000..2de6b7b008d --- /dev/null +++ b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/Main.hs @@ -0,0 +1,5 @@ +{-# LANGUAGE CPP #-} + +#ifdef HELLO +main = putStrLn "hi" +#endif diff --git a/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.out b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.out new file mode 100644 index 00000000000..879d5760318 --- /dev/null +++ b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.out @@ -0,0 +1,32 @@ +# cabal install +Wrote tarball sdist to /cabal.dist/work/./dist/sdist/t7297-89097236a-1.0.tar.gz +Resolving dependencies... +Build profile: -w ghc- -O1 +In order, the following will be built: + - t7297-89097236a-1.0 (exe:my-exe) (requires build) +Configuring t7297-89097236a-1.0... +Preprocessing executable 'my-exe' for t7297-89097236a-1.0... +Building executable 'my-exe' for t7297-89097236a-1.0... +Installing executable my-exe in +Warning: The directory /ghc-/incoming/new-/ghc-/-/bin is not in the system search path. +Symlinking 'my-exe' to '/my-exe' +# cabal install +Wrote tarball sdist to /cabal.dist/work/./dist/sdist/t7297-89097236a-1.0.tar.gz +Resolving dependencies... +Symlinking 'my-exe' to '/my-exe' +# cabal install +Wrote tarball sdist to /cabal.dist/work/./dist/sdist/t7297-89097236a-1.0.tar.gz +Resolving dependencies... +Symlinking 'my-exe' to '/my-exe' +# cabal install +Wrote tarball sdist to /cabal.dist/work/./dist/sdist/t7297-89097236a-1.0.tar.gz +Resolving dependencies... +Symlinking 'my-exe' to '/my-exe' +# cabal install +Wrote tarball sdist to /cabal.dist/work/./dist/sdist/t7297-89097236a-1.0.tar.gz +Resolving dependencies... +Symlinking 'my-exe' to '/my-exe' +# cabal install +Wrote tarball sdist to /cabal.dist/work/./dist/sdist/t7297-89097236a-1.0.tar.gz +Resolving dependencies... +Symlinking 'my-exe' to '/my-exe' diff --git a/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.project b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.project new file mode 100644 index 00000000000..e6fdbadb439 --- /dev/null +++ b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.project @@ -0,0 +1 @@ +packages: . diff --git a/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.test.hs b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.test.hs new file mode 100644 index 00000000000..9d1d223b2be --- /dev/null +++ b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/cabal.test.hs @@ -0,0 +1,14 @@ +import Test.Cabal.Prelude + +main = withShorterPathForNewBuildStore $ \storeDir -> cabalTest $ do + let + cabalOpts = [ "--store-dir=" ++ storeDir, "--installdir=" ++ storeDir] + commonOpts = ["--ghc-options=-DHELLO", "--overwrite-policy=always"] + installWithTgt tgt = cabalG cabalOpts "install" (tgt:commonOpts) + + cabalG cabalOpts "install" commonOpts -- no target + installWithTgt "t7297-89097236a" + installWithTgt "exe:my-exe" + installWithTgt "my-exe" + installWithTgt "all" + installWithTgt "all:exes" diff --git a/cabal-testsuite/PackageTests/Install/T7297-8909-7236/t7297-89097236a.cabal b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/t7297-89097236a.cabal new file mode 100644 index 00000000000..84ca78363c7 --- /dev/null +++ b/cabal-testsuite/PackageTests/Install/T7297-8909-7236/t7297-89097236a.cabal @@ -0,0 +1,8 @@ +name: t7297-89097236a +version: 1.0 +build-type: Simple +cabal-version: >= 1.2 + +executable my-exe + main-is: Main.hs + build-depends: base