让不和谐的机器人每 10 秒改变一次播放状态

Making a discord bot change playing status every 10 seconds

我正在尝试让测试 discord 机器人的状态每十秒在两条消息之间切换一次。我需要在状态消息更改时执行脚本的其余部分,但每当我尝试使其工作时,错误就会不断弹出。我的脚本中有线程,但我不完全确定在这种情况下如何使用它。

@test_bot.event
async def on_ready():
    print('Logged in as')
    print(test_bot.user.name)
    print(test_bot.user.id)
    print('------')
    await change_playing()


@test_bot.event
async def change_playing():
    threading.Timer(10, change_playing).start()
    await test_bot.change_presence(game=discord.Game(name='Currently on ' + str(len(test_bot.servers)) +
                                                          ' servers'))
    threading.Timer(10, change_playing).start()
    await test_bot.change_presence(game=discord.Game(name='Say test.help'))

错误信息如下:

C:\Python\Python36-32\lib\threading.py:1182: RuntimeWarning: coroutine 'change_playing' was never awaited
  self.function(*self.args, **self.kwargs)

看到这个:

https://github.com/Rapptz/discord.py/blob/master/examples/background_task.py

import discord
import asyncio
client = discord.Client()
async def my_background_task():
    await client.wait_until_ready()
    counter = 0
    channel = discord.Object(id='channel_id_here')
    while not client.is_closed:
        counter += 1
        await client.send_message(channel, counter)
        await asyncio.sleep(60) # task runs every 60 seconds
@client.event
async def on_ready():
    print('Logged in as')
    print(client.user.name)
    print(client.user.id)
    print('------')
client.loop.create_task(my_background_task())
client.run('token')

Discord.py 自带后台任务功能,

不幸的是,线程和 asyncio 不能很好地结合在一起。您需要跳过额外的环节才能等待线程内的协程。最简单的解决方案就是不使用线程。

您要做的是等待一段时间,然后 运行 一个协程。这可以通过后台任务完成 (example)

async def status_task():
    while True:
        await test_bot.change_presence(...)
        await asyncio.sleep(10)
        await test_bot.change_presence(...)
        await asyncio.sleep(10)

@test_bot.event
async def on_ready():
    ...
    bot.loop.create_task(status_task())

您不能使用 time.sleep(),因为这会阻止机器人的执行。 asyncio.sleep() 虽然和其他所有东西一样都是协程,因此是非阻塞的。

最后,@client.event 装饰器只能用于机器人识别为 events 的函数。比如on_ready和on_message.

version 1.1.0 introduced discord.ext.tasks,旨在简化您所描述的后台任务,并在出现连接问题时处理重新连接到 discord 的潜在复杂逻辑。

这是您使用 tasks 的任务示例:

from discord.ext import commands, tasks
from commands import Bot
from tasks import loop
from asyncio import sleep

bot = Bot("!")

@loop(seconds=10)
async def name_change():
    await bot.change_presence(...)
    await sleep(10)
    await bot.change_presence(...)

name_change.before_loop(bot.wait_until_ready())    
name_change.start()
bot.run("TOKEN")