-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlambda_function.py
278 lines (240 loc) · 10.9 KB
/
lambda_function.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
#!/usr/bin/python3
import config
import datetime
import json
import pytz
import requests
import sys
import dateutil.parser
import os.path
from google.auth.transport.requests import Request
from google.oauth2 import service_account
from google.oauth2.credentials import Credentials
from googleapiclient.discovery import build
SLACK_URL="https://slack.com/api/chat.postMessage"
SCOPES = ['https://www.googleapis.com/auth/calendar.readonly']
# time in minutes
ONE_WEEK = 10080
ONE_MONTH = 43200
# auto check time limit
AUTO_TIME = 15
from io import StringIO
from html.parser import HTMLParser
#https://stackoverflow.com/a/925630
class MLStripper(HTMLParser):
def __init__(self):
super().__init__()
self.reset()
self.strict = False
self.convert_charrefs= True
self.text = StringIO()
def handle_data(self, d):
self.text.write(d)
def get_data(self):
return self.text.getvalue()
def strip_tags(html):
s = MLStripper()
s.feed(html)
return s.get_data()
# relies on service token from google api dashboard; different than user token which we don't use anymore
def get_service():
creds = None
creds = service_account.Credentials.from_service_account_file('credentials.json', scopes=SCOPES)
service = build('calendar', 'v3', credentials=creds)
return service
# grabs all events between now and now+minutes from API, then trims events that have already started
def get_events_in_range(minutes):
service = get_service()
dt_now = datetime.datetime.utcnow()
now = dt_now.isoformat() + 'Z' # 'Z' indicates UTC time
time_limit = ((dt_now + datetime.timedelta(minutes=minutes)).isoformat()) + 'Z'
events_result = service.events().list(calendarId=config.CALENDAR_ID, timeMin=now, timeMax=time_limit,
singleEvents=True, orderBy='startTime').execute()
# we replace now and dt_now here because of timezone weirdness
now = datetime.datetime.now(datetime.timezone.utc)
dt_now = None
events = events_result.get('items', [])
ret_events = []
for event in events:
start = event['start'].get('dateTime', event['start'].get('date'))
start = dateutil.parser.isoparse(start)
delta = start - now
# should have already filtered events between now and max time; just want to make sure we do not report
# events that have already started
if (delta > datetime.timedelta(minutes=0)):
ret_events.append(event)
return ret_events
# For each channel, grabs events within time delta and removes channel tag from event subject, also
# verifies timezone which is hardcoded as CST for SC21
# minutes: how far in the future to look for events
# channels: which channels to search for events for; channels are searched and posted independently
def calendar_request_events(minutes, channels):
ret = {}
events = get_events_in_range(minutes)
print("events:", events)
for key in channels:
ret[key] = []
for event in events:
text_content = event["summary"]
text_desc = strip_tags(event.get("description", ""))
print("event_desc =", event.get("description", ""))
print("text_desc =", text_desc)
fkey = None
for key in channels:
if f"[{key}]".lower() in text_content.lower():
text_content = text_content.replace(f"[{key}]", '').strip()
fkey = key
break
else:
continue
start = event['start'].get('dateTime', event['start'].get('date'))
start = dateutil.parser.isoparse(start)
start = start.astimezone(pytz.timezone("US/Central"))
ret[fkey].append((text_content, start, text_desc))
# return should be a dict (channel key) of lists (events) containing tuple of
# (event name, datetime with CST timezone)
return ret
# Posts responses to requests to slack, depending on message content.
# text: raw text from slack message if here because of @ tag; should contain keyword
# uid: uid who sent raw text
# channelid: channel originating request
def keyword_response(text, uid, channelid):
time_str = ""
time_max = None
channel_keys = []
text = text.lower()
print("processing keyword response, text:", text)
# what kind of message are we responding to?
if "who" in text:
payload = {'channel': config.CHANNEL_TAGS[config.CHANNEL_ID_TAG[channelid]][0], 'text': f"hey <@{uid}>, my name is SVOLidarity, the volunteer who reminds you of all your upcoming events, but you can call me svollie."}
headers = {"Authorization": f"Bearer {config.BOT_OAUTH}", 'Content-Type': 'application/json'}
print("got a who request")
print("headers:", headers)
print("payload:", payload)
r = requests.post(SLACK_URL, json=payload, headers=headers)
return
elif "month" in text:
time_str = "30 days"
time_max = ONE_MONTH
base_key = config.CHANNEL_TAGS[config.CHANNEL_ID_TAG[channelid]][0]
channel_keys = [k for k,v in config.CHANNEL_TAGS.items() if base_key in v ]
elif "week" in text:
time_str = "7 days"
time_max = ONE_WEEK
#channel_keys = [k for k,v in config.CHANNEL_ID_TAG.items() if v == config.CHANNEL_ID_TAG[channelid]]
# i don't love this but in the event a channel has more than one associated tag we pull all tags
base_key = config.CHANNEL_TAGS[config.CHANNEL_ID_TAG[channelid]][0]
channel_keys = [k for k,v in config.CHANNEL_TAGS.items() if base_key in v ]
elif "hey" in text or "hello" in text or "hi " in text:
payload = {'channel': config.CHANNEL_TAGS[config.CHANNEL_ID_TAG[channelid]][0], 'text': f"Hi <@{uid}>!"}
headers = {"Authorization": f"Bearer {config.BOT_OAUTH}", 'Content-Type': 'application/json'}
print("got a hey request")
print("headers:", headers)
print("payload:", payload)
r = requests.post(SLACK_URL, json=payload, headers=headers)
return
elif uid == None:
time_str = f"{AUTO_TIME} minutes"
time_max = AUTO_TIME
channel_keys = config.CHANNEL_TAGS.keys()
else:
payload = {'channel': config.CHANNEL_TAGS[config.CHANNEL_ID_TAG[channelid]][0], 'text': f'<@{uid}> sorry, i don\'t know how to respond :( try one of these keywords: "week", "month", "who", "hello"'}
headers = {"Authorization": f"Bearer {config.BOT_OAUTH}", 'Content-Type': 'application/json'}
print("got a hey request")
print("headers:", headers)
print("payload:", payload)
r = requests.post(SLACK_URL, json=payload, headers=headers)
return
print("minutes, channels:", time_max, ",", channel_keys)
events_dicts = calendar_request_events(time_max, channel_keys)
print("events dict:", events_dicts)
print()
if uid != None:
all_events = events_dicts.values()
events_dicts = {config.CHANNEL_ID_TAG[channelid] : sorted(sum(all_events,[]), key=lambda e: e[1])}
else:
# combine all events that go to another channel for each channel
new_events_dicts = {}
for key in events_dicts.keys():
tag = config.CHANNEL_TAGS[key][0]
if tag is None:
continue
print(f"key: {key}")
channel_vals = [events_dicts[k] for k,v in config.CHANNEL_TAGS.items() if tag in v]
new_events_dicts[key] = sorted(sum(channel_vals,[]), key=lambda e: e[1])
events_dicts = new_events_dicts
print("combined_events_dicts:", events_dicts)
print()
# for all the channels, events returned, figure out a response
for key, events in events_dicts.items():
if config.CHANNEL_TAGS[key][0] is None:
continue
response = None
print ("local events:", events)
if len(events) > 0:
if uid != None:
response = f'hey <@{uid}>, here are the event(s) coming up for {config.CHANNEL_STRINGS[key]} in the next {time_str}:\n'
else:
#response = f'check out these upcoming events for {config.CHANNEL_STRINGS[key]} in the next {time_str}:\n'
response = f'check out these events coming up in the next {time_str}:\n'
for event in events:
event_name = event[0]
event_time = event[1]
event_desc = event[2]
if not event_desc or event_desc.isspace():
event_desc = "no event info found"
event_time = event_time.strftime('on %a %x at %X %Z')
response += f"- {event[0]} {event_time} - {event_desc}\n"
else:
if uid != None:
response = f"hey <@{uid}>, i didn't find any events for {config.CHANNEL_STRINGS[key]} in the next {time_str} : ("
else:
print (f"no upcoming events for autocheck in {key}")
continue
payload = {'channel': config.CHANNEL_TAGS[key][0], 'text':response, "unfurl_links": "false", "unfurl_media" : "false"}
headers = {"Authorization": f"Bearer {config.BOT_OAUTH}", 'Content-Type': 'application/json'}
r = requests.post(SLACK_URL, json=payload, headers=headers)
return
# Filters out events that are just slack url verification; otherwise whould be user event
def process_user_event(event):
mes_body = json.loads(event['body'])
print("mes_body:", mes_body)
print("mes_body[type]:", mes_body['type'])
# there has to be a better way to unpack the JSON from slack
# url verification is to ensure we actually own the endpoint
if mes_body['type'] == "url_verification":
print ("responding to challenge")
challenge = mes_body['challenge']
ret = {"content-type" : "text/plain", "challenge" : challenge}
# should cover any slash commands or direct @bot tags that we implement
elif mes_body['type'] == "event_callback":
print("event callback caught")
event = mes_body['event']
if event['type'] == "app_mention":
print("doing app mention")
uid = event['user']
text = event['text']
channel = event['channel']
ret = keyword_response(text, uid, channel)
else:
print("unhandled event_callback:", event['type'])
else:
print("unhandled event type:", mes_body['type'])
return ret
# just determines if we have a default case of timer interrupt or if user requested info; default is fastpath.
def lambda_handler(event, context):
print("event:", event)
print("context:", context)
ret = None
# should be an auto-timer event
if event == {} or event == None:
# even though this isn't keyword it's simplest to lump this in as the null case
ret = keyword_response("", None, "all")
elif "body" in event:
ret = process_user_event(event)
else:
print("no body in event?")
print("done")
return ret
if __name__== "__main__":
lambda_handler(None, None)