-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathappStart.py
366 lines (335 loc) · 13.5 KB
/
appStart.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
import os
# Use the package we installed
import logging
import random
from slack_bolt import App, Say, BoltContext
from slack_sdk import WebClient
logging.basicConfig(level=logging.DEBUG)
tips = [
"Always make sure that you update the Request URL when restarting ngrok",
"Make sure you check your scopes when an API method does not work",
"Every API call has a scope check that you use the correct ones and when calling something new add that scope on Slack",
"After changing the scope you need to install your App again",
"Every Event you want to use, you need to subscribe to on Slack",
"Every Slack command needs the Request URL",
"The Bot token is on the Basics Page",
"The Slack Signing Secret is on the OAuth & Permission page",
"Limit the scopes to what you really need",
"You can add collaborators to your Slack App under Collaborators, they need to be in the workspace",
"To use buttons you need to enable Interactive Components",
"Always go in tiny steps, use the logger or print statements to see where you are and if events reach your app",
"If the user does something on Slack always send a response, even if it is just an emoji",
"Your App can use on the users behalf, you need to get User Scopes in that case. Later this would lead to having to save user tokes",
"Do not plan to much, do simple things first, when you get these done then start to dream big",
"Storing persistent data in a DB makes sense, when you are not familiar with it maybe a dict in a file might be enough for now",
"Oauth -- so authenticating the app and user is important when distributing your app, while locally developing it is not that important yet, get some functionality going first",
"Have the Slack API, Slack Events and your Slack Build App page open in your browser to have fast access",
"Slash commands need to be unique in a workspace, so do append your bots name to them"
]
# Initialize app with bot token and signing secret
app = App(
token=os.environ.get("SLACK_BOT_TOKEN"),
signing_secret=os.environ.get("SLACK_SIGNING_SECRET")
)
@app.event("app_mention")
def app_mentioned(body, event, logger, say, client):
logger.info("Testing app_mention\n")
#logger.info(body)
channelList = client.conversations_list()
logger.info(channelList)
say(f"Hello there <@{event['user']}>")
client.chat_postEphemeral(channel=event['channel'], user=event['user'], text="Feel free to ask me anything!")
@app.event("member_joined_channel")
def event_join(ack, body, say, logger, client):
ack()
logger.info("Testing member_joined_channel event\n")
logger.info(body)
if body['event']['channel'] == 'C01HFBP03GE':
text = [
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": f"Hello <@{body['event']['user']}>,\n would you like to introduce yourself a bit? "
},
"accessory": {
"type": "image",
"image_url": "https://api.slack.com/img/blocks/bkb_template_images/approvalsNewDevice.png",
"alt_text": "computer thumbnail"
}
},
{
"type": "actions",
"elements": [
{
"type": "button",
"text": {
"type": "plain_text",
"text": "YES"
},
"action_id": "intro_yes_button",
"style": "primary",
"value": "yes"
},
{
"type": "button",
"text": {
"type": "plain_text",
"text": "NO"
},
"action_id": "intro_nope_button",
"style": "danger",
"value": "nope"
}
]
}
]
client.chat_postMessage(
channel=body['event']['channel'], user=body['event']['user'], blocks=text)
@app.command("/slackbot2.0_tips")
def tips_called(ack, body, command, logger, client):
ack()
logger.info("Testing slackbot2.0_tips")
logger.info(command)
tip = random.choice(tips)
block = [
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": "Tip: "
}
}
]
attach= [
{
"color": "#f2c744",
"blocks": [
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": tip
}
},
{
"type": "actions",
"elements": [
{
"type": "button",
"text": {
"type": "plain_text",
"text": "Show more Slack API tips",
"emoji": True
},
"value": "click_me_123",
"action_id": "show_more_tips"
}
]
}
]
}
]
client.chat_postMessage(channel=command['channel_id'], blocks=block, attachments=attach)
@app.action("show_more_tips")
def tip_button_clicked(body, action, ack, logger):
ack()
logger.info("Testing tip_button_clicked")
logger.info(action)
@app.event("app_home_opened")
def update_home_tab(client, event, logger):
try:
tip = random.choice(tips)
# views.publish is the method that your app uses to push a view to the Home tab
client.views_publish(
# the user that opened your app's app home
user_id = event["user"],
# the view object that appears in the app home
view = {
"type": "home",
"callback_id": "home_view",
# body of the view
"blocks": [
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": "*Welcome to your _App's Home_* :tada:"
}
},
{
"type": "divider"
},
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": "`Tip:` " + tip
}
},
{
"type": "actions",
"elements": [
{
"type": "button",
"text": {
"type": "plain_text",
"text": "Show more tips",
"emoji": True
},
"value": "click_me_123",
"action_id": "show_more_tips"
}
]
},
{
"type": "divider"
},
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": "This button won't do much for now but you can set up a listener for it using the `actions()` method and passing its unique `action_id`. See an example in the `examples` folder within your Bolt app."
}
},
{
"type": "actions",
"elements": [
{
"action_id": "HelloThere",
"type": "button",
"style": "primary",
"text": {
"type": "plain_text",
"text": "Click me now!"
}
},
]
}
]
}
)
except Exception as e:
logger.error(f"Error publishing home tab: {e}")
# Example button for button on App home which just posts a message to the user in the
# apps user channel
@app.action("HelloThere")
def react_to_button(body, logger, client):
logger.info(body)
client.chat_postMessage(
channel=body["user"]["id"], text="You pushed a button!")
@app.action("intro_yes_button")
def yes_button(body, action, logger, client, ack):
ack()
## triggering a modal view
res = client.views_open(
trigger_id=body["trigger_id"],
view={
"type": "modal",
"callback_id": "intro_modal",
"title": {"type": "plain_text", "text": "My Intro",},
"submit": {"type": "plain_text", "text": "Submit",}, # has a submit button is optional, that submit will trigger callback_id
"close": {"type": "plain_text", "text": "Cancel",},
"private_metadata": f"{body['channel']['id']},{body['message']['ts']}", ## stored so we can update the original message
"blocks": [ ## create 4 input fields
{
"type": "input",
"block_id":"where",
"element": {"type": "plain_text_input", "action_id": "where"},
"label": {"type": "plain_text", "text": "Where are you from?",},
},
{
"type": "input",
"block_id":"you",
"element": {"type": "plain_text_input", "action_id": "you"},
"label": {"type": "plain_text", "text": "Tell us about yourself",},
},
{
"type": "input",
"block_id":"learn",
"element": {"type": "plain_text_input", "action_id": "learn"},
"label": {"type": "plain_text", "text": "What do you want to learn",},
},
{
"type": "input",
"block_id":"project",
"element": {"type": "plain_text_input", "action_id": "project"},
"label": {"type": "plain_text", "text": "What project would you like to work on?",},
}
],
},
)
logger.info(res)
### reads out our simple modal and formats the result a bit
@app.view('intro_modal')
def intro_modal_submitted(body, ack, logger, client):
ack()
logger.info("Modal Submitted")
## reading the data from the input fields
you = body['view']['state']['values']['you']['you']['value']
learn = body['view']['state']['values']['learn']['learn']['value']
where = body['view']['state']['values']['where']['where']['value']
project = body['view']['state']['values']['project']['project']['value']
# reading metadata from the original channel and ts that was added
data = body['view']['private_metadata'].split(',')
block = [
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": f"<@{body['user']['id']}> INTRO"
}
},
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": f"*From:* \n {where}"
}
},
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": f"*Who I am:* \n {you}"
}
},
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": f"*What I want to learn:* \n {learn}"
}
},
{
"type": "section",
"text": {
"type": "mrkdwn",
"text": f"*What I want to work on:* \n {project}"
}
}
]
client.chat_update(channel=data[0],
ts=data[1], blocks=block)
# print shame on you when they do not want to introduce themselves
@app.action("intro_nope_button")
def yes_button(body, action, logger, client, ack):
ack()
# creating a new message which has parts of the old messag and adds a section
oldMessage = [body['message']['blocks'][0]]
oldMessage.append({
"type": "section",
"text": {
"type": "mrkdwn",
"text": f"You said nope, SHAME on you!"
}
})
# we are updating an existing message here and remove the buttons
client.chat_update(channel=body['channel']['id'],
ts=body['message']['ts'], blocks=oldMessage)
@app.message("Test")
def reply_to_test(say):
say("Yes, tests are important!")
# Start app
if __name__ == "__main__":
app.start(port=int(os.environ.get("PORT", 3000)))