技術筆記-用 python 叫 discord 發訊息的幾種方式

更新於 發佈於 閱讀時間約 22 分鐘

「通知」這種功能現在已經非常稀鬆平常了,三不五時手機跳出訊息或聲音,不勝其擾!但不要因噎廢食,對於「真正重要的事件」,即時通知還是非常必需的。像是刷卡通知,應該沒有人會關掉吧,這種服務早期還需要付費的呢!所以對於一個交易系統而言,牽涉到金錢的進出,通知功能是一定要的啦。有 line notify 相伴的年代,輕鬆愉快:

import resuests
def notify(msg):
url = "https://notify-api.line.me/api/notify"
token = "xxx"
headers = {"Authorization": f"Bearer {token}"}
data = {"message": msg }
res = requests.post(url, headers = headers, data = data)
if res.status_code in (200, 204):
print(f"Request fulfilled with response: {res.text}")
else:
print(f"Request failed with response: {res.status_code}-{res.text}")

以上幾行程式碼,簡單發出 http request 就完成了。但輕鬆的時代即將結束, line notify 於 2015 Q1 就要結束服務了!心想別怕,好好找找新玩具,也許送走一個輕鬆的時代,將迎來一個「更輕鬆」的全新境界,此時 discord 吸引了我的目光。


印象中的 discord,已經在遊戲玩家,幣圈,厲害的開發者社群等等之中,已經火紅很久了。研究了一天之後,發現此平台太強大,發訊息達成通知功能實在太容易了,簡直是殺雞用牛刀!但一開始不知道,走了迂迴的道路,在 multi-thread,async await 森林中迷失,但也學習到相關的寶貴知識,以下娓娓道來。

方法一:webhook

這是最簡單的方式,在建立一個自己的伺服器之後,按個鍵「產生 webhook」,就完成了。真的很絕,真的沒有額外動作,用上述呼叫 line notify 一樣的方式,甚至更簡單,因為連認證的 token 都不需要,直接完成!

raw-image
raw-image
raw-image

discord 非常慷慨,每個人都可以建立自己的「伺服器」,建立後會有預設的文字頻道,按齒輪圖示「編輯頻道」,依上述截圖點一點,即刻生成 webhook 網址,複製之貼入以下程式:

def notify_discord_webhook(msg):
url = 'https://discord.com/api/webhooks/131743762990xxxyyy'
headers = {"Content-Type": "application/json"}
data = {"content": msg, "username": "newmanBot"}
res = requests.post(url, headers = headers, json = data)
if res.status_code in (200, 204):
print(f"Request fulfilled with response: {res.text}")
else:
print(f"Request failed with response: {res.status_code}-{res.text}")

這段程式可以在系統任何地方呼叫,可即時送出訊息到此頻道,保持手機 discord app 允許通知,系統的重要事件不會漏接。若只需要通知功能,這樣已經足夠,下文也可以略過了。但強大的 discord 只玩這樣未免可惜,還是建個 bot (聊天機器人) 來玩玩吧,這就需要進一步了解 api。

方法二:原生 api

上述從 webhook 發出的訊息,並不具備身份識別,所以發完訊息後完全沒有後續互動的可能性,也無法發出「私訊」,但這些才是 discord 的主要功能。要體驗這些功能,必須先到 developer portal 建立一個 application,這是一切開發的起點。進到所建立的 application,首先取得第一項重要資訊:token,注意只會出現一次,複製起來以備程式使用:

raw-image

有了 token,相當於這個 bot 的身分證,接下來是授權給這個 bot 做事的權限,如允許他讀取群組人員上線狀態,訊息內容等等,先把下面三個勾打開:

raw-image

然後還要針對某個頻道授權,discord 的設計相當貼心,他可以幫我們產生一串 url,到另一個瀏覽頁面打開,打開時需要登入認證,然後就可選擇授權自己旗下有權限的頻道,給 bot 運作施展的空間:

raw-image

從以上畫面選擇 bot permission,自己用的當然是給足權限 Administrator,捲到下面複製 url,貼到新的瀏覽器頁面:

raw-image

到此終於完成了,而 token 就是鑰匙,將這所有的授權內容鎖住,程式只要持有 token 就可以解鎖這些資源了。因此一樣簡單的程式,差別只是帶著 token:

