2019-08-03 18:19:46 +00:00
|
|
|
<?php
|
|
|
|
|
2022-07-30 18:22:25 +00:00
|
|
|
use MediaWiki\Extension\Scribunto\Engines\LuaSandbox\LuaSandboxEngine;
|
2022-07-30 18:52:37 +00:00
|
|
|
use MediaWiki\Extension\Scribunto\Engines\LuaStandalone\LuaStandaloneEngine;
|
|
|
|
use MediaWiki\Extension\Scribunto\ScribuntoEngineBase;
|
2020-04-16 20:32:58 +00:00
|
|
|
use MediaWiki\MediaWikiServices;
|
2019-10-06 03:17:05 +00:00
|
|
|
use PHPUnit\Framework\DataProviderTestSuite;
|
|
|
|
use PHPUnit\Framework\TestSuite;
|
|
|
|
use PHPUnit\Framework\WarningTestCase;
|
|
|
|
use PHPUnit\Util\Test;
|
|
|
|
|
2019-08-03 18:19:46 +00:00
|
|
|
/**
|
|
|
|
* Trait that helps LuaEngineTestBase and LuaEngineUnitTestBase
|
|
|
|
*/
|
2022-08-03 10:03:12 +00:00
|
|
|
trait Scribunto_LuaEngineTestHelper {
|
2020-12-18 19:04:17 +00:00
|
|
|
/** @var array[] */
|
2019-08-03 18:19:46 +00:00
|
|
|
private static $engineConfigurations = [
|
|
|
|
'LuaSandbox' => [
|
2022-07-30 18:22:25 +00:00
|
|
|
'class' => LuaSandboxEngine::class,
|
2019-08-03 18:19:46 +00:00
|
|
|
'memoryLimit' => 50000000,
|
|
|
|
'cpuLimit' => 30,
|
|
|
|
'allowEnvFuncs' => true,
|
|
|
|
'maxLangCacheSize' => 30,
|
|
|
|
],
|
|
|
|
'LuaStandalone' => [
|
2022-07-30 18:52:37 +00:00
|
|
|
'class' => LuaStandaloneEngine::class,
|
2019-08-03 18:19:46 +00:00
|
|
|
'errorFile' => null,
|
|
|
|
'luaPath' => null,
|
|
|
|
'memoryLimit' => 50000000,
|
|
|
|
'cpuLimit' => 30,
|
|
|
|
'allowEnvFuncs' => true,
|
|
|
|
'maxLangCacheSize' => 30,
|
|
|
|
],
|
|
|
|
];
|
2022-09-22 07:45:47 +00:00
|
|
|
/** @var int[] */
|
|
|
|
protected $templateLoadCounts = [];
|
2019-08-03 18:19:46 +00:00
|
|
|
|
2020-01-14 18:50:34 +00:00
|
|
|
/**
|
|
|
|
* Create a PHPUnit test suite to run the test against all engines
|
|
|
|
* @param string $className Test class name
|
|
|
|
* @param string|null $group Engine to run with, or null to run all engines
|
|
|
|
* @return TestSuite
|
|
|
|
*/
|
2019-08-03 18:19:46 +00:00
|
|
|
protected static function makeSuite( $className, $group = null ) {
|
2019-10-06 03:17:05 +00:00
|
|
|
$suite = new TestSuite;
|
2019-08-03 18:19:46 +00:00
|
|
|
$suite->setName( $className );
|
|
|
|
|
|
|
|
$class = new ReflectionClass( $className );
|
|
|
|
|
|
|
|
foreach ( self::$engineConfigurations as $engineName => $opts ) {
|
|
|
|
if ( $group !== null && $group !== $engineName ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2020-04-16 20:32:58 +00:00
|
|
|
$parser = MediaWikiServices::getInstance()->getParserFactory()->create();
|
2020-03-07 02:31:53 +00:00
|
|
|
$parser->startExternalParse(
|
|
|
|
Title::newMainPage(),
|
|
|
|
ParserOptions::newFromAnon(),
|
|
|
|
Parser::OT_HTML,
|
|
|
|
true
|
|
|
|
);
|
2022-07-30 18:22:25 +00:00
|
|
|
$engineClass = $opts['class'];
|
2019-08-03 18:19:46 +00:00
|
|
|
$engine = new $engineClass(
|
|
|
|
self::$engineConfigurations[$engineName] + [ 'parser' => $parser ]
|
|
|
|
);
|
|
|
|
$parser->scribunto_engine = $engine;
|
|
|
|
$engine->setTitle( $parser->getTitle() );
|
|
|
|
$engine->getInterpreter();
|
2022-08-03 10:03:12 +00:00
|
|
|
} catch ( Scribunto_LuaInterpreterNotFoundError $e ) {
|
2019-08-03 18:19:46 +00:00
|
|
|
$suite->addTest(
|
2022-08-03 10:03:12 +00:00
|
|
|
new Scribunto_LuaEngineTestSkip(
|
2019-08-03 18:19:46 +00:00
|
|
|
$className, "interpreter for $engineName is not available"
|
|
|
|
), [ 'Lua', $engineName ]
|
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Work around PHPUnit breakage: the only straightforward way to
|
2019-10-06 03:17:05 +00:00
|
|
|
// get the data provider is to call Test::getProvidedData, but that
|
|
|
|
// instantiates the class without passing any parameters to the
|
|
|
|
// constructor. But we *need* that engine name.
|
2019-08-03 18:19:46 +00:00
|
|
|
self::$staticEngineName = $engineName;
|
|
|
|
|
2019-10-06 03:17:05 +00:00
|
|
|
$engineSuite = new DataProviderTestSuite;
|
2019-08-03 18:19:46 +00:00
|
|
|
$engineSuite->setName( "$engineName: $className" );
|
|
|
|
|
|
|
|
foreach ( $class->getMethods() as $method ) {
|
2020-01-23 16:21:53 +00:00
|
|
|
if ( Test::isTestMethod( $method ) && $method->isPublic() ) {
|
2019-08-03 18:19:46 +00:00
|
|
|
$name = $method->getName();
|
2019-10-06 03:17:05 +00:00
|
|
|
$groups = Test::getGroups( $className, $name );
|
2019-08-03 18:19:46 +00:00
|
|
|
$groups[] = 'Lua';
|
|
|
|
$groups[] = $engineName;
|
2020-04-08 21:27:28 +00:00
|
|
|
// Only run tests locally if the engine isn't the MW sandbox T125050
|
|
|
|
if ( $engineName !== 'LuaSandbox' ) {
|
|
|
|
$groups[] = 'Standalone';
|
|
|
|
}
|
2019-08-03 18:19:46 +00:00
|
|
|
$groups = array_unique( $groups );
|
|
|
|
|
2019-10-06 03:17:05 +00:00
|
|
|
$data = Test::getProvidedData( $className, $name );
|
|
|
|
if ( is_iterable( $data ) ) {
|
2019-08-03 18:19:46 +00:00
|
|
|
// with @dataProvider
|
2019-10-06 03:17:05 +00:00
|
|
|
$dataSuite = new DataProviderTestSuite(
|
2019-08-03 18:19:46 +00:00
|
|
|
$className . '::' . $name
|
|
|
|
);
|
|
|
|
foreach ( $data as $k => $v ) {
|
|
|
|
$dataSuite->addTest(
|
|
|
|
new $className( $name, $v, $k, $engineName ),
|
|
|
|
$groups
|
|
|
|
);
|
|
|
|
}
|
|
|
|
$engineSuite->addTest( $dataSuite );
|
|
|
|
} elseif ( $data === false ) {
|
|
|
|
// invalid @dataProvider
|
2019-10-06 03:17:05 +00:00
|
|
|
$engineSuite->addTest( new WarningTestCase(
|
2019-08-03 18:19:46 +00:00
|
|
|
"The data provider specified for {$className}::$name is invalid."
|
|
|
|
) );
|
|
|
|
} else {
|
|
|
|
// no @dataProvider
|
|
|
|
$engineSuite->addTest(
|
|
|
|
new $className( $name, [], '', $engineName ),
|
|
|
|
$groups
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$suite->addTest( $engineSuite );
|
|
|
|
}
|
|
|
|
|
|
|
|
return $suite;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return ScribuntoEngineBase
|
|
|
|
*/
|
|
|
|
protected function getEngine() {
|
|
|
|
if ( !$this->engine ) {
|
2020-04-16 20:32:58 +00:00
|
|
|
$parser = MediaWikiServices::getInstance()->getParserFactory()->create();
|
2020-03-07 02:31:53 +00:00
|
|
|
$options = ParserOptions::newFromAnon();
|
2019-08-03 18:19:46 +00:00
|
|
|
$options->setTemplateCallback( [ $this, 'templateCallback' ] );
|
|
|
|
$parser->startExternalParse( $this->getTestTitle(), $options, Parser::OT_HTML, true );
|
2022-07-30 18:52:37 +00:00
|
|
|
|
|
|
|
// HACK
|
|
|
|
if ( $this->engineName === 'LuaSandbox' ) {
|
|
|
|
$class = LuaSandboxEngine::class;
|
|
|
|
} elseif ( $this->engineName === 'LuaStandalone' ) {
|
|
|
|
$class = LuaStandaloneEngine::class;
|
|
|
|
}
|
|
|
|
|
2019-08-03 18:19:46 +00:00
|
|
|
$this->engine = new $class(
|
|
|
|
self::$engineConfigurations[$this->engineName] + [ 'parser' => $parser ]
|
|
|
|
);
|
|
|
|
$parser->scribunto_engine = $this->engine;
|
|
|
|
$this->engine->setTitle( $parser->getTitle() );
|
|
|
|
}
|
|
|
|
return $this->engine;
|
|
|
|
}
|
|
|
|
|
2020-01-14 18:50:34 +00:00
|
|
|
/**
|
|
|
|
* @see Parser::statelessFetchTemplate
|
|
|
|
* @param Title $title
|
|
|
|
* @param Parser|false $parser
|
|
|
|
* @return array
|
|
|
|
*/
|
2019-08-03 18:19:46 +00:00
|
|
|
public function templateCallback( $title, $parser ) {
|
2022-09-22 07:45:47 +00:00
|
|
|
$this->templateLoadCounts[$title->getFullText()] =
|
|
|
|
( $this->templateLoadCounts[$title->getFullText()] ?? 0 ) + 1;
|
2019-08-03 18:19:46 +00:00
|
|
|
if ( isset( $this->extraModules[$title->getFullText()] ) ) {
|
|
|
|
return [
|
|
|
|
'text' => $this->extraModules[$title->getFullText()],
|
|
|
|
'finalTitle' => $title,
|
|
|
|
'deps' => []
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
$modules = $this->getTestModules();
|
|
|
|
foreach ( $modules as $name => $fileName ) {
|
|
|
|
$modTitle = Title::makeTitle( NS_MODULE, $name );
|
|
|
|
if ( $modTitle->equals( $title ) ) {
|
|
|
|
return [
|
|
|
|
'text' => file_get_contents( $fileName ),
|
|
|
|
'finalTitle' => $title,
|
|
|
|
'deps' => []
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Parser::statelessFetchTemplate( $title, $parser );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the title used for unit tests
|
|
|
|
*
|
|
|
|
* @return Title
|
|
|
|
*/
|
|
|
|
protected function getTestTitle() {
|
|
|
|
return Title::newMainPage();
|
|
|
|
}
|
|
|
|
|
2020-01-02 19:51:52 +00:00
|
|
|
/**
|
|
|
|
* Reset the cached engine. The next call to getEngine() will return a new
|
|
|
|
* object.
|
|
|
|
*/
|
|
|
|
protected function resetEngine() {
|
|
|
|
$this->engine = null;
|
|
|
|
}
|
|
|
|
|
2019-08-03 18:19:46 +00:00
|
|
|
}
|