{-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE UndecidableInstances #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE TypeFamilies #-} {-# LANGUAGE ConstraintKinds #-} {-# LANGUAGE PolyKinds #-} {-# LANGUAGE DataKinds #-} {-# LANGUAGE DeriveDataTypeable #-} module Propellor.Types ( -- * Core data types Host(..) , Property(..) , property , property'' , Desc , RevertableProperty(..) , () , Propellor(..) , LiftPropellor(..) , Info -- * Types of properties , UnixLike , Linux , DebianLike , Debian , Buntish , ArchLinux , FreeBSD , HasInfo , type (+) , TightenTargets(..) , TightenTargetsAllowed -- * Combining and modifying properties , Combines(..) , CombinedType , ResultCombiner , adjustPropertySatisfy -- * Other included types , module Propellor.Types.OS , module Propellor.Types.ConfigurableValue , module Propellor.Types.Dns , module Propellor.Types.Result , module Propellor.Types.ZFS ) where import GHC.TypeLits hiding (type (+)) import GHC.Exts (Constraint) import Data.Type.Bool import qualified Data.Semigroup as Sem import Data.Monoid import Control.Applicative import Prelude import Propellor.Types.Core import Propellor.Types.Info import Propellor.Types.OS import Propellor.Types.ConfigurableValue import Propellor.Types.Dns import Propellor.Types.Result import Propellor.Types.MetaTypes import Propellor.Types.ZFS -- | The core data type of Propellor, this represents a property -- that the system should have, with a description, and an action to ensure -- it has the property. -- -- There are different types of properties that target different OS's, -- and so have different metatypes. -- For example: "Property DebianLike" and "Property FreeBSD". -- -- Also, some properties have associated `Info`, which is indicated in -- their type: "Property (HasInfo + DebianLike)" -- -- There are many associated type families, which are mostly used -- internally, so you needn't worry about them. data Property metatypes = Property metatypes Desc (Maybe (Propellor Result)) Info [ChildProperty] instance Show (Property metatypes) where show p = "property " ++ show (getDesc p) -- | Constructs a Property, from a description and an action to run to -- ensure the Property is met. -- -- Due to the polymorphic return type of this function, most uses will need -- to specify a type signature. This lets you specify what OS the property -- targets, etc. -- -- For example: -- -- > foo :: Property Debian -- > foo = property "foo" $ do -- > ... -- > return MadeChange property :: SingI metatypes => Desc -> Propellor Result -> Property (MetaTypes metatypes) property d a = Property sing d (Just a) mempty mempty property'' :: SingI metatypes => Desc -> Maybe (Propellor Result) -> Property (MetaTypes metatypes) property'' d a = Property sing d a mempty mempty -- | Changes the action that is performed to satisfy a property. adjustPropertySatisfy :: Property metatypes -> (Propellor Result -> Propellor Result) -> Property metatypes adjustPropertySatisfy (Property t d s i c) f = Property t d (f <$> s) i c -- | A property that can be reverted. The first Property is run -- normally and the second is run when it's reverted. -- -- See `Propellor.Property.Versioned.Versioned` -- for a way to use RevertableProperty to define different -- versions of a host. data RevertableProperty setupmetatypes undometatypes = RevertableProperty { setupRevertableProperty :: Property setupmetatypes , undoRevertableProperty :: Property undometatypes } instance Show (RevertableProperty setupmetatypes undometatypes) where show (RevertableProperty p _) = show p -- | Shorthand to construct a revertable property from any two Properties. () :: Property setupmetatypes -> Property undometatypes -> RevertableProperty setupmetatypes undometatypes setup undo = RevertableProperty setup undo instance IsProp (Property metatypes) where setDesc (Property t _ a i c) d = Property t d a i c getDesc (Property _ d _ _ _) = d getChildren (Property _ _ _ _ c) = c addChildren (Property t d a i c) c' = Property t d a i (c ++ c') getInfoRecursive (Property _ _ _ i c) = i <> mconcat (map getInfoRecursive c) getInfo (Property _ _ _ i _) = i toChildProperty (Property _ d a i c) = ChildProperty d a i c getSatisfy (Property _ _ a _ _) = a instance IsProp (RevertableProperty setupmetatypes undometatypes) where -- | Sets the description of both sides. setDesc (RevertableProperty p1 p2) d = RevertableProperty (setDesc p1 d) (setDesc p2 ("not " ++ d)) getDesc (RevertableProperty p1 _) = getDesc p1 getChildren (RevertableProperty p1 _) = getChildren p1 -- | Only add children to the active side. addChildren (RevertableProperty p1 p2) c = RevertableProperty (addChildren p1 c) p2 -- | Return the Info of the currently active side. getInfoRecursive (RevertableProperty p1 _p2) = getInfoRecursive p1 getInfo (RevertableProperty p1 _p2) = getInfo p1 toChildProperty (RevertableProperty p1 _p2) = toChildProperty p1 getSatisfy (RevertableProperty p1 _) = getSatisfy p1 -- | Type level calculation of the type that results from combining two -- types of properties. type family CombinedType x y where CombinedType (Property (MetaTypes x)) (Property (MetaTypes y)) = Property (MetaTypes (Combine x y)) CombinedType (RevertableProperty (MetaTypes x) (MetaTypes x')) (RevertableProperty (MetaTypes y) (MetaTypes y')) = RevertableProperty (MetaTypes (Combine x y)) (MetaTypes (Combine x' y')) -- When only one of the properties is revertable, the combined -- property is not fully revertable, so is not a RevertableProperty. CombinedType (RevertableProperty (MetaTypes x) (MetaTypes x')) (Property (MetaTypes y)) = Property (MetaTypes (Combine x y)) CombinedType (Property (MetaTypes x)) (RevertableProperty (MetaTypes y) (MetaTypes y')) = Property (MetaTypes (Combine x y)) type ResultCombiner = Maybe (Propellor Result) -> Maybe (Propellor Result) -> Maybe (Propellor Result) class Combines x y where -- | Combines together two properties, yielding a property that -- has the description and info of the first, and that has the -- second property as a child property. combineWith :: ResultCombiner -- ^ How to combine the actions to satisfy the properties. -> ResultCombiner -- ^ Used when combining revertable properties, to combine -- their reversion actions. -> x -> y -> CombinedType x y instance (CheckCombinable x y, SingI (Combine x y)) => Combines (Property (MetaTypes x)) (Property (MetaTypes y)) where combineWith f _ (Property _ d1 a1 i1 c1) (Property _ d2 a2 i2 c2) = Property sing d1 (f a1 a2) i1 (ChildProperty d2 a2 i2 c2 : c1) instance (CheckCombinable x y, CheckCombinable x' y', SingI (Combine x y), SingI (Combine x' y')) => Combines (RevertableProperty (MetaTypes x) (MetaTypes x')) (RevertableProperty (MetaTypes y) (MetaTypes y')) where combineWith sf tf (RevertableProperty s1 t1) (RevertableProperty s2 t2) = RevertableProperty (combineWith sf tf s1 s2) (combineWith tf sf t1 t2) instance (CheckCombinable x y, SingI (Combine x y)) => Combines (RevertableProperty (MetaTypes x) (MetaTypes x')) (Property (MetaTypes y)) where combineWith sf tf (RevertableProperty x _) y = combineWith sf tf x y instance (CheckCombinable x y, SingI (Combine x y)) => Combines (Property (MetaTypes x)) (RevertableProperty (MetaTypes y) (MetaTypes y')) where combineWith sf tf x (RevertableProperty y _) = combineWith sf tf x y class TightenTargets p where -- | Tightens the MetaType list of a Property (or similar), -- to contain fewer targets. -- -- For example, to make a property that uses apt-get, which is only -- available on DebianLike systems: -- -- > upgraded :: Property DebianLike -- > upgraded = tightenTargets $ cmdProperty "apt-get" ["upgrade"] tightenTargets :: ( TightenTargetsAllowed untightened tightened , SingI tightened ) => p (MetaTypes untightened) -> p (MetaTypes tightened) -- Note that this uses PolyKinds type family TightenTargetsAllowed untightened tightened :: Constraint where TightenTargetsAllowed untightened tightened = If (Targets tightened `IsSubset` Targets untightened && NonTargets untightened `IsSubset` NonTargets tightened) ('True ~ 'True) (IfStuck (Targets tightened) (DelayError ('Text "Unable to infer desired Property type in this use of tightenTargets." ':$$: ('Text "Consider adding a type annotation.") ) ) (DelayErrorFcf ('Text "This use of tightenTargets would widen, not narrow, adding: " ':$$: PrettyPrintMetaTypes (Difference (Targets tightened) (Targets untightened)) ) ) ) instance TightenTargets Property where tightenTargets (Property _ d a i c) = Property sing d a i c -- | Any type of Property is a Semigroup. When properties x and y are -- appended together, the resulting property has a description like -- "x and y". Note that when x fails to be ensured, it will not -- try to ensure y. instance SingI metatypes => Sem.Semigroup (Property (MetaTypes metatypes)) where Property _ d1 a1 i1 c1 <> Property _ d2 a2 i2 c2 = Property sing d (a1 <> a2) (i1 <> i2) (c1 <> c2) where -- Avoid including "noop property" in description -- when using eg mconcat. d = case (a1, a2) of (Just _, Just _) -> d1 <> " and " <> d2 (Just _, Nothing) -> d1 (Nothing, Just _) -> d2 (Nothing, Nothing) -> d1 -- | Any type of Property is a Monoid. instance SingI metatypes => Monoid (Property (MetaTypes metatypes)) where -- | A property that does nothing. mempty = Property sing "noop property" Nothing mempty mempty mappend = (Sem.<>) -- | Any type of RevertableProperty is a Semigroup. When revertable -- properties x and y are appended together, the resulting revertable -- property has a description like "x and y". -- Note that when x fails to be ensured, it will not try to ensure y. instance ( Sem.Semigroup (Property (MetaTypes setupmetatypes)) , Sem.Semigroup (Property (MetaTypes undometatypes)) , SingI setupmetatypes , SingI undometatypes ) => Sem.Semigroup (RevertableProperty (MetaTypes setupmetatypes) (MetaTypes undometatypes)) where RevertableProperty s1 u1 <> RevertableProperty s2 u2 = RevertableProperty (s1 <> s2) (u2 <> u1) instance ( Monoid (Property (MetaTypes setupmetatypes)) , Monoid (Property (MetaTypes undometatypes)) , SingI setupmetatypes , SingI undometatypes ) => Monoid (RevertableProperty (MetaTypes setupmetatypes) (MetaTypes undometatypes)) where mempty = RevertableProperty mempty mempty mappend = (Sem.<>)