Skip to content

Commit b86b8be

Browse files
committed
Fix environment variable handling when running executables
This fixes a bug where environment variables were duplicated when running executables. ``` overrideEnv <- fromMaybe [] <$> getEffectiveEnvironment ([("PATH", Just newPath)] ++ envOverrides) let shellEnv = overrideEnv ++ existingEnv ``` Since getEffectiveEnvironment already calls getEnvironment internally, if any overrides are passed then the result is a complete environment. Appending it to the already existing environment results in duplicated environment variables. The fix: * Added getFullEnvironment function to handle the common pattern correctly * Updated code in Bench, Test/ExeV10, Test/LibV09, and Client/Run to use this function In the future it would be good to generalise `getFullEnvironment` further so it can also handle the `addLibraryPath` case, which modifies an environment variable, rather than merely setting or unsetting it. Fixes #10718
1 parent 44086db commit b86b8be

File tree

5 files changed

+28
-21
lines changed

5 files changed

+28
-21
lines changed

Cabal/src/Distribution/Simple/Bench.hs

+1-5
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,6 @@ module Distribution.Simple.Bench
2222
import Distribution.Compat.Prelude
2323
import Prelude ()
2424

25-
import Distribution.Compat.Environment
2625
import qualified Distribution.PackageDescription as PD
2726
import Distribution.Pretty
2827
import Distribution.Simple.Build (addInternalBuildTools)
@@ -89,15 +88,12 @@ bench args pkg_descr lbi flags = do
8988
dieWithException verbosity $
9089
NoBenchMarkProgram cmd
9190

92-
existingEnv <- getEnvironment
93-
9491
-- Compute the appropriate environment for running the benchmark
9592
let progDb = LBI.withPrograms lbiForBench
9693
pathVar = progSearchPath progDb
9794
envOverrides = progOverrideEnv progDb
9895
newPath <- programSearchPathAsPATHVar pathVar
99-
overrideEnv <- fromMaybe [] <$> getEffectiveEnvironment ([("PATH", Just newPath)] ++ envOverrides)
100-
let shellEnv = overrideEnv ++ existingEnv
96+
shellEnv <- getFullEnvironment ([("PATH", Just newPath)] ++ envOverrides)
10197

10298
-- Add (DY)LD_LIBRARY_PATH if needed
10399
shellEnv' <-

Cabal/src/Distribution/Simple/Program/Run.hs

+12
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ module Distribution.Simple.Program.Run
2929
, getProgramInvocationOutputAndErrors
3030
, getProgramInvocationLBSAndErrors
3131
, getEffectiveEnvironment
32+
, getFullEnvironment
3233
) where
3334

3435
import Distribution.Compat.Prelude
@@ -248,6 +249,17 @@ getEffectiveEnvironment overrides =
248249
update (var, Nothing) = Map.delete var
249250
update (var, Just val) = Map.insert var val
250251

252+
-- | Like 'getEffectiveEnvironment', but when no overrides are specified,
253+
-- returns the full environment instead of 'Nothing'.
254+
getFullEnvironment
255+
:: [(String, Maybe String)]
256+
-> IO [(String, String)]
257+
getFullEnvironment overrides = do
258+
menv <- getEffectiveEnvironment overrides
259+
case menv of
260+
Just env -> return env
261+
Nothing -> getEnvironment
262+
251263
-- | Like the unix xargs program. Useful for when we've got very long command
252264
-- lines that might overflow an OS limit on command line length and so you
253265
-- need to invoke a command multiple times to get all the args in.

Cabal/src/Distribution/Simple/Test/ExeV10.hs

+7-5
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@ module Distribution.Simple.Test.ExeV10
88
import Distribution.Compat.Prelude
99
import Prelude ()
1010

11-
import Distribution.Compat.Environment
1211
import qualified Distribution.PackageDescription as PD
1312
import Distribution.Simple.BuildPaths
1413
import Distribution.Simple.Compiler
@@ -64,8 +63,6 @@ runTest pkg_descr lbi clbi hpcMarkupInfo flags suite = do
6463
way = guessWay lbi
6564
tixDir_ = i $ tixDir distPref way
6665

