我在heroku上运行的discord.py机器人不断停止

时间:2020-08-06 21:34:22

标签: python heroku discord.py

我在heroku上托管了一个非常基本的discord.py机器人。它的唯一功能是每24小时从可能的消息列表中发送一条消息。它总是发送第一个消息,然后停止。我在代码中找不到任何错误,当我降低计时并测试它在我的计算机上运行还是在heroku中运行时,都可以正常工作。

全部在一个python文件上,其中有一些heroku所需的文件和两个用于消息的文本文档。

以下是主要脚本:

import discord
import time
import random as rng

clnt = discord.Client()
ch = 0 #channel id
cnt = 0 #amount of minutes on timer
lp = True #i think this is just a leftover variable from another version, i can't find anywhere i used it


@clnt.event
async def on_ready():
    print('ready')


async def ph(): #function to send message
    global ch
    global lp

    qts = open('quotes.txt') #get messages
    qtz = qts.read().splitlines() #put messages in a list
    qts.close() #close file
    
    #if message file is empty, get the list from a backup file and put them into the first file, reseting the messages
    if not qtz:  
        qts2 = open('quoteslog.txt') 
        qtz2 = qts2.read().splitlines()
        qts2.close()

        with open('quotes.txt', 'w') as f:
            for i in qtz2:
                f.write("%s\n" % i)
            f.close()

        qts = open('quotes.txt')
        qtz = qts.read().splitlines()
        qts.close()

    #get random message from list
    x = rng.randint(1, len(qtz))
    x2 = x - 1
    y = qtz[x2]
    qtz.pop(x2)

    open('quotes.txt', 'w').close() #clear the list

    #rewrite the same file without the message sent
    with open('quotes.txt', 'w') as f:
        for i in qtz:
            f.write("%s\n" % i)
        f.close()
    
    #used for messages with new lines
    if y == 'ph1':
        await ch.send("this is for one of the messages, it has new lines so it can't be re-inserted into a txt file")
        await timer()

    elif y == 'ph2':
        await ch.send('same here')
        await timer()

    else:
        #send message to channel and restart the timer
        await ch.send(y)
        await timer()


@clnt.event
async def on_message(m):
    if m.author == clnt.user:
        return

    global ch

    if m.content.startswith('send here'):
        ch = clnt.get_channel(m.channel.id)
        await m.channel.send('ok')

    elif m.content.startswith('start'):
        await timer()


async def timer():  #loops every 60 seconds, 1440 times, or 24hrs
    global lp
    while lp:
        global cnt
        time.sleep(60)
        cnt += 1
        if cnt == 1440:
            cnt = 0 #reset timer and send message
            await ph() 


clnt.run('the discord bot id')

是的,我知道代码可能是垃圾格式化方面的,但是据我所知它应该可以工作,但事实并非如此。我什至不确定这是否是代码错误,也可能是heroku问题,但我不知道。

如果有人有任何可能的帮助,将不胜感激!

1 个答案:

答案 0 :(得分:3)

我建议您使用BotCog类,它们会更加高效,它们在discord.py中提供,并且具有装饰器来定义循环函数 >。它们在模块的discord.ext.commands部分中。您可以这样做:

from discord.ext import commands, tasks

我刚刚在另一篇文章中回答了一个齿轮和循环功能的例子,您会发现它here。这是适合您情况的相同结构:

# LoopCog.py
from discord.ext import commands, tasks
import random

class LoopCog(commands.Cog):
    def __init__(self, bot):
        self.bot = bot
        self.test_loop.change_interval(minutes = self.bot.cnt)

    @commands.Cog.listener("on_ready")
    async def on_ready(self):
        self.test_loop.start()

    @tasks.loop(hours=24)
    async def test_loop(self):
        # insert your ph function contents here
        # access the channel id via self.bot.ch

def setup(bot):
    bot.add_cog(LoopCog(bot))
# main.py
from discord.ext import commands

bot = commands.Bot(command_prefix = "!")
bot.ch = 0 
bot.cnt = 0
bot.load_extension("LoopCog")

@bot.event
async def on_ready():
    print("ready")

bot.run(token, reconnect = True)

我建议您去看一些教程。将Client类用于Bot不是正确的方法,如果继续使用Bot,则必须对Client中已经存在的所有内容进行编程。 / p>

您将找到hereBotCog的API文档tasks

关于 heroku 不幸的是,它将每隔24小时重启一次机器人,除非您选择付费的专业报价,其中包括24/7运行时。如果程序在30分钟内未收到任何请求,也会使您的程序进入睡眠模式。