-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathrobot.py
123 lines (102 loc) · 3.6 KB
/
robot.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
import os
import sys
sys.path.append(os.path.abspath('.'))
import time
import traceback
from importlib import import_module
from slackclient import SlackClient
from concurrent.futures import ThreadPoolExecutor, TimeoutError
from brain import RedisBrain
from loggers import logger, level
from settings import APPS, CMD_PREFIX, CMD_LENGTH, SLACK_TOKEN, MAX_WORKERS
class Robot(object):
def __init__(self):
self.client = SlackClient(SLACK_TOKEN)
self.brain = RedisBrain()
self.apps, self.docs = self.load_apps()
self.logger = logger
def load_apps(self):
docs = ['='*14, 'Usage', '='*14]
apps = {}
for name in APPS:
app = import_module('apps.%s' % name)
docs.append('{0}{1}: {2}'.format(
CMD_PREFIX, ', '.join(app.run.commands), app.run.__doc__
))
for command in app.run.commands:
apps[command] = app
return apps, docs
def handle_message(self, message):
channel, user, text = message
command, payloads = self.extract_command(text)
if not command:
return
app = self.apps.get(command, None)
if not app:
return
try:
app.run(self, channel, user, payloads)
except:
self.logger.error(traceback.format_exc())
def extract_messages(self, events):
messages = []
for event in events:
channel = event.get('channel', '')
user = event.get('user', '')
text = event.get('text', '')
if channel and user and text:
messages.append((channel, user, text))
return messages
def extract_command(self, text):
if CMD_PREFIX and CMD_PREFIX != text[0]:
return (None, None)
tokens = text.split(' ', 1)
if 1 < len(tokens):
return tokens[0][CMD_LENGTH:], tokens[1]
else:
return (text[CMD_LENGTH:], '')
def rtm_connect(self):
while not self.client.rtm_connect(with_team_state=False):
self.logger.info('RTM Connecting...')
time.sleep(1)
self.logger.info('RTM Connected.')
def read_message(self):
try:
return self.client.rtm_read()
except KeyboardInterrupt as e:
raise e
except Exception as e:
self.logger.error(traceback.format_exc())
self.rtm_connect()
def run(self):
self.brain.connect()
self.rtm_connect()
if not self.client.server.connected:
raise RuntimeError(
'Can not connect to slack client. Check your settings.'
)
while True:
events = self.read_message()
if events:
messages = self.extract_messages(events)
if messages:
with ThreadPoolExecutor(max_workers=MAX_WORKERS) as executor:
try:
executor.map(self.handle_message, messages)
except TimeoutError:
self.logger.error(traceback.format_exc())
else:
time.sleep(0.3)
def disconnect(self):
if self.client and self.client.server and self.client.server.websocket:
self.client.server.websocket.close()
self.logger.info('RTM disconnected.')
if '__main__' == __name__:
robot = Robot()
try:
robot.run()
except KeyboardInterrupt as e:
robot.logger.info('Honey Shutdown By User.')
finally:
robot.disconnect()
robot.logger.info('Honey Shutdown.')