adf8a40108cc95adb76d0403c7923f3d91bcc762
[lhc/web/wiklou.git] / tests / phpunit / includes / db / LBFactoryTest.php
1 <?php
2 /**
3 * Holds tests for LBFactory abstract MediaWiki class.
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * http://www.gnu.org/copyleft/gpl.html
19 *
20 * @group Database
21 * @file
22 * @author Antoine Musso
23 * @copyright © 2013 Antoine Musso
24 * @copyright © 2013 Wikimedia Foundation Inc.
25 */
26 class LBFactoryTest extends MediaWikiTestCase {
27
28 /**
29 * @dataProvider getLBFactoryClassProvider
30 */
31 public function testGetLBFactoryClass( $expected, $deprecated ) {
32 $mockDB = $this->getMockBuilder( 'DatabaseMysql' )
33 ->disableOriginalConstructor()
34 ->getMock();
35
36 $config = [
37 'class' => $deprecated,
38 'connection' => $mockDB,
39 # Various other parameters required:
40 'sectionsByDB' => [],
41 'sectionLoads' => [],
42 'serverTemplate' => [],
43 ];
44
45 $this->hideDeprecated( '$wgLBFactoryConf must be updated. See RELEASE-NOTES for details' );
46 $result = LBFactoryMW::getLBFactoryClass( $config );
47
48 $this->assertEquals( $expected, $result );
49 }
50
51 public function getLBFactoryClassProvider() {
52 return [
53 # Format: new class, old class
54 [ 'LBFactorySimple', 'LBFactory_Simple' ],
55 [ 'LBFactorySingle', 'LBFactory_Single' ],
56 [ 'LBFactoryMulti', 'LBFactory_Multi' ],
57 ];
58 }
59
60 public function testLBFactorySimpleServer() {
61 global $wgDBserver, $wgDBname, $wgDBuser, $wgDBpassword, $wgDBtype;
62
63 $servers = [
64 [
65 'host' => $wgDBserver,
66 'dbname' => $wgDBname,
67 'user' => $wgDBuser,
68 'password' => $wgDBpassword,
69 'type' => $wgDBtype,
70 'load' => 0,
71 'flags' => DBO_TRX // REPEATABLE-READ for consistency
72 ],
73 ];
74
75 $factory = new LBFactorySimple( [ 'servers' => $servers ] );
76 $lb = $factory->getMainLB();
77
78 $dbw = $lb->getConnection( DB_MASTER );
79 $this->assertTrue( $dbw->getLBInfo( 'master' ), 'master shows as master' );
80
81 $dbr = $lb->getConnection( DB_SLAVE );
82 $this->assertTrue( $dbr->getLBInfo( 'master' ), 'DB_SLAVE also gets the master' );
83
84 $factory->shutdown();
85 $lb->closeAll();
86 }
87
88 public function testLBFactorySimpleServers() {
89 global $wgDBserver, $wgDBname, $wgDBuser, $wgDBpassword, $wgDBtype;
90
91 $servers = [
92 [ // master
93 'host' => $wgDBserver,
94 'dbname' => $wgDBname,
95 'user' => $wgDBuser,
96 'password' => $wgDBpassword,
97 'type' => $wgDBtype,
98 'load' => 0,
99 'flags' => DBO_TRX // REPEATABLE-READ for consistency
100 ],
101 [ // emulated slave
102 'host' => $wgDBserver,
103 'dbname' => $wgDBname,
104 'user' => $wgDBuser,
105 'password' => $wgDBpassword,
106 'type' => $wgDBtype,
107 'load' => 100,
108 'flags' => DBO_TRX // REPEATABLE-READ for consistency
109 ]
110 ];
111
112 $factory = new LBFactorySimple( [
113 'servers' => $servers,
114 'loadMonitorClass' => 'LoadMonitorNull'
115 ] );
116 $lb = $factory->getMainLB();
117
118 $dbw = $lb->getConnection( DB_MASTER );
119 $this->assertTrue( $dbw->getLBInfo( 'master' ), 'master shows as master' );
120 $this->assertEquals(
121 $wgDBserver, $dbw->getLBInfo( 'clusterMasterHost' ), 'cluster master set' );
122
123 $dbr = $lb->getConnection( DB_SLAVE );
124 $this->assertTrue( $dbr->getLBInfo( 'replica' ), 'slave shows as slave' );
125 $this->assertEquals(
126 $wgDBserver, $dbr->getLBInfo( 'clusterMasterHost' ), 'cluster master set' );
127
128 $factory->shutdown();
129 $lb->closeAll();
130 }
131
132 public function testLBFactoryMulti() {
133 global $wgDBserver, $wgDBname, $wgDBuser, $wgDBpassword, $wgDBtype;
134
135 $factory = new LBFactoryMulti( [
136 'sectionsByDB' => [],
137 'sectionLoads' => [
138 'DEFAULT' => [
139 'test-db1' => 0,
140 'test-db2' => 100,
141 ],
142 ],
143 'serverTemplate' => [
144 'dbname' => $wgDBname,
145 'user' => $wgDBuser,
146 'password' => $wgDBpassword,
147 'type' => $wgDBtype,
148 'flags' => DBO_DEFAULT
149 ],
150 'hostsByName' => [
151 'test-db1' => $wgDBserver,
152 'test-db2' => $wgDBserver
153 ],
154 'loadMonitorClass' => 'LoadMonitorNull'
155 ] );
156 $lb = $factory->getMainLB();
157
158 $dbw = $lb->getConnection( DB_MASTER );
159 $this->assertTrue( $dbw->getLBInfo( 'master' ), 'master shows as master' );
160
161 $dbr = $lb->getConnection( DB_SLAVE );
162 $this->assertTrue( $dbr->getLBInfo( 'replica' ), 'slave shows as slave' );
163
164 $factory->shutdown();
165 $lb->closeAll();
166 }
167
168 public function testChronologyProtector() {
169 // (a) First HTTP request
170 $mPos = new MySQLMasterPos( 'db1034-bin.000976', '843431247' );
171
172 $now = microtime( true );
173 $mockDB = $this->getMockBuilder( 'DatabaseMysql' )
174 ->disableOriginalConstructor()
175 ->getMock();
176 $mockDB->method( 'writesOrCallbacksPending' )->willReturn( true );
177 $mockDB->method( 'lastDoneWrites' )->willReturn( $now );
178 $mockDB->method( 'getMasterPos' )->willReturn( $mPos );
179
180 $lb = $this->getMockBuilder( 'LoadBalancer' )
181 ->disableOriginalConstructor()
182 ->getMock();
183 $lb->method( 'getConnection' )->willReturn( $mockDB );
184 $lb->method( 'getServerCount' )->willReturn( 2 );
185 $lb->method( 'parentInfo' )->willReturn( [ 'id' => "main-DEFAULT" ] );
186 $lb->method( 'getAnyOpenConnection' )->willReturn( $mockDB );
187 $lb->method( 'hasOrMadeRecentMasterChanges' )->will( $this->returnCallback(
188 function () use ( $mockDB ) {
189 $p = 0;
190 $p |= call_user_func( [ $mockDB, 'writesOrCallbacksPending' ] );
191 $p |= call_user_func( [ $mockDB, 'lastDoneWrites' ] );
192
193 return (bool)$p;
194 }
195 ) );
196 $lb->method( 'getMasterPos' )->willReturn( $mPos );
197
198 $bag = new HashBagOStuff();
199 $cp = new ChronologyProtector(
200 $bag,
201 [
202 'ip' => '127.0.0.1',
203 'agent' => "Totally-Not-FireFox"
204 ]
205 );
206
207 $mockDB->expects( $this->exactly( 2 ) )->method( 'writesOrCallbacksPending' );
208 $mockDB->expects( $this->exactly( 2 ) )->method( 'lastDoneWrites' );
209
210 // Nothing to wait for
211 $cp->initLB( $lb );
212 // Record in stash
213 $cp->shutdownLB( $lb );
214 $cp->shutdown();
215
216 // (b) Second HTTP request
217 $cp = new ChronologyProtector(
218 $bag,
219 [
220 'ip' => '127.0.0.1',
221 'agent' => "Totally-Not-FireFox"
222 ]
223 );
224
225 $lb->expects( $this->once() )
226 ->method( 'waitFor' )->with( $this->equalTo( $mPos ) );
227
228 // Wait
229 $cp->initLB( $lb );
230 // Record in stash
231 $cp->shutdownLB( $lb );
232 $cp->shutdown();
233 }
234
235 private function newLBFactoryMulti( array $baseOverride = [], array $serverOverride = [] ) {
236 global $wgDBserver, $wgDBuser, $wgDBpassword, $wgDBname, $wgDBtype;
237
238 return new LBFactoryMulti( $baseOverride + [
239 'sectionsByDB' => [],
240 'sectionLoads' => [
241 'DEFAULT' => [
242 'test-db1' => 1,
243 ],
244 ],
245 'serverTemplate' => $serverOverride + [
246 'dbname' => $wgDBname,
247 'user' => $wgDBuser,
248 'password' => $wgDBpassword,
249 'type' => $wgDBtype,
250 'flags' => DBO_DEFAULT
251 ],
252 'hostsByName' => [
253 'test-db1' => $wgDBserver,
254 ],
255 'loadMonitorClass' => 'LoadMonitorNull',
256 'localDomain' => wfWikiID()
257 ] );
258 }
259
260 public function testNiceDomains() {
261 global $wgDBname;
262
263 $factory = $this->newLBFactoryMulti();
264 $lb = $factory->getMainLB();
265
266 $db = $lb->getConnectionRef( DB_MASTER );
267 $this->assertEquals(
268 $wgDBname,
269 $db->getDomainID()
270 );
271 unset( $db );
272
273 /** @var DatabaseBase $db */
274 $db = $lb->getConnection( DB_MASTER, [], '' );
275
276 $this->assertEquals(
277 '',
278 $db->getDomainID()
279 );
280
281 $this->assertEquals(
282 $db->addIdentifierQuotes( 'page' ),
283 $db->tableName( 'page' ),
284 "Correct full table name"
285 );
286
287 $this->assertEquals(
288 $db->addIdentifierQuotes( $wgDBname ) . '.' . $db->addIdentifierQuotes( 'page' ),
289 $db->tableName( "$wgDBname.page" ),
290 "Correct full table name"
291 );
292
293 $this->assertEquals(
294 $db->addIdentifierQuotes( 'nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
295 $db->tableName( 'nice_db.page' ),
296 "Correct full table name"
297 );
298
299 $factory->setDomainPrefix( 'my_' );
300 $this->assertEquals(
301 '',
302 $db->getDomainID()
303 );
304 $this->assertEquals(
305 $db->addIdentifierQuotes( 'my_page' ),
306 $db->tableName( 'page' ),
307 "Correct full table name"
308 );
309 $this->assertEquals(
310 $db->addIdentifierQuotes( 'other_nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
311 $db->tableName( 'other_nice_db.page' ),
312 "Correct full table name"
313 );
314
315 $factory->closeAll();
316 $factory->destroy();
317 }
318
319 public function testTrickyDomain() {
320 $dbname = 'unittest-domain';
321 $factory = $this->newLBFactoryMulti(
322 [ 'localDomain' => $dbname ], [ 'dbname' => $dbname ] );
323 $lb = $factory->getMainLB();
324 /** @var DatabaseBase $db */
325 $db = $lb->getConnection( DB_MASTER, [], '' );
326
327 $this->assertEquals(
328 '',
329 $db->getDomainID()
330 );
331
332 $this->assertEquals(
333 $db->addIdentifierQuotes( 'page' ),
334 $db->tableName( 'page' ),
335 "Correct full table name"
336 );
337
338 $this->assertEquals(
339 $db->addIdentifierQuotes( $dbname ) . '.' . $db->addIdentifierQuotes( 'page' ),
340 $db->tableName( "$dbname.page" ),
341 "Correct full table name"
342 );
343
344 $this->assertEquals(
345 $db->addIdentifierQuotes( 'nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
346 $db->tableName( 'nice_db.page' ),
347 "Correct full table name"
348 );
349
350 $factory->setDomainPrefix( 'my_' );
351
352 $this->assertEquals(
353 $db->addIdentifierQuotes( 'my_page' ),
354 $db->tableName( 'page' ),
355 "Correct full table name"
356 );
357 $this->assertEquals(
358 $db->addIdentifierQuotes( 'other_nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
359 $db->tableName( 'other_nice_db.page' ),
360 "Correct full table name"
361 );
362
363 \MediaWiki\suppressWarnings();
364 $this->assertFalse( $db->selectDB( 'garbage-db' ) );
365 \MediaWiki\restoreWarnings();
366
367 $this->assertEquals(
368 $db->addIdentifierQuotes( 'garbage-db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
369 $db->tableName( 'garbage-db.page' ),
370 "Correct full table name"
371 );
372
373 $factory->closeAll();
374 $factory->destroy();
375 }
376 }