使用模型工厂的 MySQL 锁定等待超时

Posted

技术标签:

【中文标题】使用模型工厂的 MySQL 锁定等待超时【英文标题】:MySQL Lock Wait Timeout using Model Factories 【发布时间】:2016-07-24 01:00:39 【问题描述】:

我在测试类的方法中生成了一个非常小的数据集,我曾尝试在 setUp() 方法中生成数据,但它导致每个测试的锁定等待超时。将此代码从setUp() 方法中移出并将其放入它自己的方法中会有所帮助。这意味着并非每个测试都抱怨锁。

我正在使用 Laravel 5.2 中的 DatabaseTransactions 特征,以便在运行每个测试用例之前重置数据库。

在我拥有的每个测试用例中,我都会调用以获取这样的模拟数据。

$data = self::getRandomCommunityWithAssociatedData();

实际方法只为创建的社区生成少量社区对象和用户对象。

public static function getRandomCommunityWithAssociatedData()

    self::$communities = factory(\Community::class, self::COMMUNITIES_TO_CREATE)->create()->each(function ($community) 
        self::$users[$community->id] = factory(User::class, self::USERS_TO_CREATE)->create()->each(function (\User $user) use ($community) 
            $user->community()->associate($community);
        );

        self::$admins[$community->id] = factory(User::class, 'superadmin', self::ADMINS_TO_CREATE)->create()->each(function (\User $admin) use ($community) 
            $admin->community()->associate($community);
        );
    );

    $community = self::$communities[mt_rand(0, count(self::$communities) - 1)];

    return ['community' => $community, 'users' => self::$users[$community->id], 'admins' => self::$admins[$community->id]];

该方法中使用了一些常量,它们用于确定要创建的每个对象的数量。目前我正在为每个社区实例创建 2 个社区和 3 个用户和 2 个管理员。

锁等待超时是不可预测的,一次运行它可能发生在第一个测试用例上,另一次运行它可能发生在第五个测试用例上。

我尝试将 mysql 等待锁定的时间增加到 500 秒,但我仍然遇到超时。增加这个时间确实不是一种选择,因为测试需要能够在所有环境中运行。

关于在 Laravel 5.2 中使用 DatabaseTransactions trait 和这么小的数据集时为什么我可能会遇到这些锁定等待超时的任何想法?

