寻找PyMySQL错误的来源-err.InterfaceError(“(0,'')”)

时间:2019-05-27 04:45:35

标签: digital-ocean discord.py peewee pymysql

我是Discord机器人开发人员,最近完成了一项订单。客户端在服务器上安装应用程序时最初没有问题,但是据他们说,在运行“大约三个小时”之后,程序开始吐出特定的堆栈跟踪错误,并且不再接受命令。

该机器人使用Discord.py构建,并使用Peewee作为ORM,并使用PyMySQL作为数据库驱动程序。客户端在其上运行的服务器由DigitalOcean托管,如果需要有关硬件等的任何信息,则客户端可以应要求向我提供该信息。我们已经尝试卸载并重新安装所有依赖项,并尝试对其进行不同的分发,但是错误仍然存​​在。

这是客户端收到的确切跟踪:

  File "/usr/local/lib/python3.6/dist-packages/peewee.py", line 2666, in __exit__
    reraise(new_type, new_type(*exc_args), traceback)
  File "/usr/local/lib/python3.6/dist-packages/peewee.py", line 179, in reraise
    raise value.with_traceback(tb)
  File "/usr/local/lib/python3.6/dist-packages/peewee.py", line 2875, in execute_sql
    cursor.execute(sql, params or ())
  File "/usr/local/lib/python3.6/dist-packages/pymysql/cursors.py", line 170, in execute
    result = self._query(query)
  File "/usr/local/lib/python3.6/dist-packages/pymysql/cursors.py", line 328, in _query
    conn.query(q)
  File "/usr/local/lib/python3.6/dist-packages/pymysql/connections.py", line 516, in query
    self._execute_command(COMMAND.COM_QUERY, sql)
  File "/usr/local/lib/python3.6/dist-packages/pymysql/connections.py", line 750, in _execute_command
    raise err.InterfaceError("(0, '')")
peewee.InterfaceError: (0, '')

我的database.py文件中打开数据库连接的相关部分:

import discord
from peewee import *

from config import db_host, db_port, mysql_db_name, \
    mysql_db_username, mysql_db_password

db_connection = MySQLDatabase(
    mysql_db_name,
    user = mysql_db_username,
    password = mysql_db_password,
    host = db_host,
    port = db_port
)

def create_db_tables():
    # create_tables() does safe creation by default, and will simply not create
    # table if it already exists
    db_connection.create_tables([User])

我bot.py文件中的相关部分,特别是首次打开bot时运行的bot启动功能,以及创建和启动bot客户端的行:

client = discord.Client()

async def bot_startup():
    # Check for config changes
    if client.user.name != config.bot_username:
        print("Username Updated To: {}".format(config.bot_username))
        await client.edit_profile(username=config.bot_username)

    # Start 'playing' message
    await client.change_presence(
        game=discord.Game( name=config.playing_message )
    )

    # Prepare database
    database.create_db_tables()
    print("Database Connected")

    print("Connected Successfully")

# ...

@client.event
async def on_ready():
    await bot_startup()

# ...

client.run(config.token)

根据客户端,重新启动bot可以暂时解决该问题,并且可以正常运行几个小时,然后再次出现错误。一旦错误开始,僵尸程序将不再响应任何传入的命令,如果引发了足够的错误,则会完全崩溃。

此错误的典型原因是什么,应采取什么步骤解决引起此错误的原因?

1 个答案:

答案 0 :(得分:0)

错误出现的时间让我想起了我在与数据库通信时也遇到的一个问题。

问题可能在于,由于连接在机器人启动时打开(甚至在程序开始执行时)并且偶尔使用,数据库可能会关闭连接< /strong>,因此任何进一步的执行都会导致错误。为了解决这个问题,我为我的数据库类的所有方法

创建了这个装饰器
def ensures_connected(f):
    def wrapper(*args):
        args[0].connection.ping(reconnect=True, attempts=3, delay=2)
        return f(*args)

    return wrapper

置于任何必须与数据库通信的方法或函数之上。

args[0].connection.ping(reconnect=True, attempts=3, delay=2) 表示我们将在 self.connection 上调用方法 ping(因为它是调用时传递给方法的第一个参数),它允许 {{3} } 如果连接断开。

self.connection 在我的代码中是一个由方法调用 reconnect 返回的对象,应该等同于你的对象 db_connection

这个装饰器应该像下面的例子一样放在方法定义之上:


def ensures_connected(f):
    def wrapper(*args):
        args[0].connection.ping(reconnect=True, attempts=3, delay=2)
        return f(*args)

    return wrapper

class Database:
    def __init__(self):
        self.connection = mysql.connector.connect(
            user=self.username, password=password, host=self.host, database=self.database)

    # ...

    @ensures_connected
    def is_member_registered(self, guild: Guild, member: Member):
        # ...
        return