wiki.techinc.nl/includes/parser/Parser_DiffTest.php
Brion Vibber c83882e96e Revert r39949 "* Revert revert r39662 of my parser changes."
Causes weird regressions on http://meta.wikimedia.org/wiki/Talk:Spam_blacklist
Couldn't isolate to a parser test in a few minutes; some kind of template interaction perhaps.

Sample bad HTML like:
The associated page is used by the Mediawiki <a href="&lt;a href=" class="external free" title="http://www.mediawiki.org/wiki/Extension:SpamBlacklist" rel="nofollow">http://www.mediawiki.org/wiki/Extension:SpamBlacklist</a>" class="extiw" title="mw:Extension:SpamBlacklist"&gt;Spam Blacklist extension, and lists strings of text that may not be used in URLs in any page in Wikimedia Foundation projects (as well as many external wikis). Any meta <a href="/wiki/Administrator" title="Administrator">administrator</a> can edit the spam blacklist. There is also a more aggressive way to block spamming through direct use of <a href="/wiki/Anti-spam_features#.24wgSpamRegex" title="Anti-spam features">$wgSpamRegex</a>. Only <a href="/wiki/Developers" title="Developers" class="mw-redirect">developers</a> can make changes to $wgSpamRegex, and its use is to be avoided whenever possible.
2008-08-25 22:19:50 +00:00

102 lines
2.4 KiB
PHP

<?php
/**
* @ingroup Parser
*/
class Parser_DiffTest
{
var $parsers, $conf;
var $shortOutput = false;
var $dfUniqPrefix;
function __construct( $conf ) {
if ( !isset( $conf['parsers'] ) ) {
throw new MWException( __METHOD__ . ': no parsers specified' );
}
$this->conf = $conf;
$this->dtUniqPrefix = "\x7fUNIQ" . Parser::getRandomString();
}
function init() {
if ( !is_null( $this->parsers ) ) {
return;
}
global $wgHooks;
static $doneHook = false;
if ( !$doneHook ) {
$doneHook = true;
$wgHooks['ParserClearState'][] = array( $this, 'onClearState' );
}
if ( isset( $this->conf['shortOutput'] ) ) {
$this->shortOutput = $this->conf['shortOutput'];
}
foreach ( $this->conf['parsers'] as $i => $parserConf ) {
if ( !is_array( $parserConf ) ) {
$class = $parserConf;
$parserConf = array( 'class' => $parserConf );
} else {
$class = $parserConf['class'];
}
$this->parsers[$i] = new $class( $parserConf );
}
}
function __call( $name, $args ) {
$this->init();
$results = array();
$mismatch = false;
$lastResult = null;
$first = true;
foreach ( $this->parsers as $i => $parser ) {
$currentResult = call_user_func_array( array( &$this->parsers[$i], $name ), $args );
if ( $first ) {
$first = false;
} else {
if ( is_object( $lastResult ) ) {
if ( $lastResult != $currentResult ) {
$mismatch = true;
}
} else {
if ( $lastResult !== $currentResult ) {
$mismatch = true;
}
}
}
$results[$i] = $currentResult;
$lastResult = $currentResult;
}
if ( $mismatch ) {
throw new MWException( "Parser_DiffTest: results mismatch on call to $name\n" .
'Arguments: ' . $this->formatArray( $args ) . "\n" .
'Results: ' . $this->formatArray( $results ) . "\n" );
}
return $lastResult;
}
function formatArray( $array ) {
if ( $this->shortOutput ) {
foreach ( $array as $key => $value ) {
if ( $value instanceof ParserOutput ) {
$array[$key] = "ParserOutput: {$value->getText()}";
}
}
}
return var_export( $array, true );
}
function setFunctionHook( $id, $callback, $flags = 0 ) {
$this->init();
foreach ( $this->parsers as $i => $parser ) {
$parser->setFunctionHook( $id, $callback, $flags );
}
}
function onClearState( &$parser ) {
// hack marker prefixes to get identical output
$parser->mUniqPrefix = $this->dtUniqPrefix;
return true;
}
}