Skip to content

Commit 01b0b13

Browse files
authored
Merge pull request commercialhaskell#6579 from commercialhaskell/re6574
Re commercialhaskell#6574 Use ghc-pkg to retreive field value
2 parents 505e884 + d29a303 commit 01b0b13

File tree

3 files changed

+52
-46
lines changed

3 files changed

+52
-46
lines changed

ChangeLog.md

+2
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,8 @@ Bug fixes:
7171
* The `config set` commands support existing keys only in the form `key: value`
7272
on a single line. The commands now recognise that a line `key:` does not have
7373
that form.
74+
* On Unix-like operating systems, the `test --coverage` command now finds
75+
package keys even for very long package names.
7476

7577
## v2.15.7 - 2024-05-12
7678

src/Stack/Coverage.hs

+27-46
Original file line numberDiff line numberDiff line change
@@ -18,14 +18,18 @@ module Stack.Coverage
1818
) where
1919

2020
import qualified Data.ByteString.Lazy.Char8 as L8
21+
import Data.Conduit ( await )
2122
import qualified Data.List as L
2223
import qualified Data.Map.Strict as Map
2324
import qualified Data.Set as Set
2425
import qualified Data.Text as T
2526
import qualified Data.Text.Lazy as LT
27+
import Distribution.Types.MungedPackageId ( computeCompatPackageId )
28+
import Distribution.Types.UnqualComponentName
29+
( mkUnqualComponentName )
2630
import Distribution.Version ( mkVersion )
2731
import Path
28-
( (</>), dirname, filename, parent, parseAbsFile, parseRelDir
32+
( (</>), dirname, parent, parseAbsFile, parseRelDir
2933
, parseRelFile, stripProperPrefix
3034
)
3135
import Path.Extra ( toFilePathNoTrailingSep )
@@ -35,7 +39,8 @@ import Path.IO
3539
, resolveDir', resolveFile'
3640
)
3741
import RIO.ByteString.Lazy ( putStrLn )
38-
import RIO.Process ( ProcessException, proc, readProcess_ )
42+
import RIO.Process
43+
( ExitCodeException, ProcessException, proc, readProcess_ )
3944
import Stack.Build.Target ( NeedTargets (..) )
4045
import Stack.Constants
4146
( relDirAll, relDirCombined, relDirCustom
@@ -44,11 +49,13 @@ import Stack.Constants
4449
)
4550
import Stack.Constants.Config ( distDirFromDir, hpcRelativeDir )
4651
import Stack.Package ( hasBuildableMainLibrary )
52+
import Stack.PackageDump ( ghcPkgField )
4753
import Stack.Prelude
4854
import Stack.Runners ( ShouldReexec (..), withConfig, withEnvConfig )
4955
import Stack.Types.BuildConfig
5056
( BuildConfig (..), HasBuildConfig (..) )
5157
import Stack.Types.Compiler ( getGhcVersion )
58+
import Stack.Types.CompilerPaths ( getGhcPkgExe )
5259
import Stack.Types.CompCollection ( getBuildableSetText )
5360
import Stack.Types.BuildOptsCLI
5461
( BuildOptsCLI (..), defaultBuildOptsCLI )
@@ -609,61 +616,35 @@ findPackageFieldForBuiltPackage ::
609616
=> Path Abs Dir -> PackageIdentifier -> Set.Set Text -> Text
610617
-> RIO env (Either Text [Text])
611618
findPackageFieldForBuiltPackage pkgDir pkgId subLibs field = do
619+
let subLibNames =
620+
Set.map (LSubLibName . mkUnqualComponentName . T.unpack) subLibs
621+
libraryNames = Set.insert LMainLibName subLibNames
622+
mungedPackageIds = Set.map (computeCompatPackageId pkgId) libraryNames
612623
distDir <- distDirFromDir pkgDir
624+
ghcPkgExe <- getGhcPkgExe
613625
let inplaceDir = distDir </> relDirPackageConfInplace
614626
pkgIdStr = packageIdentifierString pkgId
615627
notFoundErr = pure $
616628
Left $ "Failed to find package key for " <> T.pack pkgIdStr
617-
extractField path = do
618-
contents <- readFileUtf8 (toFilePath path)
619-
case asum (map (T.stripPrefix (field <> ": ")) (T.lines contents)) of
629+
extractField mungedPkgId = do
630+
mContents <- catch
631+
(ghcPkgField ghcPkgExe inplaceDir mungedPkgId (T.unpack field) await)
632+
-- A .conf file may not exist in the package database for a library or
633+
-- sub-library, if that component has not been built yet.
634+
(\(_ :: ExitCodeException) -> pure Nothing)
635+
case mContents of
620636
Just result -> pure $ Right $ T.strip result
621637
Nothing -> notFoundErr
622638
logDebug $
623639
"Scanning "
624640
<> fromString (toFilePath inplaceDir)
625-
<> " for files matching "
641+
<> " for munged packages matching "
626642
<> fromString pkgIdStr
627-
(_, files) <- handleIO (const $ pure ([], [])) $ listDir inplaceDir
628-
logDebug $ displayShow files
629-
-- From all the files obtained from the scanning process above, we need to
630-
-- identify which are .conf files and then ensure that there is at most one
631-
-- .conf file for each library and sub-library (some might be missing if that
632-
-- component has not been built yet). We should error if there are more than
633-
-- one .conf file for a component or if there are no .conf files at all in the
634-
-- searched location.
635-
let toFilename = T.pack . toFilePath . filename
636-
-- strip known prefix and suffix from the found files to determine only
637-
-- the .conf files
638-
stripKnown =
639-
T.stripSuffix ".conf" <=< T.stripPrefix (T.pack (pkgIdStr ++ "-"))
640-
stripped =
641-
mapMaybe (\file -> fmap (,file) . stripKnown . toFilename $ file) files
642-
-- which component could have generated each of these conf files
643-
stripHash n =
644-
let z = T.dropWhile (/= '-') n
645-
in if T.null z then "" else T.tail z
646-
matchedComponents = map (\(n, f) -> (stripHash n, [f])) stripped
647-
byComponents =
648-
Map.restrictKeys (Map.fromListWith (++) matchedComponents) $ Set.insert "" subLibs
649-
logDebug $ displayShow byComponents
650-
if Map.null $ Map.filter (\fs -> length fs > 1) byComponents
651-
then case concat $ Map.elems byComponents of
652-
[] -> notFoundErr
653-
-- for each of these files, we need to extract the requested field
654-
paths -> do
655-
(errors, keys) <- partitionEithers <$> traverse extractField paths
656-
case errors of
657-
(a:_) -> pure $ Left a -- the first error only, since they're repeated anyway
658-
[] -> pure $ Right keys
659-
else
660-
pure
661-
$ Left
662-
$ "Multiple files matching "
663-
<> T.pack (pkgIdStr ++ "-*.conf")
664-
<> " found in "
665-
<> T.pack (toFilePath inplaceDir)
666-
<> ". Maybe try 'stack clean' on this package?"
643+
(errors, keys) <-
644+
partitionEithers <$> traverse extractField (Set.toList mungedPackageIds)
645+
case errors of
646+
(a:_) -> pure $ Left a -- the first error only, since they're repeated anyway
647+
[] -> pure $ Right keys
667648

