-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathPySMS.py
450 lines (383 loc) · 16.4 KB
/
PySMS.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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
"""Really old Python 2 code."""
import smtplib
import imaplib
import email
import datetime
import time
import random
import inspect
import logging
import threading
class PySMSException:
def __init__(self, value):
self.value = value
def __str__(self):
return repr(self.value)
class PySMS:
def __init__(self, address, password, smtp_server, smtp_port, imap_server=None, ssl=False, window=5, delimiter=":",
identifier_length=4, max_tries=5, text_wait_time=5, check_wait_time=15, check_unit=60, debug=False):
self.carriers = {
# US
"alltel": "@mms.alltelwireless.com",
"att": "@mms.att.net",
"boost": "@myboostmobile.com",
"cricket": "@mms.cricketwireless.net",
"p_fi": "msg.fi.google.com",
"sprint": "@pm.sprint.com",
"tmobile": "@tmomail.net",
"us_cellular": "@mms.uscc.net",
"verizon": "@vzwpix.com",
"virgin": "@vmpix.com",
# Canada
"bell": "@txt.bell.ca",
"chatr": "@fido.ca",
"fido": "@fido.ca",
"freedom": "@txt.freedommobile.ca",
"koodo": "@msg.koodomobile.com",
"public_mobile": "@msg.telus.com",
"telus": "@msg.telus.com",
"rogers": "@pcs.rogers.com",
"sasktel": "@sms.sasktel.com",
"speakout": "@pcs.rogers.com",
"virgin_ca": "@vmobile.ca"
}
# Smtp
self.smtp = None
self.validate(address, password)
self.address = address.encode("utf-8")
self.password = password.encode("utf-8")
self.smtp_server = smtp_server
self.smtp_port = smtp_port
self.ssl = ssl
# Imap
self.imap = None
self.imap_server = imap_server
self.imap_mailbox = "INBOX"
self.delimiter = delimiter
self.identifier_length = identifier_length
# Parameters
self.window = window
self.max_tries = max_tries
self.text_wait_time = text_wait_time
self.check_wait_time = check_wait_time
self.check_unit = check_unit
# Daemon
self.auto_check_enabled = False
self.daemon = None
self.lock = threading.Lock()
# Format: key => [time, address, lambda]
self.hook_dict = {}
# Format: number => address
self.addresses = {}
# Format: address => [uids]
self.ignore_dict = {}
self.ignore_set = set()
self.tracked = set()
# Logger
logging.basicConfig()
self.logger = logging.getLogger(__name__)
self.logger.setLevel(logging.INFO)
if debug:
self.logger.setLevel(logging.DEBUG)
self.init_server()
# Getter/Setter Functions
# Note: Pythonically not necessary but allows for more readable code
def get_smtp_server(self):
return self.smtp
def get_imap_server(self):
return self.imap
def get_imap_mailbox(self):
return self.imap_mailbox
def set_imap_mailbox(self, mailbox):
self.imap_mailbox = mailbox
def get_hook_dict(self):
return self.hook_dict
def get_hook_address(self, key):
return self.hook_dict[key][1]
def get_delimiter(self):
return self.delimiter
def set_delimiter(self, delimiter):
self.delimiter = delimiter
def get_window(self):
return self.window
def set_window(self, window):
self.window = window
def get_max_tries(self):
return self.max_tries
def set_max_tries(self, max_tries):
self.max_tries = max_tries
def get_text_wait_time(self):
return self.text_wait_time
def set_text_wait_time(self, wait_time):
self.text_wait_time = wait_time
def get_identifier_length(self):
return self.identifier_length
def set_identifier_length(self, identifier_length):
self.identifier_length = identifier_length
def get_auto_check_enabled(self):
return self.auto_check_enabled
def get_check_wait_time(self):
return self.check_wait_time
def set_check_wait_time(self, wait_time):
self.check_wait_time = wait_time
def get_check_unit(self):
return self.check_unit
def set_check_unit(self, unit):
self.check_unit = unit
# Utility Functions
def validate(self, address, password):
try:
assert isinstance(address, basestring)
assert isinstance(password, basestring)
except AssertionError:
raise PySMSException("Please make sure address and password are strings.")
def check_callback_requirements(self, callback):
if self.imap:
if callable(callback):
if len(inspect.getargspec(callback).args) == 2:
return
else:
raise PySMSException("Callback function does not have the correct number of arguments.")
else:
raise PySMSException("Callback function is not callable.")
else:
raise PySMSException("IMAP settings not configured or valid.")
def get_current_time(self):
return time.time()
# MMS/Internal Functions
def init_server(self):
self.logger.info("Initializing SMTP/IMAP servers.")
# PySMS at minimum uses smtp server
try:
if self.ssl:
self.smtp = smtplib.SMTP_SSL(self.smtp_server, self.smtp_port)
else:
self.smtp = smtplib.SMTP(self.smtp_server, self.smtp_port)
self.smtp.starttls()
self.smtp.login(self.address, self.password)
except smtplib.SMTPException:
raise PySMSException("Unable to start smtp server, please check credentials.")
# If responding functionality is enabled
if self.imap_server:
try:
if self.ssl:
self.imap = imaplib.IMAP4_SSL(self.imap_server)
else:
self.imap = imaplib.IMAP4(self.imap_server)
r, data = self.imap.login(self.address, self.password)
if r == "OK":
r, data = self.imap.select(self.imap_mailbox)
if r != "OK":
raise PySMSException("Unable to select mailbox: {0}".format(self.imap_mailbox))
else:
raise PySMSException("Unable to login to IMAP server with given credentials.")
except imaplib.IMAP4.error:
raise PySMSException("Unable to start IMAP server, please check address and SSL/TLS settings.")
def add_number(self, number, carrier):
if carrier in self.carriers:
address = number + self.carriers[carrier]
self.addresses[number] = address
self.logger.info("Number: {0} added.".format(number))
else:
raise PySMSException("Please enter a valid carrier.")
def del_number(self, number):
if number in self.addresses:
del self.addresses[number]
self.logger.info("Number: {0} deleted.".format(number))
self.logger.error("Number: {0} not found in list of addresses, ignoring.".format(number))
def add_hook(self, identifier, address, callback_function):
self.hook_dict[identifier] = [self.get_current_time(), address, callback_function]
if address not in self.tracked:
self.tracked.add(address)
def remove_hook(self, key):
if key in self.hook_dict:
self.tracked.remove(self.hook_dict[key][1])
del self.hook_dict[key]
def add_ignore(self, mail, uid):
if mail["From"] in self.tracked:
ignore_list = [uid]
if mail["From"] in self.ignore_dict:
ignore_list += self.ignore_dict[mail["From"]]
self.ignore_dict[mail["From"]] = ignore_list
self.ignore_set.add(uid)
def del_ignore(self, address):
for uid in self.ignore_dict[address]:
self.ignore_set.remove(uid)
del self.ignore_dict[address]
def generate_identifier(self):
def generate():
ret = ""
for num in random.sample(range(0, 10), self.identifier_length):
ret += str(num)
return ret
identifier = generate()
while identifier in self.hook_dict:
identifier = generate()
return identifier
def generate_rfc_query(self):
ret = ""
for _ in range(len(self.tracked) - 1):
ret += "OR "
for track in self.tracked:
ret += "FROM " + track + " "
return ret[:-1]
def check_tracked(self):
if self.tracked:
date = (datetime.date.today() - datetime.timedelta(1)).strftime("%d-%b-%Y")
r, uids = self.imap.uid("search", None,
"(SENTSINCE {date} {query})".format(date=date, query=self.generate_rfc_query()))
if r == "OK":
email_data = self.get_emails(uids)
if email_data:
# Pass a static current time because emails might take time to execute
current_time = self.get_current_time()
for e_d in email_data:
self.check_email(e_d[0], e_d[1], current_time)
else:
self.logger.info("No new emails to check (either ignored or no new mail).")
else:
self.logger.info("No addresses being tracked")
# Clean at end to avoid race condition
self.clean_hook_dict()
def get_email(self, uid):
r, email_data = self.imap.uid('fetch', uid, '(RFC822)')
if r == "OK":
return email_data
return None
def get_emails(self, uids):
ret = []
for uid in uids[0].split():
if uid not in self.ignore_set:
ret.append((uid, self.get_email(uid)))
return ret
# TODO: use min heap to speed up runtime if a lot of keys
def clean_hook_dict(self):
for key in self.hook_dict:
if self.get_current_time() - self.hook_dict[key][0] > self.window * 60:
self.del_ignore(self.hook_dict[key][1])
self.remove_hook(key)
def check_email(self, uid, email_data, current_time):
mail = email.message_from_string(email_data[0][1])
mail_time = email.utils.mktime_tz(email.utils.parsedate_tz(mail["Date"]))
if current_time - mail_time < self.window * 60:
if mail.get_content_maintype() == "multipart":
for part in mail.walk():
if part.get_content_maintype() != 'multipart' and part.get('Content-Disposition') is not None:
response = part.get_payload(decode=True)
response = response.split(self.delimiter)
if len(response) == 2:
key = response[0].strip()
value = response[1].strip()
# If hook is not valid then also ignore
if not self.execute_hook(key, value):
self.logger.info("Adding failed hook with uid: {uid} to ignore.".format(uid=uid))
self.add_ignore(mail, uid)
return
# Clean_hook_dict will take care of this later
self.logger.info("Email with uid: {uid} is expired, ignoring in next check".format(uid=uid))
# Add uid to ignore if uid is expired so it knows not to request it next cycle
self.add_ignore(mail, uid)
def enable_auto_check(self):
self.logger.info("Auto checking enabled.")
self.lock.acquire()
self.auto_check_enabled = True
if not self.daemon:
self.logger.info("Starting daemon thread.")
self.daemon = threading.Thread(target=self.auto_check_daemon)
self.daemon.setDaemon(True)
self.daemon.start()
self.lock.release()
def disable_auto_check(self):
self.lock.acquire()
self.logger.info("Auto checking disabled.")
self.auto_check_enabled = False
self.lock.release()
def change_wait_time(self, wait_time):
self.check_wait_time = wait_time
def auto_check_daemon(self):
self.logger.info("Auto check daemon function called.")
daemon_wait_time = self.check_wait_time
self.logger.info("Initial auto check time: {0}".format(str(daemon_wait_time)))
counter = 0
while True:
# hold lock to check for both the flag and wait time
self.lock.acquire()
# updating daemon checking properties
if daemon_wait_time != self.check_wait_time:
daemon_wait_time = self.check_wait_time
self.logger.info("Auto check time changed to: {0}".format(str(daemon_wait_time)))
counter = 0
elif counter >= daemon_wait_time:
counter = 0
# perform update functions
if counter < daemon_wait_time and self.auto_check_enabled:
self.logger.debug("Auto checking tracked emails with wait interval: {0} and counter is: {1}".format(
str(daemon_wait_time), str(counter)))
self.check_tracked()
# release lock once time has been changed
self.lock.release()
# sleep to allow for interval in interval change
time.sleep(self.check_unit)
counter += 1
def execute_hook(self, key, value):
success = True
if key in self.hook_dict:
try:
self.hook_dict[key][2](self.hook_dict[key][1], value)
# General Exception here to catch user defined lambda function
except Exception:
success = False
if success:
self.logger.info("Hook with key: {key} for {address} executed.".format(key=key, address=self.hook_dict[key][1]))
else:
self.logger.info("Hook with key: {key} for {address} was not executed or failed.".format(
key=key, address=self.hook_dict[key][1]))
# Remove from ignore and remove from hook_dict
self.del_ignore(self.hook_dict[key][1])
self.remove_hook(key)
else:
self.logger.info("Hook with key: {key} not valid.".format(key=key))
success = False
return success
def text(self, msg, address=None, callback=False):
ret = []
if address:
addresses = [address]
else:
addresses = self.addresses.values()
tmp_msg = msg
for address in addresses:
success = False
for _ in range(self.max_tries):
try:
# Add call back function if enabled
identifier = None
if callback:
# Validate callback function
self.check_callback_requirements(callback)
identifier = self.generate_identifier()
tmp_msg += "\rReply with identifier {identifier} followed by a \"{delimiter}\"".format(
identifier=identifier, delimiter=self.delimiter)
# Send text message through SMS gateway of destination address
self.smtp.sendmail(self.address, address, tmp_msg)
self.logger.info("Message: {message} sent to: {address} successfully.".format(message=tmp_msg, address=address))
# Only add hook if message was sent successfully
if callback:
self.add_hook(identifier, address, callback)
# Reset msg back to original
tmp_msg = msg
success = True
break
except smtplib.SMTPException:
self.logger.info("Failed to send message, reinitializing server.")
try:
self.init_server()
except PySMSException:
self.logger.info("Server reinitialization failed.")
pass
time.sleep(self.text_wait_time)
pass
if not success:
self.logger.debug("Message: \"{message}\" sent to: {address} unsuccessfully.".format(message=msg, address=address))
ret.append(success)
return ret