{-# LANGUAGE NoImplicitPrelude #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-}
module Pantry
(
PantryConfig
, HackageSecurityConfig (..)
, defaultHackageSecurityConfig
, HasPantryConfig (..)
, withPantryConfig
, HpackExecutable (..)
, PantryApp
, runPantryApp
, runPantryAppClean
, hpackExecutableL
, PantryException (..)
, PackageName
, Version
, FlagName
, PackageIdentifier (..)
, FileSize (..)
, RelFilePath (..)
, ResolvedPath (..)
, Unresolved
, SHA256
, TreeKey (..)
, BlobKey (..)
, RawPackageMetadata (..)
, PackageMetadata (..)
, Package (..)
, CabalFileInfo (..)
, Revision (..)
, PackageIdentifierRevision (..)
, UsePreferredVersions (..)
, RawArchive (..)
, Archive (..)
, ArchiveLocation (..)
, Repo (..)
, RepoType (..)
, withRepo
, RawPackageLocation (..)
, PackageLocation (..)
, toRawPL
, RawPackageLocationImmutable (..)
, PackageLocationImmutable (..)
, RawSnapshotLocation (..)
, SnapshotLocation (..)
, toRawSL
, RawSnapshot (..)
, Snapshot (..)
, RawSnapshotPackage (..)
, SnapshotPackage (..)
, RawSnapshotLayer (..)
, SnapshotLayer (..)
, toRawSnapshotLayer
, WantedCompiler (..)
, resolvePaths
, loadPackageRaw
, loadPackage
, loadRawSnapshotLayer
, loadSnapshotLayer
, loadSnapshot
, loadAndCompleteSnapshot
, loadAndCompleteSnapshotRaw
, CompletedSL (..)
, CompletedPLI (..)
, addPackagesToSnapshot
, AddPackagesConfig (..)
, completePackageLocation
, completeSnapshotLayer
, completeSnapshotLocation
, parseWantedCompiler
, parseRawSnapshotLocation
, parsePackageIdentifierRevision
, parseHackageText
, parsePackageIdentifier
, parsePackageName
, parsePackageNameThrowing
, parseFlagName
, parseVersion
, parseVersionThrowing
, ltsSnapshotLocation
, nightlySnapshotLocation
, packageIdentifierString
, packageNameString
, flagNameString
, versionString
, moduleNameString
, CabalString (..)
, toCabalStringMap
, unCabalStringMap
, gpdPackageIdentifier
, gpdPackageName
, gpdVersion
, fetchPackages
, unpackPackageLocationRaw
, unpackPackageLocation
, getPackageLocationName
, getRawPackageLocationIdent
, packageLocationIdent
, packageLocationVersion
, getRawPackageLocationTreeKey
, getPackageLocationTreeKey
, loadCabalFileRaw
, loadCabalFile
, loadCabalFileRawImmutable
, loadCabalFileImmutable
, loadCabalFilePath
, findOrGenerateCabalFile
, PrintWarnings (..)
, updateHackageIndex
, DidUpdateOccur (..)
, RequireHackageIndex (..)
, hackageIndexTarballL
, getHackagePackageVersions
, getLatestHackageVersion
, getLatestHackageLocation
, getLatestHackageRevision
, getHackageTypoCorrections
, loadGlobalHints
, partitionReplacedDependencies
, SnapshotCacheHash (..)
, withSnapshotCache
) where
import RIO
import Conduit
import Control.Arrow (right)
import Control.Monad.State.Strict (State, execState, get, modify')
import qualified RIO.Map as Map
import qualified RIO.Set as Set
import qualified RIO.ByteString as B
import qualified RIO.Text as T
import qualified RIO.List as List
import qualified RIO.FilePath as FilePath
import Pantry.Archive
import Pantry.Repo
import qualified Pantry.SHA256 as SHA256
import Pantry.Storage hiding (TreeEntry, PackageName, Version)
import Pantry.Tree
import Pantry.Types
import Pantry.Hackage
import Path (Path, Abs, File, toFilePath, Dir, (</>), filename, parseAbsDir, parent, parseRelFile)
import Path.IO (doesFileExist, resolveDir', listDir)
import Distribution.PackageDescription (GenericPackageDescription, FlagName)
import qualified Distribution.PackageDescription as D
import Distribution.Parsec.Common (PWarning (..), showPos)
import qualified Hpack
import qualified Hpack.Config as Hpack
import Network.HTTP.Download
import RIO.PrettyPrint
import RIO.PrettyPrint.StylesUpdate
import RIO.Process
import RIO.Directory (getAppUserDataDirectory)
import qualified Data.Yaml as Yaml
import Pantry.Internal.AesonExtended (WithJSONWarnings (..), Value)
import Data.Aeson.Types (parseEither)
import Data.Monoid (Endo (..))
import Pantry.HTTP
import Data.Char (isHexDigit)
withPantryConfig
:: HasLogFunc env
=> Path Abs Dir
-> HackageSecurityConfig
-> HpackExecutable
-> Int
-> (PantryConfig -> RIO env a)
-> RIO env a
withPantryConfig root hsc he count inner = do
env <- ask
pantryRelFile <- parseRelFile "pantry.sqlite3"
runRIO (mempty :: LogFunc) $ initStorage (root </> pantryRelFile) $ \storage -> runRIO env $ do
ur <- newMVar True
ref1 <- newIORef mempty
ref2 <- newIORef mempty
inner PantryConfig
{ pcHackageSecurity = hsc
, pcHpackExecutable = he
, pcRootDir = root
, pcStorage = storage
, pcUpdateRef = ur
, pcConnectionCount = count
, pcParsedCabalFilesRawImmutable = ref1
, pcParsedCabalFilesMutable = ref2
}
defaultHackageSecurityConfig :: HackageSecurityConfig
defaultHackageSecurityConfig = HackageSecurityConfig
{ hscKeyIds =
[ "0a5c7ea47cd1b15f01f5f51a33adda7e655bc0f0b0615baa8e271f4c3351e21d"
, "1ea9ba32c526d1cc91ab5e5bd364ec5e9e8cb67179a471872f6e26f0ae773d42"
, "280b10153a522681163658cb49f632cde3f38d768b736ddbc901d99a1a772833"
, "2a96b1889dc221c17296fcc2bb34b908ca9734376f0f361660200935916ef201"
, "2c6c3627bd6c982990239487f1abd02e08a02e6cf16edb105a8012d444d870c3"
, "51f0161b906011b52c6613376b1ae937670da69322113a246a09f807c62f6921"
, "772e9f4c7db33d251d5c6e357199c819e569d130857dc225549b40845ff0890d"
, "aa315286e6ad281ad61182235533c41e806e5a787e0b6d1e7eef3f09d137d2e9"
, "fe331502606802feac15e514d9b9ea83fee8b6ffef71335479a2e68d84adc6b0"
]
, hscKeyThreshold = 3
, hscDownloadPrefix = "https://s3.amazonaws.com/hackage.fpcomplete.com/"
, hscIgnoreExpiry = False
}
getLatestHackageVersion
:: (HasPantryConfig env, HasLogFunc env)
=> RequireHackageIndex
-> PackageName
-> UsePreferredVersions
-> RIO env (Maybe PackageIdentifierRevision)
getLatestHackageVersion req name preferred =
((fmap fst . Map.maxViewWithKey) >=> go) <$> getHackagePackageVersions req preferred name
where
go (version, m) = do
(_rev, BlobKey sha size) <- fst <$> Map.maxViewWithKey m
pure $ PackageIdentifierRevision name version $ CFIHash sha $ Just size
getLatestHackageLocation
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RequireHackageIndex
-> PackageName
-> UsePreferredVersions
-> RIO env (Maybe PackageLocationImmutable)
getLatestHackageLocation req name preferred = do
mversion <-
fmap fst . Map.maxViewWithKey <$> getHackagePackageVersions req preferred name
let mVerCfKey = do
(version, revisions) <- mversion
(_rev, cfKey) <- fst <$> Map.maxViewWithKey revisions
pure (version, cfKey)
forM mVerCfKey $ \(version, cfKey@(BlobKey sha size)) -> do
let pir = PackageIdentifierRevision name version (CFIHash sha (Just size))
treeKey' <- getHackageTarballKey pir
pure $ PLIHackage (PackageIdentifier name version) cfKey treeKey'
getLatestHackageRevision
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RequireHackageIndex
-> PackageName
-> Version
-> RIO env (Maybe (Revision, BlobKey, TreeKey))
getLatestHackageRevision req name version = do
revisions <- getHackagePackageVersionRevisions req name version
case fmap fst $ Map.maxViewWithKey revisions of
Nothing -> pure Nothing
Just (revision, cfKey@(BlobKey sha size)) -> do
let cfi = CFIHash sha (Just size)
treeKey' <- getHackageTarballKey (PackageIdentifierRevision name version cfi)
return $ Just (revision, cfKey, treeKey')
fetchTreeKeys
:: (HasPantryConfig env, HasLogFunc env, Foldable f)
=> f TreeKey
-> RIO env ()
fetchTreeKeys _ =
logWarn "Network caching not yet implemented!"
fetchPackages
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env, Foldable f)
=> f PackageLocationImmutable
-> RIO env ()
fetchPackages pls = do
fetchTreeKeys $ map getTreeKey $ toList pls
traverseConcurrently_ (void . uncurry getHackageTarball) hackages
fetchArchives archives
fetchRepos repos
where
s x = Endo (x:)
run (Endo f) = f []
(hackagesE, archivesE, reposE) = foldMap go pls
hackages = run hackagesE
archives = run archivesE
repos = run reposE
go (PLIHackage ident cfHash tree) = (s (toPir ident cfHash, Just tree), mempty, mempty)
go (PLIArchive archive pm) = (mempty, s (archive, pm), mempty)
go (PLIRepo repo pm) = (mempty, mempty, s (repo, pm))
toPir (PackageIdentifier name ver) (BlobKey sha size) =
PackageIdentifierRevision name ver (CFIHash sha (Just size))
unpackPackageLocationRaw
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> Path Abs Dir
-> RawPackageLocationImmutable
-> RIO env ()
unpackPackageLocationRaw fp loc = loadPackageRaw loc >>= unpackTree loc fp . packageTree
unpackPackageLocation
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> Path Abs Dir
-> PackageLocationImmutable
-> RIO env ()
unpackPackageLocation fp loc = loadPackage loc >>= unpackTree (toRawPLI loc) fp . packageTree
loadCabalFileImmutable
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> PackageLocationImmutable
-> RIO env GenericPackageDescription
loadCabalFileImmutable loc = withCache $ do
logDebug $ "Parsing cabal file for " <> display loc
bs <- loadCabalFileBytes loc
let foundCabalKey = bsToBlobKey bs
(_warnings, gpd) <- rawParseGPD (Left $ toRawPLI loc) bs
let pm =
case loc of
PLIHackage (PackageIdentifier name version) cfHash mtree -> PackageMetadata
{ pmIdent = PackageIdentifier name version
, pmTreeKey = mtree
, pmCabal = cfHash
}
PLIArchive _ pm' -> pm'
PLIRepo _ pm' -> pm'
let exc = MismatchedPackageMetadata (toRawPLI loc) (toRawPM pm) Nothing
foundCabalKey (gpdPackageIdentifier gpd)
PackageIdentifier name ver = pmIdent pm
maybe (throwIO exc) pure $ do
guard $ name == gpdPackageName gpd
guard $ ver == gpdVersion gpd
guard $ pmCabal pm == foundCabalKey
pure gpd
where
withCache inner = do
let rawLoc = toRawPLI loc
ref <- view $ pantryConfigL.to pcParsedCabalFilesRawImmutable
m0 <- readIORef ref
case Map.lookup rawLoc m0 of
Just x -> pure x
Nothing -> do
x <- inner
atomicModifyIORef' ref $ \m -> (Map.insert rawLoc x m, x)
loadCabalFileRawImmutable
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RIO env GenericPackageDescription
loadCabalFileRawImmutable loc = withCache $ do
logDebug $ "Parsing cabal file for " <> display loc
bs <- loadRawCabalFileBytes loc
let foundCabalKey = bsToBlobKey bs
(_warnings, gpd) <- rawParseGPD (Left loc) bs
let rpm =
case loc of
RPLIHackage (PackageIdentifierRevision name version cfi) mtree -> RawPackageMetadata
{ rpmName = Just name
, rpmVersion = Just version
, rpmTreeKey = mtree
, rpmCabal =
case cfi of
CFIHash sha (Just size) -> Just $ BlobKey sha size
_ -> Nothing
}
RPLIArchive _ rpm' -> rpm'
RPLIRepo _ rpm' -> rpm'
let exc = MismatchedPackageMetadata loc rpm Nothing foundCabalKey (gpdPackageIdentifier gpd)
maybe (throwIO exc) pure $ do
guard $ maybe True (== gpdPackageName gpd) (rpmName rpm)
guard $ maybe True (== gpdVersion gpd) (rpmVersion rpm)
guard $ maybe True (== foundCabalKey) (rpmCabal rpm)
pure gpd
where
withCache inner = do
ref <- view $ pantryConfigL.to pcParsedCabalFilesRawImmutable
m0 <- readIORef ref
case Map.lookup loc m0 of
Just x -> pure x
Nothing -> do
x <- inner
atomicModifyIORef' ref $ \m -> (Map.insert loc x m, x)
loadCabalFileRaw
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocation
-> RIO env GenericPackageDescription
loadCabalFileRaw (RPLImmutable loc) = loadCabalFileRawImmutable loc
loadCabalFileRaw (RPLMutable rfp) = do
(gpdio, _, _) <- loadCabalFilePath (resolvedAbsolute rfp)
liftIO $ gpdio NoPrintWarnings
loadCabalFile
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> PackageLocation
-> RIO env GenericPackageDescription
loadCabalFile (PLImmutable loc) = loadCabalFileImmutable loc
loadCabalFile (PLMutable rfp) = do
(gpdio, _, _) <- loadCabalFilePath (resolvedAbsolute rfp)
liftIO $ gpdio NoPrintWarnings
loadCabalFilePath
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> Path Abs Dir
-> RIO env
( PrintWarnings -> IO GenericPackageDescription
, PackageName
, Path Abs File
)
loadCabalFilePath dir = do
ref <- view $ pantryConfigL.to pcParsedCabalFilesMutable
mcached <- Map.lookup dir <$> readIORef ref
case mcached of
Just triple -> pure triple
Nothing -> do
(name, cabalfp) <- findOrGenerateCabalFile dir
gpdRef <- newIORef Nothing
run <- askRunInIO
let gpdio = run . getGPD cabalfp gpdRef
triple = (gpdio, name, cabalfp)
atomicModifyIORef' ref $ \m -> (Map.insert dir triple m, triple)
where
getGPD cabalfp gpdRef printWarnings = do
mpair <- readIORef gpdRef
(warnings0, gpd) <-
case mpair of
Just pair -> pure pair
Nothing -> do
bs <- liftIO $ B.readFile $ toFilePath cabalfp
(warnings0, gpd) <- rawParseGPD (Right cabalfp) bs
checkCabalFileName (gpdPackageName gpd) cabalfp
pure (warnings0, gpd)
warnings <-
case printWarnings of
YesPrintWarnings -> mapM_ (logWarn . toPretty cabalfp) warnings0 $> []
NoPrintWarnings -> pure warnings0
writeIORef gpdRef $ Just (warnings, gpd)
pure gpd
toPretty :: Path Abs File -> PWarning -> Utf8Builder
toPretty src (PWarning _type pos msg) =
"Cabal file warning in" <>
fromString (toFilePath src) <> "@" <>
fromString (showPos pos) <> ": " <>
fromString msg
checkCabalFileName :: MonadThrow m => PackageName -> Path Abs File -> m ()
checkCabalFileName name cabalfp = do
let expected = T.unpack $ unSafeFilePath $ cabalFileName name
when (expected /= toFilePath (filename cabalfp))
$ throwM $ MismatchedCabalName cabalfp name
findOrGenerateCabalFile
:: forall env. (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> Path Abs Dir
-> RIO env (PackageName, Path Abs File)
findOrGenerateCabalFile pkgDir = do
hpack pkgDir
files <- filter (flip hasExtension "cabal" . toFilePath) . snd
<$> listDir pkgDir
let isHidden ('.':_) = True
isHidden _ = False
case filter (not . isHidden . toFilePath . filename) files of
[] -> throwIO $ NoCabalFileFound pkgDir
[x] -> maybe
(throwIO $ InvalidCabalFilePath x)
(\pn -> pure $ (pn, x)) $
List.stripSuffix ".cabal" (toFilePath (filename x)) >>=
parsePackageName
_:_ -> throwIO $ MultipleCabalFilesFound pkgDir files
where hasExtension fp x = FilePath.takeExtension fp == "." ++ x
hpack
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> Path Abs Dir
-> RIO env ()
hpack pkgDir = do
packageConfigRelFile <- parseRelFile Hpack.packageConfig
let hpackFile = pkgDir </> packageConfigRelFile
exists <- liftIO $ doesFileExist hpackFile
when exists $ do
logDebug $ "Running hpack on " <> fromString (toFilePath hpackFile)
he <- view $ pantryConfigL.to pcHpackExecutable
case he of
HpackBundled -> do
r <- liftIO $ Hpack.hpackResult $ Hpack.setProgramName "stack" $ Hpack.setTarget (toFilePath hpackFile) Hpack.defaultOptions
forM_ (Hpack.resultWarnings r) (logWarn . fromString)
let cabalFile = fromString . Hpack.resultCabalFile $ r
case Hpack.resultStatus r of
Hpack.Generated -> logDebug $ "hpack generated a modified version of " <> cabalFile
Hpack.OutputUnchanged -> logDebug $ "hpack output unchanged in " <> cabalFile
Hpack.AlreadyGeneratedByNewerHpack -> logWarn $
cabalFile <>
" was generated with a newer version of hpack,\n" <>
"please upgrade and try again."
Hpack.ExistingCabalFileWasModifiedManually -> logWarn $
cabalFile <>
" was modified manually. Ignoring " <>
fromString (toFilePath hpackFile) <>
" in favor of the cabal file.\nIf you want to use the " <>
fromString (toFilePath (filename hpackFile)) <>
" file instead of the cabal file,\n" <>
"then please delete the cabal file."
HpackCommand command ->
withWorkingDir (toFilePath pkgDir) $
proc command [] runProcess_
gpdPackageIdentifier :: GenericPackageDescription -> PackageIdentifier
gpdPackageIdentifier = D.package . D.packageDescription
gpdPackageName :: GenericPackageDescription -> PackageName
gpdPackageName = pkgName . gpdPackageIdentifier
gpdVersion :: GenericPackageDescription -> Version
gpdVersion = pkgVersion . gpdPackageIdentifier
loadCabalFileBytes
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> PackageLocationImmutable
-> RIO env ByteString
loadCabalFileBytes (PLIHackage pident cfHash _mtree) = getHackageCabalFile (pirForHash pident cfHash)
loadCabalFileBytes pl = do
package <- loadPackage pl
let sfp = cabalFileName $ pkgName $ packageIdent package
cabalBlobKey <- case (packageCabalEntry package) of
PCHpack pcHpack -> pure $ teBlob . phGenerated $ pcHpack
PCCabalFile (TreeEntry blobKey _) -> pure blobKey
mbs <- withStorage $ loadBlob cabalBlobKey
case mbs of
Nothing -> do
throwIO $ TreeReferencesMissingBlob (toRawPLI pl) sfp cabalBlobKey
Just bs -> pure bs
loadRawCabalFileBytes
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RIO env ByteString
loadRawCabalFileBytes (RPLIHackage pir _mtree) = getHackageCabalFile pir
loadRawCabalFileBytes pl = do
package <- loadPackageRaw pl
let sfp = cabalFileName $ pkgName $ packageIdent package
TreeEntry cabalBlobKey _ft = case packageCabalEntry package of
PCCabalFile cabalTE -> cabalTE
PCHpack hpackCE -> phGenerated hpackCE
mbs <- withStorage $ loadBlob cabalBlobKey
case mbs of
Nothing -> do
throwIO $ TreeReferencesMissingBlob pl sfp cabalBlobKey
Just bs -> pure bs
loadPackage
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> PackageLocationImmutable
-> RIO env Package
loadPackage (PLIHackage ident cfHash tree) =
htrPackage <$> getHackageTarball (pirForHash ident cfHash) (Just tree)
loadPackage pli@(PLIArchive archive pm) = getArchivePackage (toRawPLI pli) (toRawArchive archive) (toRawPM pm)
loadPackage (PLIRepo repo pm) = getRepo repo (toRawPM pm)
loadPackageRaw
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RIO env Package
loadPackageRaw (RPLIHackage pir mtree) = htrPackage <$> getHackageTarball pir mtree
loadPackageRaw rpli@(RPLIArchive archive pm) = getArchivePackage rpli archive pm
loadPackageRaw (RPLIRepo repo rpm) = getRepo repo rpm
completePackageLocation
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RIO env PackageLocationImmutable
completePackageLocation (RPLIHackage (PackageIdentifierRevision n v (CFIHash sha (Just size))) (Just tk)) =
pure $ PLIHackage (PackageIdentifier n v) (BlobKey sha size) tk
completePackageLocation (RPLIHackage pir0@(PackageIdentifierRevision name version cfi0) _) = do
logDebug $ "Completing package location information from " <> display pir0
(pir, cfKey) <-
case cfi0 of
CFIHash sha (Just size) -> pure (pir0, BlobKey sha size)
_ -> do
bs <- getHackageCabalFile pir0
let size = FileSize (fromIntegral (B.length bs))
sha = SHA256.hashBytes bs
cfi = CFIHash sha (Just size)
pir = PackageIdentifierRevision name version cfi
logDebug $ "Added in cabal file hash: " <> display pir
pure (pir, BlobKey sha size)
treeKey' <- getHackageTarballKey pir
pure $ PLIHackage (PackageIdentifier name version) cfKey treeKey'
completePackageLocation pl@(RPLIArchive archive rpm) = do
(sha, size, package) <- getArchive pl archive rpm
let RawArchive loc _ _ subdir = archive
pure $ PLIArchive (Archive loc sha size subdir) (packagePM package)
completePackageLocation pl@(RPLIRepo repo rpm) = do
unless (isSHA1 (repoCommit repo)) $ throwIO $ CannotCompleteRepoNonSHA1 repo
PLIRepo repo <$> completePM pl rpm
where
isSHA1 t = T.length t == 40 && T.all isHexDigit t
completePM
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RawPackageMetadata
-> RIO env PackageMetadata
completePM plOrig rpm@(RawPackageMetadata mn mv mtk mc)
| Just n <- mn, Just v <- mv, Just tk <- mtk, Just c <- mc =
pure $ PackageMetadata (PackageIdentifier n v) tk c
| otherwise = do
pm <- packagePM <$> loadPackageRaw plOrig
let isSame x (Just y) = x == y
isSame _ _ = True
allSame =
isSame (pkgName $ pmIdent pm) (rpmName rpm) &&
isSame (pkgVersion $ pmIdent pm) (rpmVersion rpm) &&
isSame (pmTreeKey pm) (rpmTreeKey rpm) &&
isSame (pmCabal pm) (rpmCabal rpm)
if allSame
then pure pm
else throwIO $ CompletePackageMetadataMismatch plOrig pm
packagePM :: Package -> PackageMetadata
packagePM package = PackageMetadata
{ pmIdent = packageIdent package
, pmTreeKey = packageTreeKey package
, pmCabal = teBlob $ case packageCabalEntry package of
PCCabalFile cfile -> cfile
PCHpack hfile -> phGenerated hfile
}
completeSnapshotLocation
:: (HasPantryConfig env, HasLogFunc env)
=> RawSnapshotLocation
-> RIO env SnapshotLocation
completeSnapshotLocation (RSLCompiler c) = pure $ SLCompiler c
completeSnapshotLocation (RSLFilePath f) = pure $ SLFilePath f
completeSnapshotLocation (RSLUrl url (Just blobKey)) = pure $ SLUrl url blobKey
completeSnapshotLocation (RSLUrl url Nothing) = do
bs <- loadFromURL url Nothing
pure $ SLUrl url (bsToBlobKey bs)
completeSnapshotLayer
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawSnapshotLayer
-> RIO env SnapshotLayer
completeSnapshotLayer rsnapshot = do
parent' <- completeSnapshotLocation $ rslParent rsnapshot
pls <- traverseConcurrently completePackageLocation $ rslLocations rsnapshot
pure SnapshotLayer
{ slParent = parent'
, slLocations = pls
, slCompiler= rslCompiler rsnapshot
, slDropPackages = rslDropPackages rsnapshot
, slFlags = rslFlags rsnapshot
, slHidden = rslHidden rsnapshot
, slGhcOptions = rslGhcOptions rsnapshot
, slPublishTime = rslPublishTime rsnapshot
}
traverseConcurrently_
:: (Foldable f, HasPantryConfig env)
=> (a -> RIO env ())
-> f a
-> RIO env ()
traverseConcurrently_ f t0 = do
cnt <- view $ pantryConfigL.to pcConnectionCount
traverseConcurrentlyWith_ cnt f t0
traverseConcurrentlyWith_
:: (MonadUnliftIO m, Foldable f)
=> Int
-> (a -> m ())
-> f a
-> m ()
traverseConcurrentlyWith_ count f t0 = do
queue <- newTVarIO $ toList t0
replicateConcurrently_ count $
fix $ \loop -> join $ atomically $ do
toProcess <- readTVar queue
case toProcess of
[] -> pure (pure ())
(x:rest) -> do
writeTVar queue rest
pure $ do
f x
loop
traverseConcurrently
:: (HasPantryConfig env, Traversable t)
=> (a -> RIO env b)
-> t a
-> RIO env (t b)
traverseConcurrently f t0 = do
cnt <- view $ pantryConfigL.to pcConnectionCount
traverseConcurrentlyWith cnt f t0
traverseConcurrentlyWith
:: (MonadUnliftIO m, Traversable t)
=> Int
-> (a -> m b)
-> t a
-> m (t b)
traverseConcurrentlyWith count f t0 = do
(queue, t1) <- atomically $ do
queueDList <- newTVar id
t1 <- for t0 $ \x -> do
res <- newEmptyTMVar
modifyTVar queueDList (. ((x, res):))
pure $ atomically $ takeTMVar res
dlist <- readTVar queueDList
queue <- newTVar $ dlist []
pure (queue, t1)
replicateConcurrently_ count $
fix $ \loop -> join $ atomically $ do
toProcess <- readTVar queue
case toProcess of
[] -> pure (pure ())
((x, res):rest) -> do
writeTVar queue rest
pure $ do
y <- f x
atomically $ putTMVar res y
loop
sequence t1
loadSnapshotRaw
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawSnapshotLocation
-> RIO env RawSnapshot
loadSnapshotRaw loc = do
eres <- loadRawSnapshotLayer loc
case eres of
Left wc ->
pure RawSnapshot
{ rsCompiler = wc
, rsPackages = mempty
, rsDrop = mempty
}
Right (rsl, _) -> do
snap0 <- loadSnapshotRaw $ rslParent rsl
(packages, unused) <-
addPackagesToSnapshot
(display loc)
(rslLocations rsl)
AddPackagesConfig
{ apcDrop = rslDropPackages rsl
, apcFlags = rslFlags rsl
, apcHiddens = rslHidden rsl
, apcGhcOptions = rslGhcOptions rsl
}
(rsPackages snap0)
warnUnusedAddPackagesConfig (display loc) unused
pure RawSnapshot
{ rsCompiler = fromMaybe (rsCompiler snap0) (rslCompiler rsl)
, rsPackages = packages
, rsDrop = apcDrop unused
}
loadSnapshot
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> SnapshotLocation
-> RIO env RawSnapshot
loadSnapshot loc = do
eres <- loadSnapshotLayer loc
case eres of
Left wc ->
pure RawSnapshot
{ rsCompiler = wc
, rsPackages = mempty
, rsDrop = mempty
}
Right rsl -> do
snap0 <- loadSnapshotRaw $ rslParent rsl
(packages, unused) <-
addPackagesToSnapshot
(display loc)
(rslLocations rsl)
AddPackagesConfig
{ apcDrop = rslDropPackages rsl
, apcFlags = rslFlags rsl
, apcHiddens = rslHidden rsl
, apcGhcOptions = rslGhcOptions rsl
}
(rsPackages snap0)
warnUnusedAddPackagesConfig (display loc) unused
pure RawSnapshot
{ rsCompiler = fromMaybe (rsCompiler snap0) (rslCompiler rsl)
, rsPackages = packages
, rsDrop = apcDrop unused
}
data CompletedPLI = CompletedPLI !RawPackageLocationImmutable !PackageLocationImmutable
data CompletedSL = CompletedSL !RawSnapshotLocation !SnapshotLocation
loadAndCompleteSnapshot
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> SnapshotLocation
-> Map RawSnapshotLocation SnapshotLocation
-> Map RawPackageLocationImmutable PackageLocationImmutable
-> RIO env (Snapshot, [CompletedSL], [CompletedPLI])
loadAndCompleteSnapshot loc cachedSL cachedPL =
loadAndCompleteSnapshotRaw (toRawSL loc) cachedSL cachedPL
loadAndCompleteSnapshotRaw
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawSnapshotLocation
-> Map RawSnapshotLocation SnapshotLocation
-> Map RawPackageLocationImmutable PackageLocationImmutable
-> RIO env (Snapshot, [CompletedSL], [CompletedPLI])
loadAndCompleteSnapshotRaw rawLoc cacheSL cachePL = do
eres <- case Map.lookup rawLoc cacheSL of
Just loc -> right (\rsl -> (rsl, (CompletedSL rawLoc loc))) <$> loadSnapshotLayer loc
Nothing -> loadRawSnapshotLayer rawLoc
case eres of
Left wc ->
let snapshot = Snapshot
{ snapshotCompiler = wc
, snapshotPackages = mempty
, snapshotDrop = mempty
}
in pure (snapshot, [CompletedSL (RSLCompiler wc) (SLCompiler wc)], [])
Right (rsl, sloc) -> do
(snap0, slocs, completed0) <- loadAndCompleteSnapshotRaw (rslParent rsl) cacheSL cachePL
(packages, completed, unused) <-
addAndCompletePackagesToSnapshot
rawLoc
cachePL
(rslLocations rsl)
AddPackagesConfig
{ apcDrop = rslDropPackages rsl
, apcFlags = rslFlags rsl
, apcHiddens = rslHidden rsl
, apcGhcOptions = rslGhcOptions rsl
}
(snapshotPackages snap0)
warnUnusedAddPackagesConfig (display rawLoc) unused
let snapshot = Snapshot
{ snapshotCompiler = fromMaybe (snapshotCompiler snap0) (rslCompiler rsl)
, snapshotPackages = packages
, snapshotDrop = apcDrop unused
}
return (snapshot, sloc : slocs,completed0 ++ completed)
data SingleOrNot a
= Single !a
| Multiple !a !a !([a] -> [a])
instance Semigroup (SingleOrNot a) where
Single a <> Single b = Multiple a b id
Single a <> Multiple b c d = Multiple a b ((c:) . d)
Multiple a b c <> Single d = Multiple a b (c . (d:))
Multiple a b c <> Multiple d e f =
Multiple a b (c . (d:) . (e:) . f)
sonToEither :: (k, SingleOrNot a) -> Either (k, a) (k, [a])
sonToEither (k, Single a) = Left (k, a)
sonToEither (k, Multiple a b c) = Right (k, (a : b : c []))
data AddPackagesConfig = AddPackagesConfig
{ apcDrop :: !(Set PackageName)
, apcFlags :: !(Map PackageName (Map FlagName Bool))
, apcHiddens :: !(Map PackageName Bool)
, apcGhcOptions :: !(Map PackageName [Text])
}
warnUnusedAddPackagesConfig
:: HasLogFunc env
=> Utf8Builder
-> AddPackagesConfig
-> RIO env ()
warnUnusedAddPackagesConfig source (AddPackagesConfig _drops flags hiddens options) = do
unless (null ls) $ do
logWarn $ "Some warnings discovered when adding packages to snapshot (" <> source <> ")"
traverse_ logWarn ls
where
ls = concat [flags', hiddens', options']
flags' =
map
(\pn ->
"Setting flags for non-existent package: " <>
fromString (packageNameString pn))
(Map.keys flags)
hiddens' =
map
(\pn ->
"Hiding non-existent package: " <>
fromString (packageNameString pn))
(Map.keys hiddens)
options' =
map
(\pn ->
"Setting options for non-existent package: " <>
fromString (packageNameString pn))
(Map.keys options)
addPackagesToSnapshot
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> Utf8Builder
-> [RawPackageLocationImmutable]
-> AddPackagesConfig
-> Map PackageName RawSnapshotPackage
-> RIO env (Map PackageName RawSnapshotPackage, AddPackagesConfig)
addPackagesToSnapshot source newPackages (AddPackagesConfig drops flags hiddens options) old = do
new' <- for newPackages $ \loc -> do
name <- getPackageLocationName loc
pure (name, RawSnapshotPackage
{ rspLocation = loc
, rspFlags = Map.findWithDefault mempty name flags
, rspHidden = Map.findWithDefault False name hiddens
, rspGhcOptions = Map.findWithDefault [] name options
})
let (newSingles, newMultiples)
= partitionEithers
$ map sonToEither
$ Map.toList
$ Map.fromListWith (<>)
$ map (second Single) new'
unless (null $ newMultiples) $ throwIO $
DuplicatePackageNames source $ map (second (map rspLocation)) newMultiples
let new = Map.fromList newSingles
allPackages0 = new `Map.union` (old `Map.difference` Map.fromSet (const ()) drops)
allPackages = flip Map.mapWithKey allPackages0 $ \name rsp ->
rsp
{ rspFlags = Map.findWithDefault (rspFlags rsp) name flags
, rspHidden = Map.findWithDefault (rspHidden rsp) name hiddens
, rspGhcOptions = Map.findWithDefault (rspGhcOptions rsp) name options
}
unused = AddPackagesConfig
(drops `Set.difference` Map.keysSet old)
(flags `Map.difference` allPackages)
(hiddens `Map.difference` allPackages)
(options `Map.difference` allPackages)
pure (allPackages, unused)
cachedSnapshotCompletePackageLocation :: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> Map RawPackageLocationImmutable PackageLocationImmutable
-> RawPackageLocationImmutable
-> RIO env PackageLocationImmutable
cachedSnapshotCompletePackageLocation cachePackages rpli = do
let xs = Map.lookup rpli cachePackages
case xs of
Nothing -> completePackageLocation rpli
Just x -> pure x
addAndCompletePackagesToSnapshot
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawSnapshotLocation
-> Map RawPackageLocationImmutable PackageLocationImmutable
-> [RawPackageLocationImmutable]
-> AddPackagesConfig
-> Map PackageName SnapshotPackage
-> RIO env (Map PackageName SnapshotPackage, [CompletedPLI], AddPackagesConfig)
addAndCompletePackagesToSnapshot loc cachedPL newPackages (AddPackagesConfig drops flags hiddens options) old = do
let source = display loc
addPackage :: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> ([(PackageName, SnapshotPackage)],[CompletedPLI])
-> RawPackageLocationImmutable
-> RIO env ([(PackageName, SnapshotPackage)], [CompletedPLI])
addPackage (ps, completed) rawLoc = do
complLoc <- cachedSnapshotCompletePackageLocation cachedPL rawLoc
let PackageIdentifier name _ = packageLocationIdent complLoc
p = (name, SnapshotPackage
{ spLocation = complLoc
, spFlags = Map.findWithDefault mempty name flags
, spHidden = Map.findWithDefault False name hiddens
, spGhcOptions = Map.findWithDefault [] name options
})
completed' = if toRawPLI complLoc == rawLoc
then completed
else CompletedPLI rawLoc complLoc:completed
pure (p:ps, completed')
(revNew, revCompleted) <- foldM addPackage ([], []) newPackages
let (newSingles, newMultiples)
= partitionEithers
$ map sonToEither
$ Map.toList
$ Map.fromListWith (<>)
$ map (second Single) (reverse revNew)
unless (null $ newMultiples) $ throwIO $
DuplicatePackageNames source $ map (second (map (toRawPLI . spLocation))) newMultiples
let new = Map.fromList newSingles
allPackages0 = new `Map.union` (old `Map.difference` Map.fromSet (const ()) drops)
allPackages = flip Map.mapWithKey allPackages0 $ \name sp ->
sp
{ spFlags = Map.findWithDefault (spFlags sp) name flags
, spHidden = Map.findWithDefault (spHidden sp) name hiddens
, spGhcOptions = Map.findWithDefault (spGhcOptions sp) name options
}
unused = AddPackagesConfig
(drops `Set.difference` Map.keysSet old)
(flags `Map.difference` allPackages)
(hiddens `Map.difference` allPackages)
(options `Map.difference` allPackages)
pure (allPackages, reverse revCompleted, unused)
loadRawSnapshotLayer
:: (HasPantryConfig env, HasLogFunc env)
=> RawSnapshotLocation
-> RIO env (Either WantedCompiler (RawSnapshotLayer, CompletedSL))
loadRawSnapshotLayer (RSLCompiler compiler) = pure $ Left compiler
loadRawSnapshotLayer rsl@(RSLUrl url blob) =
handleAny (throwIO . InvalidSnapshot rsl) $ do
bs <- loadFromURL url blob
value <- Yaml.decodeThrow bs
snapshot <- warningsParserHelperRaw rsl value Nothing
pure $ Right (snapshot, (CompletedSL rsl (SLUrl url (bsToBlobKey bs))))
loadRawSnapshotLayer rsl@(RSLFilePath fp) =
handleAny (throwIO . InvalidSnapshot rsl) $ do
value <- Yaml.decodeFileThrow $ toFilePath $ resolvedAbsolute fp
snapshot <- warningsParserHelperRaw rsl value $ Just $ parent $ resolvedAbsolute fp
pure $ Right (snapshot, CompletedSL rsl (SLFilePath fp))
loadSnapshotLayer
:: (HasPantryConfig env, HasLogFunc env)
=> SnapshotLocation
-> RIO env (Either WantedCompiler RawSnapshotLayer)
loadSnapshotLayer (SLCompiler compiler) = pure $ Left compiler
loadSnapshotLayer sl@(SLUrl url blob) =
handleAny (throwIO . InvalidSnapshot (toRawSL sl)) $ do
bs <- loadFromURL url (Just blob)
value <- Yaml.decodeThrow bs
snapshot <- warningsParserHelper sl value Nothing
pure $ Right snapshot
loadSnapshotLayer sl@(SLFilePath fp) =
handleAny (throwIO . InvalidSnapshot (toRawSL sl)) $ do
value <- Yaml.decodeFileThrow $ toFilePath $ resolvedAbsolute fp
snapshot <- warningsParserHelper sl value $ Just $ parent $ resolvedAbsolute fp
pure $ Right snapshot
loadFromURL
:: (HasPantryConfig env, HasLogFunc env)
=> Text
-> Maybe BlobKey
-> RIO env ByteString
loadFromURL url Nothing = do
mcached <- withStorage $ loadURLBlob url
case mcached of
Just bs -> return bs
Nothing -> loadWithCheck url Nothing
loadFromURL url (Just bkey) = do
mcached <- withStorage $ loadBlob bkey
case mcached of
Just bs -> return bs
Nothing -> loadWithCheck url (Just bkey)
loadWithCheck
:: (HasPantryConfig env, HasLogFunc env)
=> Text
-> Maybe BlobKey
-> RIO env ByteString
loadWithCheck url mblobkey = do
let (msha, msize) =
case mblobkey of
Nothing -> (Nothing, Nothing)
Just (BlobKey sha size) -> (Just sha, Just size)
(_, _, bss) <- httpSinkChecked url msha msize sinkList
let bs = B.concat bss
withStorage $ storeURLBlob url bs
return bs
warningsParserHelperRaw
:: HasLogFunc env
=> RawSnapshotLocation
-> Value
-> Maybe (Path Abs Dir)
-> RIO env RawSnapshotLayer
warningsParserHelperRaw rsl val mdir =
case parseEither Yaml.parseJSON val of
Left e -> throwIO $ Couldn'tParseSnapshot rsl e
Right (WithJSONWarnings x ws) -> do
unless (null ws) $ do
logWarn $ "Warnings when parsing snapshot " <> display rsl
for_ ws $ logWarn . display
resolvePaths mdir x
warningsParserHelper
:: HasLogFunc env
=> SnapshotLocation
-> Value
-> Maybe (Path Abs Dir)
-> RIO env RawSnapshotLayer
warningsParserHelper sl val mdir =
case parseEither Yaml.parseJSON val of
Left e -> throwIO $ Couldn'tParseSnapshot (toRawSL sl) e
Right (WithJSONWarnings x ws) -> do
unless (null ws) $ do
logWarn $ "Warnings when parsing snapshot " <> display sl
for_ ws $ logWarn . display
resolvePaths mdir x
getPackageLocationName
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RIO env PackageName
getPackageLocationName = fmap pkgName . getRawPackageLocationIdent
packageLocationIdent
:: PackageLocationImmutable
-> PackageIdentifier
packageLocationIdent (PLIHackage ident _ _) = ident
packageLocationIdent (PLIRepo _ pm) = pmIdent pm
packageLocationIdent (PLIArchive _ pm) = pmIdent pm
packageLocationVersion
:: PackageLocationImmutable
-> Version
packageLocationVersion (PLIHackage pident _ _) = pkgVersion pident
packageLocationVersion (PLIRepo _ pm) = pkgVersion (pmIdent pm)
packageLocationVersion (PLIArchive _ pm) = pkgVersion (pmIdent pm)
getRawPackageLocationIdent
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RIO env PackageIdentifier
getRawPackageLocationIdent (RPLIHackage (PackageIdentifierRevision name version _) _) = pure $ PackageIdentifier name version
getRawPackageLocationIdent (RPLIRepo _ RawPackageMetadata { rpmName = Just name, rpmVersion = Just version }) = pure $ PackageIdentifier name version
getRawPackageLocationIdent (RPLIArchive _ RawPackageMetadata { rpmName = Just name, rpmVersion = Just version }) = pure $ PackageIdentifier name version
getRawPackageLocationIdent rpli = packageIdent <$> loadPackageRaw rpli
getRawPackageLocationTreeKey
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> RawPackageLocationImmutable
-> RIO env TreeKey
getRawPackageLocationTreeKey pl =
case getRawTreeKey pl of
Just treeKey' -> pure treeKey'
Nothing ->
case pl of
RPLIHackage pir _ -> getHackageTarballKey pir
RPLIArchive archive pm -> getArchiveKey pl archive pm
RPLIRepo repo pm -> getRepoKey repo pm
getPackageLocationTreeKey
:: (HasPantryConfig env, HasLogFunc env, HasProcessContext env)
=> PackageLocationImmutable
-> RIO env TreeKey
getPackageLocationTreeKey pl = pure $ getTreeKey pl
getRawTreeKey :: RawPackageLocationImmutable -> Maybe TreeKey
getRawTreeKey (RPLIHackage _ mtree) = mtree
getRawTreeKey (RPLIArchive _ rpm) = rpmTreeKey rpm
getRawTreeKey (RPLIRepo _ rpm) = rpmTreeKey rpm
getTreeKey :: PackageLocationImmutable -> TreeKey
getTreeKey (PLIHackage _ _ tree) = tree
getTreeKey (PLIArchive _ pm) = pmTreeKey pm
getTreeKey (PLIRepo _ pm) = pmTreeKey pm
data PantryApp = PantryApp
{ paSimpleApp :: !SimpleApp
, paPantryConfig :: !PantryConfig
, paUseColor :: !Bool
, paTermWidth :: !Int
, paStylesUpdate :: !StylesUpdate
}
simpleAppL :: Lens' PantryApp SimpleApp
simpleAppL = lens paSimpleApp (\x y -> x { paSimpleApp = y })
hpackExecutableL :: Lens' PantryConfig HpackExecutable
hpackExecutableL k pconfig = fmap (\hpExe -> pconfig { pcHpackExecutable = hpExe }) (k (pcHpackExecutable pconfig))
instance HasLogFunc PantryApp where
logFuncL = simpleAppL.logFuncL
instance HasPantryConfig PantryApp where
pantryConfigL = lens paPantryConfig (\x y -> x { paPantryConfig = y })
instance HasProcessContext PantryApp where
processContextL = simpleAppL.processContextL
instance HasStylesUpdate PantryApp where
stylesUpdateL = lens paStylesUpdate (\x y -> x { paStylesUpdate = y })
instance HasTerm PantryApp where
useColorL = lens paUseColor (\x y -> x { paUseColor = y })
termWidthL = lens paTermWidth (\x y -> x { paTermWidth = y })
runPantryApp :: MonadIO m => RIO PantryApp a -> m a
runPantryApp f = runSimpleApp $ do
sa <- ask
stack <- getAppUserDataDirectory "stack"
root <- parseAbsDir $ stack FilePath.</> "pantry"
withPantryConfig
root
defaultHackageSecurityConfig
HpackBundled
8
$ \pc ->
runRIO
PantryApp
{ paSimpleApp = sa
, paPantryConfig = pc
, paTermWidth = 100
, paUseColor = True
, paStylesUpdate = mempty
}
f
runPantryAppClean :: MonadIO m => RIO PantryApp a -> m a
runPantryAppClean f = liftIO $ withSystemTempDirectory "pantry-clean" $ \dir -> runSimpleApp $ do
sa <- ask
root <- resolveDir' dir
withPantryConfig
root
defaultHackageSecurityConfig
HpackBundled
8
$ \pc ->
runRIO
PantryApp
{ paSimpleApp = sa
, paPantryConfig = pc
, paTermWidth = 100
, paUseColor = True
, paStylesUpdate = mempty
}
f
loadGlobalHints
:: (HasTerm env, HasPantryConfig env)
=> WantedCompiler
-> RIO env (Maybe (Map PackageName Version))
loadGlobalHints wc =
inner False
where
inner alreadyDownloaded = do
dest <- getGlobalHintsFile
req <- parseRequest "https://raw.githubusercontent.com/fpco/stackage-content/master/stack/global-hints.yaml"
downloaded <- download req dest
eres <- tryAny (inner2 dest)
mres <-
case eres of
Left e -> Nothing <$ logError ("Error when parsing global hints: " <> displayShow e)
Right x -> pure x
case mres of
Nothing | not alreadyDownloaded && not downloaded -> do
logInfo $
"Could not find local global hints for " <>
RIO.display wc <>
", forcing a redownload"
x <- redownload req dest
if x
then inner True
else do
logInfo "Redownload didn't happen"
pure Nothing
_ -> pure mres
inner2 dest
= liftIO
$ Map.lookup wc . fmap (fmap unCabalString . unCabalStringMap)
<$> Yaml.decodeFileThrow (toFilePath dest)
partitionReplacedDependencies ::
Ord id
=> Map PackageName a
-> (a -> PackageName)
-> (a -> id)
-> (a -> [id])
-> Set PackageName
-> (Map PackageName [PackageName], Map PackageName a)
partitionReplacedDependencies globals getName getId getDeps overrides =
flip execState (replaced, mempty) $
for (Map.toList globals) $ prunePackageWithDeps globals' getName getDeps
where
globals' = Map.fromList $ map (getId &&& id) (Map.elems globals)
replaced = Map.map (const []) $ Map.restrictKeys globals overrides
prunePackageWithDeps ::
Ord id
=> Map id a
-> (a -> PackageName)
-> (a -> [id])
-> (PackageName, a)
-> State (Map PackageName [PackageName], Map PackageName a) Bool
prunePackageWithDeps pkgs getName getDeps (pname, a) = do
(pruned, kept) <- get
if Map.member pname pruned
then return True
else if Map.member pname kept
then return False
else do
let deps = Map.elems $ Map.restrictKeys pkgs (Set.fromList $ getDeps a)
prunedDeps <- forMaybeM deps $ \dep -> do
let depName = getName dep
isPruned <- prunePackageWithDeps pkgs getName getDeps (depName, dep)
pure $ if isPruned then Just depName else Nothing
if null prunedDeps
then do
modify' $ second (Map.insert pname a)
else do
modify' $ first (Map.insert pname prunedDeps)
return $ not (null prunedDeps)
withSnapshotCache
:: (HasPantryConfig env, HasLogFunc env)
=> SnapshotCacheHash
-> RIO env (Map PackageName (Set ModuleName))
-> ((ModuleName -> RIO env [PackageName]) -> RIO env a)
-> RIO env a
withSnapshotCache hash getModuleMapping f = do
mres <- withStorage $ getSnapshotCacheByHash hash
cacheId <- case mres of
Nothing -> do
scId <- withStorage $ getSnapshotCacheId hash
packageModules <- getModuleMapping
logWarn "Populating snapshot module name cache"
withStorage $ storeSnapshotModuleCache scId packageModules
return scId
Just scId -> pure scId
f $ withStorage . loadExposedModulePackages cacheId