def notify_discord_channel(msg, token, channel_id):
url_base = 'https://discord.com/api/v10'
url = f"{url_base}/channels/{channel_id}/messages"
headers = {
"Authorization": f"Bot {token}",
"Content-Type": "application/json"
}
data = {
"content": msg,
}
res = requests.post(url, headers = headers, json = data)
if res.status_code in (200, 204):
print(f"Request fulfilled with response: {res.text}")
else:
print(f"Request failed with response: {res.status_code}-{res.text}")

以上 channel id,只要在 discord 的「使用者設定」的「進階」中,打開「開發者模式」,之後在頻道中按右鍵就會出現「複製頻道 ID」,可以取得。

raw-image

實測之後,有沒有發現訊息的圖示不一樣了,是一個「有身份的人」發出的訊息:

raw-image

以上已經完成 bot 的初體驗。好像沒有比 webhook 強到哪裡去?的確是的。都走到這一步了,就再多走一點吧,bot 能不能發「私訊」?可以的。不由得再度讚賞 discord 的寬容大度:

# 發訊息給「個人」,先用另一個 api 取得對該 user「動態產生的 channel id」
# 再呼叫傳送到 channel 的 function 即可​
def notify_discord_user(msg, token, user_id):
url_base = 'https://discord.com/api/v10'
url = f"{url_base}/users/@me/channels"
headers = {
"Authorization": f"Bot {token}",
"Content-Type": "application/json"
}
data = {
"content": msg,
"recipient_id": user_id
}
res = requests.post(url, headers = headers, json = data)
if res.status_code == 200:
dm_channel = res.json()
channel_id = dm_channel["id"]
CrawlService.notify_discord_channel(msg, token, channel_id)
else:
print(f"Request failed with response: {res.status_code}-{res.text}")

像阿甘跑步一樣,都跑這麼遠了,再多跑一點吧!一個 bot 應該要可以回訊息吧?當然要的,也必須可以由訊息驅動任何程式動作。這些功能若用原生 api 覺得有點吃力,但有網路大神,已經把 api 包起來了,成為 python 套件了,太強大,太感謝。

方法三:python.py

此套件用標準 pip 安裝即可: pip install python.py

坑#1

先預告一個坑,若你遇到「No module named 'audioop'.」,代表你的 python 版本太新了,這在 python@3.13 是一個 known issue,暫時未解,只好降版到 python@3.12。

坑#2

然後再預告一個坑,初始化連線就會遇到的:

Cannot connect to host discord.com:443 ssl:True \
[SSLCertVerificationError: (1, '[SSL: CERTIFICATE_VERIFY_FAILED] \
certificate verify failed: unable to get local issuer certificate \
(_ssl.c:1000)')]

這個錯誤非常隱晦,竟然有點似曾相識,咦!就是在測試富果 api 建立 socket 連線時遇到的,本來以為是憑證檔的問題,後來在社群大哥的提點下,終於搞清楚,是 ssl 底層基本設定出問題了,解法如下:

# 先執行以下程式,得到 openssl_capath
import ssl
result = ssl.get_default_verify_paths()
print(result)
-------------
DefaultVerifyPaths(cafile=None, capath=None, openssl_cafile_env='SSL_CERT_FILE', \
openssl_cafile='/Library/Frameworks/Python.framework/Versions/3.12/etc/openssl/cert.pem', \
openssl_capath_env='SSL_CERT_DIR', \
openssl_capath='/Library/Frameworks/Python.framework/Versions/3.12/etc/openssl/certs')

在 mac 中用 finder 到上述 openssl_capath 看,發現空空的,應該要有東西,我們需要複製一個憑證檔進去,檔案來源也非常隱晦,請到自己的 python 環境中,尋找 certifi 套件位置,其中有一個憑證檔:

raw-image

把憑證檔 copy 進去上述的 openssl path 中,記得還要改檔名,如我的 case 甚至是 certs 沒有副檔名!真是詭異。無論如何,這樣搞一搞就可以了,這樣只到 import discord 成功而已,才剛開始要寫程式呢!哈哈,還有路要走。

正式開始 coding

先寫一個最小版本的 bot,能讀訊息和回訊息,一個獨立的 py 檔如下:

import discord
from datetime import datetime

intents = discord.Intents.default()
intents.message_content = True
intents.messages = True
intents.guilds = True

bot = discord.Client(intents=intents)

DISCORD_TOKEN = 'MTMxNjg5NTE4MjE0NjM3xxx'

