-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcheck-puppet-reports.py
executable file
·122 lines (89 loc) · 2.98 KB
/
check-puppet-reports.py
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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
#!/usr/bin/env python3
from glob import iglob, glob
import os
import os.path
import yaml
from datetime import datetime, timedelta
import subprocess
import socket
max_age = timedelta(days=10)
report_dir = '/var/lib/puppet/reports'
def construct_ruby_object(loader, suffix, node):
return loader.construct_yaml_map(node)
def construct_ruby_sym(loader, node):
return loader.construct_yaml_str(node)
yaml.add_multi_constructor("!ruby/object:", construct_ruby_object)
yaml.add_constructor("!ruby/sym", construct_ruby_sym)
def get_report_info(filename):
"""
return: (time, status)
status can be:
failed, changed, unchanged, ...?
"""
with open(filename) as f:
report = yaml.load(f)
return (report['time'], report['status'])
def check_report(filename):
"""
return: (time, age_ok, status)
"""
(time, status) = get_report_info(filename)
# unlike the file name, time is in local time
return (time, datetime.now() - time <= max_age, status)
def is_private(hostname):
try:
address = list(map(lambda x: int(x), socket.gethostbyname(hostname).split('.')))
except:
return True
return (address[0] == 10) or \
(address[0] == 172 and address[1] >= 16 and address[1] <= 31) or \
(address[0] == 192 and address[1] == 168)
if os.path.exists('/usr/bin/fping'):
ping = [ '/usr/bin/fping', '-c1', '-t250' ]
else:
ping = [ '/usr/bin/ping', '-c1' ]
count_machines = 0
count_bad_age = 0
count_bad_status = 0
trouble = False
dirs = glob(os.path.join(report_dir, '*'))
dirs.sort()
for dir in dirs:
if not os.path.isdir(dir):
print('%s is not a directory' % dir)
continue
count_machines += 1
machine = os.path.basename(dir)
is_reachable = False
if is_private(machine):
reachable = '(private)'
else:
with open('/dev/null', 'w') as devnull:
if subprocess.call(ping + [machine], stdout=devnull, stderr=devnull) == 0:
is_reachable = True
reachable = 'reachable'
else:
reachable = 'not reach.'
if '.' in machine:
machine = machine[0:machine.find('.')]
files = glob(os.path.join(dir, '????????????.yaml'))
if len(files) == 0:
print('BAD %s: no reports found!' % machine)
count_bad_age += 1
continue
files.sort()
latest = files[-1]
(time, age_ok, status) = check_report(latest)
time = str(time)[0:19]
warning = is_reachable and ((not age_ok) or (status == 'failed'))
trouble = trouble or (status == 'failed') or warning
warning = '!' if warning else ' '
print('%s %-3s | %-8s | %-10s | %s | %-9s | age: %s' % (warning, 'OK' if age_ok and (status != 'failed') else 'BAD', machine, reachable, time, status, 'good' if age_ok else 'bad'))
if not age_ok:
count_bad_age += 1
if (status != 'changed') and (status != 'unchanged'):
count_bad_status += 1
if count_bad_age == 0 and count_bad_status == 0:
print('OK (last run: %s, %d computers)' % (datetime.now(), count_machines))
else:
print('%s Age: bad on %d/%d machines. Status: bad on %d/%d machines.' % ('ALARM!' if trouble else 'Warning:', count_bad_age, count_machines, count_bad_status, count_machines))