-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcheck_imap_receive.py
141 lines (115 loc) · 4.48 KB
/
check_imap_receive.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
#!/usr/bin/env python
__title__ = "check_emails"
__version__ = "1.0.0"
import os
import sys
import signal
import imaplib
import re
import argparse
import ConfigParser
import dateutil.parser
from datetime import datetime
import pytz
OLD_PYTHON = False
try:
from subprocess import Popen, PIPE, STDOUT
except ImportError:
OLD_PYTHON = True
import commands
from optparse import OptionParser
EXIT_OK = 0
EXIT_WARNING = 1
EXIT_CRITICAL = 2
EXIT_UNKNOWN = 3
DEFAULT_PORT = 993
DEFAULT_PROFILECONFIG = '/etc/nagios-plugins/check_email_delivery_credentials.ini'
DEFAULT_WARN = 120
DEFAULT_CRIT = 600
parser = argparse.ArgumentParser(description='This program checks if the most recent mail in an IMAP inbox is not older than a certain time. Use this check as a companion to check_smtp_send.py, which sends test email messages.')
parser.add_argument('-H', dest='host', metavar='host', required=True, help='SMTP host')
parser.add_argument('--port', type=int, default=DEFAULT_PORT, help='SMTP port (default=%i)'%DEFAULT_PORT)
parser.add_argument('--profile', required=True, help='credential profile in config file')
parser.add_argument('--profileconfig', metavar='config.ini', default=DEFAULT_PROFILECONFIG, help='location of the config file (default=%s)'%DEFAULT_PROFILECONFIG)
parser.add_argument('-w', type=int, metavar='warningseconds', dest='warn', default=DEFAULT_WARN, help='warn, if the most recent message is older than this value (default=%s)' % DEFAULT_WARN)
parser.add_argument('-c', type=int, metavar='criticalseconds', dest='crit', default=DEFAULT_CRIT, help='critical, if the most recent message is older than this value (default=%s)' % DEFAULT_CRIT)
try:
args = vars(parser.parse_args())
except SystemExit:
sys.exit(EXIT_UNKNOWN)
host = args['host']
port = args['port']
profile = args['profile']
profileconfig = args['profileconfig']
warn = args['warn']
crit = args['crit']
config = ConfigParser.SafeConfigParser()
config.read(profileconfig)
try:
username = config.get(profile,'username')
password = config.get(profile,'password')
except ConfigParser.NoSectionError:
print('Configuration error: profile %s does not exist' % profile)
sys.exit(EXIT_UNKNOWN)
except ConfigParser.NoOptionError:
print('Configuration error: profile %s does not contain username or password' % profile)
sys.exit(EXIT_UNKNOWN)
try:
imap = imaplib.IMAP4_SSL(host, port)
imap.login(username, password)
imap.select()
typ, data = imap.search(None, 'ALL')
pattern = re.compile(r'\(INTERNALDATE "(.+)"\)')
messages = []
for num in data[0].split():
typ, data = imap.fetch(num, '(INTERNALDATE)')
if typ != 'OK':
# TODO: error handling
sys.exit(EXIT_UNKNOWN)
m = pattern.search(data[0])
d = dateutil.parser.parse(m.group(1))
messages.append((d,num))
sortedmessages = sorted(messages, key=lambda x: x[0])
mostrecent = sortedmessages[-1]
for d,num in sortedmessages[:-1]:
imap.store(num, '+FLAGS', '\\Deleted')
imap.expunge()
imap.close()
imap.logout()
except IMAP4.abort as e:
# IMAP4 server errors cause this exception to be raised. This is a sub-class of IMAP4.error. Note that closing the instance and instantiating a new one will usually allow recovery from this exception.
print e
sys.exit(EXIT_CRITICAL)
except IMAP4.readonly as e:
# This exception is raised when a writable mailbox has its status changed by the server. This is a sub-class of IMAP4.error. Some other client now has write permission, and the mailbox will need to be re-opened to re-obtain write permission.
print e
sys.exit(EXIT_CRITICAL)
except IMAP4.error as e:
# Exception raised on any errors. The reason for the exception is passed to the constructor as a string.
print e
sys.exit(EXIT_CRITICAL)
sec = (datetime.now(pytz.utc) - mostrecent[0]).total_seconds()
def end(status, message, perfdata=""):
"""Exits the plugin with first arg as the return code and the second arg as the message to output."""
if perfdata:
print "%s | %s" % (message, perfdata)
else:
print "%s" % message
if status == OK:
sys.exit(OK)
elif status == WARNING:
sys.exit(WARNING)
elif status == CRITICAL:
sys.exit(CRITICAL)
else:
sys.exit(UNKNOWN)
def print_message(status):
print "%s|mostrecent=%i;%i;%i;0;" % (status, sec, warn, crit)
if sec > crit:
print_message('CRITICAL')
sys.exit(EXIT_CRITICAL)
elif sec > warn:
print_message('WARNING')
sys.exit(EXIT_WARNING)
print_message('OK')
sys.exit(EXIT_OK)