67-
existingEnv <- getEnvironment
68-
6966
let cmd =
7067
i (LBI.buildDir lbi)
7168
</> testName'
@@ -92,13 +89,18 @@ runTest pkg_descr lbi clbi hpcMarkupInfo flags suite = do
9289
pathVar = progSearchPath progDb
9390
envOverrides = progOverrideEnv progDb
9491
newPath <- programSearchPathAsPATHVar pathVar
95-
overrideEnv <- fromMaybe [] <$> getEffectiveEnvironment ([("PATH", Just newPath)] ++ envOverrides)
9692
let opts =
9793
map
9894
(testOption pkg_descr lbi suite)
9995
(testOptions flags)
10096
tixFile = packageRoot (testCommonFlags flags) </> getSymbolicPath (tixFilePath distPref way (testName'))
101-
shellEnv = [("HPCTIXFILE", tixFile) | isCoverageEnabled] ++ overrideEnv ++ existingEnv
97+
98+
shellEnv <-
99+
getFullEnvironment
100+
( [("PATH", Just newPath)]
101+
++ [("HPCTIXFILE", Just tixFile) | isCoverageEnabled]
102+
++ envOverrides
103+
)
102104

103105
-- Add (DY)LD_LIBRARY_PATH if needed
104106
shellEnv' <-

Cabal/src/Distribution/Simple/Test/LibV09.hs

+7-7
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,6 @@ import Distribution.Compat.Prelude
1717
import Distribution.Types.UnqualComponentName
1818
import Prelude ()
1919

20-
import Distribution.Compat.Environment
2120
import Distribution.Compat.Internal.TempFile
2221
import Distribution.Compat.Process (proc)
2322
import Distribution.ModuleName
@@ -70,7 +69,6 @@ runTest pkg_descr lbi clbi hpcMarkupInfo flags suite = do
7069
way = guessWay lbi
7170

7271
let mbWorkDir = LBI.mbWorkDirLBI lbi
73-
existingEnv <- getEnvironment
7472

7573
let cmd =
7674
interpretSymbolicPath mbWorkDir (LBI.buildDir lbi)
@@ -100,15 +98,17 @@ runTest pkg_descr lbi clbi hpcMarkupInfo flags suite = do
10098
pathVar = progSearchPath progDb
10199
envOverrides = progOverrideEnv progDb
102100
newPath <- programSearchPathAsPATHVar pathVar
103-
overrideEnv <- fromMaybe [] <$> getEffectiveEnvironment ([("PATH", Just newPath)] ++ envOverrides)
104101

105102
-- Run test executable
106103
let opts = map (testOption pkg_descr lbi suite) $ testOptions flags
107104
tixFile = i $ tixFilePath distPref way testName'
108-
shellEnv =
109-
[("HPCTIXFILE", tixFile) | isCoverageEnabled]
110-
++ overrideEnv
111-
++ existingEnv
105+
106+
shellEnv <-
107+
getFullEnvironment
108+
( [("PATH", Just newPath)]
109+
++ [("HPCTIXFILE", Just tixFile) | isCoverageEnabled]
110+
++ envOverrides
111+
)
112112
-- Add (DY)LD_LIBRARY_PATH if needed
113113
shellEnv' <-
114114
if LBI.withDynExe lbi

cabal-install/src/Distribution/Client/Run.hs

+1-4
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,6 @@ import Distribution.Types.UnqualComponentName
5959
import qualified Distribution.Simple.GHCJS as GHCJS
6060

6161
import Distribution.Client.Errors
62-
import Distribution.Compat.Environment (getEnvironment)
6362
import Distribution.Utils.Path
6463

6564
-- | Return the executable to run and any extra arguments that should be
@@ -178,13 +177,11 @@ run verbosity lbi exe exeArgs = do
178177
return (p, [])
179178

180179
-- Compute the appropriate environment for running the executable
181-
existingEnv <- getEnvironment
182180
let progDb = withPrograms lbiForExe
183181
pathVar = progSearchPath progDb
184182
envOverrides = progOverrideEnv progDb
185183
newPath <- programSearchPathAsPATHVar pathVar
186-
overrideEnv <- fromMaybe [] <$> getEffectiveEnvironment ([("PATH", Just newPath)] ++ envOverrides)
187-
let env = overrideEnv ++ existingEnv
184+
env <- getFullEnvironment ([("PATH", Just newPath)] ++ envOverrides)
188185

189186
-- Add (DY)LD_LIBRARY_PATH if needed
190187
env' <-

0 commit comments

Comments
 (0)