{-# LANGUAGE FunctionalDependencies #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE TypeApplications #-}
module Hiera.Server (
startHiera
, dummyHiera
, HieraQueryType (..)
, readQueryType
, HieraQueryFunc
) where
import XPrelude
import Data.Aeson
import qualified Data.Aeson as Aeson
import Data.Aeson.Lens
import qualified Data.Attoparsec.Text as AT
import qualified Data.ByteString.Lazy as BS
import qualified Data.Either.Strict as S
import qualified Data.FileCache as Cache
import qualified Data.List as List
import Data.String (fromString)
import qualified Data.Text as Text
import qualified Data.Vector as Vector
import qualified Data.Yaml as Yaml
import qualified System.Directory as Directory
import qualified System.FilePath as FilePath
import System.FilePath.Lens (directory)
import Puppet.Language
data HieraQueryType
= QFirst
| QUnique
| QHash
| QDeep { _knockoutPrefix :: Maybe Text
, _sortMerged :: Bool
, _mergeHashArray :: Bool
} deriving (Show)
readQueryType :: Text -> Maybe HieraQueryType
readQueryType s =
case s of
"first" -> Just QFirst
"unique" -> Just QUnique
"hash" -> Just QHash
_ -> Nothing
type HieraQueryFunc m = Container Text
-> Text
-> HieraQueryType
-> m (S.Either PrettyError (Maybe PValue))
data Backend
= YamlBackend FilePath
| JsonBackend FilePath
deriving (Show)
data HieraStringPart
= HPString Text
| HPVariable Text
deriving (Show)
instance Pretty HieraStringPart where
pretty (HPString t) = ppline t
pretty (HPVariable v) = dullred (ppline ("%{" <> v <> "}"))
prettyList = mconcat . map pretty
newtype InterpolableHieraString = InterpolableHieraString
{ getInterpolableHieraString :: [HieraStringPart]
} deriving (Show)
resolveString :: Container Text -> InterpolableHieraString -> Maybe Text
resolveString vars = fmap Text.concat . mapM resolve . getInterpolableHieraString
where
resolve (HPString x) = Just x
resolve (HPVariable v) = vars ^? ix v
instance FromJSON InterpolableHieraString where
parseJSON (String s) = case parseInterpolableString s of
Right x -> return (InterpolableHieraString x)
Left rr -> fail rr
parseJSON _ = fail "Invalid value type"
interpolableString :: AT.Parser [HieraStringPart]
interpolableString = AT.many1 (fmap HPString rawPart <|> fmap HPVariable interpPart)
where
rawPart = AT.takeWhile1 (/= '%')
interpPart = AT.string "%{" *> AT.takeWhile1 (/= '}') <* AT.char '}'
parseInterpolableString :: Text -> Either String [HieraStringPart]
parseInterpolableString = AT.parseOnly interpolableString
data HieraConfigFile = HieraConfigFile
{ _version :: Int
, _backends :: [Backend]
, _hierarchy :: [InterpolableHieraString]
} deriving (Show)
data QRead = QRead
{ _qvars :: Container Text
, _qtype :: HieraQueryType
, _qhier :: [Value]
}
makeLenses ''QRead
instance FromJSON HieraConfigFile where
parseJSON =
let
mkHiera5 :: Object -> Yaml.Parser HieraConfigFile
mkHiera5 v = do
let paths = Object v ^.. key "hierarchy" . values . key "paths" . values
path = Object v ^.. key "hierarchy" .values .key "path"
hierarchy_value <- case Object v ^? key "hierarchy" . nth 0 of
Just (Object h) -> pure h
_ -> fail "Hiera config should define at least one hierarchy"
datadir <- hierarchy_value .:? "datadir" >>= \case
Just (String dir) -> pure dir
Just _ -> fail "datadir should be a string"
Nothing -> pure $ Object v ^. key "defaults" . key "datadir" . _String
HieraConfigFile
<$> pure 5
<*> pure [ YamlBackend (toS datadir) ]
<*> mapM parseJSON (paths <> path)
mkHiera3 v =
HieraConfigFile
<$> pure 3
<*> (v .:? ":backends" .!= ["yaml"] >>= mapM mkBackend3)
<*> (v .:? ":hierarchy" .!= [InterpolableHieraString [HPString "common"]])
where
mkBackend3 :: Text -> Yaml.Parser Backend
mkBackend3 name = do
(backendConstructor, skey) <- case name of
"yaml" -> return (YamlBackend, ":yaml")
"json" -> return (JsonBackend, ":json")
_ -> fail ("Unknown backend " <> toS name)
datadir <- case Object v ^? key skey . key ":datadir" of
Just (String dir) -> return dir
Just _ -> fail ":datadir should be a string"
Nothing -> return "/etc/puppet/hieradata"
pure (backendConstructor (toS datadir))
in
Aeson.withObject "v3 or v5" $ \o ->
o .:? "version" >>= \case
Just (5::Int) -> mkHiera5 o
Just _ -> fail "Hiera configuration version different than 5 is not supported."
Nothing -> mkHiera3 o
type Cache = Cache.FileCacheR String Value
startHiera :: String -> FilePath -> IO (HieraQueryFunc IO)
startHiera layer fp =
Yaml.decodeFileEither fp >>= \case
Left (Yaml.AesonException "Error in $: Hiera configuration version different than 5 is not supported.") -> do
logInfoStr ("Detect a hiera configuration format in " <> fp <> " at version 4. This format is not recognized. Using a dummy hiera.")
pure dummyHiera
Left ex -> panic (show ex)
Right cfg@HieraConfigFile{..} -> do
logInfoStr ("Detect a hiera " <> layer <> " configuration format in " <> fp <> " at version " <> show _version)
cache <- Cache.newFileCache
pure (query cfg fp cache)
dummyHiera :: Monad m => HieraQueryFunc m
dummyHiera _ _ _ = return $ S.Right Nothing
query :: HieraConfigFile -> FilePath -> Cache -> HieraQueryFunc IO
query HieraConfigFile {_version, _backends, _hierarchy} fp cache vars hquery qt = do
let searchin = do
mhierarchy <- resolveString vars <$> _hierarchy
Just h <- [mhierarchy]
backend <- _backends
let decodeInfo :: (FilePath -> IO (S.Either String Value), String, String)
decodeInfo =
case backend of
JsonBackend dir -> (fmap (strictifyEither . Aeson.eitherDecode') . BS.readFile , dir, ".json")
YamlBackend dir -> (fmap (strictifyEither . (_Left %~ show)) . Yaml.decodeFileEither, dir, ".yaml")
pure (decodeInfo, toS h)
mvals <- forM searchin $ \((decodefunction, datadir, extension), h) -> do
let extension' = if snd (FilePath.splitExtension h) == ".yaml"
then ""
else extension
filename = basedir <> datadir <> "/" <> h <> extension'
basedir = case datadir of
'/' : _ -> mempty
_ -> fp ^. directory <> "/"
querycache = do
efilecontent <- Cache.query cache filename (decodefunction filename)
case efilecontent of
S.Left r -> do
logWarningStr $ "Hiera: error when reading file " <> filename <> ": "<> r
pure Nothing
S.Right val -> do
pure (Just val)
ifM (Directory.doesFileExist filename)
querycache
(pure Nothing)
let vals = catMaybes mvals
logDebugStr ("Looking up '" <> toS hquery <> "' with backends " <> List.unwords (fmap show _backends ))
return (strictifyEither $ runReader (runExceptT (recursiveQuery hquery [])) (QRead vars qt vals))
type QM a = ExceptT PrettyError (Reader QRead) a
checkLoop :: Text -> [Text] -> QM ()
checkLoop x xs =
when (x `elem` xs) (throwError ("Loop in hiera: " <> fromString (Text.unpack (Text.intercalate ", " (x:xs)))))
recursiveQuery :: Text -> [Text] -> QM (Maybe PValue)
recursiveQuery curquery prevqueries = do
checkLoop curquery prevqueries
rawlookups <- mapMaybe (preview (key curquery)) <$> view qhier
lookups <- mapM (resolveValue (curquery : prevqueries)) rawlookups
case lookups of
[] -> return Nothing
(x:xs) -> do
qt <- view qtype
let evalue = foldM (mergeWith qt) x xs
case Aeson.fromJSON <$> evalue of
Left _ -> return Nothing
Right (Aeson.Success o) -> return o
Right (Aeson.Error rr) -> throwError ("Something horrible happened in recursiveQuery: " <> fromString rr)
resolveValue :: [Text] -> Value -> QM Value
resolveValue prevqueries value =
case value of
String t -> String <$> resolveText prevqueries t
Array arr -> Array <$> mapM (resolveValue prevqueries) arr
Object hh -> Object <$> mapM (resolveValue prevqueries) hh
_ -> return value
resolveText :: [Text] -> Text -> QM Text
resolveText prevqueries t =
case parseInterpolableString t of
Right qparts -> Text.concat <$> mapM (resolveStringPart prevqueries) qparts
Left _ -> return t
resolveStringPart :: [Text] -> HieraStringPart -> QM Text
resolveStringPart prevqueries sp =
case sp of
HPString s -> return s
HPVariable varname -> do
let varsolve = fmap PString . preview (ix varname) <$> view qvars
r <- case Text.stripPrefix "lookup('" varname >>= Text.stripSuffix "')" of
Just lk -> recursiveQuery lk prevqueries
Nothing -> varsolve
case r of
Just (PString v) -> return v
_ -> return mempty
mergeWith :: HieraQueryType -> Value -> Value -> Either PrettyError Value
mergeWith qt cur new =
case qt of
QFirst -> return cur
QUnique ->
let getArray x = case x of
Array array -> Vector.toList array
_ -> [x]
curarray = getArray cur
newarray = getArray new
in case new of
Object _ -> throwError "Tried to merge a hash"
_ -> return (Array (Vector.fromList (List.nub (curarray ++ newarray))))
QHash -> case (cur, new) of
(Object curh, Object newh) -> return (Object (curh <> newh))
_ -> throwError (PrettyError ("Tried to merge things that are not hashes: " <> ppline (show cur) <+> ppline (show new)))
QDeep{} -> throwError "deep queries not supported"