2022-02-16 23:29:10 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace MediaWiki\Extension\DiscussionTools\Tests;
|
|
|
|
|
2023-12-27 22:07:24 +00:00
|
|
|
use ExtensionRegistry;
|
2022-02-16 23:29:10 +00:00
|
|
|
use ImportStringSource;
|
2024-02-03 22:10:47 +00:00
|
|
|
use MediaWiki\Extension\DiscussionTools\ThreadItemStore;
|
2022-02-16 23:29:10 +00:00
|
|
|
use MediaWiki\MediaWikiServices;
|
2024-02-03 22:10:47 +00:00
|
|
|
use MediaWiki\Title\TitleValue;
|
2022-02-16 23:29:10 +00:00
|
|
|
use TestUser;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @group DiscussionTools
|
|
|
|
* @group Database
|
2022-09-15 12:08:30 +00:00
|
|
|
* @covers \MediaWiki\Extension\DiscussionTools\ThreadItemStore
|
2022-02-16 23:29:10 +00:00
|
|
|
*/
|
|
|
|
class ThreadItemStoreTest extends IntegrationTestCase {
|
|
|
|
|
2024-02-03 22:10:47 +00:00
|
|
|
protected function setUp(): void {
|
|
|
|
parent::setUp();
|
|
|
|
|
2023-07-04 15:34:33 +00:00
|
|
|
if (
|
2023-07-26 13:29:40 +00:00
|
|
|
$this->db->getType() === 'mysql' &&
|
2023-07-04 15:34:33 +00:00
|
|
|
strpos( $this->db->getSoftwareLink(), 'MySQL' ) &&
|
|
|
|
!$this->getCliArg( 'use-normal-tables' )
|
|
|
|
) {
|
|
|
|
$this->markTestSkipped( 'Set PHPUNIT_USE_NORMAL_TABLES=1 env variable to run these tests, ' .
|
|
|
|
'otherwise they would fail due to a MySQL bug with temporary tables (T256006)' );
|
|
|
|
}
|
2023-12-27 22:07:24 +00:00
|
|
|
if ( ExtensionRegistry::getInstance()->isLoaded( 'Liquid Threads' ) ) {
|
2024-07-09 10:40:34 +00:00
|
|
|
$this->overrideConfigValue( 'LqtTalkPages', false );
|
2023-12-27 22:07:24 +00:00
|
|
|
}
|
2024-02-03 22:10:47 +00:00
|
|
|
}
|
2023-07-04 15:34:33 +00:00
|
|
|
|
2024-02-03 22:10:47 +00:00
|
|
|
private function importTestCase( string $dir ) {
|
2022-02-16 23:29:10 +00:00
|
|
|
// Create users for the imported revisions
|
|
|
|
new TestUser( 'X' );
|
|
|
|
new TestUser( 'Y' );
|
|
|
|
new TestUser( 'Z' );
|
|
|
|
|
|
|
|
// Import revisions
|
|
|
|
$source = new ImportStringSource( static::getText( "$dir/dump.xml" ) );
|
|
|
|
$importer = MediaWikiServices::getInstance()
|
|
|
|
->getWikiImporterFactory()
|
2023-12-08 21:14:58 +00:00
|
|
|
->getWikiImporter( $source, $this->getTestSysop()->getAuthority() );
|
2022-02-16 23:29:10 +00:00
|
|
|
// `true` means to assign edits to the users we created above
|
|
|
|
$importer->setUsernamePrefix( 'import', true );
|
|
|
|
$importer->doImport();
|
2024-02-03 22:10:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dataProvider provideInsertCases
|
|
|
|
*/
|
|
|
|
public function testInsertThreadItems( string $dir ): void {
|
|
|
|
$this->importTestCase( $dir );
|
2022-02-16 23:29:10 +00:00
|
|
|
|
|
|
|
// Check that expected data has been stored in the database
|
|
|
|
$expected = [];
|
|
|
|
$actual = [];
|
|
|
|
$tables = [
|
2022-10-19 14:11:02 +00:00
|
|
|
'discussiontools_items' => [ 'it_id' ],
|
|
|
|
'discussiontools_item_pages' => [ 'itp_id' ],
|
|
|
|
// We reuse rows causing the primary key to be all out of order.
|
|
|
|
// Use a consistent ordering for the output here.
|
|
|
|
'discussiontools_item_revisions' => [ 'itr_revision_id', 'itr_items_id', 'itr_itemid_id' ],
|
|
|
|
'discussiontools_item_ids' => [ 'itid_id' ],
|
2022-02-16 23:29:10 +00:00
|
|
|
];
|
2022-10-19 14:11:02 +00:00
|
|
|
foreach ( $tables as $table => $order ) {
|
2022-02-16 23:29:10 +00:00
|
|
|
$expected[$table] = static::getJson( "../$dir/$table.json", true );
|
|
|
|
|
2024-03-26 11:29:26 +00:00
|
|
|
$res = $this->getDb()->newSelectQueryBuilder()
|
2022-08-12 22:15:24 +00:00
|
|
|
->from( $table )
|
|
|
|
->field( '*' )
|
|
|
|
->caller( __METHOD__ )
|
2022-10-19 14:11:02 +00:00
|
|
|
->orderBy( $order )
|
2022-08-12 22:15:24 +00:00
|
|
|
->fetchResultSet();
|
2022-02-16 23:29:10 +00:00
|
|
|
foreach ( $res as $i => $row ) {
|
|
|
|
foreach ( $row as $key => $val ) {
|
|
|
|
$actual[$table][$i][$key] = $val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Optionally write updated content to the JSON files
|
|
|
|
if ( getenv( 'DISCUSSIONTOOLS_OVERWRITE_TESTS' ) ) {
|
2022-10-19 14:11:02 +00:00
|
|
|
foreach ( $tables as $table => $order ) {
|
2022-02-16 23:29:10 +00:00
|
|
|
static::overwriteJsonFile( "../$dir/$table.json", $actual[$table] );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static::assertEquals( $expected, $actual );
|
|
|
|
}
|
|
|
|
|
2023-05-20 13:57:13 +00:00
|
|
|
public static function provideInsertCases(): array {
|
2022-02-16 23:29:10 +00:00
|
|
|
return [
|
|
|
|
[ 'cases/ThreadItemStore/1simple-example' ],
|
|
|
|
[ 'cases/ThreadItemStore/2archived-section' ],
|
|
|
|
[ 'cases/ThreadItemStore/3indistinguishable-comments' ],
|
|
|
|
[ 'cases/ThreadItemStore/4transcluded-section' ],
|
|
|
|
[ 'cases/ThreadItemStore/5changed-comment-indentation' ],
|
|
|
|
[ 'cases/ThreadItemStore/6changed-heading-level' ],
|
2022-10-19 23:57:56 +00:00
|
|
|
[ 'cases/ThreadItemStore/7identical-rev-timestamp' ],
|
2022-10-19 03:31:39 +00:00
|
|
|
[ 'cases/ThreadItemStore/8indistinguishable-comments-same-page' ],
|
2022-02-16 23:29:10 +00:00
|
|
|
];
|
|
|
|
}
|
2024-02-03 22:10:47 +00:00
|
|
|
|
|
|
|
public function testFindSimple(): void {
|
|
|
|
$this->importTestCase( 'cases/ThreadItemStore/1simple-example' );
|
|
|
|
/** @var ThreadItemStore $itemStore */
|
|
|
|
$itemStore = $this->getServiceContainer()->getService( 'DiscussionTools.ThreadItemStore' );
|
|
|
|
|
|
|
|
// BY NAME
|
|
|
|
// Valid name finds the item
|
|
|
|
$set = $itemStore->findNewestRevisionsByName( 'c-Y-20220720010200' );
|
|
|
|
$this->assertCount( 1, $set );
|
|
|
|
$this->assertEquals( 'ThreadItemStore1', $set[0]->getPage()->getDBkey() );
|
|
|
|
$this->assertEquals( 'c-Y-20220720010200', $set[0]->getName() );
|
|
|
|
|
|
|
|
// Wrong name - no results
|
|
|
|
$set = $itemStore->findNewestRevisionsByName( 'c-blah' );
|
|
|
|
$this->assertCount( 0, $set );
|
|
|
|
|
|
|
|
// BY ID
|
|
|
|
// Valid ID finds the item
|
|
|
|
$set = $itemStore->findNewestRevisionsById( 'c-Y-20220720010200-X-20220720010100' );
|
|
|
|
$this->assertCount( 1, $set );
|
|
|
|
$this->assertEquals( 'ThreadItemStore1', $set[0]->getPage()->getDBkey() );
|
|
|
|
$this->assertEquals( 'c-Y-20220720010200-X-20220720010100', $set[0]->getId() );
|
|
|
|
|
|
|
|
// Wrong ID - no results
|
|
|
|
$set = $itemStore->findNewestRevisionsById( 'c-blah' );
|
|
|
|
$this->assertCount( 0, $set );
|
|
|
|
|
|
|
|
// BY HEADING
|
|
|
|
// Valid heading + page title finds the item
|
|
|
|
$page = $this->getServiceContainer()->getPageStore()->getPageByName( NS_TALK, 'ThreadItemStore1' );
|
|
|
|
$set = $itemStore->findNewestRevisionsByHeading( 'A', $page->getId(), TitleValue::newFromPage( $page ) );
|
|
|
|
$this->assertCount( 1, $set );
|
|
|
|
$this->assertEquals( 'ThreadItemStore1', $set[0]->getPage()->getDBkey() );
|
|
|
|
|
|
|
|
// Wrong heading - no results
|
|
|
|
$set = $itemStore->findNewestRevisionsByHeading( 'blah', $page->getId(), TitleValue::newFromPage( $page ) );
|
|
|
|
$this->assertCount( 0, $set );
|
|
|
|
|
|
|
|
// Wrong page - but we still get a result, since it's unique (case 3.)
|
|
|
|
$set = $itemStore->findNewestRevisionsByHeading( 'A', 12345, new TitleValue( NS_TALK, 'Blah' ) );
|
|
|
|
$this->assertCount( 1, $set );
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFindArchived(): void {
|
|
|
|
$this->importTestCase( 'cases/ThreadItemStore/2archived-section' );
|
|
|
|
/** @var ThreadItemStore $itemStore */
|
|
|
|
$itemStore = $this->getServiceContainer()->getService( 'DiscussionTools.ThreadItemStore' );
|
|
|
|
|
|
|
|
// Valid name finds the original item in old revision, and the item in the archive
|
|
|
|
$set = $itemStore->findNewestRevisionsByName( 'c-X-20220720020100' );
|
|
|
|
$this->assertCount( 2, $set );
|
|
|
|
|
|
|
|
// Valid ID finds the original item in old revision, and the item in the archive
|
|
|
|
$set = $itemStore->findNewestRevisionsById( 'c-X-20220720020100-A' );
|
|
|
|
$this->assertCount( 2, $set );
|
|
|
|
|
|
|
|
// Valid heading + page title finds the original item in old revision, and the item in the archive
|
|
|
|
$page = $this->getServiceContainer()->getPageStore()->getPageByName( NS_TALK, 'ThreadItemStore2' );
|
|
|
|
$set = $itemStore->findNewestRevisionsByHeading( 'A', $page->getId(), TitleValue::newFromPage( $page ) );
|
|
|
|
$this->assertCount( 2, $set );
|
|
|
|
}
|
2022-02-16 23:29:10 +00:00
|
|
|
}
|