module Language.PureScript.Make.BuildPlan
( BuildPlan()
, construct
, getResult
, collectErrors
, collectResults
, markComplete
, needsRebuild
) where
import Prelude
import Control.Concurrent.Async.Lifted as A
import Control.Concurrent.Lifted as C
import Control.Monad hiding (sequence)
import Control.Monad.Trans.Class (lift)
import Control.Monad.Trans.Control (MonadBaseControl(..))
import Control.Monad.Trans.Maybe (MaybeT(..), runMaybeT)
import Data.Foldable (foldl')
import qualified Data.Map as M
import Data.Maybe (catMaybes, fromMaybe)
import Data.Time.Clock (UTCTime)
import Language.PureScript.AST
import Language.PureScript.Crash
import qualified Language.PureScript.CST as CST
import Language.PureScript.Errors
import Language.PureScript.Externs
import Language.PureScript.Make.Actions as Actions
import Language.PureScript.Names (ModuleName)
data BuildPlan = BuildPlan
{ bpPrebuilt :: M.Map ModuleName Prebuilt
, bpBuildJobs :: M.Map ModuleName BuildJob
}
data Prebuilt = Prebuilt
{ pbModificationTime :: UTCTime
, pbExternsFile :: ExternsFile
}
data BuildJob = BuildJob
{ bjResult :: C.MVar (Maybe (MultipleErrors, ExternsFile))
, bjErrors :: C.MVar (Maybe MultipleErrors)
}
markComplete
:: (MonadBaseControl IO m)
=> BuildPlan
-> ModuleName
-> Maybe (MultipleErrors, ExternsFile)
-> Maybe MultipleErrors
-> m ()
markComplete buildPlan moduleName result errors = do
let BuildJob rVar eVar = fromMaybe (internalError "make: markComplete no barrier") $ M.lookup moduleName (bpBuildJobs buildPlan)
putMVar rVar result
putMVar eVar errors
needsRebuild :: BuildPlan -> ModuleName -> Bool
needsRebuild bp moduleName = M.member moduleName (bpBuildJobs bp)
collectErrors
:: (MonadBaseControl IO m)
=> BuildPlan
-> m [MultipleErrors]
collectErrors buildPlan = do
errors <- traverse readMVar $ map bjErrors $ M.elems (bpBuildJobs buildPlan)
pure (catMaybes errors)
collectResults
:: (MonadBaseControl IO m)
=> BuildPlan
-> m (M.Map ModuleName ExternsFile)
collectResults buildPlan = do
let externs = M.map pbExternsFile (bpPrebuilt buildPlan)
barrierResults <- traverse (takeMVar . bjResult) $ bpBuildJobs buildPlan
let barrierExterns = M.map (snd . fromMaybe (internalError "make: externs were missing but no errors reported.")) barrierResults
pure (M.union externs barrierExterns)
getResult
:: (MonadBaseControl IO m)
=> BuildPlan
-> ModuleName
-> m (Maybe (MultipleErrors, ExternsFile))
getResult buildPlan moduleName =
case M.lookup moduleName (bpPrebuilt buildPlan) of
Just es ->
pure (Just (MultipleErrors [], pbExternsFile es))
Nothing ->
readMVar $ bjResult $ fromMaybe (internalError "make: no barrier") $ M.lookup moduleName (bpBuildJobs buildPlan)
construct
:: forall m. (Monad m, MonadBaseControl IO m)
=> MakeActions m
-> ([CST.PartialResult Module], [(ModuleName, [ModuleName])])
-> m BuildPlan
construct MakeActions{..} (sorted, graph) = do
prebuilt <- foldl' collectPrebuiltModules M.empty . catMaybes <$> A.forConcurrently sorted findExistingExtern
let toBeRebuilt = filter (not . flip M.member prebuilt . getModuleName . CST.resPartial) sorted
buildJobs <- foldM makeBuildJob M.empty (map (getModuleName . CST.resPartial) toBeRebuilt)
pure $ BuildPlan prebuilt buildJobs
where
makeBuildJob prev moduleName = do
buildJob <- BuildJob <$> C.newEmptyMVar <*> C.newEmptyMVar
pure (M.insert moduleName buildJob prev)
findExistingExtern :: CST.PartialResult Module -> m (Maybe (ModuleName, Bool, Prebuilt))
findExistingExtern (getModuleName . CST.resPartial -> moduleName) = runMaybeT $ do
inputTimestamp <- lift $ getInputTimestamp moduleName
(rebuildNever, existingTimestamp) <-
case inputTimestamp of
Left RebuildNever ->
fmap (True,) $ MaybeT $ getOutputTimestamp moduleName
Right (Just t1) -> do
outputTimestamp <- MaybeT $ getOutputTimestamp moduleName
guard (t1 < outputTimestamp)
pure (False, outputTimestamp)
_ -> mzero
externsFile <- MaybeT $ decodeExterns . snd <$> readExterns moduleName
pure (moduleName, rebuildNever, Prebuilt existingTimestamp externsFile)
collectPrebuiltModules :: M.Map ModuleName Prebuilt -> (ModuleName, Bool, Prebuilt) -> M.Map ModuleName Prebuilt
collectPrebuiltModules prev (moduleName, rebuildNever, pb)
| rebuildNever = M.insert moduleName pb prev
| otherwise = do
let deps = fromMaybe (internalError "make: module not found in dependency graph.") (lookup moduleName graph)
case traverse (fmap pbModificationTime . flip M.lookup prev) deps of
Nothing ->
prev
Just modTimes ->
case maximumMaybe modTimes of
Just depModTime | pbModificationTime pb < depModTime ->
prev
_ -> M.insert moduleName pb prev
maximumMaybe :: Ord a => [a] -> Maybe a
maximumMaybe [] = Nothing
maximumMaybe xs = Just $ maximum xs