-
Notifications
You must be signed in to change notification settings - Fork 3
/
slack_bot.py
279 lines (214 loc) · 8.31 KB
/
slack_bot.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
import os
import signal
import sys
import time
import threading
import datetime as dt
from apscheduler.schedulers.background import BackgroundScheduler
from slack_bolt.adapter.socket_mode import SocketModeHandler
from slack_sdk import WebClient
from slack_bolt import App, Ack, Respond
from core.agent import agent_executor, agent_llm
from core.slack import Conversation, save_session, restore_session
from langchain.agents.openai_functions_agent.agent_token_buffer_memory import (
AgentTokenBufferMemory,
)
from http.server import BaseHTTPRequestHandler, HTTPServer
from multiprocessing import Process
import random
import re
# --
# Event API & Web API
app = App(token=os.environ['SLACK_BOT_TOKEN'])
client = WebClient(os.environ['SLACK_BOT_TOKEN'])
socket = SocketModeHandler(app, os.environ['SLACK_APP_TOKEN'])
scheduler = BackgroundScheduler()
active_conversations = []
conversation_lock = threading.RLock()
def find_conversation(channel, thread_ts):
match = None
with conversation_lock:
for ref in active_conversations:
if(ref["channel"]==channel and ref["thread"]==str(thread_ts)):
match = ref["conversation"]
break
return match
def retire_inactive_conversation():
with conversation_lock:
# dump state
#print("Total conversations: "+str(len(active_conversations)))
#[print(ref) for ref in active_conversations]
for ref in active_conversations:
conversation = ref["conversation"]
if(conversation.is_expired()):
if(conversation.current_state!='answered'):
handle_retirement(conversation)
active_conversations.remove(ref)
def handle_retirement(conversation):
# persist session
save_session(conversation)
# noity client
conversation.retire()
# This gets activated when the bot is tagged in a channel
# it will start a new thread that will hold the conversation
@app.event("app_mention")
def handle_message_events(body, logger):
thread = body["event"].get("thread_ts")
if thread is not None:
# handle direct mention in thread
message_sender = body["event"].get("user")
channel = body["event"].get("channel")
restart_pattern = re.search('@\w+> restore', body["event"]["text"])
if(restart_pattern):
# any active conversation?
conversation = find_conversation(
channel=channel,
thread_ts=thread
)
if(conversation is None):
conversation = restore_session(
client=client,
channel=channel,
thread_ts=thread,
owner=message_sender
)
if(conversation is not None):
with conversation_lock:
active_conversations.append({
"channel": channel,
"thread": thread,
"conversation": conversation
})
conversation.listening()
else:
response = "Known commands are: `restore`"
blocks= [{
"type": "context",
"elements": [
{
"type": "mrkdwn",
"text": response
}
]
}]
client.chat_postMessage(
channel=channel,
thread_ts=thread,
blocks=blocks,
text=response
)
else:
# direct mention outside thread starts a new conversation within a thread
message_sender = body["event"]["user"]
channel = body["event"]["channel"]
thread = body["event"]["event_ts"]
# register new conversation
conversation = Conversation(
owner=message_sender,
slack_client=client,
channel=channel,
thread_ts=thread,
memory=AgentTokenBufferMemory(llm=agent_llm)
)
with conversation_lock:
active_conversations.append({
"channel": channel,
"thread": str(thread),
"conversation": conversation
})
conversation.listening()
# the main loop for interaction with the bot
# the bot ignores messages outsides threads
@app.event("message")
def handle_message_events(event):
try:
mention_pattern = re.search('@\w+', event.get('text'))
except:
print("Failed to parse message body: ", str(event))
return
if(mention_pattern):
# this case is handled by @app.event("app_mention")
# we use direct mentions to instruct the bot within thread (i.e. to restore sessions)
return
if event.get("thread_ts"):
# within threads we listen to messages
print("handle message within thread")
message_sender = event.get("user")
response_channel = event.get("channel")
response_thread = event.get("thread_ts")
# any active conversation?
conversation = find_conversation(
channel=response_channel,
thread_ts=response_thread
)
if(conversation is not None):
if(conversation.owner != message_sender):
print("Ignore message from sender who not owner")
return
elif(conversation.current_state == conversation.answered):
text = event.get('text')
conversation.inquire(text)
else:
business = [
"Give me a minute ...",
"Working on it ...",
"Just a sec ...",
"Just a moment ...",
"Hang on ..."
]
client.chat_postMessage(
channel=response_channel,
thread_ts=response_thread,
text=random.choice(business)
)
else:
print("Cannot find conversation")
else:
# outside thread we ignore messages
pass
class HealthcheckHandler(BaseHTTPRequestHandler):
def do_GET(self):
# Sending a 200 OK response
self.send_response(200)
# Setting the header
self.send_header('Content-type', 'text/plain')
self.end_headers()
# Sending the response message
message = "Hello, World! This is a simple HTTP server."
# Writing the message body
self.wfile.write(message.encode())
def run_healthcheck():
server_address = ("0.0.0.0", 8080)
httpd = HTTPServer(server_address, HealthcheckHandler)
print(f"Healthcheck running on {server_address}")
# Running the server
try:
httpd.serve_forever()
except KeyboardInterrupt:
pass
finally:
httpd.server_close()
healthcheck_process = Process(target=run_healthcheck, daemon=True)
# make sure conversation are retried when bot stops
def graceful_shutdown(signum, frame):
print("Shutdown bot ...")
# retire all active conversations
[handle_retirement(ref["conversation"]) for ref in active_conversations]
time.sleep(3)
# stop the scheduler
scheduler.shutdown(wait=True)
# stop the listener
socket.disconnect()
socket.close()
healthcheck_process.kill()
sys.exit(0)
if __name__ == "__main__":
signal.signal(signal.SIGINT, graceful_shutdown)
signal.signal(signal.SIGTERM, graceful_shutdown)
# scheduler reaper
scheduler.add_job(retire_inactive_conversation, 'interval', seconds=5, id='retirement_job')
scheduler.start()
# start healthecheck listener
healthcheck_process.start()
# start listening for messages
socket.start()