@bot.event
async def on_ready():
print(f'Logged in as {bot.user.name}')
guild = bot.get_guild(GUILD_ID)
if not guild:
print('Guild not found')
return
channel = guild.get_channel(CHANNEL_ID)
if not channel:
print('Channel not found')
return
await channel.send('Hello from server')

@bot.event
async def on_message(message):
print(f'{datetime.now()} {message.author}: {message.content}')
text = message.content
if text.startswith('$'):
channel = message.channel
await channel.send(f'Thanks for sending {text[1:]}')

bot.run(DISCORD_TOKEN)

此 bot 能讀訊息,若發現前贅字是 $,則回應 $ 之後的字元。

raw-image

這段程式完成了非常多任務,包括與伺服器建立 socket 連線,持續監聽頻道,on_message 事件處理器,在接到訊息之後可以串連驅動任何後端的程式模組,進行任何高深的計算,且回應任何訊息!其實已經非常強大了,大可以停在這裡,好好去開發應用了。又是因為手癢的原因,想要把他整合到 ui,這又讓我遇到了 multi-thread 的問題!技術細節再論下去實在太繁瑣,就紀錄下最終的結果吧。


以下的 class 可以由 ui 的主程式呼叫,建立 bot 的監聽訊息服務,也可隨時呼叫傳送訊息。只是經過前面的探討,傳送訊息只要額外呼叫 http request 就可完成了,
大可不必像這裏硬要呼叫套件的 send 功能,導致需要再度加開 thread 去處理,實在是疊床架屋的做法。但既然做了,就記下來欣賞一下,當作對 thread async await 的學習。

class DiscordService():
def __init__(self):
self.TOKEN = 'MTMxNjg5NTE4MjE0NjM3MTxxx'
self.GUILD_ID = 00000
self.CHANNEL_ID = 00000
self.ADMIN_USER_ID = 00000 # newman.church
self.loop = asyncio.new_event_loop()

# creating bot
self.intents = discord.Intents.default()
self.intents.message_content = True
self.intents.messages = True
self.intents.guilds = True
self.client = discord.Client(intents=self.intents)

# event handling
self.client.event(self.on_ready)
self.client.event(self.on_message)

# bot will run in seperate thread
self.bot_thread = threading.Thread(target=self.do_trigger_run_bot
, daemon=True)
self.bot_thread.start()

def send_message_channel(self, msg):
# 簡單的 http request
CrawlService.notify_discord_channel(msg, self.TOKEN, self.CHANNEL_ID)

def send_message_user(self, msg):
# 簡單的 http request
CrawlService.notify_discord_user(msg, self.TOKEN, self.ADMIN_USER_ID)

async def on_ready(self):
print(f'{datetime.now()} Logged on as {self.client.user}!')

async def on_message(self, message):
print(f'{datetime.now()} {message.author}: {message.content}')

async def trigger_send_message(self, message):
guild = self.client.get_guild(self.GUILD_ID)
if not guild:
print(f"Guild with ID {self.GUILD_ID} not found.")
return
channel = guild.get_channel(self.CHANNEL_ID)
if not channel:
print(f"Channel with ID {self.CHANNEL_ID} not found.")
return
# 這是套件的傳訊息功能
await channel.send(message)

​# 呼叫套件的傳訊息功能,必須再開立新的 thread 處理
# 因為是 async function,呼叫方式必須借用 asyncio 幫忙用安全方式呼叫
def send_message_foolish(self, message):
asyncio.run_coroutine_threadsafe(self.trigger_send_message(message), self.loop)

# 這已經轉太多彎了,暈了
def do_trigger_run_bot(self):
asyncio.set_event_loop(self.loop)
self.loop.create_task(self.trigger_run_bot())
self.loop.run_forever()

async def trigger_run_bot(self):
await self.client.start(self.TOKEN) # start is the shorthand coroutine for login() + connect().

# def run_bot(self):
# self.client.run(self.TOKEN) # run is a blocking call

以上終於完整紀錄一天以來發生的事了。達成目的只需第一段,只是既然走過就留下痕跡吧,學到經驗很值得。

Newman 2024/12/16

導覽頁:紐曼的技術筆記-索引








