Automate the Boring Stuff with Slackbot (ver. 2)
Takanori Suzuki
PyCon HK 2023 / 2023 Nov 11
Agenda
Background and Motivation for Slackbot
How to create simple bot
How to create interactive bot
How to extend bot using libs and APIs
Photos 📷 Tweets 🐦 👍
#pyconhk
/ @takanory
Slides 💻
Why ver. 2 in the title?
Back to 2019
Title: “Automate the Boring Stuff with Slackbot”
Talk in 🇵🇭 🇹🇭 🇲🇾 🇯🇵 🇹🇼 🇸🇬 🇮🇩
data:image/s3,"s3://crabby-images/d491b/d491b5b71e0f109286056c6647b68d57a8374f61" alt="../_images/pycon2019-collage3.jpg"
And the 2023
Updated with latest information 🆕
-> Ver.2 !
Who am I? 👤
Takanori Suzuki / 鈴木 たかのり ( @takanory)
PyCon JP Association: Chair
BeProud Inc.: Director / Python Climber
Python Boot Camp, Python mini Hack-a-thon, Python Bouldering Club
Love: Ferrets, LEGO, 🍺 / Hobby: 🎺, 🧗♀️
PyCon APAC Booth
data:image/s3,"s3://crabby-images/19f20/19f20429e4aa2fc8d0cc05370be36aca353af67b" alt="../_images/pyconapacbooth.jpg"
Hong Kong and I
My first time in PyCon HK and Hong Kong
Thanks to organizers and volunteers 👏
data:image/s3,"s3://crabby-images/ea822/ea822fcdf323306fdb172b1626abe5311090a64f" alt="../_images/hongkong.jpg"
Background and Motivation
Conference Tasks
I held PyCon JP(2014-2016) as Chair
Conference tasks:
👨💻 Keynotes, Talks and Trainings
🎫 Ticket sales and reception
🏬 Venue and facility(WiFi, Video…)
🍱 Foods, ☕️ Coffee, 🧁 Snacks and 🍺 Beers
Volunteers ask me same things
40+ staff
🐣 NEW : 🐔 OLD = 50 : 50
🐣 NEW volunteers ask me similar questions
Programmers are Lazy 😱
I have an Idea! 💡
Let’s create a Secretary!! 🤩
Goal 🥅
How to create simple bot
How to create interactive bot
How to extend bot using libs and APIs
Why Slack ?
Launching the Slack app at any time 💻 📱
Easy to access
To do everything
data:image/s3,"s3://crabby-images/07f18/07f18310f9513836f239c3171f34e355348ac413" alt="../_images/slack1.png"
You can create interactive bot
data:image/s3,"s3://crabby-images/4fbd0/4fbd0a41987e0699840fc269cc687aaefde8a2b9" alt="../_images/bot-result14.png"
data:image/s3,"s3://crabby-images/c1855/c1855baa5330c6cf26ba2b34168438b98a7d4884" alt="../_images/bot-result24.png"
Simple integration with Incoming Webhooks 🪝
System overview
data:image/s3,"s3://crabby-images/5bd6d/5bd6d52a6191a6f1105ee32e2065803c4e0c5662" alt="../_images/diagram-webhook4.png"
Create Incoming Webhooks Integration 🔧
Create Incoming Webhooks Integration
Generate Webhook URL
Create a Slack app
Activate Incoming Webhooks in the app
Add Webhook to Workspace
1. Create a Slack app
data:image/s3,"s3://crabby-images/c4ad0/c4ad014483d800595af5389815eb0a3a406c4205" alt="../_images/create-webhook1-14.png"
data:image/s3,"s3://crabby-images/0a9cc/0a9ccba592c9358aaae876acb0af037a1205b73a" alt="../_images/create-webhook1-24.png"
Enter name and choose workspace
data:image/s3,"s3://crabby-images/3d9ef/3d9ef1c609780ea80f8d12fee26440350b1e1b6a" alt="../_images/create-webhook24.png"
Set app icon (optional)
data:image/s3,"s3://crabby-images/d1fca/d1fca81c5600793598b19a9f674cfed03a78d4f3" alt="../_images/create-webhook34.png"
2. Activate Incoming Webhooks
data:image/s3,"s3://crabby-images/d70cb/d70cb987ca2986fce3dba4f98c08f70cf3b8c749" alt="../_images/create-webhook4-14.png"
3. Add Webhook to Workspace
Click “Add New Webhook to Workspace”
data:image/s3,"s3://crabby-images/aa00b/aa00b56e632ac3f9f7de3ffa5c6f35a25af6f6ce" alt="../_images/create-webhook4-24.png"
Choose channel → Click “Allow”
data:image/s3,"s3://crabby-images/382fc/382fc175559997fb7c047ee6a2b42c7c0e1d0f95" alt="../_images/create-webhook54.png"
Get Webhook URL:
https://hooks.slack.com/...
data:image/s3,"s3://crabby-images/7b32a/7b32a196232f7b2524ad2d0169391ccf8233db58" alt="../_images/create-webhook64.png"
Post message via Webhook URL 📬
Post message with cURL
$ curl -X POST -H 'Content-type: application/json' \
> --data '{"text":"Hello Slack!"}' \
> https://hooks.slack.com/services/T000...
data:image/s3,"s3://crabby-images/da70e/da70ebaee2e73bf8bed2f128ab754875d57a661e" alt="../_images/webhook-curl4.png"
Post message with Python
see: urllib.request
import json
from urllib import request
url = "https://hooks.slack.com/services/T000..."
message = {"text": "Hello from Python!"}
data = json.dumps(message).encode()
request.urlopen(url, data=data)
data:image/s3,"s3://crabby-images/e0449/e044965012238819c5e8d39b112c7d873015ba2c" alt="../_images/webhook-python4.png"
Post message with Requests
see: Requests
$ pip install requests
import requests
url = "https://hooks.slack.com/services/T000..."
message = {"text": "Hello from Requests!"}
r = requests.post(url, json=message)
data:image/s3,"s3://crabby-images/e54e2/e54e2adf696407679e9ec23e35a4b78d7292f3ee" alt="../_images/webhook-requests4.png"
Post message with Slack SDK
see: Python Slack SDK
$ pip install slack-sdk
from slack_sdk.webhook import WebhookClient
url = "https://hooks.slack.com/services/T000..."
webhook = WebhookClient(url)
r = webhook.send(text="Hello from Slack SDK!")
data:image/s3,"s3://crabby-images/17734/177347f14fcd2408f17ffb34b5e8cac1f11c55c4" alt="../_images/webhook-slacksdk4.png"
Formatting text
from slack_sdk.webhook import WebhookClient
url = "https://hooks.slack.com/services/T000..."
webhook = WebhookClient(url)
# *bold*, <url|text>, :emoji: and etc.
r = webhook.send(text="*Hello* from "
"<https://slack.dev/python-slack-sdk/|Slack SDK>! :beer:")
data:image/s3,"s3://crabby-images/f1f43/f1f43c1d6da8bac2e622881577b4472267fe82d1" alt="../_images/webhook-formatting4.gif"
Block Kit 🧱
Block Kit
data:image/s3,"s3://crabby-images/ee74d/ee74d145289868d53b44f90926206a5988f55814" alt="../_images/block-kit4.png"
see: Block Kit
Example of Block Kit
blocks = [{
"type": "section",
"text": {"type": "mrkdwn",
"text": "*THANK YOU* for coming to my talk !:tada: Please tell me good *craft beer bar* :bow:",
},
"fields": [
{"type": "mrkdwn", "text": "*Love*"},
{"type": "mrkdwn", "text": "*From*"},
{"type": "plain_text", "text": ":beer:, Ferrets, LEGO"},
{"type": "plain_text", "text": "Japan :jp:"},
],
}]
response = webhook.send(blocks=blocks)
data:image/s3,"s3://crabby-images/7afcb/7afcb36ce8ee0b1787a07f1f430fb06061661d46" alt="../_images/webhook-blocks4.png"
Block Kit Builder
data:image/s3,"s3://crabby-images/6a6d6/6a6d6ed5e9b1e8d90b3dcd76c4c19554fd1b6ef3" alt="../_images/block-kit-builder4.gif"
Summary of Incoming Webhooks
Easy to post messages from programs 📬
Create complex messages with Block Kit 🧱
But one-way (program➡️Webhook➡️Slack)
Interactive bot 🤝
Connection protocols
Events API over HTTP
Socket Mode
Events API over HTTP
data:image/s3,"s3://crabby-images/74ad3/74ad359bfa5a52b3f121894c5ecdb01f3a0a3cf9" alt="../_images/diagram-eventsapi4.png"
Socket Mode
data:image/s3,"s3://crabby-images/0aac1/0aac1bcf58297c6cc81bac7e2c4f6f381bb90b6d" alt="../_images/diagram-socketmode4.png"
see: Intro to Socket Mode
Connection protocols
Events API over HTTP
Socket Mode 👈
Create bot user 🤖
Create bot user
Create bot user with Socket Mode
Create a Slack app (same procedure)
Enable Socket Mode
Subscribe bot event
Add Bot Token Scopes
Install App to Workspace
Invite bot user to Slack channels
1. Create a Slack app
data:image/s3,"s3://crabby-images/c4ad0/c4ad014483d800595af5389815eb0a3a406c4205" alt="../_images/create-webhook1-14.png"
data:image/s3,"s3://crabby-images/0a9cc/0a9ccba592c9358aaae876acb0af037a1205b73a" alt="../_images/create-webhook1-24.png"
Enter name and choose workspace
data:image/s3,"s3://crabby-images/b883e/b883e59da62b3972f06a573d737280ee2af65961" alt="../_images/create-bot24.png"
Set app icon (optional)
data:image/s3,"s3://crabby-images/64166/64166ac932bdc5527498e66b04707c0ec146d9c7" alt="../_images/create-bot34.png"
2. Enable Socket Mode
Select “Socket Mode” → Turn toggle on
data:image/s3,"s3://crabby-images/d3c09/d3c098f96812e4d516a0c915f1bb0a5756a90679" alt="../_images/create-bot44.png"
Enter token name → Click “Generate”
data:image/s3,"s3://crabby-images/d9638/d96389756cfb6422d48deffca13f60f47036800d" alt="../_images/create-bot54.png"
Get app-level token:
xapp-...
data:image/s3,"s3://crabby-images/a2e1c/a2e1cd2ef2459cf170f5c76099d592722b54a8a8" alt="../_images/create-bot64.png"
3. Subscribe bot event
Select “Event Subscriptions” → Turn toggle on
data:image/s3,"s3://crabby-images/98cb8/98cb8caa858e82765327ac0390cc728f2cb21efa" alt="../_images/create-bot3-14.png"
Add “message.channels” to bot events
data:image/s3,"s3://crabby-images/45371/453719cd46030c98975c947fbdc58a81a073f3be" alt="../_images/create-bot3-2-14.png"
data:image/s3,"s3://crabby-images/0b995/0b995d8f049918bddf8f58636ca81e0082e5db49" alt="../_images/create-bot3-2-24.png"
4. Add Bot Token Scopes
Select “OAuth & Permissions”
data:image/s3,"s3://crabby-images/140a3/140a33adf7eef2fcc9c9fd84e773002348ce5671" alt="../_images/create-bot74.png"
Click “Add on OAuth Scope”
data:image/s3,"s3://crabby-images/0e125/0e125b31c2550ec478867f2dfd402e09534ba7d9" alt="../_images/create-bot84.png"
Add “chat:write” to Bot Token Scopes
data:image/s3,"s3://crabby-images/158dc/158dcc6241231defd147d97b201447157abc624a" alt="../_images/create-bot94.png"
data:image/s3,"s3://crabby-images/7b0ae/7b0aebffbf9970bb4836b69c0031c5779c8eddba" alt="../_images/create-bot104.png"
5. Install App to Workspace
Select “Install App” → Click “Install to Workspace”
data:image/s3,"s3://crabby-images/a84d3/a84d3e27123709ae732ffd8acb853db06b5a8c5d" alt="../_images/create-bot114.png"
Switch OAuth screen → Click “Allow” button
data:image/s3,"s3://crabby-images/3e7e2/3e7e2b502cf8619388fb9fc8792c003659551ffe" alt="../_images/create-bot124.png"
Get Bot Token:
xoxb-...
data:image/s3,"s3://crabby-images/0a710/0a710d435fa0cd73615edb6056c2b3413b298334" alt="../_images/create-bot134.png"
Invite bot user to channels
data:image/s3,"s3://crabby-images/b7563/b7563d0bd507476f1575dbf88ba7692d5e5fb61a" alt="../_images/invite-bot4.png"
Long and Complex !! 🤯
App Manifest ⚙️
App Manifest
YAML-formatted configuration for Slack apps
Example of App Manifest
display_information:
name: beerbot2
features:
bot_user:
display_name: beerbot2
always_online: false
oauth_config:
scopes:
bot:
- channels:history
- chat:write
settings:
event_subscriptions:
bot_events:
- message.channels
interactivity:
is_enabled: true
org_deploy_enabled: false
socket_mode_enabled: true
token_rotation_enabled: false
Get App Manifest
Select “App Manifest” menu
data:image/s3,"s3://crabby-images/854b9/854b973b54a6fa62203bc0fe9dd0b47b2159a29e" alt="../_images/get-app-manifest4.png"
Create new app with App Manifest
Select “From an app manifest”
Select workspace → Click “Next”
data:image/s3,"s3://crabby-images/d0fc2/d0fc2aa6515720b23aedc9051b67c9deaf77bd06" alt="../_images/app-manifest14.png"
data:image/s3,"s3://crabby-images/44c8e/44c8eb34aad44aec91e6b0774147bc60be71423a" alt="../_images/app-manifest24.png"
Enter app manifest YAML
data:image/s3,"s3://crabby-images/69fad/69fad27676a72fd97b73571d7ad083cd224ac0cd" alt="../_images/app-manifest34.png"
Review app summary → Click “Create”
data:image/s3,"s3://crabby-images/44b27/44b270a3df686aa7e544c2563638c2fa477a3356" alt="../_images/app-manifest44.png"
data:image/s3,"s3://crabby-images/36ad0/36ad06fa47abc6443e6261114e0bdb6c1b724b51" alt="../_images/app-manifest54.png"
data:image/s3,"s3://crabby-images/b4e84/b4e84b2ec9e881cb472fc7f869dd62f5539245d1" alt="../_images/app-manifest64.png"
Install App to Workspace
data:image/s3,"s3://crabby-images/0c3fe/0c3fe3e67a8b27ccb3cf520f248a02abaa2d11e9" alt="../_images/app-manifest74.png"
Generate App-Level Token
data:image/s3,"s3://crabby-images/e775e/e775ef760acbc5a3d70974f2abf271ddbcc86b9a" alt="../_images/app-manifest84.png"
data:image/s3,"s3://crabby-images/88989/88989855416607ad1839a52d7466b17fe3822550" alt="../_images/app-manifest94.png"
Short and Reusable !! 🥳
Create bot with Bolt ⚡️
Bolt for Python
Python framework to build Slack app in a flash
Developped by Slack
see:
The Bolt family of SDKs (JavaScript, Java)
Install Bolt for Python
$ mkdir beerbot
$ cd beerbot
$ python3.11 -m venv env
$ . env/bin/activate
(env) $ pip install slack-bolt
Create a simple bot with Bolt
import os
from slack_bolt import App
from slack_bolt.adapter.socket_mode import SocketModeHandler
app = App(token=os.environ["SLACK_BOT_TOKEN"])
# match any message contains "Hi"
@app.message("Hi")
def handle_hi_message(message, say):
say("Hi!!! I am beerbot! :beer:")
if __name__ == "__main__":
app_token = os.environ["SLACK_APP_TOKEN"]
SocketModeHandler(app, app_token).start()
Running bot
# Set 2 tokens in environment variables
(env) $ export SLACK_APP_TOKEN=xapp-...
(env) $ export SLACK_BOT_TOKEN=xoxb-...
(env) $ python app.py
⚡️ Bolt app is running!
I can interact with the bot ! 🎉
data:image/s3,"s3://crabby-images/6da32/6da32473e2bc2e6725562f91f46b56864f8d6ee6" alt="../_images/bot-hi4.png"
Extend bot 🔧
@app.message()
decolator
# match any message contains "Hi"
@app.message("Hi")
def handle_hi_message(message, say):
say("Hi!!! I am beerbot! :beer:")
# match any message contains "cheers"
@app.message("cheers")
def handle_cheers_message(mesasge, say):
say("Cheers! :beers:")
data:image/s3,"s3://crabby-images/a1ae5/a1ae59c4b13b0309d09d178e105f9992d373d91c" alt="../_images/bot-decolator4.png"
mention
# mention
@app.message("morning")
def handle_morning_message(message, say):
user = message["user"] # get user ID
say(f"Good morning <@{user}>!")
data:image/s3,"s3://crabby-images/9ba22/9ba22fc558aa0d5bb7b40c0333f0a585e1c9e998" alt="../_images/bot-mention4.png"
Using regular expression
import random
import re
@app.message(re.compile(r"choice (.*)"))
def handle_choice(say, context):
# get matched text from context["matches"]
words = context["matches"][0].split()
say(random.choice(words))
data:image/s3,"s3://crabby-images/2f05b/2f05b3ba07bd3d4fa1c2c32d9548901a09e18057" alt="../_images/bot-choice4.png"
Using regular expression
@app.message(re.compile(r"(\d+)\s*(beer|tea)"))
def handle_beer_or_tea(say, context):
count = int(context["matches"][0])
drink = context["matches"][1]
say(f":{drink}:" * count)
data:image/s3,"s3://crabby-images/c37bd/c37bd2eaae30147f34676d22db4141cf7271f80d" alt="../_images/bot-beers4.png"
Block Kit support
@app.message("follow me")
def handle_follow_me(message, say):
blocks = [{
"type": "section",
"text": {
"type": "mrkdwn",
"text": "*Takanori Suzuki*\nFollow me! <https://twitter.com/takanory|@takanory>"
},
"accessory": {
"type": "image",
"image_url": "https://pbs.twimg.com/profile_images/192722095/kurokuri_400x400.jpg",
"alt_text": "takanory"
}}]
say(blocks=blocks)
data:image/s3,"s3://crabby-images/77bdd/77bdd61db67a12ec4e0904b3f3ade419b4ea3351" alt="../_images/bot-followme4.png"
Logging
import logging
logging.basicConfig(level=logging.DEBUG)
@app.message("log")
def handle_log(message, say, logger):
logger.debug(f"message: {message['text']}")
say("logs exported!")
$ python app.py
⚡️ Bolt app is running!
DEBUG:app.py:handle_log:message: log
DEBUG:app.py:handle_log:message: please write log
data:image/s3,"s3://crabby-images/340d5/340d5f1193fbf3e6819606679ffd475d1f70d313" alt="../_images/bot-logging4.png"
Events and Scopes 🔭
Events and Scopes
Can only receive events in Bot Events
Can only execute APIs allowed by Bot Token Scopes
Current Bot Events and Scopes
Events
message.channels
Scopes
channels:history
View messages in public channels
chat:write
Post message
Current Bot Events and Scopes
Cannot read/write messages on private channels
data:image/s3,"s3://crabby-images/75fb0/75fb0dd3605ae6d19444d7a99e01db7bdf4ffc66" alt="../_images/bot-private-cannot-view4.png"
Add Events and Scopes for private channels
Select “Event Subscriptions” → Click “Add Bot User Event”
Add message.groups event→ Click “Save Changes”
data:image/s3,"s3://crabby-images/4ce74/4ce74975ee49ce5e57a4afaa08cc8a56447bc1e8" alt="../_images/add-events-and-scopes14.png"
Add Events and Scopes for private channels
Select “OAuth & Permissions”
groups:history scope is automatically added
data:image/s3,"s3://crabby-images/f04ee/f04eee8c69bf9d33d1be229fbf4dc86492ac7931" alt="../_images/add-events-and-scopes24.png"
Add Events and Scopes for private channels
Reinstall app to workspace
data:image/s3,"s3://crabby-images/9112b/9112bb271adb27827204ca7bc3edc4dc1f6f95cb" alt="../_images/add-events-and-scopes34.png"
data:image/s3,"s3://crabby-images/a316f/a316fdfe91fae2b5536e1a326e1086f47b9fc619" alt="../_images/add-events-and-scopes44.png"
Add Events and Scopes for private channels
Bot can read/write messages in private channel
data:image/s3,"s3://crabby-images/7dc3a/7dc3a656256d2a5b9104fc28170616ed9da3e35e" alt="../_images/add-events-and-scopes64.png"
To know user joined a channel
Add member_joined_channel event → Reinstall app
# A user joined a public or private channel
@app.event("member_joined_channel")
def member_joined(event, say):
user = event["user"] # get user ID
say(f"Welcome <@{user}>! :tada:")
data:image/s3,"s3://crabby-images/ca2ce/ca2ce6ff6f966efd5e3088fd45c2ae88cd0908e4" alt="../_images/event-member-joined4.png"
Add Emoji reaction
Add reactions:write scope → Reinstall app
@app.message("beer")
def add_beer_emoji(client, message):
# add reaction beer emoji
client.reactions_add(
channel=message["channel"],
timestamp=message["ts"],
name="beer",
)
data:image/s3,"s3://crabby-images/b4603/b46036d8878a2ed4e679a3ec19147e27eedc82fd" alt="../_images/scope-reactions-write4.png"
Summary of Events and Scopes
To receive new events
To use new API with new scopes
Add events and/or scopes → Reinstall app
see: Events API types
see: Permission scopes
Case studies 📚
Calculator function using SymPy 🔢
Calculator function using SymPy
Motivation
I feel heavy to call a calculator app on my smartphone
It seems useful if Slack as a calculator
System overview
data:image/s3,"s3://crabby-images/0d0c8/0d0c85a80bd7978ca64e0b7ff194323480b46667" alt="../_images/diagram-sympy4.png"
about SymPy
data:image/s3,"s3://crabby-images/1f4df/1f4df02810c22df0f51aaf64ca8232c6681170bf" alt="https://www.sympy.org/static/images/logo.png"
SymPy: Python library for symbolic mathematics
$ pip install sympy
calc() function using Sympy
from sympy import sympify, SympifyError
@app.message(re.compile(r"^([-+*/^%!().\d\s]+)$"))
def calc(message, context, say):
try:
formula = context["matches"][0]
result = sympify(formula) # Simplifies formula
if result.is_Integer:
answer = int(result) # Convert to integer
else:
answer = float(result) # Convert to float
say(f"{answer:,}")
except SympifyError:
pass
Slack as a calculator !! 🎉
data:image/s3,"s3://crabby-images/f4b9f/f4b9f728f529d56c3628867c9bb96356f8022d52" alt="../_images/case-sympy4.png"
Plus-plus feature using Peewee ORM 👍
Plus-plus feature using Peewee ORM
Motivation
In PyCon JP, I want to make a culture that appreciates each other staff 👍
System overview
data:image/s3,"s3://crabby-images/da412/da4126f42431856df2ff0c039c2fb53abde7dc77" alt="../_images/diagram-peewee4.png"
about Peewee
data:image/s3,"s3://crabby-images/2f9a7/2f9a74b5c0723c9b4d9697f78dc00fcff55dde71" alt="https://docs.peewee-orm.com/en/latest/_images/peewee3-logo.png"
Simple and small ORM.
a small, expressive ORM
supports sqlite, mysql and postgresql
$ pip install peewee
plusplus_model.py
from peewee import SqliteDatabase, Model, CharField, IntegerField
db = SqliteDatabase("plusplus.db")
class Plusplus(Model):
name = CharField(primary_key=True) # fields
counter = IntegerField(default=0)
class Meta:
database = db
db.connect()
db.create_tables([Plusplus], safe=True)
plusplus() function using Peewee
from plusplus_model import Plusplus
# match "word++" pattern
@app.message(re.compile(r"^(\w+)\+\+"))
def plusplus(say, context):
name = context["matches"][0]
# Get or create object
plus, created = Plusplus.get_or_create(
name=name.lower(),
defaults={'counter': 0}
)
plus.counter += 1
plus.save()
say(f"Thank you {name}! (count: {plus.counter})")
I can appreciate it! 🎉
data:image/s3,"s3://crabby-images/07918/079185c456111ba1ac3e4d248113e00b3d22d21e" alt="../_images/case-peewee4.png"
Search issues with Jira APIs 🔎
Search issues with Jira APIs
Motivation
Jira is very useful
Jira Web is slow
Search issues without Jira Web
System overview
data:image/s3,"s3://crabby-images/5c572/5c57270bf7403b540ed83f24da4f0f70554d8290" alt="../_images/diagram-jira4.png"
about Python Jira
Python library to work with Jira APIs
$ pip install jira
Authentication
Create an API token
from jira import JIRA
url = 'https://jira.atlassian.com/'
jira = JIRA(url, basic_auth=('email', 'API token'))
Search issues
@app.message(re.compile(r"^jira (.*)$"))
def jira_search(message, context, say):
keywords = context["matches"][0]
jql = f'text ~ "{keywords}" order by created desc'
text = ""
# get 5 recent issues
for issue in jira.search_issues(jql, maxResults=5):
issue_id = issue.key
url = issue.permalink()
summary = issue.fields.summary
text += f"* <{url}|{issue_id}> {summary}\n"
if not text:
text = "No issues found"
say(text)
Free from Jira web! 🎉
data:image/s3,"s3://crabby-images/a96b8/a96b8d1b55d26d8672f5301ecf9fdeb6679ce837" alt="../_images/bot-jira4.png"
Create multiple issues from a template 📝
Create multiple issues from a template
Motivation
In pycamp event, 20+ issues are required for each event
Copying issues by hand is painful
Jira Web is slow (again)
System overview
data:image/s3,"s3://crabby-images/e9866/e98660715ea08b50ffe7b56a2916cc2450a971ea" alt="../_images/diagram-template4.png"
Google Authorization is Complex
Create a Google Cloud project
Enable API(in this case: Google Sheets API)
Download
credentials.json
Install Google Client Library
$ pip install google-api-python-client \
google-auth-httplib2 google-auth-oauthlib
Download quickstart.py from GitHub
Google Authorization is Complex
Run
quickstart.py
Select your Google account in Web browser
Click “Accept” button
Get
token.json
(finish!!)
$ python quickstart.py
Please visit this URL to authorize this application: https://accounts.google.com/o/oauth2/auth?....
Name, Major:
Alexandra, English
:
Issue template
data:image/s3,"s3://crabby-images/6dd14/6dd14e600a34f8a312c0eb40b6eb843b1cef9a98" alt="../_images/bot-issue-template4.png"
Get Spreadsheet data
from google.oauth2.credentials import Credentials
from googleapiclient.discovery import build
SCOPES = ["https://www.googleapis.com/auth/spreadsheets.readonly"]
SHEET = "1i3YQPObe3ZC_i1Jalo44_2_..."
@app.message("create issues")
def creaete_issues(message, say):
creds = Credentials.from_authorized_user_file('token.json', SCOPES)
service = build("sheets", "v4", credentials=creds)
sheet = service.spreadsheets()
result = sheet.values().get(spreadsheetId=SHEET, range="A2:C4").execute()
for row in result.get("values", []):
say(f"* Title: {row[0]}, Delta: {row[1]}")
Get Spreadsheet data
data:image/s3,"s3://crabby-images/84b9f/84b9f9ea6bd770fc34a7e2df99b8d7bccded942c" alt="../_images/bot-sheet14.png"
Create Jira issues
def creaete_issues(message, say):
# --snip--
today = datetime.date.today()
for row in result.get("values", []):
duedate = today + datetime.timedelta(days=int(row[1]))
issue_dict = {"project": {"key": "ISSHA"},
"summary": row[0],
"description": row[2],
"duedate": f"{duedate:%Y-%m-%d}",
"issuetype": {"name": "Task"}}
issue = jira.create_issue(fields=issue_dict)
url = issue.permalink()
say(f"* Create: <{url}|{issue.key}> {row[0]}\n")
see: 2.1.4. Issues
Free from copying issues! 🎉
data:image/s3,"s3://crabby-images/a871b/a871b9409e5bda7a927ccece7992a63a56bd6ac0" alt="../_images/bot-sheet24.png"
data:image/s3,"s3://crabby-images/4e42d/4e42dcf87f5778bb1a17a887b264e5f0efa99015" alt="../_images/bot-sheet34.png"
Account management of Google Workspace 👥
Account management of Google Workspace
Motivation
PyCon JP Association use
pycon.jp
domain with Google WorkspaceI only use Google Admin web occasionally
I forgot to use admin screen
System overview
data:image/s3,"s3://crabby-images/3d4a7/3d4a7bd043b412ae412e22063e152d2b7a2a54a1" alt="../_images/diagram-directory4.png"
Update Google Authorization
Update a Google Cloud project
add Directory API
re-download
credentials.json
Remove
token.json
Add Directory API
quickstart.py
Re-run
quickstart.py
Get new
token.json
Get user list
SCOPES = ["https://www.googleapis.com/auth/admin.directory.user"]
@app.message("user_list")
def user_list(message, say):
creds = Credentials.from_authorized_user_file("token.json", SCOPES)
service = build("admin", "directory_v1", credentials=creds)
users_list = service.users().list(
orderBy="email", maxResults=10, customer="my_customer").execute()
for user in users_list.get("users", []):
email = user["primaryEmail"]
fullname = user["name"]["fullName"]
say(f"* {email} {fullname}")
Get user list
data:image/s3,"s3://crabby-images/aeffd/aeffdd4831e68696943a700f61d3be9082f83045" alt="../_images/bot-user-list4.png"
Add user
@app.message(re.compile(r"user_add (\w+) (\w+) (\w+) (\w+)"))
def insert_user(message, say, context):
creds = Credentials.from_authorized_user_file("token.json", SCOPES)
service = build("admin", "directory_v1", credentials=creds)
body = {
"primaryEmail": context["matches"][0] + "@pycon.jp",
"password": context["matches"][1],
"name": {
"givenName": context["matches"][2],
"familyName": context["matches"][3]
}}
service.users().insert(body=body).execute()
say(f"User {body['primaryEmail']} created")
Add user
data:image/s3,"s3://crabby-images/c5f90/c5f90115028cd9de5f08ce729e440d15025e0b59" alt="../_images/bot-user-add5.png"
data:image/s3,"s3://crabby-images/f4f57/f4f577da865be7f325670b006adf578b2d9f666a" alt="../_images/bot-user-add24.png"
I can forget Google Admin! 🎉
Security Issue 🔓
Anyone can run it
Run only Slack Admin 🔒
Not-Admin cannot run
Add
users:read
scope, use users.info API
@app.message("user_list")
def user_list(message, say, client):
# get user info: https://api.slack.com/methods/users.info
response = client.users_info(user=message["user"])
user = response.data["user"]
if not user["is_admin"]:
say("You are not an admin!")
return
data:image/s3,"s3://crabby-images/60e47/60e47831cef0c5fb2b3890fe66655538a8de2b21" alt="../_images/bot-not-admin3.png"
Resolve a security issue 🎊
Summary
Simple bot using Incoming Webhooks
Interactive bot using Bolt for Python
Extend bot using libraries and APIs
Next Step 🪜
Let’s make your Slackbot
Let’s connect with libraries and APIs
Automate your Boring Stuff with Slackbot
Thank you! 🙏
data:image/s3,"s3://crabby-images/e1d66/e1d66c867bcc3b975bb900468cf265372f2ea20d" alt="../_images/bot-translate4.png"
translate command
$ pip install deepl
import deepl
translator = deepl.Translator(os.environ["DEEPL_AUTH_KEY"])
@app.message(re.compile(r"^translate (.*)"))
def translate(message, context, say):
text = context["matches"][0]
result = translator.translate_text(
text, target_lang="EN-US")
say(result.text)
Thank you! 🙏
data:image/s3,"s3://crabby-images/e1d66/e1d66c867bcc3b975bb900468cf265372f2ea20d" alt="../_images/bot-translate4.png"
@takanory takanory takanory takanory