-- |Load dependencies for a project using `ghc-pkg`. module Data.Prune.Dependency where import Prelude hiding (words) import Data.Foldable (find) import Data.Map (Map) import Data.Maybe (mapMaybe) import Data.Text (Text, isPrefixOf, pack, strip, unpack, words) import Data.Traversable (for) import System.Exit (ExitCode(ExitFailure, ExitSuccess)) import Turtle (shellStrict) import qualified Data.Map as Map import qualified Data.Set as Set import Data.Prune.ImportParser (parseExposedModules) import qualified Data.Prune.Types as T -- |Run a shell command or exit if it fails. runOrFail :: Text -> IO Text runOrFail cmd = shellStrict cmd mempty >>= \case (ExitSuccess, out) -> pure out (ExitFailure _, out) -> fail . unpack $ "Failed to \"" <> cmd <> "\" due to " <> out -- |For the dependencies listed in the specified packages, load `ghc-pkg` and inspect the `exposed-modules` field. -- Return a map of module to dependency name. getDependencyByModule :: FilePath -> [T.Package] -> IO (Map T.ModuleName T.DependencyName) getDependencyByModule stackYamlFile packages = do let allDependencies = foldMap T.packageBaseDependencies packages <> foldMap T.compilableDependencies (foldMap T.packageCompilables packages) tupleDependency x = (, x) <$> find (\d -> isPrefixOf (T.unDependencyName d) x) allDependencies compilerBin <- strip <$> runOrFail ("stack --stack-yaml " <> pack stackYamlFile <> " path --compiler-bin") snapshotPkgDb <- strip <$> runOrFail ("stack --stack-yaml " <> pack stackYamlFile <> " path --snapshot-pkg-db") globalPkgDb <- strip <$> runOrFail ("stack --stack-yaml " <> pack stackYamlFile <> " path --global-pkg-db") localPkgDb <- strip <$> runOrFail ("stack --stack-yaml " <> pack stackYamlFile <> " path --local-pkg-db") let snapshotGhcPkg = compilerBin <> "/ghc-pkg --package-db " <> snapshotPkgDb globalGhcPkg = compilerBin <> "/ghc-pkg --package-db " <> globalPkgDb localGhcPkg = compilerBin <> "/ghc-pkg --package-db " <> localPkgDb snapshotPkgs <- mapMaybe tupleDependency . words . strip <$> runOrFail (snapshotGhcPkg <> " list --simple-output") globalPkgs <- mapMaybe tupleDependency . words . strip <$> runOrFail (globalGhcPkg <> " list --simple-output") localPkgs <- mapMaybe tupleDependency . words . strip <$> runOrFail (localGhcPkg <> " list --simple-output") snapshotDependencyByModule <- fmap mconcat . for snapshotPkgs $ \(dependencyName, pkg) -> do moduleNames <- parseExposedModules . unpack . strip =<< runOrFail (snapshotGhcPkg <> " field " <> pkg <> " exposed-modules") pure . Map.fromList . map (, dependencyName) . Set.toList $ moduleNames globalDependencyByModule <- fmap mconcat . for globalPkgs $ \(dependencyName, pkg) -> do moduleNames <- parseExposedModules . unpack . strip =<< runOrFail (globalGhcPkg <> " field " <> pkg <> " exposed-modules") pure . Map.fromList . map (, dependencyName) . Set.toList $ moduleNames localDependencyByModule <- fmap mconcat . for localPkgs $ \(dependencyName, pkg) -> do moduleNames <- parseExposedModules . unpack . strip =<< runOrFail (localGhcPkg <> " field " <> pkg <> " exposed-modules") pure . Map.fromList . map (, dependencyName) . Set.toList $ moduleNames pure $ snapshotDependencyByModule <> globalDependencyByModule <> localDependencyByModule