aed2d83a7d7f5b35f1e39bd5af2da0c865e0b954
[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 = MWLBFactory::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 Database $db */
274 $db = $lb->getConnection( DB_MASTER, [], '' );
275 $lb->reuseConnection( $db ); // don't care
276
277 $this->assertEquals(
278 '',
279 $db->getDomainID()
280 );
281
282 $this->assertEquals(
283 $db->addIdentifierQuotes( 'page' ),
284 $db->tableName( 'page' ),
285 "Correct full table name"
286 );
287
288 $this->assertEquals(
289 $db->addIdentifierQuotes( $wgDBname ) . '.' . $db->addIdentifierQuotes( 'page' ),
290 $db->tableName( "$wgDBname.page" ),
291 "Correct full table name"
292 );
293
294 $this->assertEquals(
295 $db->addIdentifierQuotes( 'nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
296 $db->tableName( 'nice_db.page' ),
297 "Correct full table name"
298 );
299
300 $factory->setDomainPrefix( 'my_' );
301 $this->assertEquals(
302 '',
303 $db->getDomainID()
304 );
305 $this->assertEquals(
306 $db->addIdentifierQuotes( 'my_page' ),
307 $db->tableName( 'page' ),
308 "Correct full table name"
309 );
310 $this->assertEquals(
311 $db->addIdentifierQuotes( 'other_nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
312 $db->tableName( 'other_nice_db.page' ),
313 "Correct full table name"
314 );
315
316 $factory->closeAll();
317 $factory->destroy();
318 }
319
320 public function testTrickyDomain() {
321 $dbname = 'unittest-domain';
322 $factory = $this->newLBFactoryMulti(
323 [ 'localDomain' => $dbname ], [ 'dbname' => $dbname ] );
324 $lb = $factory->getMainLB();
325 /** @var Database $db */
326 $db = $lb->getConnection( DB_MASTER, [], '' );
327 $lb->reuseConnection( $db ); // don't care
328
329 $this->assertEquals(
330 '',
331 $db->getDomainID()
332 );
333
334 $this->assertEquals(
335 $db->addIdentifierQuotes( 'page' ),
336 $db->tableName( 'page' ),
337 "Correct full table name"
338 );
339
340 $this->assertEquals(
341 $db->addIdentifierQuotes( $dbname ) . '.' . $db->addIdentifierQuotes( 'page' ),
342 $db->tableName( "$dbname.page" ),
343 "Correct full table name"
344 );
345
346 $this->assertEquals(
347 $db->addIdentifierQuotes( 'nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
348 $db->tableName( 'nice_db.page' ),
349 "Correct full table name"
350 );
351
352 $factory->setDomainPrefix( 'my_' );
353
354 $this->assertEquals(
355 $db->addIdentifierQuotes( 'my_page' ),
356 $db->tableName( 'page' ),
357 "Correct full table name"
358 );
359 $this->assertEquals(
360 $db->addIdentifierQuotes( 'other_nice_db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
361 $db->tableName( 'other_nice_db.page' ),
362 "Correct full table name"
363 );
364
365 \MediaWiki\suppressWarnings();
366 $this->assertFalse( $db->selectDB( 'garbage-db' ) );
367 \MediaWiki\restoreWarnings();
368
369 $this->assertEquals(
370 $db->addIdentifierQuotes( 'garbage-db' ) . '.' . $db->addIdentifierQuotes( 'page' ),
371 $db->tableName( 'garbage-db.page' ),
372 "Correct full table name"
373 );
374
375 $factory->closeAll();
376 $factory->destroy();
377 }
378 }