Cron module upgrades
- added cron_file attribute: if specified, the file with appropriate job is created in /etc/cron.d directory. Also, you can store multiple jobs in one file. state='absent' attribute is handled in the following way in this case: if after the deletion of the job from the file specified by cron_file variable the file is empty, the file is deleted, otherwise not. - fixed the behaviour, when the backupfile is saved forever in /tmp folder, even if the backup= atribute is not set (os.unlink() is called if backup is not True). - added some comments to the unobvious places
This commit is contained in:
parent
50495680d5
commit
171a01deac
2 changed files with 132 additions and 24 deletions
114
library/cron
114
library/cron
|
@ -2,6 +2,7 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
#
|
#
|
||||||
# (c) 2012, Dane Summers <dsummers@pinedesk.biz>
|
# (c) 2012, Dane Summers <dsummers@pinedesk.biz>
|
||||||
|
# (c) 2013, Mike Grozak <mike.grozak@gmail.com>
|
||||||
#
|
#
|
||||||
# This file is part of Ansible
|
# This file is part of Ansible
|
||||||
#
|
#
|
||||||
|
@ -58,9 +59,18 @@ options:
|
||||||
state:
|
state:
|
||||||
description:
|
description:
|
||||||
- Whether to ensure the job is present or absent.
|
- Whether to ensure the job is present or absent.
|
||||||
|
- If the cron_file setting is specified and the state is 'absent', if after the deletion
|
||||||
|
of the job the file is empty, the file is deleted
|
||||||
required: false
|
required: false
|
||||||
default: present
|
default: present
|
||||||
aliases: []
|
aliases: []
|
||||||
|
cron_file:
|
||||||
|
descrition:
|
||||||
|
- The file with appropriate job is created in /etc/cron.d directory. Also, you can store
|
||||||
|
multiple jobs in one file.
|
||||||
|
required: fasle
|
||||||
|
default:
|
||||||
|
aliases: []
|
||||||
backup:
|
backup:
|
||||||
description:
|
description:
|
||||||
- If set, then create a backup of the crontab before it is modified.
|
- If set, then create a backup of the crontab before it is modified.
|
||||||
|
@ -114,20 +124,30 @@ examples:
|
||||||
description: 'Ensure an old job is no longer present. Removes any job that is preceded by "#Ansible: an old job" in the crontab'
|
description: 'Ensure an old job is no longer present. Removes any job that is preceded by "#Ansible: an old job" in the crontab'
|
||||||
- code: 'cron: name="a job for reboot" reboot=True job="/some/job.sh"'
|
- code: 'cron: name="a job for reboot" reboot=True job="/some/job.sh"'
|
||||||
description: 'Creates an entry like "@reboot /some/job.sh"'
|
description: 'Creates an entry like "@reboot /some/job.sh"'
|
||||||
|
- code: 'cron: name="yum autoupdate" weekday="2" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate" cron_file=ansible_yum-autoupdate
|
||||||
|
|
||||||
requirements: cron
|
requirements: cron
|
||||||
author: Dane Summers
|
author: Dane Summers
|
||||||
|
updates: Mike Grozak
|
||||||
'''
|
'''
|
||||||
|
|
||||||
import re
|
import re
|
||||||
import tempfile
|
import tempfile
|
||||||
|
|
||||||
def get_jobs_file(module, user, tmpfile):
|
def get_jobs_file(module, user, tmpfile, cron_file):
|
||||||
cmd = "crontab -l %s > %s" % (user,tmpfile)
|
if cron_file:
|
||||||
|
cmd = "cp -fp /etc/cron.d/%s %s" % (cron_file, tmpfile)
|
||||||
|
else:
|
||||||
|
cmd = "crontab -l %s > %s" % (user,tmpfile)
|
||||||
|
|
||||||
return module.run_command(cmd)
|
return module.run_command(cmd)
|
||||||
|
|
||||||
def install_jobs(module, user, tmpfile):
|
def install_jobs(module, user, tmpfile, cron_file):
|
||||||
cmd = "crontab %s %s" % (user,tmpfile)
|
if cron_file:
|
||||||
|
cmd = "ln -f %s /etc/cron.d/%s" % (tmpfile, cron_file)
|
||||||
|
else:
|
||||||
|
cmd = "crontab %s %s" % (user, tmpfile)
|
||||||
|
|
||||||
return module.run_command(cmd)
|
return module.run_command(cmd)
|
||||||
|
|
||||||
def get_jobs(tmpfile):
|
def get_jobs(tmpfile):
|
||||||
|
@ -166,6 +186,10 @@ def remove_job(name,tmpfile):
|
||||||
def do_remove_job(lines,comment,job):
|
def do_remove_job(lines,comment,job):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
def remove_job_file(cron_file):
|
||||||
|
fname = "/etc/cron.d/%s" % (cron_file)
|
||||||
|
os.unlink(fname)
|
||||||
|
|
||||||
def _update_job(name,job,tmpfile,addlinesfunction):
|
def _update_job(name,job,tmpfile,addlinesfunction):
|
||||||
ansiblename="#Ansible: %s" % (name)
|
ansiblename="#Ansible: %s" % (name)
|
||||||
f = open(tmpfile)
|
f = open(tmpfile)
|
||||||
|
@ -186,7 +210,25 @@ def _update_job(name,job,tmpfile,addlinesfunction):
|
||||||
f.write(l)
|
f.write(l)
|
||||||
f.write('\n')
|
f.write('\n')
|
||||||
f.close()
|
f.close()
|
||||||
return (0,"","") # TODO add some more error testing
|
|
||||||
|
if len(newlines) == 0:
|
||||||
|
return (0,"","",True)
|
||||||
|
else:
|
||||||
|
return (0,"","",False) # TODO add some more error testing
|
||||||
|
|
||||||
|
def get_cron_job(minute,hour,day,month,weekday,job,user,cron_file,reboot):
|
||||||
|
if reboot:
|
||||||
|
if cron_file:
|
||||||
|
return "@reboot %s %s" % (user, job)
|
||||||
|
else:
|
||||||
|
return "@reboot %s" % (job)
|
||||||
|
else:
|
||||||
|
if cron_file:
|
||||||
|
return "%s %s %s %s %s %s %s" % (minute,hour,day,month,weekday,user,job)
|
||||||
|
else:
|
||||||
|
return "%s %s %s %s %s %s" % (minute,hour,day,month,weekday,job)
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
# The following example playbooks:
|
# The following example playbooks:
|
||||||
|
@ -217,6 +259,7 @@ def main():
|
||||||
name=dict(required=True),
|
name=dict(required=True),
|
||||||
user=dict(required=False),
|
user=dict(required=False),
|
||||||
job=dict(required=False),
|
job=dict(required=False),
|
||||||
|
cron_file=dict(required=False),
|
||||||
state=dict(default='present', choices=['present', 'absent']),
|
state=dict(default='present', choices=['present', 'absent']),
|
||||||
backup=dict(default=False, choices=BOOLEANS),
|
backup=dict(default=False, choices=BOOLEANS),
|
||||||
minute=dict(default='*'),
|
minute=dict(default='*'),
|
||||||
|
@ -232,40 +275,45 @@ def main():
|
||||||
name = module.params['name']
|
name = module.params['name']
|
||||||
user = module.params['user']
|
user = module.params['user']
|
||||||
job = module.params['job']
|
job = module.params['job']
|
||||||
|
cron_file = module.params['cron_file']
|
||||||
minute = module.params['minute']
|
minute = module.params['minute']
|
||||||
hour = module.params['hour']
|
hour = module.params['hour']
|
||||||
day = module.params['day']
|
day = module.params['day']
|
||||||
month = module.params['month']
|
month = module.params['month']
|
||||||
weekday = module.params['weekday']
|
weekday = module.params['weekday']
|
||||||
reboot = module.boolean(module.params.get('reboot', False))
|
reboot = module.boolean(module.params.get('reboot', False))
|
||||||
|
state = module.params['state']
|
||||||
do_install = module.params['state'] == 'present'
|
do_install = module.params['state'] == 'present'
|
||||||
changed = False
|
changed = False
|
||||||
|
|
||||||
if reboot and (True in [(x != '*') for x in [minute, hour, day, month, weekday]]):
|
if reboot and (True in [(x != '*') for x in [minute, hour, day, month, weekday]]):
|
||||||
module.fail_json(msg="You must specify either reboot=True or any of minute, hour, day, month, weekday")
|
module.fail_json(msg="You must specify either reboot=True or any of minute, hour, day, month, weekday")
|
||||||
|
|
||||||
if reboot:
|
if cron_file:
|
||||||
job = "@reboot %s" % (job)
|
if not user:
|
||||||
|
module.fail_json(msg="To use file=... parameter you must specify user=... as well")
|
||||||
else:
|
else:
|
||||||
job = "%s %s %s %s %s %s" % (minute,hour,day,month,weekday,job)
|
if not user:
|
||||||
|
user = ""
|
||||||
if not user:
|
else:
|
||||||
user = ""
|
user = "-u %s" % (user)
|
||||||
else:
|
|
||||||
user = "-u %s" % (user)
|
|
||||||
|
|
||||||
rc, out, err, status = (0, None, None, None)
|
|
||||||
|
|
||||||
|
job = get_cron_job(minute,hour,day,month,weekday,job,user,cron_file,reboot)
|
||||||
|
rc, out, err, rm, status = (0, None, None, None, None)
|
||||||
if job is None and do_install:
|
if job is None and do_install:
|
||||||
module.fail_json(msg="You must specify 'job' to install a new cron job")
|
module.fail_json(msg="You must specify 'job' to install a new cron job")
|
||||||
|
|
||||||
tmpfile = tempfile.NamedTemporaryFile()
|
tmpfile = tempfile.NamedTemporaryFile()
|
||||||
(rc, out, err) = get_jobs_file(module,user,tmpfile.name)
|
(rc, out, err) = get_jobs_file(module,user,tmpfile.name, cron_file)
|
||||||
|
|
||||||
if rc != 0 and rc != 1: # 1 can mean that there are no jobs.
|
if rc != 0 and rc != 1: # 1 can mean that there are no jobs.
|
||||||
module.fail_json(msg=err)
|
module.fail_json(msg=err)
|
||||||
|
|
||||||
(handle,backupfile) = tempfile.mkstemp(prefix='crontab')
|
(handle,backupfile) = tempfile.mkstemp(prefix='crontab')
|
||||||
(rc, out, err) = get_jobs_file(module,user,backupfile)
|
(rc, out, err) = get_jobs_file(module,user,backupfile, cron_file)
|
||||||
if rc != 0 and rc != 1:
|
if rc != 0 and rc != 1:
|
||||||
module.fail_json(msg=err)
|
module.fail_json(msg=err)
|
||||||
|
|
||||||
old_job = find_job(name,backupfile)
|
old_job = find_job(name,backupfile)
|
||||||
if do_install:
|
if do_install:
|
||||||
if len(old_job) == 0:
|
if len(old_job) == 0:
|
||||||
|
@ -276,27 +324,45 @@ def main():
|
||||||
changed = True
|
changed = True
|
||||||
else:
|
else:
|
||||||
if len(old_job) > 0:
|
if len(old_job) > 0:
|
||||||
(rc, out, err) = remove_job(name,tmpfile.name)
|
# if rm is true after the next line, file will be deleted afterwards
|
||||||
|
(rc, out, err, rm) = remove_job(name,tmpfile.name)
|
||||||
changed = True
|
changed = True
|
||||||
|
else:
|
||||||
|
# there is no old_jobs for deletion - we should leave everything
|
||||||
|
# as is. If the file is empty, it will be removed later
|
||||||
|
tmpfile.close()
|
||||||
|
# the file created by mks should be deleted explicitly
|
||||||
|
os.unlink(backupfile)
|
||||||
|
module.exit_json(changed=changed,cron_file=cron_file,state=state)
|
||||||
|
|
||||||
if (rc != 0):
|
if (rc != 0):
|
||||||
module.fail_json(msg=err)
|
module.fail_json(msg=err)
|
||||||
if changed:
|
|
||||||
if backup:
|
|
||||||
module.backup_local(backupfile)
|
|
||||||
(rc, out, err) = install_jobs(module,user,tmpfile.name)
|
|
||||||
if (rc != 0):
|
|
||||||
module.fail_json(msg=err)
|
|
||||||
|
|
||||||
|
if changed:
|
||||||
|
# If the file is empty - remove it
|
||||||
|
if rm:
|
||||||
|
remove_job_file(cron_file)
|
||||||
|
else:
|
||||||
|
if backup:
|
||||||
|
module.backup_local(backupfile)
|
||||||
|
(rc, out, err) = install_jobs(module,user,tmpfile.name, cron_file)
|
||||||
|
if (rc != 0):
|
||||||
|
module.fail_json(msg=err)
|
||||||
|
|
||||||
|
# get the list of jobs in file
|
||||||
jobnames = []
|
jobnames = []
|
||||||
for j in get_jobs(tmpfile.name):
|
for j in get_jobs(tmpfile.name):
|
||||||
jobnames.append(j[0])
|
jobnames.append(j[0])
|
||||||
tmpfile.close()
|
tmpfile.close()
|
||||||
|
|
||||||
if not backup:
|
if not backup:
|
||||||
|
os.unlink(backupfile)
|
||||||
module.exit_json(changed=changed,jobs=jobnames)
|
module.exit_json(changed=changed,jobs=jobnames)
|
||||||
else:
|
else:
|
||||||
module.exit_json(changed=changed,jobs=jobnames,backup=backupfile)
|
module.exit_json(changed=changed,jobs=jobnames,backup=backupfile)
|
||||||
|
|
||||||
# include magic from lib/ansible/module_common.py
|
# include magic from lib/ansible/module_common.py
|
||||||
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
|
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
|
||||||
|
|
||||||
main()
|
main()
|
||||||
|
|
||||||
|
|
42
test/cron_test.yml
Normal file
42
test/cron_test.yml
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
-
|
||||||
|
hosts: all
|
||||||
|
gather_facts: no
|
||||||
|
user: root
|
||||||
|
vars:
|
||||||
|
color: brown
|
||||||
|
tasks:
|
||||||
|
- name: test 1
|
||||||
|
cron: name="execution test 1" weekday="2,3" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate" cron_file=yum-autoupdate state=absent
|
||||||
|
tags:
|
||||||
|
- cron
|
||||||
|
|
||||||
|
- name: test 1-1
|
||||||
|
cron: name="execution test 1" weekday="2,3" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate" cron_file=yum-autoupdate state=absent
|
||||||
|
tags:
|
||||||
|
- cron
|
||||||
|
|
||||||
|
- name: test 2-1
|
||||||
|
cron: name="execution test 2" weekday="2,3" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate" state=absent
|
||||||
|
tags:
|
||||||
|
- cron
|
||||||
|
|
||||||
|
- name: test 2-2
|
||||||
|
cron: name="execution test 2" weekday="2,3" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate" state=absent
|
||||||
|
tags:
|
||||||
|
- cron
|
||||||
|
|
||||||
|
- name: test 2-3
|
||||||
|
cron: name="execution test 2" weekday="2,3" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate"
|
||||||
|
tags:
|
||||||
|
- cron
|
||||||
|
|
||||||
|
- name: test 3-1
|
||||||
|
cron: name="execution test 3" weekday="2,3" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate" cron_file=yum-autoupdate state=absent
|
||||||
|
tags:
|
||||||
|
- cron
|
||||||
|
|
||||||
|
- name: test 3-2
|
||||||
|
cron: name="execution test 3" weekday="2,3" minute=0 hour=12 user="root" job="YUMINTERACTIVE=0 /usr/sbin/yum-autoupdate" cron_file=yum-autoupdate
|
||||||
|
tags:
|
||||||
|
- cron
|
||||||
|
|
Loading…
Reference in a new issue