forked from moodle/moodle
-
Notifications
You must be signed in to change notification settings - Fork 0
/
lock_test.php
143 lines (120 loc) · 5.5 KB
/
lock_test.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
<?php
// This file is part of Moodle - http://moodle.org/
//
// Moodle is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Moodle is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
/**
* lock unit tests
*
* @package core
* @category test
* @copyright 2013 Damyon Wiese
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
defined('MOODLE_INTERNAL') || die();
/**
* Unit tests for our locking implementations.
*
* @package core
* @category test
* @copyright 2013 Damyon Wiese
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
class lock_testcase extends advanced_testcase {
/**
* Some lock types will store data in the database.
*/
protected function setUp(): void {
$this->resetAfterTest(true);
}
/**
* Run a suite of tests on a lock factory class.
*
* @param class $lockfactoryclass - A lock factory class to test
*/
protected function run_on_lock_factory($lockfactoryclass) {
$modassignfactory = new $lockfactoryclass('mod_assign');
$tooltaskfactory = new $lockfactoryclass('tool_task');
// Test for lock clashes between lock stores.
$assignlock = $modassignfactory->get_lock('abc', 0);
$this->assertNotEmpty($assignlock, 'Get a lock "abc" from store "mod_assign"');
$tasklock = $tooltaskfactory->get_lock('abc', 0);
$this->assertNotEmpty($tasklock, 'Get a lock "abc" from store "tool_task"');
$assignlock->release();
$tasklock->release();
$lockfactory = new $lockfactoryclass('default');
if ($lockfactory->is_available()) {
// This should work.
$lock1 = $lockfactory->get_lock('abc', 2);
$this->assertNotEmpty($lock1, 'Get a lock');
if ($lockfactory->supports_timeout()) {
// Attempt to obtain a lock within a 2 sec timeout.
$durationlock2 = -microtime(true);
$lock2 = $lockfactory->get_lock('abc', 2);
$durationlock2 += microtime(true);
if (!$lock2) { // If the lock was not obtained.
$this->assertFalse($lock2, 'Cannot get a stacked lock');
// This should timeout after 2 seconds.
$this->assertTrue($durationlock2 < 2.5, 'Lock should timeout after no more than 2 seconds');
} else {
$this->assertNotEmpty($lock2, 'Get a stacked lock');
$this->assertTrue($lock2->release(), 'Release a stacked lock');
}
// Attempt to obtain a lock within a 0 sec timeout.
$durationlock2 = -microtime(true);
$lock2 = $lockfactory->get_lock('abc', 0);
$durationlock2 += microtime(true);
if (!$lock2) { // If the lock was not obtained.
// This should timeout almost instantly.
$this->assertTrue($durationlock2 < 0.100, 'Lock should timeout almost instantly < 100ms');
} else {
// This stacked lock should be gained almost instantly.
$this->assertTrue($durationlock2 < 0.100, 'Lock should be gained almost instantly');
$lock2->release();
// We should also assert that locks fail instantly if locked
// from another process but this is hard to unit test.
}
}
// Release the lock.
$this->assertTrue($lock1->release(), 'Release a lock');
// Get it again.
$lock3 = $lockfactory->get_lock('abc', 2);
$this->assertNotEmpty($lock3, 'Get a lock again');
// Release the lock again.
$this->assertTrue($lock3->release(), 'Release a lock again');
// Release the lock again (shouldn't hurt).
$this->assertFalse($lock3->release(), 'Release a lock that is not held');
if (!$lockfactory->supports_auto_release()) {
// Test that a lock can be claimed after the timeout period.
$lock4 = $lockfactory->get_lock('abc', 2, 2);
$this->assertNotEmpty($lock4, 'Get a lock');
sleep(3);
$lock5 = $lockfactory->get_lock('abc', 2, 2);
$this->assertNotEmpty($lock5, 'Get another lock after a timeout');
$this->assertTrue($lock5->release(), 'Release the lock');
$this->assertTrue($lock4->release(), 'Release the lock');
}
}
}
/**
* Tests the testable lock factories classes.
* @return void
*/
public function test_locks() {
// Run the suite on the current configured default (may be non-core).
$this->run_on_lock_factory(\core\lock\lock_config::get_lock_factory_class());
// Manually create the core no-configuration factories.
$this->run_on_lock_factory(\core\lock\db_record_lock_factory::class);
$this->run_on_lock_factory(\core\lock\file_lock_factory::class);
}
}