module Language.PureScript.ModuleDependencies (
sortModules
) where
import Data.Data
import Data.Graph
import Data.Generics
import Data.List (nub)
import Language.PureScript.Declarations
import Language.PureScript.Names
sortModules :: [Module] -> Either String [Module]
sortModules ms = do
let verts = map (\m -> (m, getModuleName m, usedModules m)) ms
mapM toModule $ stronglyConnComp verts
usedModules :: (Data d) => d -> [ModuleName]
usedModules = nub . everything (++) (mkQ [] qualifiedIdents `extQ` qualifiedProperNames `extQ` imports)
where
qualifiedIdents :: Qualified Ident -> [ModuleName]
qualifiedIdents (Qualified (Just mn) _) = [mn]
qualifiedIdents _ = []
qualifiedProperNames :: Qualified ProperName -> [ModuleName]
qualifiedProperNames (Qualified (Just mn) _) = [mn]
qualifiedProperNames _ = []
imports :: Declaration -> [ModuleName]
imports (ImportDeclaration mn _) = [mn]
imports _ = []
getModuleName :: Module -> ModuleName
getModuleName (Module mn _) = mn
toModule :: SCC Module -> Either String Module
toModule (AcyclicSCC m) = return m
toModule (CyclicSCC [m]) = return m
toModule (CyclicSCC _) = Left "Cycle in module dependencies"