@@ -18,14 +18,18 @@ module Stack.Coverage
18
18
) where
19
19
20
20
import qualified Data.ByteString.Lazy.Char8 as L8
21
+ import Data.Conduit ( await )
21
22
import qualified Data.List as L
22
23
import qualified Data.Map.Strict as Map
23
24
import qualified Data.Set as Set
24
25
import qualified Data.Text as T
25
26
import qualified Data.Text.Lazy as LT
27
+ import Distribution.Types.MungedPackageId ( computeCompatPackageId )
28
+ import Distribution.Types.UnqualComponentName
29
+ ( mkUnqualComponentName )
26
30
import Distribution.Version ( mkVersion )
27
31
import Path
28
- ( (</>) , dirname , filename , parent , parseAbsFile , parseRelDir
32
+ ( (</>) , dirname , parent , parseAbsFile , parseRelDir
29
33
, parseRelFile , stripProperPrefix
30
34
)
31
35
import Path.Extra ( toFilePathNoTrailingSep )
@@ -35,7 +39,8 @@ import Path.IO
35
39
, resolveDir' , resolveFile'
36
40
)
37
41
import RIO.ByteString.Lazy ( putStrLn )
38
- import RIO.Process ( ProcessException , proc , readProcess_ )
42
+ import RIO.Process
43
+ ( ExitCodeException , ProcessException , proc , readProcess_ )
39
44
import Stack.Build.Target ( NeedTargets (.. ) )
40
45
import Stack.Constants
41
46
( relDirAll , relDirCombined , relDirCustom
@@ -44,11 +49,13 @@ import Stack.Constants
44
49
)
45
50
import Stack.Constants.Config ( distDirFromDir , hpcRelativeDir )
46
51
import Stack.Package ( hasBuildableMainLibrary )
52
+ import Stack.PackageDump ( ghcPkgField )
47
53
import Stack.Prelude
48
54
import Stack.Runners ( ShouldReexec (.. ), withConfig , withEnvConfig )
49
55
import Stack.Types.BuildConfig
50
56
( BuildConfig (.. ), HasBuildConfig (.. ) )
51
57
import Stack.Types.Compiler ( getGhcVersion )
58
+ import Stack.Types.CompilerPaths ( getGhcPkgExe )
52
59
import Stack.Types.CompCollection ( getBuildableSetText )
53
60
import Stack.Types.BuildOptsCLI
54
61
( BuildOptsCLI (.. ), defaultBuildOptsCLI )
@@ -609,61 +616,35 @@ findPackageFieldForBuiltPackage ::
609
616
=> Path Abs Dir -> PackageIdentifier -> Set. Set Text -> Text
610
617
-> RIO env (Either Text [Text ])
611
618
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
612
623
distDir <- distDirFromDir pkgDir
624
+ ghcPkgExe <- getGhcPkgExe
613
625
let inplaceDir = distDir </> relDirPackageConfInplace
614
626
pkgIdStr = packageIdentifierString pkgId
615
627
notFoundErr = pure $
616
628
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
620
636
Just result -> pure $ Right $ T. strip result
621
637
Nothing -> notFoundErr
622
638
logDebug $
623
639
" Scanning "
624
640
<> fromString (toFilePath inplaceDir)
625
- <> " for files matching "
641
+ <> " for munged packages matching "
626
642
<> 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
667
648
668
649
displayReportPath ::
669
650
HasTerm env
0 commit comments