summaryrefslogtreecommitdiff
path: root/persistent_crontab_minutes.rb
blob: cd800942830c9ac221f189964fa22c36a4d5c557 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
#
# persistent_crontab_minutes.rb
#

module Puppet::Parser::Functions
  newfunction(:persistent_crontab_minutes, :type => :rvalue, :doc => <<-EOS
    EOS
  ) do |arguments|

    raise(Puppet::ParseError, "Wrong number of arguments " +
      "given (#{arguments.size} for 2)") if arguments.size < 2

    require 'md5'

    value = 0

    job  = arguments[0]
    host = arguments[1]

    environment = Puppet[:environment]

    # We select first directory that exists.  This might not be the best idea ...
    modules = Puppet[:modulepath].split(':').select { |i| File.exists?(i) }.first

    raise(Puppet::ParseError, "Unable to determine the storage " +
      "directory for Puppet modules") unless modules

    # Prepare the file where we store current value ...
    file = "/puppet/state/crontab/#{host}-#{job}.minutes"
    file = File.join(modules, file)

    # Get the directory portion from the file name ...
    directory = File.dirname(file)

    FileUtils.mkdir_p(directory) unless File.directory?(directory)

    if FileTest.exists?(file)
      File.open(file, 'r') { |f| value = f.read.to_i }

      raise(Puppet::ParseError, "The value for minutes in the file `%s' " +
        "is out of the range from 0 to 59 inclusive") unless value < 60
    else
      #
      # Pick a random number based on the job and host name.  This will yield
      # the same value for exactly the same combination of the job and host name.
      #
      value = MD5.new(job_name + host).to_s.hex % 60

      # Minutes are from 0 to 59 inclusive ...
      value = value < 60 ? value : 59

      File.open(file, 'w') { |f| f.write(value) }
    end

    # Tell Puppet to keep an eye on this file ...
    parser = Puppet::Parser::Parser.new(environment)
    parser.watch_file(file) if File.exists?(file)

    return value
  end
end

# vim: set ts=2 sw=2 et :