wiki.techinc.nl/tests/phpunit/maintenance/CategoryChangesAsRdfTest.php
Timo Tijhof c02513c97e phpunit: Fix tests relying on implicit wgScript/wgArticlePath
A number of tests have hardcoded expections that pass only in WMF CI
where Quibble has LocalSettings.php with $wgScript and $wgArticlePath
set a certain way.

We could fix these by adding setMwGlobals() in their tests, as we
often do, but these are so often forgotten that I'd rather we just
add them to TestSetup.php so that it is simply impossible to write a
test that that passes locally for you (if you have the same config)
but not for someone else.

There is a larger project in there somewhere about expanding this
slowly such that we basically only pluck DB-settings and extension
enablement from LocalSettings and otherwise run the tests with the
default settings in PHPUnit. Pretty much by definition, any (other)
setting you have in LocalSettings is irrelevant because it either:
1. has no effect on the test (majority, harmless either way),
2. has a custom default via TestSetup.php (which has precedence over
   LocalSettings.php),
3. is relevant to the code being tested and the test case correctly
   calls setMwGlobals() to ensure a consistent value during test.
4. is relevant to the tested code but has no override, thus only
   passes if you happen to have the "right" value set for it
   (undesirable).

Case 4 is already categorically impossible for the most common config
settings that influence random code because we give them a value
in TestSetup.php. This patch expands that to include $wgScript
and $wgArticlePath. Perhaps in the future we can think about a way
to do this automatically by either re-applying MainConfigSchema
(sans db settings) or by only selectively applying LocalSettings.php
in the first place.

This patch follows-up I072ddf89562fe, which added a test case in
WikitextContentHandlerIntegrationTest.php that assumed "/index.php"
as the value of $wgScript. This passes in WMF CI since Quibble uses
that value, but the tests failed in most local development installs
since those tend to use "/w" instead.

Rather than one-off fixing that one test with overrideConfigValues(),
switch to a more general fixture, since the precise values don't
matter for this test.

Bug: T349087
Bug: T277470
Change-Id: If4304b7ca4a838bd892d4516a0b5c6dfbc30986e
2024-05-05 00:00:01 +00:00

309 lines
8.5 KiB
PHP

