mirror of
https://github.com/Universal-Omega/DynamicPageList3
synced 2024-12-04 20:48:35 +00:00
1048 lines
34 KiB
PHP
1048 lines
34 KiB
PHP
<?php
|
|
|
|
namespace MediaWiki\Extension\DynamicPageList3;
|
|
|
|
use ExtVariables;
|
|
use MediaWiki\Extension\DynamicPageList3\Heading\Heading;
|
|
use MediaWiki\Extension\DynamicPageList3\Lister\Lister;
|
|
use MediaWiki\MediaWikiServices;
|
|
use MWException;
|
|
use Parser;
|
|
use RequestContext;
|
|
use Title;
|
|
use WebRequest;
|
|
|
|
class Parse {
|
|
/**
|
|
* Parameters Object
|
|
*
|
|
* @var Parameters
|
|
*/
|
|
private $parameters = null;
|
|
|
|
/**
|
|
* Logger Object
|
|
*
|
|
* @var Logger
|
|
*/
|
|
private $logger = null;
|
|
|
|
/**
|
|
* Array of prequoted table names.
|
|
*
|
|
* @var string[]
|
|
*/
|
|
private $tableNames = [];
|
|
|
|
/**
|
|
* Header Output
|
|
*
|
|
* @var string
|
|
*/
|
|
private $header = '';
|
|
|
|
/**
|
|
* Footer Output
|
|
*
|
|
* @var string
|
|
*/
|
|
private $footer = '';
|
|
|
|
/**
|
|
* Body Output
|
|
*
|
|
* @var string
|
|
*/
|
|
private $output = '';
|
|
|
|
/**
|
|
* Replacement Variables
|
|
*
|
|
* @var array
|
|
*/
|
|
private $replacementVariables = [];
|
|
|
|
/**
|
|
* WebRequest object
|
|
*
|
|
* @var WebRequest
|
|
*/
|
|
private $request;
|
|
|
|
/**
|
|
* Array of possible URL arguments.
|
|
*
|
|
* @var array
|
|
*/
|
|
private $urlArguments = [
|
|
'DPL_offset',
|
|
'DPL_count',
|
|
'DPL_fromTitle',
|
|
'DPL_findTitle',
|
|
'DPL_toTitle'
|
|
];
|
|
|
|
public function __construct() {
|
|
$this->parameters = new Parameters();
|
|
$this->logger = new Logger();
|
|
$this->tableNames = Query::getTableNames();
|
|
$this->request = RequestContext::getMain()->getRequest();
|
|
}
|
|
|
|
/**
|
|
* The real callback function for converting the input text to wiki text output
|
|
*
|
|
* @param string $input
|
|
* @param Parser $parser
|
|
* @param array &$reset
|
|
* @param array &$eliminate
|
|
* @param bool $isParserTag
|
|
* @return string
|
|
*
|
|
* @suppress PhanUndeclaredProperty Use of Parser::mTemplatePath
|
|
*/
|
|
public function parse( $input, Parser $parser, &$reset, &$eliminate, $isParserTag = false ) {
|
|
$dplStartTime = microtime( true );
|
|
|
|
// Reset headings when being ran more than once in the same page load.
|
|
Article::resetHeadings();
|
|
|
|
// Check that we are not in an infinite transclusion loop
|
|
if ( isset( $parser->mTemplatePath[$parser->getTitle()->getPrefixedText()] ) ) {
|
|
$this->logger->addMessage( Hooks::WARN_TRANSCLUSIONLOOP, $parser->getTitle()->getPrefixedText() );
|
|
|
|
return $this->getFullOutput();
|
|
}
|
|
|
|
// Check if DPL shall only be executed from protected pages.
|
|
if ( Config::getSetting( 'runFromProtectedPagesOnly' ) === true && !$parser->getTitle()->isProtected( 'edit' ) ) {
|
|
// Ideally we would like to allow using a DPL query if the query istelf is coded on a template page which is protected. Then there would be no need for the article to be protected. However, how can one find out from which wiki source an extension has been invoked???
|
|
$this->logger->addMessage( Hooks::FATAL_NOTPROTECTED, $parser->getTitle()->getPrefixedText() );
|
|
|
|
return $this->getFullOutput();
|
|
}
|
|
|
|
/************************************/
|
|
/* Check for URL Arguments in Input */
|
|
/************************************/
|
|
if ( strpos( $input, '{%DPL_' ) >= 0 ) {
|
|
for ( $i = 1; $i <= 5; $i++ ) {
|
|
$this->urlArguments[] = 'DPL_arg' . $i;
|
|
}
|
|
}
|
|
|
|
$input = $this->resolveUrlArguments( $input, $this->urlArguments );
|
|
$this->getUrlArgs( $parser );
|
|
|
|
$this->parameters->setParameter( 'offset', $this->request->getInt( 'DPL_offset', $this->parameters->getData( 'offset' )['default'] ) );
|
|
|
|
/***************************************/
|
|
/* User Input preparation and parsing. */
|
|
/***************************************/
|
|
$cleanParameters = $this->prepareUserInput( $input );
|
|
if ( !is_array( $cleanParameters ) ) {
|
|
// Short circuit for dumb things.
|
|
$this->logger->addMessage( Hooks::FATAL_NOSELECTION );
|
|
|
|
return $this->getFullOutput();
|
|
}
|
|
|
|
$cleanParameters = Parameters::sortByPriority( $cleanParameters );
|
|
|
|
// To check if pseudo-category of Uncategorized pages is included
|
|
$this->parameters->setParameter( 'includeuncat', false );
|
|
|
|
foreach ( $cleanParameters as $parameter => $option ) {
|
|
foreach ( $option as $_option ) {
|
|
// Parameter functions return true or false. The full parameter data will be passed into the Query object later.
|
|
if ( $this->parameters->$parameter( $_option ) === false ) {
|
|
// Do not build this into the output just yet. It will be collected at the end.
|
|
$this->logger->addMessage( Hooks::WARN_WRONGPARAM, $parameter, $_option );
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************/
|
|
/* Execute and Exit Only */
|
|
/*************************/
|
|
if ( $this->parameters->getParameter( 'execandexit' ) !== null ) {
|
|
// The keyword "geturlargs" is used to return the Url arguments and do nothing else.
|
|
if ( $this->parameters->getParameter( 'execandexit' ) == 'geturlargs' ) {
|
|
return '';
|
|
}
|
|
|
|
// In all other cases we return the value of the argument which may contain parser function calls.
|
|
return $this->parameters->getParameter( 'execandexit' );
|
|
}
|
|
|
|
// Construct internal keys for TableRow according to the structure of "include". This will be needed in the output phase.
|
|
$secLabels = $this->parameters->getParameter( 'seclabels' );
|
|
if ( is_array( $secLabels ) && !empty( $this->parameters->getParameter( 'seclabels' ) ) ) {
|
|
$this->parameters->setParameter( 'tablerow', $this->updateTableRowKeys( $this->parameters->getParameter( 'tablerow' ), $this->parameters->getParameter( 'seclabels' ) ) );
|
|
}
|
|
|
|
/****************/
|
|
/* Check Errors */
|
|
/****************/
|
|
$errors = $this->doQueryErrorChecks();
|
|
if ( $errors === false ) {
|
|
// WHAT HAS HAPPENED OH NOOOOOOOOOOOOO.
|
|
return $this->getFullOutput();
|
|
}
|
|
|
|
$calcRows = false;
|
|
if ( !Config::getSetting( 'allowUnlimitedResults' ) && $this->parameters->getParameter( 'goal' ) != 'categories' && strpos( $this->parameters->getParameter( 'resultsheader' ) . $this->parameters->getParameter( 'noresultsheader' ) . $this->parameters->getParameter( 'resultsfooter' ), '%TOTALPAGES%' ) !== false ) {
|
|
$calcRows = true;
|
|
}
|
|
|
|
/*********/
|
|
/* Query */
|
|
/*********/
|
|
try {
|
|
$query = new Query( $this->parameters );
|
|
|
|
$foundRows = null;
|
|
$rows = $query->buildAndSelect( $calcRows, $foundRows );
|
|
if ( $rows === false ) {
|
|
// This error path is very fast (We exit immediately if poolcounter is full)
|
|
// Thus it should be safe to try again in ~5 minutes.
|
|
$parser->getOutput()->updateCacheExpiry( 4 * 60 + mt_rand( 0, 120 ) );
|
|
|
|
// Pool counter all threads in use.
|
|
$this->logger->addMessage( Hooks::FATAL_POOLCOUNTER );
|
|
return $this->getFullOutput( true );
|
|
}
|
|
} catch ( MWException $e ) {
|
|
$this->logger->addMessage( Hooks::FATAL_SQLBUILDERROR, $e->getMessage() );
|
|
return $this->getFullOutput();
|
|
}
|
|
|
|
$numRows = count( $rows );
|
|
$articles = $this->processQueryResults( $rows, $parser );
|
|
|
|
if ( $query->getSqlQuery() ) {
|
|
$this->addOutput( $query->getSqlQuery() . "\n" );
|
|
}
|
|
|
|
$this->addOutput( '{{Extension DPL}}' );
|
|
|
|
// Preset these to defaults.
|
|
$this->setVariable( 'TOTALPAGES', '0' );
|
|
$this->setVariable( 'PAGES', '0' );
|
|
$this->setVariable( 'VERSION', Hooks::getVersion() );
|
|
|
|
/*********************/
|
|
/* Handle No Results */
|
|
/*********************/
|
|
if ( $numRows == 0 || empty( $articles ) ) {
|
|
return $this->getFullOutput( 0, false );
|
|
}
|
|
|
|
// Backward scrolling: If the user specified only titlelt with descending reverse the output order.
|
|
if ( $this->parameters->getParameter( 'titlelt' ) && !$this->parameters->getParameter( 'titlegt' ) && $this->parameters->getParameter( 'order' ) == 'descending' ) {
|
|
$articles = array_reverse( $articles );
|
|
}
|
|
|
|
// Special sort for card suits (Bridge)
|
|
if ( $this->parameters->getParameter( 'ordersuitsymbols' ) ) {
|
|
$articles = $this->cardSuitSort( $articles );
|
|
}
|
|
|
|
/*******************/
|
|
/* Generate Output */
|
|
/*******************/
|
|
$lister = Lister::newFromStyle( $this->parameters->getParameter( 'mode' ), $this->parameters, $parser );
|
|
$heading = Heading::newFromStyle( $this->parameters->getParameter( 'headingmode' ), $this->parameters );
|
|
if ( $heading !== null ) {
|
|
$this->addOutput( $heading->format( $articles, $lister ) );
|
|
} else {
|
|
$this->addOutput( $lister->format( $articles ) );
|
|
}
|
|
|
|
// $this->addOutput($lister->format($articles));
|
|
if ( $foundRows === null ) {
|
|
// Get row count after calling format() otherwise the count will be inaccurate.
|
|
$foundRows = $lister->getRowCount();
|
|
}
|
|
|
|
/*******************************/
|
|
/* Replacement Variables */
|
|
/*******************************/
|
|
|
|
// Guaranteed to be an accurate count if SQL_CALC_FOUND_ROWS was used. Otherwise only accurate if results are less than the SQL LIMIT.
|
|
$this->setVariable( 'TOTALPAGES', (string)$foundRows );
|
|
|
|
// This could be different than TOTALPAGES. PAGES represents the total results within the constraints of SQL LIMIT.
|
|
$this->setVariable( 'PAGES', $lister->getRowCount() );
|
|
|
|
// Replace %DPLTIME% by execution time and timestamp in header and footer
|
|
$nowTimeStamp = date( 'Y/m/d H:i:s' );
|
|
$dplElapsedTime = sprintf( '%.3f sec.', microtime( true ) - $dplStartTime );
|
|
$dplTime = "{$dplElapsedTime} ({$nowTimeStamp})";
|
|
$this->setVariable( 'DPLTIME', $dplTime );
|
|
|
|
$firstNamespaceFound = '';
|
|
$firstTitleFound = '';
|
|
$lastNamespaceFound = '';
|
|
$lastTitleFound = '';
|
|
|
|
// Replace %LASTTITLE% / %LASTNAMESPACE% by the last title found in header and footer
|
|
$n = count( $articles );
|
|
if ( $n > 0 ) {
|
|
$firstNamespaceFound = str_replace( ' ', '_', $articles[0]->mTitle->getNamespace() );
|
|
$firstTitleFound = str_replace( ' ', '_', $articles[0]->mTitle->getText() );
|
|
$lastNamespaceFound = str_replace( ' ', '_', $articles[$n - 1]->mTitle->getNamespace() );
|
|
$lastTitleFound = str_replace( ' ', '_', $articles[$n - 1]->mTitle->getText() );
|
|
}
|
|
|
|
$this->setVariable( 'FIRSTNAMESPACE', $firstNamespaceFound );
|
|
$this->setVariable( 'FIRSTTITLE', $firstTitleFound );
|
|
$this->setVariable( 'LASTNAMESPACE', $lastNamespaceFound );
|
|
$this->setVariable( 'LASTTITLE', $lastTitleFound );
|
|
$this->setVariable( 'SCROLLDIR', $this->parameters->getParameter( 'scrolldir' ) );
|
|
|
|
/*******************************/
|
|
/* Scroll Variables */
|
|
/*******************************/
|
|
$scrollVariables = [
|
|
'DPL_firstNamespace' => $firstNamespaceFound,
|
|
'DPL_firstTitle' => $firstTitleFound,
|
|
'DPL_lastNamespace' => $lastNamespaceFound,
|
|
'DPL_lastTitle' => $lastTitleFound,
|
|
'DPL_scrollDir' => $this->parameters->getParameter( 'scrolldir' ),
|
|
'DPL_time' => $dplTime,
|
|
'DPL_count' => $this->parameters->getParameter( 'count' ),
|
|
'DPL_totalPages' => $foundRows,
|
|
'DPL_pages' => $lister->getRowCount()
|
|
];
|
|
|
|
$this->defineScrollVariables( $scrollVariables, $parser );
|
|
|
|
if ( $this->parameters->getParameter( 'allowcachedresults' ) || Config::getSetting( 'alwaysCacheResults' ) ) {
|
|
$parser->getOutput()->updateCacheExpiry( $this->parameters->getParameter( 'cacheperiod' ) ?? 3600 );
|
|
} else {
|
|
$parser->getOutput()->updateCacheExpiry( 0 );
|
|
}
|
|
|
|
$finalOutput = $this->getFullOutput( $foundRows, false );
|
|
|
|
$this->triggerEndResets( $finalOutput, $reset, $eliminate, $isParserTag, $parser );
|
|
|
|
return $finalOutput;
|
|
}
|
|
|
|
/**
|
|
* Process Query Results
|
|
*
|
|
* @param $rows
|
|
* @param Parser $parser
|
|
* @return array
|
|
*/
|
|
private function processQueryResults( $rows, Parser $parser ) {
|
|
/*******************************/
|
|
/* Random Count Pick Generator */
|
|
/*******************************/
|
|
$randomCount = $this->parameters->getParameter( 'randomcount' );
|
|
if ( $randomCount > 0 ) {
|
|
$nResults = count( $rows );
|
|
|
|
// Constrain the total amount of random results to not be greater than the total results.
|
|
if ( $randomCount > $nResults ) {
|
|
$randomCount = $nResults;
|
|
}
|
|
|
|
// Generate pick numbers for results.
|
|
$pick = range( 1, $nResults );
|
|
|
|
// Shuffle the pick numbers.
|
|
shuffle( $pick );
|
|
|
|
// Select pick numbers from the beginning to the maximum of $randomCount.
|
|
$pick = array_slice( $pick, 0, $randomCount );
|
|
}
|
|
|
|
$articles = [];
|
|
|
|
/**********************/
|
|
/* Article Processing */
|
|
/**********************/
|
|
$i = 0;
|
|
foreach ( $rows as $row ) {
|
|
$i++;
|
|
|
|
// In random mode skip articles which were not chosen.
|
|
if ( $randomCount > 0 && !in_array( $i, $pick ?? [] ) ) {
|
|
continue;
|
|
}
|
|
|
|
if ( $this->parameters->getParameter( 'goal' ) == 'categories' ) {
|
|
$pageNamespace = NS_CATEGORY;
|
|
$pageTitle = $row->cl_to;
|
|
} elseif ( $this->parameters->getParameter( 'openreferences' ) ) {
|
|
if ( count( $this->parameters->getParameter( 'imagecontainer' ) ?? [] ) > 0 ) {
|
|
$pageNamespace = NS_FILE;
|
|
$pageTitle = $row->il_to;
|
|
} else {
|
|
// Maybe non-existing title
|
|
$pageNamespace = $row->pl_namespace;
|
|
$pageTitle = $row->pl_title;
|
|
}
|
|
} else {
|
|
// Existing PAGE TITLE
|
|
$pageNamespace = $row->page_namespace;
|
|
$pageTitle = $row->page_title;
|
|
}
|
|
|
|
// if subpages are to be excluded: skip them
|
|
if ( !$this->parameters->getParameter( 'includesubpages' ) && strpos( $pageTitle, '/' ) !== false ) {
|
|
continue;
|
|
}
|
|
|
|
$title = Title::makeTitle( $pageNamespace, $pageTitle );
|
|
$thisTitle = $parser->getTitle();
|
|
|
|
// Block recursion from happening by seeing if this result row is the page the DPL query was ran from.
|
|
if ( $this->parameters->getParameter( 'skipthispage' ) && $thisTitle->equals( $title ) ) {
|
|
continue;
|
|
}
|
|
|
|
$articles[] = Article::newFromRow( $row, $this->parameters, $title, $pageNamespace, $pageTitle );
|
|
}
|
|
|
|
return $articles;
|
|
}
|
|
|
|
/**
|
|
* Do basic clean up and structuring of raw user input.
|
|
*
|
|
* @param string $input
|
|
* @return array
|
|
*/
|
|
private function prepareUserInput( $input ) {
|
|
// We replace double angle brackets with single angle brackets to avoid premature tag expansion in the input.
|
|
// The ¦ symbol is an alias for |.
|
|
// The combination '²{' and '}²'will be translated to double curly braces; this allows postponed template execution which is crucial for DPL queries which call other DPL queries.
|
|
$input = str_replace( [ '«', '»', '¦', '²{', '}²' ], [ '<', '>', '|', '{{', '}}' ], $input );
|
|
|
|
// Standard new lines into the standard \n and clean up any hanging new lines.
|
|
$input = str_replace( [ "\r\n", "\r" ], "\n", $input );
|
|
$input = trim( $input, "\n" );
|
|
$rawParameters = explode( "\n", $input );
|
|
|
|
$parameters = [];
|
|
foreach ( $rawParameters as $parameterOption ) {
|
|
if ( empty( $parameterOption ) ) {
|
|
// Softly ignore blank lines.
|
|
continue;
|
|
}
|
|
|
|
if ( strpos( $parameterOption, '=' ) === false ) {
|
|
$this->logger->addMessage( Hooks::WARN_PARAMNOOPTION, $parameterOption );
|
|
|
|
continue;
|
|
}
|
|
|
|
[ $parameter, $option ] = explode( '=', $parameterOption, 2 );
|
|
$parameter = trim( $parameter );
|
|
$option = trim( $option );
|
|
|
|
if ( strpos( $parameter, '<' ) !== false || strpos( $parameter, '>' ) !== false ) {
|
|
// Having the actual less than and greater than symbols is nasty for programatic look up. The old parameter is still supported along with the new, but we just fix it here before calling it.
|
|
$parameter = str_replace( '<', 'lt', $parameter );
|
|
$parameter = str_replace( '>', 'gt', $parameter );
|
|
}
|
|
|
|
// Force lower case for ease of use.
|
|
$parameter = strtolower( $parameter );
|
|
if ( empty( $parameter ) || substr( $parameter, 0, 1 ) == '#' || ( $this->parameters->exists( $parameter ) && !$this->parameters->testRichness( $parameter ) ) ) {
|
|
continue;
|
|
}
|
|
|
|
if ( !$this->parameters->exists( $parameter ) ) {
|
|
$this->logger->addMessage( Hooks::WARN_UNKNOWNPARAM, $parameter, implode( ', ', $this->parameters->getParametersForRichness() ) );
|
|
|
|
continue;
|
|
}
|
|
|
|
// Ignore parameter settings without argument (except namespace and category).
|
|
if ( !strlen( $option ) ) {
|
|
if ( $parameter != 'namespace' && $parameter != 'notnamespace' && $parameter != 'category' && $this->parameters->exists( $parameter ) ) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
$parameters[$parameter][] = $option;
|
|
}
|
|
|
|
return $parameters;
|
|
}
|
|
|
|
/**
|
|
* Concatenate output
|
|
*
|
|
* @param string $output
|
|
*/
|
|
private function addOutput( $output ) {
|
|
$this->output .= $output;
|
|
}
|
|
|
|
/**
|
|
* Set the output text.
|
|
*
|
|
* @return string
|
|
*/
|
|
private function getOutput() {
|
|
// @TODO: 2015-08-28 Consider calling $this->replaceVariables() here. Might cause issues with text returned in the results.
|
|
return $this->output;
|
|
}
|
|
|
|
/**
|
|
* Return output optionally including header and footer.
|
|
*
|
|
* @param bool|int $totalResults
|
|
* @param bool $skipHeaderFooter
|
|
* @return string
|
|
*/
|
|
private function getFullOutput( $totalResults = false, $skipHeaderFooter = true ) {
|
|
if ( !$skipHeaderFooter ) {
|
|
$header = '';
|
|
$footer = '';
|
|
|
|
// Only override header and footers if specified.
|
|
$_headerType = $this->getHeaderFooterType( 'header', (int)$totalResults );
|
|
if ( $_headerType !== false ) {
|
|
$header = $this->parameters->getParameter( $_headerType );
|
|
}
|
|
|
|
$_footerType = $this->getHeaderFooterType( 'footer', (int)$totalResults );
|
|
if ( $_footerType !== false ) {
|
|
$footer = $this->parameters->getParameter( $_footerType );
|
|
}
|
|
|
|
$this->setHeader( $header );
|
|
$this->setFooter( $footer );
|
|
}
|
|
|
|
if ( !$totalResults && !strlen( $this->getHeader() ) && !strlen( $this->getFooter() ) ) {
|
|
$this->logger->addMessage( Hooks::WARN_NORESULTS );
|
|
}
|
|
|
|
$messages = $this->logger->getMessages( false );
|
|
|
|
return ( count( $messages ) ? implode( "<br/>\n", $messages ) : null ) . $this->getHeader() . $this->getOutput() . $this->getFooter();
|
|
}
|
|
|
|
/**
|
|
* Set the header text.
|
|
*
|
|
* @param string $header
|
|
*/
|
|
private function setHeader( $header ) {
|
|
if ( Hooks::getDebugLevel() == 5 ) {
|
|
$header = '<pre><nowiki>' . $header;
|
|
}
|
|
|
|
$this->header = $this->replaceVariables( $header );
|
|
}
|
|
|
|
/**
|
|
* Set the header text.
|
|
*
|
|
* @return string
|
|
*/
|
|
private function getHeader() {
|
|
return $this->header;
|
|
}
|
|
|
|
/**
|
|
* Set the footer text.
|
|
*
|
|
* @param string $footer
|
|
*/
|
|
private function setFooter( $footer ) {
|
|
if ( Hooks::getDebugLevel() == 5 ) {
|
|
$footer .= '</nowiki></pre>';
|
|
}
|
|
|
|
$this->footer = $this->replaceVariables( $footer );
|
|
}
|
|
|
|
/**
|
|
* Set the footer text.
|
|
*
|
|
* @return string
|
|
*/
|
|
private function getFooter() {
|
|
return $this->footer;
|
|
}
|
|
|
|
/**
|
|
* Determine the header/footer type to use based on what output format parameters were chosen and the number of results.
|
|
*
|
|
* @param string $position
|
|
* @param int $count
|
|
* @return mixed Type to use: 'results', 'oneresult', or 'noresults'. False if invalid or none should be used.
|
|
*/
|
|
private function getHeaderFooterType( $position, $count ) {
|
|
$count = intval( $count );
|
|
|
|
if ( $position != 'header' && $position != 'footer' ) {
|
|
return false;
|
|
}
|
|
|
|
if ( $this->parameters->getParameter( 'results' . $position ) !== null && ( $count >= 2 || ( $this->parameters->getParameter( 'oneresult' . $position ) === null && $count >= 1 ) ) ) {
|
|
$_type = 'results' . $position;
|
|
} elseif ( $count === 1 && $this->parameters->getParameter( 'oneresult' . $position ) !== null ) {
|
|
$_type = 'oneresult' . $position;
|
|
} elseif ( $count === 0 && $this->parameters->getParameter( 'noresults' . $position ) !== null ) {
|
|
$_type = 'noresults' . $position;
|
|
} else {
|
|
$_type = false;
|
|
}
|
|
|
|
return $_type;
|
|
}
|
|
|
|
/**
|
|
* Set a variable to be replaced with the provided text later at the end of the output.
|
|
*
|
|
* @param string $variable
|
|
* @param string $replacement
|
|
*/
|
|
private function setVariable( $variable, $replacement ) {
|
|
$variable = "%" . mb_strtoupper( $variable, "UTF-8" ) . "%";
|
|
$this->replacementVariables[$variable] = $replacement;
|
|
}
|
|
|
|
/**
|
|
* Return text with variables replaced.
|
|
*
|
|
* @param string $text
|
|
* @return string
|
|
*/
|
|
private function replaceVariables( $text ) {
|
|
$text = self::replaceNewLines( $text );
|
|
|
|
foreach ( $this->replacementVariables as $variable => $replacement ) {
|
|
$text = str_replace( $variable, $replacement, $text );
|
|
}
|
|
|
|
return $text;
|
|
}
|
|
|
|
/**
|
|
* Return text with custom new line characters replaced.
|
|
*
|
|
* @param string $text
|
|
* @return string
|
|
*/
|
|
public static function replaceNewLines( $text ) {
|
|
return str_replace( [ '\n', "¶" ], "\n", $text );
|
|
}
|
|
|
|
/**
|
|
* Work through processed parameters and check for potential issues.
|
|
*
|
|
* @return bool
|
|
*/
|
|
private function doQueryErrorChecks() {
|
|
/**************************/
|
|
/* Parameter Error Checks */
|
|
/**************************/
|
|
|
|
$totalCategories = 0;
|
|
if ( is_array( $this->parameters->getParameter( 'category' ) ) ) {
|
|
foreach ( $this->parameters->getParameter( 'category' ) as $comparisonType => $operatorTypes ) {
|
|
foreach ( $operatorTypes as $operatorType => $categoryGroups ) {
|
|
foreach ( $categoryGroups as $categories ) {
|
|
if ( is_array( $categories ) ) {
|
|
$totalCategories += count( $categories );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( is_array( $this->parameters->getParameter( 'notcategory' ) ) ) {
|
|
foreach ( $this->parameters->getParameter( 'notcategory' ) as $comparisonType => $operatorTypes ) {
|
|
foreach ( $operatorTypes as $operatorType => $categories ) {
|
|
if ( is_array( $categories ) ) {
|
|
$totalCategories += count( $categories );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Too many categories.
|
|
if ( $totalCategories > Config::getSetting( 'maxCategoryCount' ) && !Config::getSetting( 'allowUnlimitedCategories' ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_TOOMANYCATS, Config::getSetting( 'maxCategoryCount' ) );
|
|
|
|
return false;
|
|
}
|
|
|
|
// Not enough categories.(Really?)
|
|
if ( $totalCategories < Config::getSetting( 'minCategoryCount' ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_TOOFEWCATS, Config::getSetting( 'minCategoryCount' ) );
|
|
|
|
return false;
|
|
}
|
|
|
|
// Selection criteria needs to be found.
|
|
if ( !$totalCategories && !$this->parameters->isSelectionCriteriaFound() ) {
|
|
$this->logger->addMessage( Hooks::FATAL_NOSELECTION );
|
|
|
|
return false;
|
|
}
|
|
|
|
// ordermethod=sortkey requires ordermethod=category
|
|
// Delayed to the construction of the SQL query, see near line 2211, gs
|
|
// if (in_array('sortkey',$aOrderMethods) && ! in_array('category',$aOrderMethods)) $aOrderMethods[] = 'category';
|
|
|
|
$orderMethods = (array)$this->parameters->getParameter( 'ordermethod' );
|
|
// Throw an error in no categories were selected when using category sorting modes or requesting category information.
|
|
if ( $totalCategories == 0 && ( in_array( 'categoryadd', $orderMethods ) || $this->parameters->getParameter( 'addfirstcategorydate' ) === true ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_CATDATEBUTNOINCLUDEDCATS );
|
|
|
|
return false;
|
|
}
|
|
|
|
// No more than one type of date at a time!
|
|
// @TODO: Can this be fixed to allow all three later after fixing the article class?
|
|
if ( ( intval( $this->parameters->getParameter( 'addpagetoucheddate' ) ) + intval( $this->parameters->getParameter( 'addfirstcategorydate' ) ) + intval( $this->parameters->getParameter( 'addeditdate' ) ) ) > 1 ) {
|
|
$this->logger->addMessage( Hooks::FATAL_MORETHAN1TYPEOFDATE );
|
|
|
|
return false;
|
|
}
|
|
|
|
// the dominant section must be one of the sections mentioned in includepage
|
|
if ( $this->parameters->getParameter( 'dominantsection' ) > 0 && count( $this->parameters->getParameter( 'seclabels' ) ) < $this->parameters->getParameter( 'dominantsection' ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_DOMINANTSECTIONRANGE, count( $this->parameters->getParameter( 'seclabels' ) ) );
|
|
|
|
return false;
|
|
}
|
|
|
|
// category-style output requested with not compatible order method
|
|
if ( $this->parameters->getParameter( 'mode' ) == 'category' && !array_intersect( $orderMethods, [ 'sortkey', 'title', 'titlewithoutnamespace' ] ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_WRONGORDERMETHOD, 'mode=category', 'sortkey | title | titlewithoutnamespace' );
|
|
|
|
return false;
|
|
}
|
|
|
|
// addpagetoucheddate=true with unappropriate order methods
|
|
if ( $this->parameters->getParameter( 'addpagetoucheddate' ) && !array_intersect( $orderMethods, [ 'pagetouched', 'title' ] ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_WRONGORDERMETHOD, 'addpagetoucheddate=true', 'pagetouched | title' );
|
|
|
|
return false;
|
|
}
|
|
|
|
// addeditdate=true but not (ordermethod=...,firstedit or ordermethod=...,lastedit)
|
|
// firstedit (resp. lastedit) -> add date of first (resp. last) revision
|
|
if ( $this->parameters->getParameter( 'addeditdate' ) && !array_intersect( $orderMethods, [ 'firstedit', 'lastedit' ] ) && ( $this->parameters->getParameter( 'allrevisionsbefore' ) || $this->parameters->getParameter( 'allrevisionssince' ) || $this->parameters->getParameter( 'firstrevisionsince' ) || $this->parameters->getParameter( 'lastrevisionbefore' ) ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_WRONGORDERMETHOD, 'addeditdate=true', 'firstedit | lastedit' );
|
|
|
|
return false;
|
|
}
|
|
|
|
// adduser=true but not (ordermethod=...,firstedit or ordermethod=...,lastedit)
|
|
/**
|
|
* @todo allow to add user for other order methods.
|
|
* The fact is a page may be edited by multiple users. Which user(s) should we show? all? the first or the last one?
|
|
* Ideally, we could use values such as 'all', 'first' or 'last' for the adduser parameter.
|
|
*/
|
|
if ( $this->parameters->getParameter( 'adduser' ) && !array_intersect( $orderMethods, [ 'firstedit', 'lastedit' ] ) && !$this->parameters->getParameter( 'allrevisionsbefore' ) && !$this->parameters->getParameter( 'allrevisionssince' ) && !$this->parameters->getParameter( 'firstrevisionsince' ) && !$this->parameters->getParameter( 'lastrevisionbefore' ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_WRONGORDERMETHOD, 'adduser=true', 'firstedit | lastedit' );
|
|
|
|
return false;
|
|
}
|
|
|
|
if ( $this->parameters->getParameter( 'minoredits' ) && !array_intersect( $orderMethods, [ 'firstedit', 'lastedit' ] ) ) {
|
|
$this->logger->addMessage( Hooks::FATAL_WRONGORDERMETHOD, 'minoredits', 'firstedit | lastedit' );
|
|
|
|
return false;
|
|
}
|
|
|
|
// add*** parameters have no effect with 'mode=category' (only namespace/title can be viewed in this mode)
|
|
if ( $this->parameters->getParameter( 'mode' ) == 'category' && ( $this->parameters->getParameter( 'addcategories' ) || $this->parameters->getParameter( 'addeditdate' ) || $this->parameters->getParameter( 'addfirstcategorydate' ) || $this->parameters->getParameter( 'addpagetoucheddate' ) || $this->parameters->getParameter( 'incpage' ) || $this->parameters->getParameter( 'adduser' ) || $this->parameters->getParameter( 'addauthor' ) || $this->parameters->getParameter( 'addcontribution' ) || $this->parameters->getParameter( 'addlasteditor' ) ) ) {
|
|
$this->logger->addMessage( Hooks::WARN_CATOUTPUTBUTWRONGPARAMS );
|
|
}
|
|
|
|
// headingmode has effects with ordermethod on multiple components only
|
|
if ( $this->parameters->getParameter( 'headingmode' ) !== 'none' && count( $orderMethods ) < 2 ) {
|
|
$this->logger->addMessage( Hooks::WARN_HEADINGBUTSIMPLEORDERMETHOD, $this->parameters->getParameter( 'headingmode' ), 'none' );
|
|
$this->parameters->setParameter( 'headingmode', 'none' );
|
|
}
|
|
|
|
// The 'openreferences' parameter is incompatible with many other options.
|
|
if ( $this->parameters->isOpenReferencesConflict() && $this->parameters->getParameter( 'openreferences' ) === true ) {
|
|
$this->logger->addMessage( Hooks::FATAL_OPENREFERENCES );
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Create keys for TableRow which represent the structure of the "include=" arguments.
|
|
*
|
|
* @param array $tableRow
|
|
* @param array $sectionLabels
|
|
* @return array
|
|
*/
|
|
private static function updateTableRowKeys( $tableRow, $sectionLabels ) {
|
|
$_tableRow = (array)$tableRow;
|
|
$tableRow = [];
|
|
$groupNr = -1;
|
|
$t = -1;
|
|
|
|
foreach ( $sectionLabels as $label ) {
|
|
$t++;
|
|
$groupNr++;
|
|
$cols = explode( '}:', $label );
|
|
|
|
if ( count( $cols ) <= 1 ) {
|
|
if ( array_key_exists( $t, $_tableRow ) ) {
|
|
$tableRow[$groupNr] = $_tableRow[$t];
|
|
}
|
|
} else {
|
|
$n = count( explode( ':', $cols[1] ) );
|
|
$colNr = -1;
|
|
$t--;
|
|
|
|
for ( $i = 1; $i <= $n; $i++ ) {
|
|
$colNr++;
|
|
$t++;
|
|
|
|
if ( array_key_exists( $t, $_tableRow ) ) {
|
|
$tableRow[$groupNr . '.' . $colNr] = $_tableRow[$t];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return $tableRow;
|
|
}
|
|
|
|
/**
|
|
* Resolve arguments in the input that would normally be in the URL.
|
|
*
|
|
* @param string $input
|
|
* @param array $arguments
|
|
* @return string
|
|
*/
|
|
private function resolveUrlArguments( $input, $arguments ) {
|
|
$arguments = (array)$arguments;
|
|
|
|
foreach ( $arguments as $arg ) {
|
|
$dplArg = $this->request->getVal( $arg, '' );
|
|
|
|
if ( $dplArg == '' ) {
|
|
$input = preg_replace( '/\{%' . $arg . ':(.*)%\}/U', '\1', $input );
|
|
$input = str_replace( '{%' . $arg . '%}', '', $input );
|
|
} else {
|
|
$input = preg_replace( '/\{%' . $arg . ':.*%\}/U ', $dplArg, $input );
|
|
$input = str_replace( '{%' . $arg . '%}', $dplArg, $input );
|
|
}
|
|
}
|
|
|
|
return $input;
|
|
}
|
|
|
|
/**
|
|
* This function uses the Variables extension to provide URL-arguments like &DPL_xyz=abc in the form of a variable which can be accessed as {{#var:xyz}} if Extension:Variables is installed.
|
|
*
|
|
* @param Parser $parser
|
|
*/
|
|
private function getUrlArgs( Parser $parser ) {
|
|
$args = $this->request->getValues();
|
|
|
|
foreach ( $args as $argName => $argValue ) {
|
|
if ( strpos( $argName, 'DPL_' ) === false ) {
|
|
continue;
|
|
}
|
|
|
|
Variables::setVar( [ '', '', $argName, $argValue ] );
|
|
|
|
if ( defined( 'ExtVariables::VERSION' ) ) {
|
|
ExtVariables::get( $parser )->setVarValue( $argName, $argValue );
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This function uses the Variables extension to provide navigation aids such as DPL_firstTitle, DPL_lastTitle, or DPL_findTitle. These variables can be accessed as {{#var:DPL_firstTitle}} if Extension:Variables is installed.
|
|
*
|
|
* @param array $scrollVariables
|
|
* @param Parser $parser
|
|
*/
|
|
private function defineScrollVariables( $scrollVariables, Parser $parser ) {
|
|
$scrollVariables = (array)$scrollVariables;
|
|
|
|
foreach ( $scrollVariables as $variable => $value ) {
|
|
Variables::setVar( [ '', '', $variable, $value ] );
|
|
|
|
if ( defined( 'ExtVariables::VERSION' ) ) {
|
|
ExtVariables::get( $parser )->setVarValue( $variable, $value );
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Trigger Resets and Eliminates that run at the end of parsing.
|
|
*
|
|
* @param string $output
|
|
* @param array &$reset
|
|
* @param array &$eliminate
|
|
* @param bool $isParserTag
|
|
* @param Parser $parser
|
|
*/
|
|
private function triggerEndResets( $output, &$reset, &$eliminate, $isParserTag, Parser $parser ) {
|
|
global $wgHooks;
|
|
|
|
$localParser = MediaWikiServices::getInstance()->getParserFactory()->create();
|
|
$parserOutput = $localParser->parse( $output, $parser->getTitle(), $parser->getOptions() );
|
|
|
|
if ( !is_array( $reset ) ) {
|
|
$reset = [];
|
|
}
|
|
|
|
$reset = array_merge( $reset, (array)$this->parameters->getParameter( 'reset' ) );
|
|
|
|
if ( !is_array( $eliminate ) ) {
|
|
$eliminate = [];
|
|
}
|
|
|
|
$eliminate = array_merge( $eliminate, (array)$this->parameters->getParameter( 'eliminate' ) );
|
|
|
|
if ( $isParserTag === true ) {
|
|
// In tag mode 'eliminate' is the same as 'reset' for templates, categories, and images.
|
|
if ( isset( $eliminate['templates'] ) && $eliminate['templates'] ) {
|
|
$reset['templates'] = true;
|
|
$eliminate['templates'] = false;
|
|
}
|
|
|
|
if ( isset( $eliminate['categories'] ) && $eliminate['categories'] ) {
|
|
$reset['categories'] = true;
|
|
$eliminate['categories'] = false;
|
|
}
|
|
|
|
if ( isset( $eliminate['images'] ) && $eliminate['images'] ) {
|
|
$reset['images'] = true;
|
|
$eliminate['images'] = false;
|
|
}
|
|
} else {
|
|
if ( isset( $reset['templates'] ) && $reset['templates'] ) {
|
|
Hooks::$createdLinks['resetTemplates'] = true;
|
|
}
|
|
|
|
if ( isset( $reset['categories'] ) && $reset['categories'] ) {
|
|
Hooks::$createdLinks['resetCategories'] = true;
|
|
}
|
|
|
|
if ( isset( $reset['images'] ) && $reset['images'] ) {
|
|
Hooks::$createdLinks['resetImages'] = true;
|
|
}
|
|
}
|
|
|
|
if ( ( $isParserTag === true && isset( $reset['links'] ) ) || $isParserTag === false ) {
|
|
if ( isset( $reset['links'] ) ) {
|
|
Hooks::$createdLinks['resetLinks'] = true;
|
|
}
|
|
|
|
// Register a hook to reset links which were produced during parsing DPL output.
|
|
if ( !isset( $wgHooks['ParserAfterTidy'] ) || !is_array( $wgHooks['ParserAfterTidy'] ) || !in_array( 'MediaWiki\\Extension\\DynamicPageList3\\Hooks::endReset', $wgHooks['ParserAfterTidy'] ) ) {
|
|
$wgHooks['ParserAfterTidy'][] = 'MediaWiki\\Extension\\DynamicPageList3\\Hooks::endReset';
|
|
}
|
|
}
|
|
|
|
if ( array_sum( $eliminate ) ) {
|
|
// Register a hook to reset links which were produced during parsing DPL output
|
|
if ( !isset( $wgHooks['ParserAfterTidy'] ) || !is_array( $wgHooks['ParserAfterTidy'] ) || !in_array( 'MediaWiki\\Extension\\DynamicPageList3\\Hooks::endEliminate', $wgHooks['ParserAfterTidy'] ) ) {
|
|
$wgHooks['ParserAfterTidy'][] = 'MediaWiki\\Extension\\DynamicPageList3\\Hooks::endEliminate';
|
|
}
|
|
|
|
if ( isset( $eliminate['links'] ) && $eliminate['links'] ) {
|
|
// Trigger the mediawiki parser to find links, images, categories etc. which are contained in the DPL output. This allows us to remove these links from the link list later. If the article containing the DPL statement itself uses one of these links they will be thrown away!
|
|
Hooks::$createdLinks[0] = [];
|
|
|
|
foreach ( $parserOutput->getLinks() as $nsp => $link ) {
|
|
Hooks::$createdLinks[0][$nsp] = $link;
|
|
}
|
|
}
|
|
|
|
if ( isset( $eliminate['templates'] ) && $eliminate['templates'] ) {
|
|
Hooks::$createdLinks[1] = [];
|
|
|
|
foreach ( $parserOutput->getTemplates() as $nsp => $tpl ) {
|
|
Hooks::$createdLinks[1][$nsp] = $tpl;
|
|
}
|
|
}
|
|
|
|
if ( isset( $eliminate['categories'] ) && $eliminate['categories'] ) {
|
|
Hooks::$createdLinks[2] = $parserOutput->mCategories;
|
|
}
|
|
|
|
if ( isset( $eliminate['images'] ) && $eliminate['images'] ) {
|
|
Hooks::$createdLinks[3] = $parserOutput->mImages;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sort an array of Article objects by the card suit symbol.
|
|
*
|
|
* @param array $articles
|
|
* @return array
|
|
*/
|
|
private function cardSuitSort( $articles ) {
|
|
$sortKeys = [];
|
|
|
|
foreach ( $articles as $key => $article ) {
|
|
$title = preg_replace( '/.*:/', '', $article->mTitle );
|
|
$tokens = preg_split( '/ - */', $title );
|
|
$newKey = '';
|
|
|
|
foreach ( $tokens as $token ) {
|
|
$initial = substr( $token, 0, 1 );
|
|
|
|
if ( $initial >= '1' && $initial <= '7' ) {
|
|
$newKey .= $initial;
|
|
$suit = substr( $token, 1 );
|
|
|
|
if ( $suit == '♣' ) {
|
|
$newKey .= '1';
|
|
} elseif ( $suit == '♦' ) {
|
|
$newKey .= '2';
|
|
} elseif ( $suit == '♥' ) {
|
|
$newKey .= '3';
|
|
} elseif ( $suit == '♠' ) {
|
|
$newKey .= '4';
|
|
} elseif ( strtolower( $suit ) == 'sa' || strtolower( $suit ) == 'nt' ) {
|
|
$newKey .= '5 ';
|
|
} else {
|
|
$newKey .= $suit;
|
|
}
|
|
} elseif ( strtolower( $initial ) == 'p' ) {
|
|
$newKey .= '0 ';
|
|
} elseif ( strtolower( $initial ) == 'x' ) {
|
|
$newKey .= '8 ';
|
|
} else {
|
|
$newKey .= $token;
|
|
}
|
|
}
|
|
|
|
$sortKeys[$key] = $newKey;
|
|
}
|
|
|
|
asort( $sortKeys );
|
|
|
|
$sortedArticles = [];
|
|
|
|
foreach ( $sortKeys as $oldKey => $newKey ) {
|
|
$sortedArticles[] = $articles[$oldKey];
|
|
}
|
|
|
|
return $sortedArticles;
|
|
}
|
|
}
|