4 * Test class for Revision storage.
6 * @group ContentHandler
8 * ^--- important, causes temporary tables to be used instead of the real database
11 * ^--- important, causes tests not to fail with timeout
13 class RevisionStorageTest
extends MediaWikiTestCase
{
16 * @var WikiPage $the_page
20 function __construct( $name = null, array $data = array(), $dataName = '' ) {
21 parent
::__construct( $name, $data, $dataName );
23 $this->tablesUsed
= array_merge( $this->tablesUsed
,
41 public function setUp() {
42 global $wgExtraNamespaces, $wgNamespaceContentModels, $wgContentHandlers, $wgContLang;
46 $wgExtraNamespaces[12312] = 'Dummy';
47 $wgExtraNamespaces[12313] = 'Dummy_talk';
49 $wgNamespaceContentModels[12312] = 'DUMMY';
50 $wgContentHandlers['DUMMY'] = 'DummyContentHandlerForTesting';
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
);
59 public function tearDown() {
60 global $wgExtraNamespaces, $wgNamespaceContentModels, $wgContentHandlers, $wgContLang;
64 unset( $wgExtraNamespaces[12312] );
65 unset( $wgExtraNamespaces[12313] );
67 unset( $wgNamespaceContentModels[12312] );
68 unset( $wgContentHandlers['DUMMY'] );
70 MWNamespace
::getCanonicalNamespaces( true ); # reset namespace cache
71 $wgContLang->resetNamespaces(); # reset namespace cache
74 protected function makeRevision( $props = null ) {
75 if ( $props === null ) {
79 if ( !isset( $props['content'] ) && !isset( $props['text'] ) ) {
80 $props['text'] = 'Lorem Ipsum';
83 if ( !isset( $props['comment'] ) ) {
84 $props['comment'] = 'just a test';
87 if ( !isset( $props['page'] ) ) {
88 $props['page'] = $this->the_page
->getId();
91 $rev = new Revision( $props );
93 $dbw = wfgetDB( DB_MASTER
);
94 $rev->insertOn( $dbw );
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
)
104 $ns = $this->getDefaultWikitextNS();
105 $page = MWNamespace
::getCanonicalName( $ns ) . ':' . $page;
108 $page = Title
::newFromText( $page );
111 if ( $page instanceof Title
) {
112 $page = new WikiPage( $page );
115 if ( $page->exists() ) {
116 $page->doDeleteArticle( "done" );
119 $content = ContentHandler
::makeContent( $text, $page->getTitle(), $model );
120 $page->doEditContent( $content, "testing", EDIT_NEW
);
125 protected function assertRevEquals( Revision
$orig, Revision
$rev = null ) {
126 $this->assertNotNull( $rev, 'missing revision' );
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() );
138 * @covers Revision::__construct
140 public function testConstructFromRow() {
141 $orig = $this->makeRevision();
143 $dbr = wfgetDB( DB_SLAVE
);
144 $res = $dbr->select( 'revision', '*', array( 'rev_id' => $orig->getId() ) );
145 $this->assertTrue( is_object( $res ), 'query failed' );
147 $row = $res->fetchObject();
150 $rev = new Revision( $row );
152 $this->assertRevEquals( $orig, $rev );
156 * @covers Revision::newFromRow
158 public function testNewFromRow() {
159 $orig = $this->makeRevision();
161 $dbr = wfgetDB( DB_SLAVE
);
162 $res = $dbr->select( 'revision', '*', array( 'rev_id' => $orig->getId() ) );
163 $this->assertTrue( is_object( $res ), 'query failed' );
165 $row = $res->fetchObject();
168 $rev = Revision
::newFromRow( $row );
170 $this->assertRevEquals( $orig, $rev );
175 * @covers Revision::newFromArchiveRow
177 public function testNewFromArchiveRow() {
178 $page = $this->createPage( 'RevisionStorageTest_testNewFromArchiveRow', 'Lorem Ipsum', CONTENT_MODEL_WIKITEXT
);
179 $orig = $page->getRevision();
180 $page->doDeleteArticle( 'test Revision::newFromArchiveRow' );
182 $dbr = wfgetDB( DB_SLAVE
);
183 $res = $dbr->select( 'archive', '*', array( 'ar_rev_id' => $orig->getId() ) );
184 $this->assertTrue( is_object( $res ), 'query failed' );
186 $row = $res->fetchObject();
189 $rev = Revision
::newFromArchiveRow( $row );
191 $this->assertRevEquals( $orig, $rev );
195 * @covers Revision::newFromId
197 public function testNewFromId() {
198 $orig = $this->makeRevision();
200 $rev = Revision
::newFromId( $orig->getId() );
202 $this->assertRevEquals( $orig, $rev );
206 * @covers Revision::fetchRevision
208 public function testFetchRevision() {
209 $page = $this->createPage( 'RevisionStorageTest_testFetchRevision', 'one', CONTENT_MODEL_WIKITEXT
);
210 $id1 = $page->getRevision()->getId();
212 $page->doEditContent( new WikitextContent( 'two' ), 'second rev' );
213 $id2 = $page->getRevision()->getId();
215 $res = Revision
::fetchRevision( $page->getTitle() );
217 #note: order is unspecified
219 while ( ( $row = $res->fetchObject() ) ) {
220 $rows[$row->rev_id
] = $row;
223 $row = $res->fetchObject();
224 $this->assertEquals( 1, count( $rows ), 'expected exactly one revision' );
225 $this->assertArrayHasKey( $id2, $rows, 'missing revision with id ' . $id2 );
229 * @covers Revision::selectFields
231 public function testSelectFields() {
232 global $wgContentHandlerUseDB;
234 $fields = Revision
::selectFields();
236 $this->assertTrue( in_array( 'rev_id', $fields ), 'missing rev_id in list of fields' );
237 $this->assertTrue( in_array( 'rev_page', $fields ), 'missing rev_page in list of fields' );
238 $this->assertTrue( in_array( 'rev_timestamp', $fields ), 'missing rev_timestamp in list of fields' );
239 $this->assertTrue( in_array( 'rev_user', $fields ), 'missing rev_user in list of fields' );
241 if ( $wgContentHandlerUseDB ) {
242 $this->assertTrue( in_array( 'rev_content_model', $fields ),
243 'missing rev_content_model in list of fields' );
244 $this->assertTrue( in_array( 'rev_content_format', $fields ),
245 'missing rev_content_format in list of fields' );
250 * @covers Revision::getPage
252 public function testGetPage() {
253 $page = $this->the_page
;
255 $orig = $this->makeRevision( array( 'page' => $page->getId() ) );
256 $rev = Revision
::newFromId( $orig->getId() );
258 $this->assertEquals( $page->getId(), $rev->getPage() );
262 * @covers Revision::getText
264 public function testGetText() {
265 $this->hideDeprecated( 'Revision::getText' );
267 $orig = $this->makeRevision( array( 'text' => 'hello hello.' ) );
268 $rev = Revision
::newFromId( $orig->getId() );
270 $this->assertEquals( 'hello hello.', $rev->getText() );
274 * @covers Revision::getContent
276 public function testGetContent_failure() {
277 $rev = new Revision( array(
278 'page' => $this->the_page
->getId(),
279 'content_model' => $this->the_page
->getContentModel(),
280 'text_id' => 123456789, // not in the test DB
283 $this->assertNull( $rev->getContent(),
284 "getContent() should return null if the revision's text blob could not be loaded." );
286 //NOTE: check this twice, once for lazy initialization, and once with the cached value.
287 $this->assertNull( $rev->getContent(),
288 "getContent() should return null if the revision's text blob could not be loaded." );
292 * @covers Revision::getContent
294 public function testGetContent() {
295 $orig = $this->makeRevision( array( 'text' => 'hello hello.' ) );
296 $rev = Revision
::newFromId( $orig->getId() );
298 $this->assertEquals( 'hello hello.', $rev->getContent()->getNativeData() );
302 * @covers Revision::revText
304 public function testRevText() {
305 $this->hideDeprecated( 'Revision::revText' );
306 $orig = $this->makeRevision( array( 'text' => 'hello hello rev.' ) );
307 $rev = Revision
::newFromId( $orig->getId() );
309 $this->assertEquals( 'hello hello rev.', $rev->revText() );
313 * @covers Revision::getRawText
315 public function testGetRawText() {
316 $this->hideDeprecated( 'Revision::getRawText' );
318 $orig = $this->makeRevision( array( 'text' => 'hello hello raw.' ) );
319 $rev = Revision
::newFromId( $orig->getId() );
321 $this->assertEquals( 'hello hello raw.', $rev->getRawText() );
325 * @covers Revision::getContentModel
327 public function testGetContentModel() {
328 global $wgContentHandlerUseDB;
330 if ( !$wgContentHandlerUseDB ) {
331 $this->markTestSkipped( '$wgContentHandlerUseDB is disabled' );
334 $orig = $this->makeRevision( array( 'text' => 'hello hello.',
335 'content_model' => CONTENT_MODEL_JAVASCRIPT
) );
336 $rev = Revision
::newFromId( $orig->getId() );
338 $this->assertEquals( CONTENT_MODEL_JAVASCRIPT
, $rev->getContentModel() );
342 * @covers Revision::getContentFormat
344 public function testGetContentFormat() {
345 global $wgContentHandlerUseDB;
347 if ( !$wgContentHandlerUseDB ) {
348 $this->markTestSkipped( '$wgContentHandlerUseDB is disabled' );
351 $orig = $this->makeRevision( array(
352 'text' => 'hello hello.',
353 'content_model' => CONTENT_MODEL_JAVASCRIPT
,
354 'content_format' => CONTENT_FORMAT_JAVASCRIPT
356 $rev = Revision
::newFromId( $orig->getId() );
358 $this->assertEquals( CONTENT_FORMAT_JAVASCRIPT
, $rev->getContentFormat() );
362 * @covers Revision::isCurrent
364 public function testIsCurrent() {
365 $page = $this->createPage( 'RevisionStorageTest_testIsCurrent', 'Lorem Ipsum', CONTENT_MODEL_WIKITEXT
);
366 $rev1 = $page->getRevision();
368 # @todo find out if this should be true
369 # $this->assertTrue( $rev1->isCurrent() );
371 $rev1x = Revision
::newFromId( $rev1->getId() );
372 $this->assertTrue( $rev1x->isCurrent() );
374 $page->doEditContent( ContentHandler
::makeContent( 'Bla bla', $page->getTitle(), CONTENT_MODEL_WIKITEXT
), 'second rev' );
375 $rev2 = $page->getRevision();
377 # @todo find out if this should be true
378 # $this->assertTrue( $rev2->isCurrent() );
380 $rev1x = Revision
::newFromId( $rev1->getId() );
381 $this->assertFalse( $rev1x->isCurrent() );
383 $rev2x = Revision
::newFromId( $rev2->getId() );
384 $this->assertTrue( $rev2x->isCurrent() );
388 * @covers Revision::getPrevious
390 public function testGetPrevious() {
391 $page = $this->createPage( 'RevisionStorageTest_testGetPrevious', 'Lorem Ipsum testGetPrevious', CONTENT_MODEL_WIKITEXT
);
392 $rev1 = $page->getRevision();
394 $this->assertNull( $rev1->getPrevious() );
396 $page->doEditContent( ContentHandler
::makeContent( 'Bla bla', $page->getTitle(), CONTENT_MODEL_WIKITEXT
),
397 'second rev testGetPrevious' );
398 $rev2 = $page->getRevision();
400 $this->assertNotNull( $rev2->getPrevious() );
401 $this->assertEquals( $rev1->getId(), $rev2->getPrevious()->getId() );
405 * @covers Revision::getNext
407 public function testGetNext() {
408 $page = $this->createPage( 'RevisionStorageTest_testGetNext', 'Lorem Ipsum testGetNext', CONTENT_MODEL_WIKITEXT
);
409 $rev1 = $page->getRevision();
411 $this->assertNull( $rev1->getNext() );
413 $page->doEditContent( ContentHandler
::makeContent( 'Bla bla', $page->getTitle(), CONTENT_MODEL_WIKITEXT
),
414 'second rev testGetNext' );
415 $rev2 = $page->getRevision();
417 $this->assertNotNull( $rev1->getNext() );
418 $this->assertEquals( $rev2->getId(), $rev1->getNext()->getId() );
422 * @covers Revision::newNullRevision
424 public function testNewNullRevision() {
425 $page = $this->createPage( 'RevisionStorageTest_testNewNullRevision', 'some testing text', CONTENT_MODEL_WIKITEXT
);
426 $orig = $page->getRevision();
428 $dbw = wfGetDB( DB_MASTER
);
429 $rev = Revision
::newNullRevision( $dbw, $page->getId(), 'a null revision', false );
431 $this->assertNotEquals( $orig->getId(), $rev->getId(),
432 'new null revision shold have a different id from the original revision' );
433 $this->assertEquals( $orig->getTextId(), $rev->getTextId(),
434 'new null revision shold have the same text id as the original revision' );
435 $this->assertEquals( 'some testing text', $rev->getContent()->getNativeData() );
438 public static function provideUserWasLastToEdit() {
441 3, true, # actually the last edit
444 2, true, # not the current edit, but still by this user
447 1, false, # edit by another user
450 0, false, # first edit, by this user, but another user edited in the mean time
456 * @dataProvider provideUserWasLastToEdit
458 public function testUserWasLastToEdit( $sinceIdx, $expectedLast ) {
459 $userA = \User
::newFromName( "RevisionStorageTest_userA" );
460 $userB = \User
::newFromName( "RevisionStorageTest_userB" );
462 if ( $userA->getId() === 0 ) {
463 $userA = \User
::createNew( $userA->getName() );
466 if ( $userB->getId() === 0 ) {
467 $userB = \User
::createNew( $userB->getName() );
470 $ns = $this->getDefaultWikitextNS();
472 $dbw = wfGetDB( DB_MASTER
);
473 $revisions = array();
475 // create revisions -----------------------------
476 $page = WikiPage
::factory( Title
::newFromText(
477 'RevisionStorageTest_testUserWasLastToEdit', $ns ) );
480 $revisions[0] = new Revision( array(
481 'page' => $page->getId(),
482 'title' => $page->getTitle(), // we need the title to determine the page's default content model
483 'timestamp' => '20120101000000',
484 'user' => $userA->getId(),
486 'content_model' => CONTENT_MODEL_WIKITEXT
,
487 'summary' => 'edit zero'
489 $revisions[0]->insertOn( $dbw );
492 $revisions[1] = new Revision( array(
493 'page' => $page->getId(),
494 'title' => $page->getTitle(), // still need the title, because $page->getId() is 0 (there's no entry in the page table)
495 'timestamp' => '20120101000100',
496 'user' => $userA->getId(),
498 'content_model' => CONTENT_MODEL_WIKITEXT
,
499 'summary' => 'edit one'
501 $revisions[1]->insertOn( $dbw );
504 $revisions[2] = new Revision( array(
505 'page' => $page->getId(),
506 'title' => $page->getTitle(),
507 'timestamp' => '20120101000200',
508 'user' => $userB->getId(),
510 'content_model' => CONTENT_MODEL_WIKITEXT
,
511 'summary' => 'edit two'
513 $revisions[2]->insertOn( $dbw );
516 $revisions[3] = new Revision( array(
517 'page' => $page->getId(),
518 'title' => $page->getTitle(),
519 'timestamp' => '20120101000300',
520 'user' => $userA->getId(),
522 'content_model' => CONTENT_MODEL_WIKITEXT
,
523 'summary' => 'edit three'
525 $revisions[3]->insertOn( $dbw );
528 $revisions[4] = new Revision( array(
529 'page' => $page->getId(),
530 'title' => $page->getTitle(),
531 'timestamp' => '20120101000200',
532 'user' => $userA->getId(),
534 'content_model' => CONTENT_MODEL_WIKITEXT
,
535 'summary' => 'edit four'
537 $revisions[4]->insertOn( $dbw );
539 // test it ---------------------------------
540 $since = $revisions[$sinceIdx]->getTimestamp();
542 $wasLast = Revision
::userWasLastToEdit( $dbw, $page->getId(), $userA->getId(), $since );
544 $this->assertEquals( $expectedLast, $wasLast );