<?php
use MediaWiki\MainConfigNames;
use MediaWiki\Title\Title;
use MediaWiki\Utils\MWTimestamp;
use Wikimedia\Rdbms\IDatabase;
use Wikimedia\Timestamp\ConvertibleTimestamp;
/**
* Tests for CategoryChangesAsRdf recent changes exporter.
* @covers \CategoryChangesAsRdf
* @group Database
*/
class CategoryChangesAsRdfTest extends MediaWikiLangTestCase {
protected function setUp(): void {
parent::setUp();
$this->overrideConfigValues( [
MainConfigNames::Server => 'http://acme.test',
MainConfigNames::CanonicalServer => 'http://acme.test',
] );
}
public static function provideCategoryData() {
return [
'delete category' => [
__DIR__ . "/../data/categoriesrdf/delete.sparql",
'getDeletedCatsIterator',
'handleDeletes',
[
(object)[ 'rc_title' => 'Test', 'rc_cur_id' => 1, '_processed' => 1 ],
(object)[ 'rc_title' => 'Test 2', 'rc_cur_id' => 2, '_processed' => 2 ],
],
],
'move category' => [
__DIR__ . "/../data/categoriesrdf/move.sparql",
'getMovedCatsIterator',
'handleMoves',
[
(object)[
'rc_title' => 'Test',
'rc_cur_id' => 4,
'page_title' => 'MovedTo',
'page_namespace' => NS_CATEGORY,
'_processed' => 4,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'MovedTo',
'rc_cur_id' => 4,
'page_title' => 'MovedAgain',
'page_namespace' => NS_CATEGORY,
'pp_propname' => 'hiddencat',
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Test 2',
'rc_cur_id' => 5,
'page_title' => 'AlsoMoved',
'page_namespace' => NS_CATEGORY,
'_processed' => 5,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Test 3',
'rc_cur_id' => 6,
'page_title' => 'MovedOut',
'page_namespace' => NS_MAIN,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Test 4',
'rc_cur_id' => 7,
'page_title' => 'Already Done',
'page_namespace' => NS_CATEGORY,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
],
[ 7 => true ],
],
'restore deleted category' => [
__DIR__ . "/../data/categoriesrdf/restore.sparql",
'getRestoredCatsIterator',
'handleRestores',
[
(object)[
'rc_title' => 'Restored cat',
'rc_cur_id' => 10,
'_processed' => 10,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Restored again',
'rc_cur_id' => 10,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Already seen',
'rc_cur_id' => 11,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
],
[ 11 => true ],
],
'new page' => [
__DIR__ . "/../data/categoriesrdf/new.sparql",
'getNewCatsIterator',
'handleAdds',
[
(object)[
'rc_title' => 'New category',
'rc_cur_id' => 20,
'_processed' => 20,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Новая категория 😃',
'rc_cur_id' => 21,
'_processed' => 21,
'pp_propname' => 'hiddencat',
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Processed already',
'rc_cur_id' => 22,
],
],
[ 22 => true ],
],
'edit category' => [
__DIR__ . "/../data/categoriesrdf/edit.sparql",
'getChangedCatsIterator',
'handleEdits',
[
(object)[
'rc_title' => 'Changed category',
'rc_cur_id' => 30,
'_processed' => 30,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Changed again',
'rc_cur_id' => 30,
'pp_propname' => null,
'cat_pages' => 12,
'cat_subcats' => 2,
'cat_files' => 1,
],
(object)[
'rc_title' => 'Processed already',
'rc_cur_id' => 31,
'pp_propname' => null,
'cat_pages' => 10,
'cat_subcats' => 2,
'cat_files' => 1,
],
],
[ 31 => true ],
],
// TODO: not sure how to test categorization changes, it uses the database select...
];
}
/**
* Mock category links iterator.
* @param IDatabase $dbr
* @param array $ids
* @return array
*/
public function getCategoryLinksIterator( $dbr, array $ids ) {
$res = [];
foreach ( $ids as $pageid ) {
$res[] = (object)[ 'cl_from' => $pageid, 'cl_to' => "Parent of $pageid" ];
}
return $res;
}
/**
* @dataProvider provideCategoryData
* @param string $testFileName Name of the test, defines filename with expected results.
* @param string $iterator Iterator method name to mock
* @param string $handler Handler method to call
* @param array $result Result to be returned from mock iterator
* @param array $preProcessed List of pre-processed items
*/
public function testSparqlUpdate( $testFileName, $iterator, $handler, $result,
array $preProcessed = [] ) {
$dumpScript =
$this->getMockBuilder( CategoryChangesAsRdf::class )
->onlyMethods( [ $iterator, 'getCategoryLinksIterator' ] )
->getMock();
$dumpScript->method( 'getCategoryLinksIterator' )
->willReturnCallback( [ $this, 'getCategoryLinksIterator' ] );
$dumpScript->expects( $this->once() )
->method( $iterator )
->willReturn( [ $result ] );
$ref = new ReflectionObject( $dumpScript );
$processedProperty = $ref->getProperty( 'processed' );
$processedProperty->setAccessible( true );
$processedProperty->setValue( $dumpScript, $preProcessed );
$output = fopen( "php://memory", "w+b" );
$dbr = $this->getDb();
/** @var CategoryChangesAsRdf $dumpScript */
$dumpScript->initialize();
$dumpScript->getRdf();
$dumpScript->$handler( $dbr, $output );
rewind( $output );
$sparql = stream_get_contents( $output );
$this->assertFileContains( $testFileName, $sparql );
$processed = $processedProperty->getValue( $dumpScript );
$expectedProcessed = array_keys( $preProcessed );
foreach ( $result as $row ) {
if ( isset( $row->_processed ) ) {
$this->assertArrayHasKey( $row->_processed, $processed,
"ID {$row->_processed} was not processed!" );
$expectedProcessed[] = $row->_processed;
}
}
$this->assertSame( $expectedProcessed, array_keys( $processed ),
'Processed array has wrong items' );
}
public function testUpdateTs() {
$dumpScript = new CategoryChangesAsRdf();
$dumpScript->initialize();
$update = $dumpScript->updateTS( 1503620949 );
$outFile = __DIR__ . '/../data/categoriesrdf/updatets.txt';
$this->assertFileContains( $outFile, $update );
}
public function testCategorization() {
$this->overrideConfigValue(
MainConfigNames::RCWatchCategoryMembership,
true
);
$start = new MWTimestamp( "2020-07-31T10:00:00" );
$end = new MWTimestamp( "2020-07-31T10:01:00" );
ConvertibleTimestamp::setFakeTime( "2020-07-31T10:00:00" );
$l1 = Title::makeTitle( NS_CATEGORY, __CLASS__ . "_L1" );
$l2 = Title::makeTitle( NS_CATEGORY, __CLASS__ . "_L2" );
$pageInL2 = Title::makeTitle( NS_CATEGORY, __CLASS__ . "_Page" );
$this->editPage( $l1, "" );
$this->editPage( $l2, "[[{$l1->getPrefixedText()}]]" );
$this->editPage( $pageInL2, "[[{$l2->getPrefixedText()}]]" );
$output = fopen( "php://memory", "w+b" );
$this->runJobs( [], [
'type' => 'categoryMembershipChange'
] );
$dbr = $this->getDb();
$categoryChangesAsRdf = new CategoryChangesAsRdf();
$ref = new ReflectionObject( $categoryChangesAsRdf );
$startTSProp = $ref->getProperty( 'startTS' );
$startTSProp->setAccessible( true );
$startTSProp->setValue( $categoryChangesAsRdf, $start->getTimestamp() );
$endTSProp = $ref->getProperty( 'endTS' );
$endTSProp->setAccessible( true );
$endTSProp->setValue( $categoryChangesAsRdf, $end->getTimestamp() );
$categoryChangesAsRdf->initialize();
$categoryChangesAsRdf->getRdf();
$categoryChangesAsRdf->handleCategorization( $dbr, $output );
rewind( $output );
$sparql = stream_get_contents( $output );
$outFile = __DIR__ . '/../data/categoriesrdf/categorization.txt';
$this->assertFileContains( $outFile, $sparql );
}
}