module Stackctl.Spec.Discover ( discoverSpecs , buildSpecPath ) where import Stackctl.Prelude import Data.List.Extra (dropPrefix) import qualified Data.List.NonEmpty as NE import Stackctl.AWS import Stackctl.AWS.Scope import Stackctl.Config (HasConfig) import Stackctl.DirectoryOption (HasDirectoryOption(..), unDirectoryOption) import Stackctl.FilterOption (HasFilterOption(..), filterStackSpecs) import Stackctl.StackSpec import Stackctl.StackSpecPath import System.FilePath (isPathSeparator) import System.FilePath.Glob discoverSpecs :: ( MonadMask m , MonadResource m , MonadLogger m , MonadReader env m , HasAwsScope env , HasConfig env , HasDirectoryOption env , HasFilterOption env ) => m [StackSpec] discoverSpecs = do dir <- unDirectoryOption <$> view directoryOptionL scope <- view awsScopeL paths <- globRelativeTo dir $ awsScopeSpecPatterns scope filterOption <- view filterOptionL let toSpecPath = stackSpecPathFromFilePath scope (errs, specPaths) = partitionEithers $ map toSpecPath paths context = [ "path" .= dir , "filters" .= filterOption , "paths" .= length paths , "errors" .= length errs , "specs" .= length specPaths ] withThreadContext context $ do checkForDuplicateStackNames specPaths specs <- sortStackSpecs . filterStackSpecs filterOption <$> traverse (readStackSpec dir) specPaths when (null specs) $ logWarn "No specs found" specs <$ logDebug ("Discovered specs" :# ["matched" .= length specs]) checkForDuplicateStackNames :: (MonadIO m, MonadLogger m) => [StackSpecPath] -> m () checkForDuplicateStackNames = traverse_ reportCollisions . NE.nonEmpty . filter ((> 1) . length) . NE.groupAllWith stackSpecPathStackName where reportCollisions :: (MonadIO m, MonadLogger m) => NonEmpty (NonEmpty StackSpecPath) -> m () reportCollisions errs = do for_ errs $ \specPaths -> do let collidingPaths = stackSpecPathBasePath <$> specPaths logError $ "Multiple specifications produced the same Stack name" :# [ "name" .= stackSpecPathStackName (NE.head specPaths) , "paths" .= collidingPaths ] exitFailure buildSpecPath :: (MonadReader env m, HasAwsScope env) => StackName -> FilePath -> m StackSpecPath buildSpecPath stackName stackPath = do scope <- view awsScopeL pure $ stackSpecPath scope stackName stackPath globRelativeTo :: MonadIO m => FilePath -> [Pattern] -> m [FilePath] globRelativeTo dir ps = liftIO $ do map (dropWhile isPathSeparator . dropPrefix dir) . concat <$> globDir ps dir