propellor/src/Propellor/Property/Cron.hs

62 lines
2.1 KiB
Haskell
Raw Normal View History

2014-03-31 05:29:47 +00:00
module Propellor.Property.Cron where
import Propellor
import qualified Propellor.Property.File as File
import qualified Propellor.Property.Apt as Apt
2014-04-17 23:43:12 +00:00
import Utility.SafeCommand
2014-07-18 16:35:00 +00:00
import Utility.FileMode
2014-03-31 05:29:47 +00:00
2014-04-13 02:36:36 +00:00
import Data.Char
2014-03-31 05:29:47 +00:00
type CronTimes = String
2014-04-17 23:43:12 +00:00
-- | Installs a cron job, run as a specified user, in a particular
-- directory. Note that the Desc must be unique, as it is used for the
-- cron.d/ filename.
--
-- Only one instance of the cron job is allowed to run at a time, no matter
-- how long it runs. This is accomplished using flock locking of the cron
-- job file.
--
-- The cron job's output will only be emailed if it exits nonzero.
job :: Desc -> CronTimes -> UserName -> FilePath -> String -> Property NoInfo
2014-07-18 16:35:00 +00:00
job desc times user cddir command = combineProperties ("cronned " ++ desc)
[ cronjobfile `File.hasContent`
[ "# Generated by propellor"
, ""
, "SHELL=/bin/sh"
, "PATH=/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin"
, ""
2014-07-18 16:39:45 +00:00
, times ++ "\t" ++ user ++ "\tchronic " ++ shellEscape scriptfile
2014-07-18 16:35:00 +00:00
]
-- Use a separate script because it makes the cron job name
-- prettier in emails, and also allows running the job manually.
, scriptfile `File.hasContent`
[ "#!/bin/sh"
, "# Generated by propellor"
, "set -e"
2014-07-18 16:39:45 +00:00
, "flock -n " ++ shellEscape cronjobfile
2014-07-18 16:35:00 +00:00
++ " sh -c " ++ shellEscape cmdline
]
, scriptfile `File.mode` combineModes (readModes ++ executeModes)
2014-03-31 05:29:47 +00:00
]
2014-04-08 23:31:03 +00:00
`requires` Apt.serviceInstalledRunning "cron"
`requires` Apt.installed ["util-linux", "moreutils"]
2014-04-13 02:36:36 +00:00
where
2014-04-18 03:32:42 +00:00
cmdline = "cd " ++ cddir ++ " && ( " ++ command ++ " )"
2014-07-18 16:35:00 +00:00
cronjobfile = "/etc/cron.d/" ++ name
scriptfile = "/usr/local/bin/" ++ name ++ "_cronjob"
name = map sanitize desc
2014-04-13 02:36:36 +00:00
sanitize c
| isAlphaNum c = c
| otherwise = '_'
2014-04-01 21:54:20 +00:00
-- | Installs a cron job, and runs it niced and ioniced.
niceJob :: Desc -> CronTimes -> UserName -> FilePath -> String -> Property NoInfo
2014-04-01 21:54:20 +00:00
niceJob desc times user cddir command = job desc times user cddir
2014-07-17 19:33:02 +00:00
("nice ionice -c 3 sh -c " ++ shellEscape command)
2014-04-01 21:54:20 +00:00
-- | Installs a cron job to run propellor.
runPropellor :: CronTimes -> Property NoInfo
runPropellor times = niceJob "propellor" times "root" localdir "./propellor"