Discord.py 中避免 Presence 更新触发速率限制的正确实践

本文详解如何在 discord.py 中安全轮换 bot 活动状态(presence),规避因 `change_presence` 调用过于频繁导致的 websocket 速率限制警告(429 rate limited),重点修正 `asyncio.sleep` 同步误用、补充错误重试机制,并提供健壮、可长期运行的轮播方案。

在 Discord.py 中,通过 bot.change_presence() 动态更新 Bot 的在线状态(如“正在观看…”)是一种常见需求。但许多开发者会遇到如下警告:

WARNING  discord.gateway WebSocket in shard ID None is ratelimited, waiting 57.3 seconds

该警告表明:你的 Bot 已被 Discord 网关限流——并非因为 5 分钟间隔太短,而是因为代码中误用了 asyncio.sleep() 的同步形式,导致循环未真正暂停,从而在极短时间内重复发起请求,瞬间触达速率上限(Discord 对 /users/@me/settings 类操作有严格限频,通常为 5 次/5 分钟)

? 根本问题定位

原始代码中这一行是致命错误:

asyncio.sleep(300)  # ❌ 错误!这是协程对象,未 await,不产生实际延迟

它仅创建了一个 asyncio.sleep 协程对象,却未 await 执行,因此 while True 循环几乎零延迟地反复调用 change_presence(),等效于“疯狂刷请求”,远超 Discord 的容忍阈值。

✅ 正确写法必须是:

await asyncio.sleep(300)  # ✅ 真正挂起协程,让出控制权

✅ 推荐实现:带错误恢复的稳健轮播

以下是生产环境推荐的完整方案,已整合异常捕获与自动退避:

import asyncio
import random
import discord
from discord.ext import commands

actaray = ["PcktWtchr's Videos", "Cams", "and Listening Always", "or Listening or Both"]

@bot.event
async def on_ready():
    print(f'Logged in as {bot.user}')

    # 使用后台任务(推荐)或 while 循环均可,此处保持简洁
    while True:
        try:
            activity = discord.Activity(
                type=discord.ActivityType.watching,
                name=random.choice(actaray)
            )
            await bot.change_presence(activity=activity)
            await asyncio.sleep(300)  # ✅ 正确 await,精确 5 分钟间隔
        except discord.HTTPException as e:
            if e.status == 429:  # 遇到限流
                retry_after = float(e.response.headers.get("Retry-After", "5"))
                print(f"Rate limited! Retrying after {retry_after:.1f}s...")
                await asyncio.sleep(retry_

after + 1) # 加 1 秒缓冲,避免边界重试 else: print(f"HTTP error during presence update: {e}") await asyncio.sleep(10) # 其他 HTTP 错误,降频重试 except Exception as e: print(f"Unexpected error in presence loop: {e}") await asyncio.sleep(60) # 可选:注册一个独立任务(更优雅,便于管理) @bot.event async def on_connect(): if not hasattr(bot, '_presence_task') or bot._presence_task.done(): bot._presence_task = bot.loop.create_task(_presence_rotator()) async def _presence_rotator(): while True: await _update_random_presence() await asyncio.sleep(300) async def _update_random_presence(): try: await bot.change_presence( activity=discord.Activity( type=discord.ActivityType.watching, name=random.choice(actaray) ) ) except discord.HTTPException as e: if e.status == 429: retry_after = float(e.response.headers.get("Retry-After", "5")) await asyncio.sleep(retry_after + 1)

⚠️ 关键注意事项

  • 不要滥用 change_presence:Discord 明确限制用户级设置类操作(含状态更新)为 5 次/5 分钟/每个用户(Bot 账户即用户)。即使你设为 300s,若前序请求因网络延迟、重试失败而堆积,仍可能触发限流。
  • 永远 await 异步函数:asyncio.sleep()、bot.change_presence() 均为协程,必须 await,否则逻辑失效。
  • on_error 事件不可靠:Discord.py 的 on_error 并非总能捕获所有 HTTPException(尤其在 on_ready 内部抛出时),因此强烈建议在业务逻辑内直接 try/except,如上例所示。
  • 考虑使用 Activity 缓存或去重:若 actaray 列表较短,连续两次选中相同文案虽无害,但影响体验。可加入简单去重逻辑:
    last_name = None
    # 在循环内:
    name = random.choice([a for a in actaray if a != last_name] or actaray)
    last_name = name

✅ 总结

解决 Discord.py Presence 限流问题的核心在于:
修正 await asyncio.sleep() 用法,确保真实延时;
在 change_presence 调用周围包裹 try/except,主动处理 429 并遵循 Retry-After 头;
避免在 on_ready 中裸写无限循环,优先采用任务化(create_task)方式提升可维护性与可观测性。

遵循以上实践,你的 Bot 将稳定、安静地轮播状态,再也不会被网关警告“轰炸”。