{-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE TupleSections #-} {-# LANGUAGE ScopedTypeVariables #-} -- Load information on package sources module Stack.Build.Source ( loadSourceMap , SourceMap , PackageSource (..) , localFlags , loadLocals ) where import Network.HTTP.Client.Conduit (HasHttpManager) import Control.Applicative ((<|>), (<$>), (<*>)) import Control.Exception (catch) import Control.Monad import Control.Monad.Catch (MonadCatch) import Control.Monad.IO.Class import Control.Monad.Logger import Control.Monad.Reader (MonadReader, asks) import Control.Monad.Trans.Resource import Crypto.Hash (Digest, SHA256) import Crypto.Hash.Conduit (sinkHash) import Data.Byteable (toBytes) import qualified Data.ByteString as S import Data.Conduit (($$), ZipSink (..)) import qualified Data.Conduit.Binary as CB import qualified Data.Conduit.List as CL import Data.Either import qualified Data.Foldable as F import Data.Function import qualified Data.HashSet as HashSet import Data.List import qualified Data.Map as Map import Data.Map.Strict (Map) import Data.Maybe import Data.Monoid ((<>), Any (..), mconcat) import Data.Set (Set) import qualified Data.Set as Set import Data.Text (Text) import qualified Data.Text as T import Path import Prelude hiding (writeFile) import Stack.Build.Cache import Stack.Build.Types import Stack.BuildPlan (loadMiniBuildPlan, shadowMiniBuildPlan) import Stack.Constants (wiredInPackages) import Stack.Package import Stack.PackageIndex import Stack.Types import System.Directory hiding (findExecutable, findFiles) import System.IO (withBinaryFile, IOMode (ReadMode)) import System.IO.Error (isDoesNotExistError) type SourceMap = Map PackageName PackageSource -- | Where the package's source is located: local directory or package index data PackageSource = PSLocal LocalPackage | PSUpstream Version InstallLocation (Map FlagName Bool) -- ^ Upstream packages could be installed in either local or snapshot -- databases; this is what 'InstallLocation' specifies. deriving Show instance PackageInstallInfo PackageSource where piiVersion (PSLocal lp) = packageVersion $ lpPackage lp piiVersion (PSUpstream v _ _) = v piiLocation (PSLocal _) = Local piiLocation (PSUpstream _ loc _) = loc loadSourceMap :: (MonadIO m, MonadCatch m, MonadReader env m, HasBuildConfig env, MonadBaseControl IO m, HasHttpManager env, MonadLogger m, HasEnvConfig env) => BuildOpts -> m ( MiniBuildPlan , [LocalPackage] , Set PackageName -- non-local targets , SourceMap ) loadSourceMap bopts = do bconfig <- asks getBuildConfig mbp0 <- case bcResolver bconfig of ResolverSnapshot snapName -> do $logDebug $ "Checking resolver: " <> renderSnapName snapName loadMiniBuildPlan snapName ResolverGhc ghc -> return MiniBuildPlan { mbpGhcVersion = fromMajorVersion ghc , mbpPackages = Map.empty } menv <- getMinimalEnvOverride caches <- getPackageCaches menv let latestVersion = Map.fromList $ map toTuple $ Map.keys caches (locals, extraNames, extraIdents) <- loadLocals bopts latestVersion let -- loadLocals returns PackageName (foo) and PackageIdentifier (bar-1.2.3) targets separately; -- here we combine them into nonLocalTargets. This is one of the -- return values of this function. nonLocalTargets :: Set PackageName nonLocalTargets = extraNames <> Set.map packageIdentifierName extraIdents -- Extend extra-deps to encompass targets requested on the command line -- that are not in the snapshot. extraDeps0 = extendExtraDeps (bcExtraDeps bconfig) mbp0 latestVersion extraNames extraIdents let shadowed = Set.fromList (map (packageName . lpPackage) locals) <> Map.keysSet extraDeps0 (mbp, extraDeps1) = shadowMiniBuildPlan mbp0 shadowed -- Add the extra deps from the stack.yaml file to the deps grabbed from -- the snapshot extraDeps2 = Map.union (Map.map (\v -> (v, Map.empty)) extraDeps0) (Map.map (\mpi -> (mpiVersion mpi, mpiFlags mpi)) extraDeps1) -- Overwrite any flag settings with those from the config file extraDeps3 = Map.mapWithKey (\n (v, f) -> PSUpstream v Local $ fromMaybe f $ Map.lookup n $ bcFlags bconfig) extraDeps2 let sourceMap = Map.unions [ Map.fromList $ flip map locals $ \lp -> let p = lpPackage lp in (packageName p, PSLocal lp) , extraDeps3 , flip fmap (mbpPackages mbp) $ \mpi -> (PSUpstream (mpiVersion mpi) Snap (mpiFlags mpi)) ] `Map.difference` Map.fromList (map (, ()) (HashSet.toList wiredInPackages)) let unknown = Set.difference nonLocalTargets $ Map.keysSet sourceMap unless (Set.null unknown) $ do let toEither name = case Map.lookup name latestVersion of Nothing -> Left name Just version -> Right (name, version) eithers = map toEither $ Set.toList unknown (unknown', notInIndex) = partitionEithers eithers throwM $ UnknownTargets (Set.fromList unknown') (Map.fromList notInIndex) (bcStackYaml bconfig) return (mbp, locals, nonLocalTargets, sourceMap) -- | 'loadLocals' combines two pieces of information: -- -- 1. Targets, i.e. arguments passed to stack such as @foo@ and @bar@ in the @stack foo bar@ invocation -- -- 2. Local packages listed in @stack.yaml@ -- -- It returns: -- -- 1. For every local package, a 'LocalPackage' structure -- -- 2. If a target does not correspond to a local package but is a valid -- 'PackageName' or 'PackageIdentifier', it is returned as such. -- -- NOTE: as the function is written right now, it may "drop" targets if -- they correspond to existing directories not listed in stack.yaml. This -- may be a bug. loadLocals :: forall m env . (MonadReader env m, HasBuildConfig env, MonadIO m, MonadLogger m, MonadThrow m, MonadCatch m,HasEnvConfig env) => BuildOpts -> Map PackageName Version -> m ([LocalPackage], Set PackageName, Set PackageIdentifier) loadLocals bopts latestVersion = do targets <- mapM parseTarget $ case boptsTargets bopts of [] -> ["."] x -> x -- Group targets by their kind (dirs, names, idents) <- case partitionEithers targets of ([], targets') -> return $ partitionTargetSpecs targets' (bad, _) -> throwM $ Couldn'tParseTargets bad econfig <- asks getEnvConfig bconfig <- asks getBuildConfig -- Iterate over local packages declared in stack.yaml and turn them -- into LocalPackage structures. The targets affect whether these -- packages will be marked as wanted. lps <- forM (Map.toList $ bcPackages bconfig) $ \(dir, validWanted) -> do cabalfp <- getCabalFileName dir name <- parsePackageNameFromFilePath cabalfp let wanted = validWanted && isWanted dirs names dir name config = PackageConfig { packageConfigEnableTests = False , packageConfigEnableBenchmarks = False , packageConfigFlags = localFlags (boptsFlags bopts) bconfig name , packageConfigGhcVersion = envConfigGhcVersion econfig , packageConfigPlatform = configPlatform $ getConfig bconfig } configFinal = config { packageConfigEnableTests = case boptsFinalAction bopts of DoTests _ -> wanted _ -> False , packageConfigEnableBenchmarks = wanted && boptsFinalAction bopts == DoBenchmarks } pkg <- readPackage config cabalfp pkgFinal <- readPackage configFinal cabalfp when (packageName pkg /= name) $ throwM $ MismatchedCabalName cabalfp (packageName pkg) mbuildCache <- tryGetBuildCache dir files <- getPackageFiles (packageFiles pkg) AllFiles cabalfp (isDirty, newBuildCache) <- checkBuildCache (fromMaybe Map.empty mbuildCache) (map toFilePath $ Set.toList files) return LocalPackage { lpPackage = pkg , lpPackageFinal = pkgFinal , lpWanted = wanted , lpFiles = files , lpDirtyFiles = isDirty , lpNewBuildCache = newBuildCache , lpCabalFile = cabalfp , lpDir = dir , lpComponents = fromMaybe Set.empty $ Map.lookup name names } let known = Set.fromList $ map (packageName . lpPackage) lps unknown = Set.difference (Map.keysSet names) known return (lps, unknown, idents) where -- Attempt to parse a TargetSpec based on its textual form and on -- whether it is a name of an existing directory. -- -- If a TargetSpec is not recognized, return it verbatim as Left. parseTarget :: Text -> m (Either Text TargetSpec) parseTarget t = do let s = T.unpack t isDir <- liftIO $ doesDirectoryExist s if isDir then liftM (Right . TSDir) $ liftIO (canonicalizePath s) >>= parseAbsDir else return $ maybe (Left t) Right $ (flip TSName Set.empty <$> parsePackageNameFromString s) <|> (TSIdent <$> parsePackageIdentifierFromString s) <|> (do t' <- T.stripSuffix ":latest" t name <- parsePackageNameFromString $ T.unpack t' version <- Map.lookup name latestVersion Just $ TSIdent $ PackageIdentifier name version) <|> (do let (name', rest) = T.break (== ':') t component <- T.stripPrefix ":" rest name <- parsePackageNameFromString $ T.unpack name' Just $ TSName name $ Set.singleton component) isWanted dirs names dir name = name `Map.member` names || any (`isParentOf` dir) dirs || any (== dir) dirs data TargetSpec = TSName PackageName (Set Text) | TSIdent PackageIdentifier | TSDir (Path Abs Dir) partitionTargetSpecs :: [TargetSpec] -> ([Path Abs Dir], Map PackageName (Set Text), Set PackageIdentifier) partitionTargetSpecs = loop id Map.empty Set.empty where loop dirs names idents ts0 = case ts0 of [] -> (dirs [], names, idents) TSName name comps:ts -> loop dirs (Map.insertWith Set.union name comps names) idents ts TSIdent ident:ts -> loop dirs names (Set.insert ident idents) ts TSDir dir:ts -> loop (dirs . (dir:)) names idents ts -- | All flags for a local package localFlags :: (Map (Maybe PackageName) (Map FlagName Bool)) -> BuildConfig -> PackageName -> Map FlagName Bool localFlags boptsflags bconfig name = Map.unions [ fromMaybe Map.empty $ Map.lookup (Just name) $ boptsflags , fromMaybe Map.empty $ Map.lookup Nothing $ boptsflags , fromMaybe Map.empty $ Map.lookup name $ bcFlags bconfig ] -- | Add in necessary packages to extra dependencies -- -- See https://github.com/commercialhaskell/stack/issues/272 for the requirements of this function extendExtraDeps :: Map PackageName Version -- ^ original extra deps -> MiniBuildPlan -> Map PackageName Version -- ^ latest versions in indices -> Set PackageName -- ^ extra package names desired -> Set PackageIdentifier -- ^ extra package identifiers desired -> Map PackageName Version -- ^ new extradeps extendExtraDeps extraDeps0 mbp latestVersion extraNames extraIdents = F.foldl' addIdent (F.foldl' addName extraDeps0 extraNames) extraIdents where snapshot = fmap mpiVersion $ mbpPackages mbp addName m name = case Map.lookup name m <|> Map.lookup name snapshot of -- alright exists in snapshot or extra-deps Just _ -> m Nothing -> case Map.lookup name latestVersion of -- use the latest version in the index Just v -> Map.insert name v m -- does not exist, will be reported as an error Nothing -> m addIdent m (PackageIdentifier name version) = case Map.lookup name snapshot of -- the version matches what's in the snapshot, so just use the snapshot version Just version' | version == version' -> m _ -> Map.insert name version m -- | Compare the current filesystem state to the cached information, and -- determine (1) if the files are dirty, and (2) the new cache values. checkBuildCache :: MonadIO m => Map FilePath FileCacheInfo -- ^ old cache -> [FilePath] -- ^ files in package -> m (Bool, Map FilePath FileCacheInfo) checkBuildCache oldCache files = liftIO $ do (Any isDirty, m) <- fmap mconcat $ mapM go files return (isDirty, m) where go fp = do mmodTime <- getModTimeMaybe fp case mmodTime of Nothing -> return (Any False, Map.empty) Just modTime' -> do (isDirty, newFci) <- case Map.lookup fp oldCache of Just fci | fciModTime fci == modTime' -> return (False, fci) | otherwise -> do newFci <- calcFci modTime' fp let isDirty = fciSize fci /= fciSize newFci || fciHash fci /= fciHash newFci return (isDirty, newFci) Nothing -> do newFci <- calcFci modTime' fp return (True, newFci) return (Any isDirty, Map.singleton fp newFci) getModTimeMaybe fp = liftIO (catch (liftM (Just . modTime) (getModificationTime fp)) (\e -> if isDoesNotExistError e then return Nothing else throwM e)) calcFci modTime' fp = withBinaryFile fp ReadMode $ \h -> do (size, digest) <- CB.sourceHandle h $$ getZipSink ((,) <$> ZipSink (CL.fold (\x y -> x + fromIntegral (S.length y)) 0) <*> ZipSink sinkHash) return FileCacheInfo { fciModTime = modTime' , fciSize = size , fciHash = toBytes (digest :: Digest SHA256) }