1) UserEmailNotificationsTest::testActiveAdminReceivesNewCommentEmailNotification
Illuminate\Database\QueryException: SQLSTATE[HY000]: General error: 1205 Lock wait timeout exceeded; try restarting transaction (SQL: insert into `communities` (`

上面执行的查询是对社区表的简单插入,在这个数据生成过程中没有复杂的子查询或类似的东西。

启用查询日志,结果。

       30 Query START TRANSACTION
       30 Query SAVEPOINT trans2
       30 Prepare   insert into `communities` (`viddler_id`, `domain`, `subdomain`, `active`, `max_seats`, `created_at`, `updated_at`, `assignment_seats`, `general_settings`, `access_settings`, `branding_settings`) values (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
       30 Execute   insert into `communities` (`viddler_id`, `domain`, `subdomain`, `active`, `max_seats`, `created_at`, `updated_at`, `assignment_seats`, `general_settings`, `access_settings`, `branding_settings`) values ('74', 'gaylord.com', 'qui', '1', '24', '2016-03-30 16:25:45', '2016-04-04 02:27:04', '23', 'a:11:s:5:\"title\";s:0:\"\";s:11:\"description\";s:0:\"\";s:10:\"meta_title\";s:0:\"\";s:16:\"meta_description\";s:0:\"\";s:13:\"meta_keywords\";s:0:\"\";s:17:\"thumbnail_display\";s:0:\"\";s:18:\"group_nomenclature\";s:8:\"Channels\";s:13:\"home_template\";s:0:\"\";s:11:\"api_version\";s:1:\"2\";s:8:\"language\";s:2:\"en\";s:21:\"use_nested_navigation\";i:0;', 'a:10:s:12:\"restrictions\";s:10:\"Restricted\";s:17:\"auto_registration\";i:0;s:14:\"default_active\";i:0;s:12:\"oauth_google\";i:0;s:14:\"oauth_facebook\";i:0;s:14:\"oauth_linkedin\";i:0;s:16:\"oauth_reg_google\";i:0;s:18:\"oauth_reg_facebook\";i:0;s:18:\"oauth_reg_linkedin\";i:0;s:11:\"lti_enabled\";i:0;', 'a:14:s:13:\"contact_email\";s:0:\"\";s:17:\"contact_link_text\";s:0:\"\";s:9:\"logo_file\";s:0:\"\";s:14:\"carousel_items\";s:0:\"\";s:11:\"html_header\";s:0:\"\";s:16:\"footer_copyright\";s:45:\"© 2015 Viddler Inc. All Rights Reserved.\";s:19:\"footer_privacy_link\";s:37:\"http://www.viddler.com/privacy-policy\";s:17:\"footer_terms_link\";s:35:\"http://www.viddler.com/terms-of-use\";s:14:\"help_link_text\";s:4:\"Help\";s:9:\"help_link\";s:0:\"\";s:7:\"color_1\";s:7:\"#ffffff\";s:7:\"color_2\";s:7:\"#2C333C\";s:7:\"color_3\";s:7:\"#2C333C\";s:7:\"color_4\";s:7:\"#60a1d7\";')
160404  9:30:25    30 Close stmt
       21 Query ROLLBACK
       21 Quit
       22 Query ROLLBACK
       22 Quit
       23 Query ROLLBACK
       23 Quit
       24 Query ROLLBACK
       24 Quit
       25 Query ROLLBACK
       25 Quit
       26 Query ROLLBACK
       26 Quit
       27 Query ROLLBACK
       27 Quit
       28 Query ROLLBACK
       28 Quit
       29 Query ROLLBACK
       29 Quit
       30 Query ROLLBACK
       30 Quit

【问题讨论】:

启用查询日志并查看导致死锁的查询。您应该将超时设置回正常水平。 插入社区表失败。我已经更新了上面的问题以显示来自 mysql 查询日志的数据。 虽然没有回答您的问题,但我建议仅使用 sqlite 进行测试。它将显着加快测试的运行时间。 我认为这通常是由针对同一个表运行的多个查询引起的。您可以在每次通话时结束交易吗?如果您登录数据库并在测试运行时调用show open tables where in_use>0;,会发生什么情况?如果您终止测试过程会发生什么? +1 @Mysteryos 对 sqlite 的评论。您可以设置您的 phpunit 以使用 sqlite 进行测试并存储在内存中,从而大大提高性能。看看vimeo.com/151390908 【参考方案1】:

因此,增加锁定等待超时不是最佳做法; the best practice is to catch the error and recover。

正式来说,这是你应该做的:

死锁和锁等待超时在繁忙的服务器上都是正常的,并且 应用程序必须意识到它们可能会发生,并且 通过重试来处理它们。您可以通过这样做来降低它们的可能性 在第一次更改数据之间尽可能少的工作 事务和提交,所以锁的持有时间最短 可能的时间和尽可能少的行数。 有时 在不同事务之间拆分工作可能是实用的,并且 有帮助。

【讨论】:

以上是关于使用模型工厂的 MySQL 锁定等待超时的主要内容,如果未能解决你的问题,请参考以下文章

保存/更新模型时,Django“超过锁定等待超时;尝试重新启动事务”

使用 C# 在 Mysql 上死锁 - “超过锁定等待超时;尝试重新启动事务”

为啥在尝试更新 MySQL 条目时出现“超过锁定等待超时”错误?

MySQL 错误:超过锁定等待超时;尝试重新启动事务查询

超过锁定等待超时;尝试重新启动事务 MYSQL Python

如何避免锁定等待超时超出异常。?