2014-04-14 19:35:29 +00:00
|
|
|
module Propellor.Property.Postfix where
|
|
|
|
|
|
|
|
import Propellor
|
|
|
|
import qualified Propellor.Property.Apt as Apt
|
2014-07-18 02:20:16 +00:00
|
|
|
import Propellor.Property.File
|
|
|
|
|
|
|
|
import qualified Data.Map as M
|
|
|
|
import Data.List
|
2014-04-14 19:35:29 +00:00
|
|
|
|
|
|
|
installed :: Property
|
|
|
|
installed = Apt.serviceInstalledRunning "postfix"
|
|
|
|
|
|
|
|
-- | Configures postfix as a satellite system, which
|
|
|
|
-- relats all mail through a relay host, which defaults to smtp.domain.
|
|
|
|
--
|
|
|
|
-- The smarthost may refuse to relay mail on to other domains, without
|
|
|
|
-- futher coniguration/keys. But this should be enough to get cron job
|
|
|
|
-- mail flowing to a place where it will be seen.
|
|
|
|
satellite :: Property
|
2014-04-14 19:42:26 +00:00
|
|
|
satellite = setup `requires` installed
|
|
|
|
where
|
2014-04-18 07:59:06 +00:00
|
|
|
setup = trivial $ property "postfix satellite system" $ do
|
2014-06-01 00:48:23 +00:00
|
|
|
hn <- asks hostName
|
2014-04-14 19:42:26 +00:00
|
|
|
ensureProperty $ Apt.reConfigure "postfix"
|
|
|
|
[ ("postfix/main_mailer_type", "select", "Satellite system")
|
|
|
|
, ("postfix/root_address", "string", "root")
|
|
|
|
, ("postfix/destinations", "string", " ")
|
|
|
|
, ("postfix/mailname", "string", hn)
|
|
|
|
]
|
2014-07-18 01:16:03 +00:00
|
|
|
|
|
|
|
-- | Sets up a file by running a property (which the filename is passed
|
|
|
|
-- to). If the setup property makes a change, postmap will be run on the
|
|
|
|
-- file, and postfix will be reloaded.
|
|
|
|
mappedFile :: FilePath -> (FilePath -> Property) -> Property
|
|
|
|
mappedFile f setup = setup f
|
2014-07-18 02:20:16 +00:00
|
|
|
`onChange` cmdProperty "postmap" [f]
|
|
|
|
|
|
|
|
-- | Parses main.cf, and removes any initial configuration lines that are
|
|
|
|
-- overridden to other values later in the file.
|
|
|
|
--
|
|
|
|
-- For example, to add some settings, removing any old settings:
|
|
|
|
--
|
|
|
|
-- > mainCf `File.containsLines`
|
|
|
|
-- > [ "# I like bars."
|
|
|
|
-- > , "foo = bar"
|
|
|
|
-- > ] `onChange` dedupMainCf
|
|
|
|
--
|
|
|
|
-- Note that multiline configurations that continue onto the next line
|
|
|
|
-- are not currently supported.
|
|
|
|
dedupMainCf :: Property
|
|
|
|
dedupMainCf = fileProperty "postfix main.cf dedupped" go mainCf
|
2014-07-18 01:16:03 +00:00
|
|
|
where
|
2014-07-18 02:20:16 +00:00
|
|
|
go ls =
|
|
|
|
let parsed = map parse ls
|
|
|
|
in dedup [] (keycounts $ rights parsed) parsed
|
|
|
|
|
|
|
|
parse l
|
|
|
|
| "#" `isPrefixOf` l = Left l
|
|
|
|
| "=" `isInfixOf` l = Right (separate (== '=') l)
|
|
|
|
| otherwise = Left l
|
|
|
|
fmt k v = k ++ "=" ++ v
|
|
|
|
|
|
|
|
keycounts = M.fromListWith (+) . map (\(k, _v) -> (k, (1 :: Integer)))
|
|
|
|
|
|
|
|
dedup c _ [] = reverse c
|
|
|
|
dedup c kc ((Left v):rest) = dedup (v:c) kc rest
|
|
|
|
dedup c kc ((Right (k, v)):rest) = case M.lookup k kc of
|
|
|
|
Just n | n > 1 -> dedup c (M.insert k (n - 1) kc) rest
|
|
|
|
_ -> dedup (fmt k v:c) kc rest
|
|
|
|
|
|
|
|
-- | The main config file for postfix.
|
|
|
|
mainCf :: FilePath
|
|
|
|
mainCf = "/etc/postfix/main.cf"
|