Automate the Boring Stuff with Slackbot (ver. 2)
Takanori Suzuki
PyCon JP 2022 / 2022 Oct 15
退屈なことは Slackbot にやらせよう (ver. 2)
Takanori Suzuki
PyCon JP 2022 / 2022 Oct 15
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 🐦 👍
#pyconjp
/ #pyconjp_1
/ @takanory
Slide 💻
今日はPyCon JP 2022で13:00からTrack 1で「Automate the Boring Stuff with Slackbot(ver.2) 」で発表します。退屈なことをbotにやらせたい人はぜひ聞きに来てくださいー。発表資料はこちら https://t.co/bUrVXtqR5I #pyconjp #pyconjp_1
— Takanori Suzuki (@takanory) October 15, 2022
Why ver. 2 in the title? ✌️
なぜタイトルに ver. 2 が入ってるの?
Back to 2019 / 2019年に遡る ⏪
Title: “Automate the Boring Stuff with Slackbot”
Talk in 🇵🇭 🇹🇭 🇲🇾 🇯🇵 🇹🇼 🇸🇬 🇮🇩
data:image/s3,"s3://crabby-images/10cf3/10cf39acce18d5d92570ee105cd5211475cf29ee" alt="../_images/pycon2019-collage1.jpg"
And the 2022 / そして2022年 ⏩
Updated with latest information 🆕
In-person event after COVID-19 in Japan 🇯🇵
Thanks to PyCon JP staff and volunteers!! 👏
Who am I? / お前誰よ 👤
Takanori Suzuki / 鈴木 たかのり ( @takanory)
PyCon JP Association Vice Chair
BeProud Inc. Director / Python Climber
data:image/s3,"s3://crabby-images/a21b3/a21b3d57d011ff82807bf825d07acea98ac8b74d" alt="../_images/sokidan-square.jpg"
data:image/s3,"s3://crabby-images/970b5/970b5b6688fbd18e5fcafce3f3b2ed3862b735ce" alt="../_images/kurokuri.jpg"
BeProud inc. 🏢
BeProud: Pythonシステム開発、Consulting
connpass: IT勉強会支援プラットフォーム
PyQ: Python独学プラットフォーム
TRACERY: システム開発ドキュメントサービス
data:image/s3,"s3://crabby-images/83bc4/83bc4055a6ee09f7f3ab89aeb5c8c27c28a5694e" alt="../_images/beproud-logos.png"
BeProud Booth
data:image/s3,"s3://crabby-images/37427/37427e684b8fdff02ea18c9ede2a824e295cf06e" alt="../_images/beproud-booth.jpg"
AD is over / 宣伝は終了
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
Staff ask me the same things
スタッフは 同じこと を質問する
40+ staff
🐣 NEW staff : 🐔 OLD staff = 50 : 50
Programmer is Lazy
プログラマーは 怠惰
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 ? / なぜ 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
対話 botが作れるようになる
data:image/s3,"s3://crabby-images/5218e/5218e7734a482e2153ba262dcf689b87751d7282" alt="../_images/bot-result12.png"
data:image/s3,"s3://crabby-images/4fcb6/4fcb6140d4b477d462443caac2235bdbe82d6bdc" alt="../_images/bot-result22.png"
Simple integration with Incoming Webhooks 🪝
Incoming Webhooks での簡単な連携
System overview / システム概要
data:image/s3,"s3://crabby-images/94780/9478060d1232be187145e03f3cb06809f3956872" alt="../_images/diagram-webhook2.png"
Create Incoming Webhooks Integration 🔧
Incoming Webhooks連携を 作成
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 / Slack appを作成
data:image/s3,"s3://crabby-images/bdd6b/bdd6bdb11e3c1e5ea12e7d0e7858bc43c1524a93" alt="../_images/create-webhook1-12.png"
data:image/s3,"s3://crabby-images/eff7b/eff7b983604e6c3928c8f80e2d94f0ef9b7ca88f" alt="../_images/create-webhook1-22.png"
Enter name and choose workspace
data:image/s3,"s3://crabby-images/093fa/093faf5912ed8f7cf995ec14b60148bf800ab0da" alt="../_images/create-webhook22.png"
Set app icon (optional)
data:image/s3,"s3://crabby-images/fbb95/fbb95666b5ccf694c7a8fbb124cb2bf7f18aca1c" alt="../_images/create-webhook32.png"
2. Activate Incoming Webhooks / 有効化
data:image/s3,"s3://crabby-images/6c5f7/6c5f778ca3064730ffbbb30bfaf44275ae4466e9" alt="../_images/create-webhook4-12.png"
3. Add Webhook to Workspace / ワークスペースに追加
Click “Add New Webhook to Workspace”
data:image/s3,"s3://crabby-images/db6a1/db6a1b1153e9a894656700d62a1f8a127a4c685b" alt="../_images/create-webhook4-22.png"
Choose channel → Click “Allow”
data:image/s3,"s3://crabby-images/312af/312af689b5b371784fd3cd28f37e7413ea4493b9" alt="../_images/create-webhook52.png"
Get Webhook URL:
https://hooks.slack.com/...
data:image/s3,"s3://crabby-images/9bddf/9bddfc34790234bfc523240b91f9b20903e9da42" alt="../_images/create-webhook62.png"
Post message via Webhook URL 📬
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/ad6ba/ad6bae67c25dd95832079749adb5823a66968458" alt="../_images/webhook-curl2.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/8d934/8d9347ff5cd629c8f8f6caffe7c6154c44f081c2" alt="../_images/webhook-python2.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/6940e/6940e2650de5e5e3ffadf21c712c96c46591df64" alt="../_images/webhook-requests2.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/aba71/aba71398e49efecfacf168b7ccffd39001596175" alt="../_images/webhook-slacksdk2.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.
sdk_url = "https://slack.dev/python-slack-sdk/"
r = webhook.send(
text=f"*Hello* from <{sdk_url}|Slack SDK>! :beer:")
data:image/s3,"s3://crabby-images/deade/deade70f2271c9dd46a6ce57cc62bb1f8a659c25" alt="../_images/webhook-formatting2.gif"
Block Kit 🧱
Block Kit
data:image/s3,"s3://crabby-images/97bd0/97bd052c9301033cfdeec652d3a179479697d6d9" alt="../_images/block-kit2.png"
see: Block Kit
Example of Block Kit
blocks = [{
"type": "section",
"text": {
"type": "mrkdwn"
"text": "*THANK YOU* for coming to my talk !:tada: Please give me *feedback* about this talk :bow:",
},
"fields": [
{"type": "mrkdwn", "text": "*Love*"},
{"type": "mrkdwn", "text": "*Hobby*"},
{"type": "plain_text", "text": "Ferrets, :beer:, LEGO"},
{"type": "plain_text", "text": ":trumpet:, :man_climbing:"},
],
}]
response = webhook.send(blocks=blocks)
data:image/s3,"s3://crabby-images/ddc97/ddc977205bd94a85b2e9d0cb15be3489f15d4909" alt="../_images/webhook-blocks2.png"
Block Kit Builder
data:image/s3,"s3://crabby-images/ad93d/ad93d2597089eac0430b6e667266e4840b787ba5" alt="../_images/block-kit-builder2.gif"
Summary of Incoming Webhooks
まとめ: Incoming Webhooks
Easy to post messages from programs 📬
Create complex messages with Block Kit 🧱
But one-way (program➡️Webhook➡️Slack)
Interactive bot 🤝
対話型 のbot
Connection protocols / 接続方式
Events API over HTTP
Socket Mode
Events API over HTTP
data:image/s3,"s3://crabby-images/ef5e1/ef5e1093ab033e1afec7073ede848333b159ee51" alt="../_images/diagram-eventsapi2.png"
Socket Mode
data:image/s3,"s3://crabby-images/5d884/5d8844617b78cb73e83294e4ce468d3ab7025e13" alt="../_images/diagram-socketmode2.png"
see: Intro to Socket Mode
Connection protocols / 接続方式
Events API over HTTP
Socket Mode 👈
Create bot user 🤖
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 / Slack appを作成
data:image/s3,"s3://crabby-images/bdd6b/bdd6bdb11e3c1e5ea12e7d0e7858bc43c1524a93" alt="../_images/create-webhook1-12.png"
data:image/s3,"s3://crabby-images/eff7b/eff7b983604e6c3928c8f80e2d94f0ef9b7ca88f" alt="../_images/create-webhook1-22.png"
Enter name and choose workspace
data:image/s3,"s3://crabby-images/2ad1e/2ad1ed8c68fff8bd0459d4f01f32f363896f928c" alt="../_images/create-bot22.png"
Set app icon (optional)
data:image/s3,"s3://crabby-images/42616/426167d79a40f9b4bcdfcb5794be93daa9b3e71c" alt="../_images/create-bot32.png"
2. Enable Socket Mode / Socket Mode有効化
Select “Socket Mode” → Turn toggle on
data:image/s3,"s3://crabby-images/15d77/15d7783447bd3f1618a9e4c4546401a723e0da0c" alt="../_images/create-bot42.png"
Enter token name → Click “Generate”
data:image/s3,"s3://crabby-images/edaee/edaee1500450d4a61b1cbc299808025294e1002b" alt="../_images/create-bot52.png"
Get app-level token:
xapp-...
data:image/s3,"s3://crabby-images/858ae/858ae33c2fbecef19fc657cad792106353da23f5" alt="../_images/create-bot62.png"
3. Subscribe bot event / イベント登録
Select “Event Subscriptions” → Turn toggle on
data:image/s3,"s3://crabby-images/aa9f0/aa9f02f5a429ac722ed675256c1c1aca99d3515b" alt="../_images/create-bot3-12.png"
Add “message.channels” to bot events
data:image/s3,"s3://crabby-images/9ad88/9ad8859bd1eba9cd8ebdc2d4a13b743ed63ab77d" alt="../_images/create-bot3-2-12.png"
data:image/s3,"s3://crabby-images/d28d8/d28d8bce87e2d5570f46dd2858869ba98d6c2d15" alt="../_images/create-bot3-2-22.png"
4. Add Bot Token Scopes / スコープ追加
Select “OAuth & Permissions”
data:image/s3,"s3://crabby-images/f32f1/f32f1385393611d272cf73479a7b755853753039" alt="../_images/create-bot72.png"
Click “Add on OAuth Scope”
data:image/s3,"s3://crabby-images/0636a/0636a8870ebc1ecc195a37e2afb19ddaa0cc3e59" alt="../_images/create-bot82.png"
Add “chat:write” to Bot Token Scopes
data:image/s3,"s3://crabby-images/13e1a/13e1ad6567c9ea8f6584d531fcb65d649093187c" alt="../_images/create-bot92.png"
data:image/s3,"s3://crabby-images/c29d7/c29d72c90a9ce78e8ea5d3244bf5b3077d802742" alt="../_images/create-bot102.png"
5. Install App to Workspace / アプリをインストール
Select “Install App” → Click “Install to Workspace”
data:image/s3,"s3://crabby-images/f16ac/f16ace0db73dcd2b0fa27782faef3710c151f868" alt="../_images/create-bot112.png"
Switch OAuth screen → Click “Allow” button
data:image/s3,"s3://crabby-images/d73bc/d73bc8172f46861dcc9ec8240e08e9edfa6662de" alt="../_images/create-bot122.png"
Get Bot Token:
xoxb-...
data:image/s3,"s3://crabby-images/c4023/c40236ca3876550cac6a46d127f2f0138d1314d5" alt="../_images/create-bot132.png"
Invite bot user to channels
botユーザーをチャンネルに招待
data:image/s3,"s3://crabby-images/639fe/639feea03ae4e968b999b5530066366f1bcf2b67" alt="../_images/invite-bot2.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 / App Manifestを取得
Select “App Manifest” menu
data:image/s3,"s3://crabby-images/114bd/114bd340208b08cc0cd1e3e59d210ff3471ed059" alt="../_images/get-app-manifest2.png"
Create new app with App Manifest
App ManifestでSlack appを作成
Select “From an app manifest”
Select workspace → Click “Next”
data:image/s3,"s3://crabby-images/9086a/9086a5e1a4a1dc6c7063de9716a5dc5f27454a3f" alt="../_images/app-manifest12.png"
data:image/s3,"s3://crabby-images/59c7b/59c7be8392d69263fdf91f8fcef8b94e1c9fb319" alt="../_images/app-manifest22.png"
Enter app manifest YAML
data:image/s3,"s3://crabby-images/f6178/f61784e6740c6bb164c5bfaacbed04bcfcb0e198" alt="../_images/app-manifest32.png"
Review app summary → Click “Create”
data:image/s3,"s3://crabby-images/8f49a/8f49a0f4c4bb35e67e485c60cde5354039ca420b" alt="../_images/app-manifest42.png"
data:image/s3,"s3://crabby-images/1e263/1e263a75515610aab19f44fa03061cb1096d97a4" alt="../_images/app-manifest52.png"
data:image/s3,"s3://crabby-images/82f40/82f401e78c17c1244989ecba5e807289cfacdef4" alt="../_images/app-manifest62.png"
Install App to Workspace
data:image/s3,"s3://crabby-images/0e857/0e857379ef2fd08f18186db94aff37499a23f576" alt="../_images/app-manifest72.png"
Generate App-Level Token
data:image/s3,"s3://crabby-images/18a41/18a41131e4459ef89deb7d6f7714f0519f9d188f" alt="../_images/app-manifest82.png"
data:image/s3,"s3://crabby-images/4f239/4f2398131ae91059ff2b532ba5eaac7559bdf2de" alt="../_images/app-manifest92.png"
Short and Reusable !! 🥳
短い し 再利用 できる!!
Create bot with Bolt ⚡️
Bolt を使ってbotを作成
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.10 -m venv env
$ . env/bin/activate
(env) $ pip install slack-bolt
Create a simple bot with Bolt
簡単なbotを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 / 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 ! 🎉
botと 会話 できた!
data:image/s3,"s3://crabby-images/c1601/c1601ff30d8881cf0e1b02eb3bb89f5eb4db7fde" alt="../_images/bot-hi2.png"
Extend bot 🔧
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/3450f/3450f25b549159f96b5ccedf4941082076602bc2" alt="../_images/bot-decolator2.png"
mention / メンション
@app.message("morning")
def handle_morning_message(message, say):
"""reply morning greetings with mention"""
user = message["user"] # get user ID
say(f"Good morning <@{user}>!")
data:image/s3,"s3://crabby-images/89a6a/89a6a4c2975028f7785d53b126abc2f249ead718" alt="../_images/bot-mention2.png"
Use regular expression / 正規表現を使う
import random
import re
@app.message(re.compile(r"choice (.*)"))
def handle_choice(say, context):
"""random choice from words"""
# get matched text from context["matches"]
words = context["matches"][0].split()
say(random.choice(words))
data:image/s3,"s3://crabby-images/34591/34591ceb3e9a947b5b523bed5f2ae39910c02e88" alt="../_images/bot-choice2.png"
Use regular expression / 正規表現を使う
@app.message(re.compile(r"(\d+)\s*(beer|tea)"))
def handle_beer_or_tea(say, context):
"""serve a specified number of beers or teas"""
count = int(context["matches"][0])
beer_or_tea = context["matches"][1]
say(f":{beer_or_tea}:" * count)
data:image/s3,"s3://crabby-images/ae913/ae913a40436d16ea362b8145fd314aee8e27ae78" alt="../_images/bot-beers2.png"
Block Kit support
@app.message("follow me")
def handle_follow_me(message, say):
"""send follow me message with icon"""
blocks = [{
"type": "section",
"text": {
"type": "mrkdwn",
"text": "*Takanori Suzuki*\nFollow me! <https://twitter.com/takanory|@takanory>",
},
"accessory": {
"type": "image", "alt_text": "takanory",
"image_url": "https://pbs.twimg.com/profile_images/192722095/kurokuri_400x400.jpg",
}}]
say(blocks=blocks)
data:image/s3,"s3://crabby-images/567e5/567e54c5bfef502407a371dd1d1f567f3f724d73" alt="../_images/bot-followme2.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/83585/8358506a1338a27debf643bee202b3f4df127baf" alt="../_images/bot-logging2.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:
message posted to public 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/5c074/5c074f72a5a3d9044df8385c7ea5e5cb142ac01c" alt="../_images/bot-private-cannot-view2.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/71687/71687d30aba475d040a3623fee597453f2fca9bf" alt="../_images/add-events-and-scopes12.png"
Add Events and Scopes for private channels
Select “OAuth & Permissions”
groups:history scope is automatically added
data:image/s3,"s3://crabby-images/0ba44/0ba44cd477e4aa8d0a7ceb93632ae0602a0f0111" alt="../_images/add-events-and-scopes22.png"
Add Events and Scopes for private channels
Reinstall app to workspace
data:image/s3,"s3://crabby-images/21fe2/21fe268e384dad3f71681d4cfb6fb17f7397094e" alt="../_images/add-events-and-scopes32.png"
data:image/s3,"s3://crabby-images/c3447/c3447649049318a98a8c758e960288d875fddb00" alt="../_images/add-events-and-scopes42.png"
Add Events and Scopes for private channels
Bot can read/write messages in private channel
data:image/s3,"s3://crabby-images/e2297/e2297360b6ab5d4dfba29d1e9d738be6ea31f7ef" alt="../_images/add-events-and-scopes62.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):
"""send a welcome message to a new member"""
user = event["user"] # get user ID
say(f"Welcome <@{user}>! :tada:")
data:image/s3,"s3://crabby-images/07f1a/07f1a981cae8863e38f1049056890c589b207537" alt="../_images/event-member-joined2.png"
Add Emoji reaction / emojiリアクション
Add reactions:write scope → Reinstall app
@app.message("beer")
def add_beer_emoji(client, message):
"""add :beer: emoji reaciton"""
client.reactions_add(
channel=message["channel"],
timestamp=message["ts"],
name="beer",
)
data:image/s3,"s3://crabby-images/2f8a3/2f8a30ddb2f332b786c9434c3a883c10e207d7f7" alt="../_images/scope-reactions-write2.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 🔢
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/ca1be/ca1becdab57ae545acc176c0b826c3a23ce6ea80" alt="../_images/diagram-sympy2.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):
"""calculator function"""
try:
formula = context["matches"][0]
num = sympify(formula) # Simplifies the formula
# convert into or float
answer = int(num) if num.is_Integer else float(num)
say(f"{answer:,}")
except SympifyError:
pass
Slack as a calculator!! 🎉
Slackが 電卓 になった!!
data:image/s3,"s3://crabby-images/e50bf/e50bf1dd97793669117023fe0270de10b11fd2be" alt="../_images/case-sympy2.png"
Plus-plus feature using Peewee ORM 👍
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/3bba5/3bba57ae2823306159e3f5676ce1fcb1878bfca0" alt="../_images/diagram-peewee2.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
@app.message(re.compile(r"^(\w+)\+\+")) # match "word++" pattern
def plusplus(say, context):
"""increments a counter with a name"""
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/ce591/ce5917bc52d0cf0bed1f1b6b60b4c977e1b24241" alt="../_images/case-peewee2.png"
Search issues with Jira APIs 🔎
Jira API で課題を検索
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/33899/338997fec8c64276a02cfc353a11d8a7f34773b7" alt="../_images/diagram-jira2.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! 🎉
Jira webからの 解放!
data:image/s3,"s3://crabby-images/9b92e/9b92ed78bb8bbfd8fe3e18f01490001a144b6ce4" alt="../_images/bot-jira2.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/022e1/022e1b36a22a2c7823bdfe117b2d191f9d6c99c6" alt="../_images/diagram-template2.png"
Google Authorization is Complex
Googleの認証は複雑
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/e8da0/e8da042eeded7d88150ffe787775dc55dab1de86" alt="../_images/bot-issue-template2.png"
Get Spreadsheet data
スプレッドシートからデータを取得
from google.oauth2.credentials import Credentials
from googleapiclient.discovery import build
SCOPES = ["https://www.googleapis.com/auth/spreadsheets.readonly"]
SHEET = "SHEET_ID..."
@app.message("create issues")
def create_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/22648/226488a477bcfd7ed1991011c60edfa2204ba48a" alt="../_images/bot-sheet12.png"
data:image/s3,"s3://crabby-images/a8409/a84099051dc6d28ec03084299eebd47346025d49" alt="../_images/bot-issue-template21.png"
Create Jira issues / 課題を作成
def create_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/9f02f/9f02f7e62a3913b78c09f78ec3f8e6c7cc03ce7a" alt="../_images/bot-sheet22.png"
data:image/s3,"s3://crabby-images/00555/00555c0f39c3b8a98b9b11f69b21bde62b78c2af" alt="../_images/bot-sheet32.png"
Account management of Google Workspace 👥
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/8049d/8049ddb0575858eb3448a1edfbb6005b742b97d0" alt="../_images/diagram-directory2.png"
Update Google Authorization
Google 認証を更新
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"]
creds = Credentials.from_authorized_user_file("token.json", SCOPES)
service = build("admin", "directory_v1", credentials=creds)
@app.message("user_list")
def user_list(message, say):
"""get Google Workspace users list"""
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/6b032/6b03250a37ba22bb7325d01254122447c1b30cb7" alt="../_images/bot-user-list2.png"
Add user / ユーザー追加
# user_add takanory hogehoge Takanori Suzuki
@app.message(re.compile(r"user_add (\w+) (\w+) (\w+) (\w+)"))
def insert_user(message, say, context):
"""insert Google Workspace user"""
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/48310/4831095db26c65b0a9c9c6f5a43260a46ab4e7b6" alt="../_images/bot-user-add3.png"
data:image/s3,"s3://crabby-images/db537/db537b1584284edef20744188334411e1448802c" alt="../_images/bot-user-add22.png"
I can forget Google Admin! 🎉
Google Adminを 忘れられる!
Security Issue / セキュリティ課題 🔓
Anyone can run it
Run only Slack Admin 🔒
Only Admin can run / 管理者 のみ実行可
Add
users:read
scope, use users.info API
def is_admin(client, user_id):
# get user info: https://api.slack.com/methods/users.info
response = client.users_info(user=user_id)
return response.data["user"]["is_admin"]
@app.message("user_list")
def user_list(message, say, client):
"""get Google Workspace users list"""
if not is_admin(client, message["user"]):
say("You are not an admin!")
return
data:image/s3,"s3://crabby-images/bf4c9/bf4c9c90c67dba9972b0639fdff977e8cacdb274" alt="../_images/bot-not-admin1.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/9cd7e/9cd7e54ba1de08f3056c15e6b3ee0b592178dd6b" alt="../_images/bot-translate2.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):
"""translate text into English"""
text = context["matches"][0]
result = translator.translate_text(
text,
target_lang="EN-US",
)
say(result.text)
Thank you! 🙏
data:image/s3,"s3://crabby-images/9cd7e/9cd7e54ba1de08f3056c15e6b3ee0b592178dd6b" alt="../_images/bot-translate2.png"