668649
displayReportPath ::
669650
HasTerm env

src/Stack/PackageDump.hs

+23
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ module Stack.PackageDump
1010
, conduitDumpPackage
1111
, ghcPkgDump
1212
, ghcPkgDescribe
13+
, ghcPkgField
1314
, sinkMatching
1415
, pruneDeps
1516
) where
@@ -22,6 +23,7 @@ import qualified Data.Conduit.List as CL
2223
import qualified Data.Conduit.Text as CT
2324
import qualified Data.Map as Map
2425
import qualified Data.Set as Set
26+
import qualified Distribution.Pretty as C
2527
import qualified Distribution.Text as C
2628
import Distribution.Types.MungedPackageName
2729
( decodeCompatPackageName )
@@ -91,6 +93,27 @@ ghcPkgDescribe pkgexe pkgName' = ghcPkgCmdArgs
9193
pkgexe
9294
["describe", "--simple-output", packageNameString pkgName']
9395

96+
-- | Call @ghc-pkg field@ with appropriate flags and stream to the given
97+
-- sink, using the given package database. Throws 'ExitCodeException' if the
98+
-- process fails (for example, if the package is not found in the package
99+
-- database or the field is not found in the package's *.conf file).
100+
ghcPkgField ::
101+
(HasCompiler env, HasProcessContext env, HasTerm env)
102+
=> GhcPkgExe
103+
-> Path Abs Dir
104+
-- ^ A package database.
105+
-> MungedPackageId
106+
-- ^ A munged package identifier.
107+
-> String
108+
-- ^ A field name.
109+
-> ConduitM Text Void (RIO env) a
110+
-- ^ Sink.
111+
-> RIO env a
112+
ghcPkgField pkgexe pkgDb mungedPkgId fieldName = ghcPkgCmdArgs
113+
pkgexe
114+
["field", C.prettyShow mungedPkgId, fieldName, "--simple-output" ]
115+
[pkgDb]
116+
94117
-- | Call @ghc-pkg@ and stream to the given sink, using the either the global
95118
-- package database or the given package databases.
96119
ghcPkgCmdArgs ::

0 commit comments

Comments
 (0)