2023-11-06 23:28:03 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
namespace MediaWiki\Tests\ResourceLoader;
|
|
|
|
|
|
2024-01-18 21:03:08 +00:00
|
|
|
use InvalidArgumentException;
|
2023-11-06 23:28:03 +00:00
|
|
|
use MediaWiki\ResourceLoader\CodexModule;
|
2024-01-26 00:03:47 +00:00
|
|
|
use Wikimedia\TestingAccessWrapper;
|
2023-11-06 23:28:03 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @group ResourceLoader
|
|
|
|
|
* @covers \MediaWiki\ResourceLoader\CodexModule
|
|
|
|
|
*/
|
|
|
|
|
class CodexModuleTest extends ResourceLoaderTestCase {
|
|
|
|
|
|
2024-05-01 23:41:05 +00:00
|
|
|
public const FIXTURE_PATH = 'tests/phpunit/data/resourceloader/codex/';
|
2024-01-26 00:03:47 +00:00
|
|
|
|
2023-11-06 23:28:03 +00:00
|
|
|
public static function provideModuleConfig() {
|
2024-05-02 00:14:26 +00:00
|
|
|
yield 'Codex subset' => [
|
2023-11-06 23:28:03 +00:00
|
|
|
[
|
|
|
|
|
'codexComponents' => [ 'CdxButton', 'CdxMessage', 'useModelWrapper' ],
|
|
|
|
|
'codexStyleOnly' => false,
|
|
|
|
|
'codexScriptOnly' => false
|
|
|
|
|
],
|
|
|
|
|
[
|
|
|
|
|
'packageFiles' => [
|
|
|
|
|
'codex.js',
|
|
|
|
|
'_codex/constants.js',
|
2024-01-17 17:17:17 +00:00
|
|
|
'_codex/useSlotContents2.js',
|
2023-11-06 23:28:03 +00:00
|
|
|
'_codex/useWarnOnce.js',
|
|
|
|
|
'_codex/useIconOnlyButton.js',
|
|
|
|
|
'_codex/_plugin-vue_export-helper.js',
|
|
|
|
|
'_codex/CdxButton.js',
|
2024-01-17 17:17:17 +00:00
|
|
|
'_codex/useComputedDirection.js',
|
|
|
|
|
'_codex/useComputedLanguage.js',
|
2023-11-06 23:28:03 +00:00
|
|
|
'_codex/Icon.js',
|
2024-01-17 17:17:17 +00:00
|
|
|
'_codex/CdxMessage.js',
|
|
|
|
|
'_codex/useModelWrapper.js'
|
2023-11-06 23:28:03 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
'styles' => [ 'modules/CdxButton.css', 'modules/CdxIcon.css', 'modules/CdxMessage.css' ]
|
2023-11-06 23:28:03 +00:00
|
|
|
]
|
2024-05-02 00:14:26 +00:00
|
|
|
];
|
|
|
|
|
yield 'Codex subset, style only' => [
|
|
|
|
|
[
|
|
|
|
|
'codexComponents' => [ 'CdxButton', 'CdxMessage' ],
|
|
|
|
|
'codexStyleOnly' => true,
|
|
|
|
|
'codexScriptOnly' => false
|
2023-11-06 23:28:03 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'packageFiles' => [],
|
|
|
|
|
'styles' => [ 'modules/CdxButton.css', 'modules/CdxIcon.css', 'modules/CdxMessage.css' ]
|
|
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
yield 'Codex subset, script only' => [
|
|
|
|
|
[
|
|
|
|
|
'codexComponents' => [ 'CdxButton', 'CdxMessage', 'useModelWrapper' ],
|
|
|
|
|
'codexStyleOnly' => false,
|
|
|
|
|
'codexScriptOnly' => true
|
|
|
|
|
],
|
|
|
|
|
[
|
|
|
|
|
'packageFiles' => [
|
|
|
|
|
'codex.js',
|
|
|
|
|
'_codex/constants.js',
|
|
|
|
|
'_codex/useSlotContents2.js',
|
|
|
|
|
'_codex/useWarnOnce.js',
|
|
|
|
|
'_codex/useIconOnlyButton.js',
|
|
|
|
|
'_codex/_plugin-vue_export-helper.js',
|
|
|
|
|
'_codex/CdxButton.js',
|
|
|
|
|
'_codex/useComputedDirection.js',
|
|
|
|
|
'_codex/useComputedLanguage.js',
|
|
|
|
|
'_codex/Icon.js',
|
|
|
|
|
'_codex/CdxMessage.js',
|
|
|
|
|
'_codex/useModelWrapper.js'
|
2023-11-06 23:28:03 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
'styles' => []
|
|
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
yield 'Exception thrown when a chunk is requested' => [
|
|
|
|
|
[
|
|
|
|
|
'codexComponents' => [ 'CdxButton', 'buttonHelpers' ],
|
|
|
|
|
],
|
|
|
|
|
[
|
|
|
|
|
'exception' => [
|
|
|
|
|
'class' => InvalidArgumentException::class,
|
|
|
|
|
'message' => '"buttonHelpers" is not an export of Codex and cannot be included in the "codexComponents" array.'
|
2023-11-06 23:28:03 +00:00
|
|
|
]
|
2024-05-02 00:14:26 +00:00
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
yield 'Exception thrown when a nonexistent file is requested' => [
|
|
|
|
|
[
|
|
|
|
|
'codexComponents' => [ 'CdxButton', 'blahblahidontexistblah' ],
|
2023-11-06 23:28:03 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'exception' => [
|
|
|
|
|
'class' => InvalidArgumentException::class,
|
|
|
|
|
'message' => '"blahblahidontexistblah" is not an export of Codex and cannot be included in the "codexComponents" array.'
|
2023-11-06 23:28:03 +00:00
|
|
|
]
|
2024-05-02 00:14:26 +00:00
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
yield 'Exception thrown when codexComponents is empty in the module definition' => [
|
|
|
|
|
[
|
|
|
|
|
'codexComponents' => []
|
2024-01-18 21:03:08 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'exception' => [
|
|
|
|
|
'class' => InvalidArgumentException::class,
|
|
|
|
|
'message' => "All 'codexComponents' properties in your module definition file " .
|
|
|
|
|
'must either be omitted or be an array with at least one component name'
|
2024-01-18 21:03:08 +00:00
|
|
|
]
|
2024-05-02 00:14:26 +00:00
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
yield 'Exception thrown when codexComponents is not an array in the module definition' => [
|
|
|
|
|
[
|
|
|
|
|
'codexComponents' => ''
|
2024-01-18 03:52:38 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'exception' => [
|
|
|
|
|
'class' => InvalidArgumentException::class,
|
|
|
|
|
'message' => "All 'codexComponents' properties in your module definition file " .
|
|
|
|
|
'must either be omitted or be an array with at least one component name'
|
2024-01-26 00:03:47 +00:00
|
|
|
]
|
2024-05-02 00:14:26 +00:00
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
yield 'Exception thrown when the @wikimedia/codex module is required' => [
|
|
|
|
|
[
|
|
|
|
|
'codexComponents' => [ 'CdxButton', 'buttonHelpers' ],
|
|
|
|
|
'dependencies' => [ '@wikimedia/codex' ]
|
2024-01-26 00:03:47 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'exception' => [
|
|
|
|
|
'class' => InvalidArgumentException::class,
|
|
|
|
|
'message' => 'ResourceLoader modules using the CodexModule class cannot ' .
|
|
|
|
|
"list the '@wikimedia/codex' module as a dependency. " .
|
|
|
|
|
"Instead, use 'codexComponents' to require a subset of components."
|
2024-01-18 03:52:38 +00:00
|
|
|
]
|
2024-05-02 00:14:26 +00:00
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
yield 'Full library' => [
|
|
|
|
|
[
|
|
|
|
|
'codexFullLibrary' => true
|
2024-01-18 03:52:38 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'packageFiles' => [
|
|
|
|
|
'codex.js'
|
2024-01-18 03:52:38 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
'styles' => [
|
|
|
|
|
'codex.style.css'
|
2024-01-18 03:52:38 +00:00
|
|
|
]
|
2024-05-02 00:14:26 +00:00
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
yield 'Full library, script only' => [
|
|
|
|
|
[
|
|
|
|
|
'codexFullLibrary' => true,
|
|
|
|
|
'codexScriptOnly' => true
|
2024-01-18 03:52:38 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'packageFiles' => [
|
|
|
|
|
'codex.js'
|
2024-02-08 21:55:47 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
'styles' => []
|
|
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
yield 'Full library, style only' => [
|
|
|
|
|
[
|
|
|
|
|
'codexFullLibrary' => true,
|
|
|
|
|
'codexStyleOnly' => true
|
2024-02-08 21:55:47 +00:00
|
|
|
],
|
2024-05-02 00:14:26 +00:00
|
|
|
[
|
|
|
|
|
'packageFiles' => [],
|
|
|
|
|
'styles' => [
|
|
|
|
|
'codex.style.css'
|
|
|
|
|
]
|
|
|
|
|
]
|
2023-11-06 23:28:03 +00:00
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @dataProvider provideModuleConfig
|
|
|
|
|
*/
|
2024-05-02 00:14:26 +00:00
|
|
|
public function testCodexSubset( $moduleDefinition, $expected ) {
|
2024-01-18 21:03:08 +00:00
|
|
|
if ( isset( $expected['exception'] ) ) {
|
|
|
|
|
$this->expectException( $expected['exception']['class'] );
|
|
|
|
|
$this->expectExceptionMessage( $expected['exception']['message'] );
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-06 23:28:03 +00:00
|
|
|
$testModule = new class( $moduleDefinition ) extends CodexModule {
|
2024-05-01 23:41:05 +00:00
|
|
|
public const CODEX_LIBRARY_DIR = CodexModuleTest::FIXTURE_PATH;
|
2023-11-06 23:28:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
$context = $this->getResourceLoaderContext();
|
|
|
|
|
$config = $context->getResourceLoader()->getConfig();
|
|
|
|
|
$testModule->setConfig( $config );
|
|
|
|
|
|
|
|
|
|
$packageFiles = $testModule->getPackageFiles( $context );
|
2024-01-18 21:03:08 +00:00
|
|
|
$styleFiles = $testModule->getStyleFiles( $context );
|
|
|
|
|
|
2023-11-06 23:28:03 +00:00
|
|
|
// Style-only module will not have any packageFiles.
|
|
|
|
|
$packageFilenames = isset( $packageFiles ) ? array_keys( $packageFiles[ 'files' ] ) : [];
|
2024-05-02 00:14:26 +00:00
|
|
|
$this->assertEquals( $expected[ 'packageFiles' ] ?? [], $packageFilenames, 'Correct packageFiles added' );
|
2023-11-06 23:28:03 +00:00
|
|
|
|
|
|
|
|
// Script-only module will not have any styleFiles.
|
|
|
|
|
$styleFilenames = [];
|
|
|
|
|
if ( count( $styleFiles ) > 0 ) {
|
|
|
|
|
$styleFilenames = array_map( static function ( $filepath ) use ( $testModule ) {
|
2024-05-02 00:14:26 +00:00
|
|
|
return str_replace( $testModule::CODEX_LIBRARY_DIR, '', $filepath->getPath() );
|
2023-11-06 23:28:03 +00:00
|
|
|
}, $styleFiles[ 'all' ] );
|
|
|
|
|
}
|
2024-05-02 00:14:26 +00:00
|
|
|
$this->assertEquals( $expected[ 'styles' ] ?? [], $styleFilenames, 'Correct styleFiles added' );
|
2023-11-06 23:28:03 +00:00
|
|
|
}
|
2024-01-17 05:11:48 +00:00
|
|
|
|
|
|
|
|
public function testMissingCodexComponentsDefinition() {
|
|
|
|
|
$moduleDefinition = [
|
|
|
|
|
'codexComponents' => [ 'CdxButton', 'CdxMessage' ]
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
$testModule = new class( $moduleDefinition ) extends CodexModule {
|
2024-05-01 23:41:05 +00:00
|
|
|
public const CODEX_LIBRARY_DIR = CodexModuleTest::FIXTURE_PATH;
|
2024-01-17 05:11:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
$context = $this->getResourceLoaderContext();
|
|
|
|
|
$config = $context->getResourceLoader()->getConfig();
|
|
|
|
|
$testModule->setConfig( $config );
|
|
|
|
|
|
|
|
|
|
$packageFiles = $testModule->getPackageFiles( $context );
|
|
|
|
|
|
|
|
|
|
$codexPackageFileContent = $packageFiles[ 'files' ][ 'codex.js' ][ 'content' ];
|
|
|
|
|
$expectedProxiedExports = '{"CdxButton":require( "./_codex/CdxButton.js" ),'
|
|
|
|
|
. '"CdxMessage":require( "./_codex/CdxMessage.js" )}';
|
|
|
|
|
|
|
|
|
|
// Components defined in the 'codexComponents' array should be proxied in the codex.js
|
|
|
|
|
// package file so that missing components will throw a custom error when required.
|
|
|
|
|
// By asserting what components are proxied, we are indirectly asserting that missing
|
|
|
|
|
// components would throw an error when required.
|
|
|
|
|
$this->assertStringContainsString( $expectedProxiedExports, $codexPackageFileContent );
|
|
|
|
|
}
|
2024-01-26 00:03:47 +00:00
|
|
|
|
|
|
|
|
public function testGetManifestFile() {
|
|
|
|
|
$moduleDefinition = [ 'codexComponents' => [ 'CdxButton', 'CdxMessage' ] ];
|
|
|
|
|
$testModule = new class( $moduleDefinition ) extends CodexModule {
|
2024-05-01 23:41:05 +00:00
|
|
|
public const CODEX_LIBRARY_DIR = CodexModuleTest::FIXTURE_PATH;
|
2024-01-26 00:03:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
$context = $this->getResourceLoaderContext();
|
|
|
|
|
$testWrapper = TestingAccessWrapper::newFromObject( $testModule );
|
|
|
|
|
|
|
|
|
|
// By default, look for a manifest file called "manifest.json"
|
|
|
|
|
$this->assertEquals(
|
2024-05-01 23:41:05 +00:00
|
|
|
MW_INSTALL_PATH . '/' . self::FIXTURE_PATH . 'modules/manifest.json',
|
2024-01-26 00:03:47 +00:00
|
|
|
$testWrapper->getManifestFilePath( $context )
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-02 22:14:37 +00:00
|
|
|
/**
|
|
|
|
|
* Test that the manifest data structure is transformed correctly.
|
|
|
|
|
* This test relies on the fixture manifest data that lives in
|
|
|
|
|
* tests/phpunit/data/resourceloader/codexModules
|
|
|
|
|
*/
|
2024-01-26 00:03:47 +00:00
|
|
|
public function testGetCodexFiles() {
|
|
|
|
|
$moduleDefinition = [ 'codexComponents' => [ 'CdxButton', 'CdxMessage' ] ];
|
|
|
|
|
$testModule = new class( $moduleDefinition ) extends CodexModule {
|
2024-05-01 23:41:05 +00:00
|
|
|
public const CODEX_LIBRARY_DIR = CodexModuleTest::FIXTURE_PATH;
|
2024-01-26 00:03:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
$context = $this->getResourceLoaderContext();
|
|
|
|
|
$testWrapper = TestingAccessWrapper::newFromObject( $testModule );
|
|
|
|
|
$codexFiles = $testWrapper->getCodexFiles( $context );
|
|
|
|
|
|
|
|
|
|
// The transformed data structure should have a "files" and a "components" array.
|
|
|
|
|
$this->assertIsArray( $codexFiles );
|
|
|
|
|
$this->assertArrayHasKey( 'files', $codexFiles );
|
|
|
|
|
$this->assertArrayHasKey( 'components', $codexFiles );
|
|
|
|
|
|
|
|
|
|
// The "components" array should contain keys like "CdxButton"
|
|
|
|
|
// with values like "CdxButton.js" (matching the names in the manifest)
|
|
|
|
|
$this->assertArrayHasKey( 'CdxButton', $codexFiles[ 'components' ] );
|
|
|
|
|
$this->assertEquals( 'CdxButton.js', $codexFiles[ 'components' ][ 'CdxButton' ] );
|
|
|
|
|
|
|
|
|
|
// The "files" array should contains keys like "CdxButton.js"
|
|
|
|
|
// Items in this array are themselves arrays with "styles" and "dependencies" keys.
|
|
|
|
|
$this->assertArrayHasKey( 'CdxButton.js', $codexFiles[ 'files' ] );
|
|
|
|
|
$this->assertArrayHasKey( 'styles', $codexFiles[ 'files' ][ 'CdxButton.js' ] );
|
|
|
|
|
$this->assertArrayHasKey( 'dependencies', $codexFiles[ 'files' ][ 'CdxButton.js' ] );
|
|
|
|
|
}
|
2023-11-06 23:28:03 +00:00
|
|
|
}
|