avatar-img
22會員
106內容數
漫步是一種境界。
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
newman的沙龍 的其他內容
如題,這應該是很容易犯的錯誤,據說 git 能使時光倒流!它就是這麼厲害,所以光是刪除檔案重新 commit 是沒用的,必須執行以下步驟: # 1. 將檔案移出納管範圍,但注意加 --cached 就不會實體刪除檔案
這一趟越走越興奮的旅程,從 測試富邦新一代 api 開始,很自然的開始 用 python 造出自己的 gui 程式,本來也不想學太多有的沒的,頂多三五分鐘用 api 監控一次市場行情,就可以玩出很多策略了,沒想到 api 一個小小的不完美,迫使我把手伸進 socket 領域了。 原來我想監控的
開始玩 python gui,首先稍微交代一下背景,這種 client-server 的應用架構,一直以來是我所唾棄的方式,因為:每個前端都需要安裝,所以若要部署給多人使用,很麻煩,每個人的電腦環境千奇百怪,難保平台與各種軟體均能相容。所以出問題時原因難查,若軟體是銷售的,也容易有爭
雖然公眾資料庫和網路爬蟲,已經可以玩得不亦樂乎了,但所有研究最終的目的,還是要實施於一個真實賬戶,真金白銀的實際效益才有意義。尤其是「即時」性要求較高的策略,一定需要用程式補上「自動化下單」的部分,快速變動的資料如「行情報價」,「訂單狀態」,「帳戶庫存」等等,也需要程式嚴密監控,這就需要券商提供
「最佳化」是很酷的觀念,因為現實世界中許多問題,並沒有嚴謹一致的公式解,但可以利用計算機高速運算能力,透過巧妙的演算法,迭代式反覆逼近最佳解,應用領域非常廣。若能多瞭解一點原理,一定可以提昇解決問題的能力。今天從網路上發現一堂手把手的教學課程,就來演練一下整個過程。期望徹底了解之後,後面可以
現有的家是 azure,功能多又有免費額度,算是堪用,只是因為雲端產業蓬勃發展,想體驗一下新東西,所以這次來玩玩新東西 fly.io。從開啟以下賞心悅目的官網,第一次接觸到成功佈署上線,大概半小時就搞定了,體驗不錯,以下稍做紀錄。
如題,這應該是很容易犯的錯誤,據說 git 能使時光倒流!它就是這麼厲害,所以光是刪除檔案重新 commit 是沒用的,必須執行以下步驟: # 1. 將檔案移出納管範圍,但注意加 --cached 就不會實體刪除檔案
這一趟越走越興奮的旅程,從 測試富邦新一代 api 開始,很自然的開始 用 python 造出自己的 gui 程式,本來也不想學太多有的沒的,頂多三五分鐘用 api 監控一次市場行情,就可以玩出很多策略了,沒想到 api 一個小小的不完美,迫使我把手伸進 socket 領域了。 原來我想監控的
開始玩 python gui,首先稍微交代一下背景,這種 client-server 的應用架構,一直以來是我所唾棄的方式,因為:每個前端都需要安裝,所以若要部署給多人使用,很麻煩,每個人的電腦環境千奇百怪,難保平台與各種軟體均能相容。所以出問題時原因難查,若軟體是銷售的,也容易有爭
雖然公眾資料庫和網路爬蟲,已經可以玩得不亦樂乎了,但所有研究最終的目的,還是要實施於一個真實賬戶,真金白銀的實際效益才有意義。尤其是「即時」性要求較高的策略,一定需要用程式補上「自動化下單」的部分,快速變動的資料如「行情報價」,「訂單狀態」,「帳戶庫存」等等,也需要程式嚴密監控,這就需要券商提供
「最佳化」是很酷的觀念,因為現實世界中許多問題,並沒有嚴謹一致的公式解,但可以利用計算機高速運算能力,透過巧妙的演算法,迭代式反覆逼近最佳解,應用領域非常廣。若能多瞭解一點原理,一定可以提昇解決問題的能力。今天從網路上發現一堂手把手的教學課程,就來演練一下整個過程。期望徹底了解之後,後面可以
現有的家是 azure,功能多又有免費額度,算是堪用,只是因為雲端產業蓬勃發展,想體驗一下新東西,所以這次來玩玩新東西 fly.io。從開啟以下賞心悅目的官網,第一次接觸到成功佈署上線,大概半小時就搞定了,體驗不錯,以下稍做紀錄。
你可能也想看
Google News 追蹤
Thumbnail
最近國泰世華CUBE App推出的「美股定期定額」功能,讓使用者可以方便地進行跨境理財(但讀者仍需根據自身需求審慎考量),除了享有美股定期定額的新功能,也同時享有台股定期定額的功能,可以一站滿足我們理財的需求! 透過國泰世華CUBE App線上開台股證券戶+複委託戶,流程最快僅需要5分鐘。
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
好的,很高興能為您服務!街口支付是台灣相當普及的行動支付平台,提供方便快速的付款、轉帳等服務。 街口支付的主要功能與特色: * 付款: * 掃碼付款: 在店家出示的QR code上掃描,即可完成付款。 * 出示付款碼: 店家掃描您手機上的付款碼,同樣能完成交易。 * 轉帳:
Thumbnail
《聊天吧》是一款專為視訊交友設計的APP,為了幫助用戶更方便的購買鑽石,提供了多種儲值方式。無論是使用信用卡、超商代碼儲值、虛擬帳戶轉帳還是Apple Pay,都能享受方便快捷的服務。此外,本文還提供了儲值教學,讓用戶可以快速瞭解如何購買優惠的鑽石。欲瞭解更多詳情,歡迎至《聊天吧》官網。
Thumbnail
Line Pay 是一個行動支付平台,像微信裡面的WeChat Pay一樣,用樣地在綁定信用卡後,透過QR Code,甚至NFC的方法向商店結帳,現時可於現於台灣、泰國、印尼及日本使用,可惜香港暫時未有得用,筆者都非常喜愛Line Brown、Conty等的公仔。 LINE Corporation
Thumbnail
只要以手指快速輕輕點擊螢幕,就能獲取免費代幣,成為人們新註冊telegram最大誘因。
Thumbnail
PDing新推出的應用程式讓創作者和粉絲之間可以透過文字、語音/視頻通話進行更親密的交流,為創作者提供了一個新收益模式。知名網紅可以在APP內建立自己的頻道,並透過簡單的認證過程實現高額收益。
Thumbnail
ChatyN 最新推出的積分獎勵系統,為用戶帶來無盡樂趣和共享獎勵。 用戶提供真實社交互動的鏈上數據,並可透過每日簽到活動獲得積分獎勵,與 ChatyN 共同打造 Web3 共享經濟生態。
Thumbnail
最近國泰世華CUBE App推出的「美股定期定額」功能,讓使用者可以方便地進行跨境理財(但讀者仍需根據自身需求審慎考量),除了享有美股定期定額的新功能,也同時享有台股定期定額的功能,可以一站滿足我們理財的需求! 透過國泰世華CUBE App線上開台股證券戶+複委託戶,流程最快僅需要5分鐘。
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
好的,很高興能為您服務!街口支付是台灣相當普及的行動支付平台,提供方便快速的付款、轉帳等服務。 街口支付的主要功能與特色: * 付款: * 掃碼付款: 在店家出示的QR code上掃描,即可完成付款。 * 出示付款碼: 店家掃描您手機上的付款碼,同樣能完成交易。 * 轉帳:
Thumbnail
《聊天吧》是一款專為視訊交友設計的APP,為了幫助用戶更方便的購買鑽石,提供了多種儲值方式。無論是使用信用卡、超商代碼儲值、虛擬帳戶轉帳還是Apple Pay,都能享受方便快捷的服務。此外,本文還提供了儲值教學,讓用戶可以快速瞭解如何購買優惠的鑽石。欲瞭解更多詳情,歡迎至《聊天吧》官網。
Thumbnail
Line Pay 是一個行動支付平台,像微信裡面的WeChat Pay一樣,用樣地在綁定信用卡後,透過QR Code,甚至NFC的方法向商店結帳,現時可於現於台灣、泰國、印尼及日本使用,可惜香港暫時未有得用,筆者都非常喜愛Line Brown、Conty等的公仔。 LINE Corporation
Thumbnail
只要以手指快速輕輕點擊螢幕,就能獲取免費代幣,成為人們新註冊telegram最大誘因。
Thumbnail
PDing新推出的應用程式讓創作者和粉絲之間可以透過文字、語音/視頻通話進行更親密的交流,為創作者提供了一個新收益模式。知名網紅可以在APP內建立自己的頻道,並透過簡單的認證過程實現高額收益。
Thumbnail
ChatyN 最新推出的積分獎勵系統,為用戶帶來無盡樂趣和共享獎勵。 用戶提供真實社交互動的鏈上數據,並可透過每日簽到活動獲得積分獎勵,與 ChatyN 共同打造 Web3 共享經濟生態。