当开启general log后,该文件会记录用户的所有操作,包括每条SQL语句的执行细节。当访问量大或者长时间不清理general log文件时,会占用大量的存储空间,导致存储空间耗尽。本文为您列举general log相关的常见问题和解决方法。
背景信息
基于以下原因,RDS MySQL选择TABLE作为general log的默认存储格式:
保存为FILE格式用户无法进行查询,因为用户无法直接访问RDS MySQL的文件。
general log和slow log同时受log_output参数影响,RDS MySQL在采集slow log时使用了rotate的机制,需要保存为TABLE格式。
General log占据大量存储空间
问题描述
RDS MySQL实例存储空间已满,通过如下排查,确认为general log文件过大导致。
查看实例存储空间使用量,sys_data_size文件过大。详情请参见查看监控信息。
查看实例参数,实例已开启general_log(运行参数值为ON)。详情请参见查看实例参数。
连接RDS MySQL实例并执行如下语句,查询发现general log文件过大。连接实例的详细请参见连接RDS MySQL实例。
SELECT table_schema AS '数据库', table_name,SUM(data_length + index_length + data_free)/1024/1024 AS "表大小MB",SUM(DATA_FREE)/1024/1024 AS "碎片大小MB" FROM information_schema.TABLES WHERE table_name='general_log'
说明此SQL语句会从
information_schema
数据库的TABLES
表中检索mysql.general_log
表的数据,然后将其转换成以MB为单位的大小。此SQL语句获得的数据为抽样数据,和实际数据存在一定误差。
问题原因
当RDS MySQL开启了general log后,该文件记录了用户的所有操作,包括每条SQL语句的执行细节,无论是查询、插入、更新还是删除操作。当访问量大或者长时间不清理general log文件时,会占用大量的存储空间,导致存储空间耗尽。
General log导致性能问题
问题描述
连接数上升,CPU使用率升高。通过SHOW PROCESSLIST
或者查看innodb_trx表等方式,可见大量连接处于Waiting for table level lock状态。
问题原因
RDS MySQL选择TABLE作为general log的默认存储格式,各线程写general log是串行写入的。这是因为写general log除了需要加MDL锁以外,还需要加表锁。也正是该表锁导致了“Waiting for table level lock”状态的出现。
General log导致RTO变长
问题描述
实例崩溃恢复时间变长,在此期间实例处于无法连接状态。
问题原因
实例非正常Shutdown,general log的Crash标记位会置为true,导致实例重启后进入到自动恢复的逻辑,当表很大时,恢复时间很长,此过程中实例无法连接。
解决方法
清理general log文件
关闭general log(运行参数值设为OFF),以防止产生新的日志。详情请参见设置实例参数。
使用高权限账号连接RDS MySQL实例并执行如下语句,清理general log文件。连接实例的详细信息请参见连接RDS MySQL实例。
说明RDS MySQL 5.6版本实例不支持通过TRUNCATE命令清理general log文件。
TRUNCATE TABLE mysql.general_log;
后续维护
建议只在调试或跟踪问题时临时开启general log,使用完成之后请及时关闭general log。
建议您开启SQL洞察和审计,该功能可以自动记录和分析来自数据库内核的SQL语句,以及SQL语句的执行账号、IP地址、执行详情等信息,对实例性能没有影响。并且SQL洞察和审计数据存储在数据库自治服务DAS中,不占用RDS MySQL存储空间。
扩容实例存储空间,详情请参见变更配置。您也可以开启存储空间自动扩容,在实例存储空间达到设定的阈值时,系统会自动扩容存储空间,详情请参见设置存储空间自动扩容。