Merge "Declare visibility on class properties of ImageGalleryBase"
[lhc/web/wiklou.git] / tests / phpunit / includes / RevisionStorageTest.php
1 <?php
2
3 /**
4 * Test class for Revision storage.
5 *
6 * @group ContentHandler
7 * @group Database
8 * ^--- important, causes temporary tables to be used instead of the real database
9 *
10 * @group medium
11 * ^--- important, causes tests not to fail with timeout
12 */
13 class RevisionStorageTest extends MediaWikiTestCase {
14
15 /**
16 * @var WikiPage $the_page
17 */
18 var $the_page;
19
20 function __construct( $name = null, array $data = array(), $dataName = '' ) {
21 parent::__construct( $name, $data, $dataName );
22
23 $this->tablesUsed = array_merge( $this->tablesUsed,
24 array( 'page',
25 'revision',
26 'text',
27
28 'recentchanges',
29 'logging',
30
31 'page_props',
32 'pagelinks',
33 'categorylinks',
34 'langlinks',
35 'externallinks',
36 'imagelinks',
37 'templatelinks',
38 'iwlinks' ) );
39 }
40
41 protected function setUp() {
42 global $wgExtraNamespaces, $wgNamespaceContentModels, $wgContentHandlers, $wgContLang;
43
44 parent::setUp();
45
46 $wgExtraNamespaces[12312] = 'Dummy';
47 $wgExtraNamespaces[12313] = 'Dummy_talk';
48
49 $wgNamespaceContentModels[12312] = 'DUMMY';
50 $wgContentHandlers['DUMMY'] = 'DummyContentHandlerForTesting';
51
52 MWNamespace::getCanonicalNamespaces( true ); # reset namespace cache
53 $wgContLang->resetNamespaces(); # reset namespace cache
54 if ( !$this->the_page ) {
55 $this->the_page = $this->createPage( 'RevisionStorageTest_the_page', "just a dummy page", CONTENT_MODEL_WIKITEXT );
56 }
57 }
58
59 protected function tearDown() {
60 global $wgExtraNamespaces, $wgNamespaceContentModels, $wgContentHandlers, $wgContLang;
61
62 parent::tearDown();
63
64 unset( $wgExtraNamespaces[12312] );
65 unset( $wgExtraNamespaces[12313] );
66
67 unset( $wgNamespaceContentModels[12312] );
68 unset( $wgContentHandlers['DUMMY'] );
69
70 MWNamespace::getCanonicalNamespaces( true ); # reset namespace cache
71 $wgContLang->resetNamespaces(); # reset namespace cache
72 }
73
74 protected function makeRevision( $props = null ) {
75 if ( $props === null ) {
76 $props = array();
77 }
78
79 if ( !isset( $props['content'] ) && !isset( $props['text'] ) ) {
80 $props['text'] = 'Lorem Ipsum';
81 }
82
83 if ( !isset( $props['comment'] ) ) {
84 $props['comment'] = 'just a test';
85 }
86
87 if ( !isset( $props['page'] ) ) {
88 $props['page'] = $this->the_page->getId();
89 }
90
91 $rev = new Revision( $props );
92
93 $dbw = wfgetDB( DB_MASTER );
94 $rev->insertOn( $dbw );
95
96 return $rev;
97 }
98
99 protected function createPage( $page, $text, $model = null ) {
100 if ( is_string( $page ) ) {
101 if ( !preg_match( '/:/', $page ) &&
102 ( $model === null || $model === CONTENT_MODEL_WIKITEXT )
103 ) {
104 $ns = $this->getDefaultWikitextNS();
105 $page = MWNamespace::getCanonicalName( $ns ) . ':' . $page;
106 }
107
108 $page = Title::newFromText( $page );
109 }
110
111 if ( $page instanceof Title ) {
112 $page = new WikiPage( $page );
113 }
114
115 if ( $page->exists() ) {
116 $page->doDeleteArticle( "done" );
117 }
118
119 $content = ContentHandler::makeContent( $text, $page->getTitle(), $model );
120 $page->doEditContent( $content, "testing", EDIT_NEW );
121
122 return $page;
123 }
124
125 protected function assertRevEquals( Revision $orig, Revision $rev = null ) {
126 $this->assertNotNull( $rev, 'missing revision' );
127
128 $this->assertEquals( $orig->getId(), $rev->getId() );
129 $this->assertEquals( $orig->getPage(), $rev->getPage() );
130 $this->assertEquals( $orig->getTimestamp(), $rev->getTimestamp() );
131 $this->assertEquals( $orig->getUser(), $rev->getUser() );
132 $this->assertEquals( $orig->getContentModel(), $rev->getContentModel() );
133 $this->assertEquals( $orig->getContentFormat(), $rev->getContentFormat() );
134 $this->assertEquals( $orig->getSha1(), $rev->getSha1() );
135 }
136
137 /**
138 * @covers Revision::__construct
139 */
140 public function testConstructFromRow() {
141 $orig = $this->makeRevision();
142
143 $dbr = wfgetDB( DB_SLAVE );
144 $res = $dbr->select( 'revision', '*', array( 'rev_id' => $orig->getId() ) );
145 $this->assertTrue( is_object( $res ), 'query failed' );
146
147 $row = $res->fetchObject();
148 $res->free();
149
150 $rev = new Revision( $row );
151
152 $this->assertRevEquals( $orig, $rev );
153 }
154
155 /**
156 * @covers Revision::newFromRow
157 */
158 public function testNewFromRow() {
159 $orig = $this->makeRevision();
160
161 $dbr = wfgetDB( DB_SLAVE );
162 $res = $dbr->select( 'revision', '*', array( 'rev_id' => $orig->getId() ) );
163 $this->assertTrue( is_object( $res ), 'query failed' );
164
165 $row = $res->fetchObject();
166 $res->free();
167
168 $rev = Revision::newFromRow( $row );
169
170 $this->assertRevEquals( $orig, $rev );
171 }
172
173 /**
174 * @covers Revision::newFromArchiveRow
175 */
176 public function testNewFromArchiveRow() {
177 $page = $this->createPage( 'RevisionStorageTest_testNewFromArchiveRow', 'Lorem Ipsum', CONTENT_MODEL_WIKITEXT );
178 $orig = $page->getRevision();
179 $page->doDeleteArticle( 'test Revision::newFromArchiveRow' );
180
181 $dbr = wfgetDB( DB_SLAVE );
182 $res = $dbr->select( 'archive', '*', array( 'ar_rev_id' => $orig->getId() ) );
183 $this->assertTrue( is_object( $res ), 'query failed' );
184
185 $row = $res->fetchObject();
186 $res->free();
187
188 $rev = Revision::newFromArchiveRow( $row );
189
190 $this->assertRevEquals( $orig, $rev );
191 }
192
193 /**
194 * @covers Revision::newFromId
195 */
196 public function testNewFromId() {
197 $orig = $this->makeRevision();
198
199 $rev = Revision::newFromId( $orig->getId() );
200
201 $this->assertRevEquals( $orig, $rev );
202 }
203
204 /**
205 * @covers Revision::fetchRevision
206 */
207 public function testFetchRevision() {
208 $page = $this->createPage( 'RevisionStorageTest_testFetchRevision', 'one', CONTENT_MODEL_WIKITEXT );
209 $id1 = $page->getRevision()->getId();
210
211 $page->doEditContent( new WikitextContent( 'two' ), 'second rev' );
212 $id2 = $page->getRevision()->getId();
213
214 $res = Revision::fetchRevision( $page->getTitle() );
215
216 #note: order is unspecified
217 $rows = array();
218 while ( ( $row = $res->fetchObject() ) ) {
219 $rows[$row->rev_id] = $row;
220 }
221
222 $row = $res->fetchObject();
223 $this->assertEquals( 1, count( $rows ), 'expected exactly one revision' );
224 $this->assertArrayHasKey( $id2, $rows, 'missing revision with id ' . $id2 );
225 }
226
227 /**
228 * @covers Revision::selectFields
229 */
230 public function testSelectFields() {
231 global $wgContentHandlerUseDB;
232
233 $fields = Revision::selectFields();
234
235 $this->assertTrue( in_array( 'rev_id', $fields ), 'missing rev_id in list of fields' );
236 $this->assertTrue( in_array( 'rev_page', $fields ), 'missing rev_page in list of fields' );
237 $this->assertTrue( in_array( 'rev_timestamp', $fields ), 'missing rev_timestamp in list of fields' );
238 $this->assertTrue( in_array( 'rev_user', $fields ), 'missing rev_user in list of fields' );
239
240 if ( $wgContentHandlerUseDB ) {
241 $this->assertTrue( in_array( 'rev_content_model', $fields ),
242 'missing rev_content_model in list of fields' );
243 $this->assertTrue( in_array( 'rev_content_format', $fields ),
244 'missing rev_content_format in list of fields' );
245 }
246 }
247
248 /**
249 * @covers Revision::getPage
250 */
251 public function testGetPage() {
252 $page = $this->the_page;
253
254 $orig = $this->makeRevision( array( 'page' => $page->getId() ) );
255 $rev = Revision::newFromId( $orig->getId() );
256
257 $this->assertEquals( $page->getId(), $rev->getPage() );
258 }
259
260 /**
261 * @covers Revision::getText
262 */
263 public function testGetText() {
264 $this->hideDeprecated( 'Revision::getText' );
265
266 $orig = $this->makeRevision( array( 'text' => 'hello hello.' ) );
267 $rev = Revision::newFromId( $orig->getId() );
268
269 $this->assertEquals( 'hello hello.', $rev->getText() );
270 }
271
272 /**
273 * @covers Revision::getContent
274 */
275 public function testGetContent_failure() {
276 $rev = new Revision( array(
277 'page' => $this->the_page->getId(),
278 'content_model' => $this->the_page->getContentModel(),
279 'text_id' => 123456789, // not in the test DB
280 ) );
281
282 $this->assertNull( $rev->getContent(),
283 "getContent() should return null if the revision's text blob could not be loaded." );
284
285 //NOTE: check this twice, once for lazy initialization, and once with the cached value.
286 $this->assertNull( $rev->getContent(),
287 "getContent() should return null if the revision's text blob could not be loaded." );
288 }
289
290 /**
291 * @covers Revision::getContent
292 */
293 public function testGetContent() {
294 $orig = $this->makeRevision( array( 'text' => 'hello hello.' ) );
295 $rev = Revision::newFromId( $orig->getId() );
296
297 $this->assertEquals( 'hello hello.', $rev->getContent()->getNativeData() );
298 }
299
300 /**
301 * @covers Revision::revText
302 */
303 public function testRevText() {
304 $this->hideDeprecated( 'Revision::revText' );
305 $orig = $this->makeRevision( array( 'text' => 'hello hello rev.' ) );
306 $rev = Revision::newFromId( $orig->getId() );
307
308 $this->assertEquals( 'hello hello rev.', $rev->revText() );
309 }
310
311 /**
312 * @covers Revision::getRawText
313 */
314 public function testGetRawText() {
315 $this->hideDeprecated( 'Revision::getRawText' );
316
317 $orig = $this->makeRevision( array( 'text' => 'hello hello raw.' ) );
318 $rev = Revision::newFromId( $orig->getId() );
319
320 $this->assertEquals( 'hello hello raw.', $rev->getRawText() );
321 }
322
323 /**
324 * @covers Revision::getContentModel
325 */
326 public function testGetContentModel() {
327 global $wgContentHandlerUseDB;
328
329 if ( !$wgContentHandlerUseDB ) {
330 $this->markTestSkipped( '$wgContentHandlerUseDB is disabled' );
331 }
332
333 $orig = $this->makeRevision( array( 'text' => 'hello hello.',
334 'content_model' => CONTENT_MODEL_JAVASCRIPT ) );
335 $rev = Revision::newFromId( $orig->getId() );
336
337 $this->assertEquals( CONTENT_MODEL_JAVASCRIPT, $rev->getContentModel() );
338 }
339
340 /**
341 * @covers Revision::getContentFormat
342 */
343 public function testGetContentFormat() {
344 global $wgContentHandlerUseDB;
345
346 if ( !$wgContentHandlerUseDB ) {
347 $this->markTestSkipped( '$wgContentHandlerUseDB is disabled' );
348 }
349
350 $orig = $this->makeRevision( array(
351 'text' => 'hello hello.',
352 'content_model' => CONTENT_MODEL_JAVASCRIPT,
353 'content_format' => CONTENT_FORMAT_JAVASCRIPT
354 ) );
355 $rev = Revision::newFromId( $orig->getId() );
356
357 $this->assertEquals( CONTENT_FORMAT_JAVASCRIPT, $rev->getContentFormat() );
358 }
359
360 /**
361 * @covers Revision::isCurrent
362 */
363 public function testIsCurrent() {
364 $page = $this->createPage( 'RevisionStorageTest_testIsCurrent', 'Lorem Ipsum', CONTENT_MODEL_WIKITEXT );
365 $rev1 = $page->getRevision();
366
367 # @todo find out if this should be true
368 # $this->assertTrue( $rev1->isCurrent() );
369
370 $rev1x = Revision::newFromId( $rev1->getId() );
371 $this->assertTrue( $rev1x->isCurrent() );
372
373 $page->doEditContent( ContentHandler::makeContent( 'Bla bla', $page->getTitle(), CONTENT_MODEL_WIKITEXT ), 'second rev' );
374 $rev2 = $page->getRevision();
375
376 # @todo find out if this should be true
377 # $this->assertTrue( $rev2->isCurrent() );
378
379 $rev1x = Revision::newFromId( $rev1->getId() );
380 $this->assertFalse( $rev1x->isCurrent() );
381
382 $rev2x = Revision::newFromId( $rev2->getId() );
383 $this->assertTrue( $rev2x->isCurrent() );
384 }
385
386 /**
387 * @covers Revision::getPrevious
388 */
389 public function testGetPrevious() {
390 $page = $this->createPage( 'RevisionStorageTest_testGetPrevious', 'Lorem Ipsum testGetPrevious', CONTENT_MODEL_WIKITEXT );
391 $rev1 = $page->getRevision();
392
393 $this->assertNull( $rev1->getPrevious() );
394
395 $page->doEditContent( ContentHandler::makeContent( 'Bla bla', $page->getTitle(), CONTENT_MODEL_WIKITEXT ),
396 'second rev testGetPrevious' );
397 $rev2 = $page->getRevision();
398
399 $this->assertNotNull( $rev2->getPrevious() );
400 $this->assertEquals( $rev1->getId(), $rev2->getPrevious()->getId() );
401 }
402
403 /**
404 * @covers Revision::getNext
405 */
406 public function testGetNext() {
407 $page = $this->createPage( 'RevisionStorageTest_testGetNext', 'Lorem Ipsum testGetNext', CONTENT_MODEL_WIKITEXT );
408 $rev1 = $page->getRevision();
409
410 $this->assertNull( $rev1->getNext() );
411
412 $page->doEditContent( ContentHandler::makeContent( 'Bla bla', $page->getTitle(), CONTENT_MODEL_WIKITEXT ),
413 'second rev testGetNext' );
414 $rev2 = $page->getRevision();
415
416 $this->assertNotNull( $rev1->getNext() );
417 $this->assertEquals( $rev2->getId(), $rev1->getNext()->getId() );
418 }
419
420 /**
421 * @covers Revision::newNullRevision
422 */
423 public function testNewNullRevision() {
424 $page = $this->createPage( 'RevisionStorageTest_testNewNullRevision', 'some testing text', CONTENT_MODEL_WIKITEXT );
425 $orig = $page->getRevision();
426
427 $dbw = wfGetDB( DB_MASTER );
428 $rev = Revision::newNullRevision( $dbw, $page->getId(), 'a null revision', false );
429
430 $this->assertNotEquals( $orig->getId(), $rev->getId(),
431 'new null revision shold have a different id from the original revision' );
432 $this->assertEquals( $orig->getTextId(), $rev->getTextId(),
433 'new null revision shold have the same text id as the original revision' );
434 $this->assertEquals( 'some testing text', $rev->getContent()->getNativeData() );
435 }
436
437 public static function provideUserWasLastToEdit() {
438 return array(
439 array( #0
440 3, true, # actually the last edit
441 ),
442 array( #1
443 2, true, # not the current edit, but still by this user
444 ),
445 array( #2
446 1, false, # edit by another user
447 ),
448 array( #3
449 0, false, # first edit, by this user, but another user edited in the mean time
450 ),
451 );
452 }
453
454 /**
455 * @dataProvider provideUserWasLastToEdit
456 */
457 public function testUserWasLastToEdit( $sinceIdx, $expectedLast ) {
458 $userA = User::newFromName( "RevisionStorageTest_userA" );
459 $userB = User::newFromName( "RevisionStorageTest_userB" );
460
461 if ( $userA->getId() === 0 ) {
462 $userA = User::createNew( $userA->getName() );
463 }
464
465 if ( $userB->getId() === 0 ) {
466 $userB = User::createNew( $userB->getName() );
467 }
468
469 $ns = $this->getDefaultWikitextNS();
470
471 $dbw = wfGetDB( DB_MASTER );
472 $revisions = array();
473
474 // create revisions -----------------------------
475 $page = WikiPage::factory( Title::newFromText(
476 'RevisionStorageTest_testUserWasLastToEdit', $ns ) );
477
478 # zero
479 $revisions[0] = new Revision( array(
480 'page' => $page->getId(),
481 'title' => $page->getTitle(), // we need the title to determine the page's default content model
482 'timestamp' => '20120101000000',
483 'user' => $userA->getId(),
484 'text' => 'zero',
485 'content_model' => CONTENT_MODEL_WIKITEXT,
486 'summary' => 'edit zero'
487 ) );
488 $revisions[0]->insertOn( $dbw );
489
490 # one
491 $revisions[1] = new Revision( array(
492 'page' => $page->getId(),
493 'title' => $page->getTitle(), // still need the title, because $page->getId() is 0 (there's no entry in the page table)
494 'timestamp' => '20120101000100',
495 'user' => $userA->getId(),
496 'text' => 'one',
497 'content_model' => CONTENT_MODEL_WIKITEXT,
498 'summary' => 'edit one'
499 ) );
500 $revisions[1]->insertOn( $dbw );
501
502 # two
503 $revisions[2] = new Revision( array(
504 'page' => $page->getId(),
505 'title' => $page->getTitle(),
506 'timestamp' => '20120101000200',
507 'user' => $userB->getId(),
508 'text' => 'two',
509 'content_model' => CONTENT_MODEL_WIKITEXT,
510 'summary' => 'edit two'
511 ) );
512 $revisions[2]->insertOn( $dbw );
513
514 # three
515 $revisions[3] = new Revision( array(
516 'page' => $page->getId(),
517 'title' => $page->getTitle(),
518 'timestamp' => '20120101000300',
519 'user' => $userA->getId(),
520 'text' => 'three',
521 'content_model' => CONTENT_MODEL_WIKITEXT,
522 'summary' => 'edit three'
523 ) );
524 $revisions[3]->insertOn( $dbw );
525
526 # four
527 $revisions[4] = new Revision( array(
528 'page' => $page->getId(),
529 'title' => $page->getTitle(),
530 'timestamp' => '20120101000200',
531 'user' => $userA->getId(),
532 'text' => 'zero',
533 'content_model' => CONTENT_MODEL_WIKITEXT,
534 'summary' => 'edit four'
535 ) );
536 $revisions[4]->insertOn( $dbw );
537
538 // test it ---------------------------------
539 $since = $revisions[$sinceIdx]->getTimestamp();
540
541 $wasLast = Revision::userWasLastToEdit( $dbw, $page->getId(), $userA->getId(), $since );
542
543 $this->assertEquals( $expectedLast, $wasLast );
544 }
545 }