class ComposerIntegrationTest in Drupal 10
Same name and namespace in other branches
- 8 core/tests/Drupal/Tests/ComposerIntegrationTest.php \Drupal\Tests\ComposerIntegrationTest
- 9 core/tests/Drupal/Tests/ComposerIntegrationTest.php \Drupal\Tests\ComposerIntegrationTest
Tests Composer integration.
@group Composer
Hierarchy
- class \Drupal\Tests\UnitTestCase extends \PHPUnit\Framework\TestCase uses \Drupal\Tests\PhpUnitCompatibilityTrait, \Symfony\Bridge\PhpUnit\ExpectDeprecationTrait, PhpUnitWarnings
- class \Drupal\Tests\ComposerIntegrationTest uses ComposerIntegrationTrait
Expanded class hierarchy of ComposerIntegrationTest
File
- core/
tests/ Drupal/ Tests/ ComposerIntegrationTest.php, line 14
Namespace
Drupal\TestsView source
class ComposerIntegrationTest extends UnitTestCase {
use ComposerIntegrationTrait;
/**
* Tests composer.lock content-hash.
*
* If you have made a change to composer.json, you may need to reconstruct
* composer.lock. Follow the link below for further instructions.
*
* @see https://www.drupal.org/about/core/policies/core-dependencies-policies/managing-composer-updates-for-drupal-core
*/
public function testComposerLockHash() {
$content_hash = self::getContentHash(file_get_contents($this->root . '/composer.json'));
$lock = json_decode(file_get_contents($this->root . '/composer.lock'), TRUE);
$this
->assertSame($content_hash, $lock['content-hash']);
// @see \Composer\Repository\PathRepository::initialize()
$core_lock_file_hash = '';
$options = [];
foreach ($lock['packages'] as $package) {
if ($package['name'] === 'drupal/core') {
$core_lock_file_hash = $package['dist']['reference'];
$options = $package['transport-options'] ?? [];
break;
}
}
$core_content_hash = sha1(file_get_contents($this->root . '/core/composer.json') . serialize($options));
$this
->assertSame($core_content_hash, $core_lock_file_hash);
}
/**
* Tests composer.json versions.
*
* @param string $path
* Path to a composer.json to test.
*
* @dataProvider providerTestComposerJson
*/
public function testComposerTilde($path) {
if (preg_match('#composer/Metapackage/CoreRecommended/composer.json$#', $path)) {
$this
->markTestSkipped("{$path} has tilde");
}
$content = json_decode(file_get_contents($path), TRUE);
$composer_keys = array_intersect([
'require',
'require-dev',
], array_keys($content));
if (empty($composer_keys)) {
$this
->markTestSkipped("{$path} has no keys to test");
}
foreach ($composer_keys as $composer_key) {
foreach ($content[$composer_key] as $dependency => $version) {
// We allow tildes if the dependency is a Symfony component.
// @see https://www.drupal.org/node/2887000
if (strpos($dependency, 'symfony/') === 0) {
continue;
}
$this
->assertStringNotContainsString('~', $version, "Dependency {$dependency} in {$path} contains a tilde, use a caret.");
}
}
}
/**
* Data provider for all the composer.json provided by Drupal core.
*
* @return array
*/
public function providerTestComposerJson() {
$data = [];
$composer_json_finder = $this
->getComposerJsonFinder(realpath(__DIR__ . '/../../../../'));
foreach ($composer_json_finder
->getIterator() as $composer_json) {
$data[$composer_json
->getPathname()] = [
$composer_json
->getPathname(),
];
}
return $data;
}
/**
* Tests core's composer.json replace section.
*
* Verify that all core components are also listed in the 'replace' section of
* core's composer.json.
*/
public function testAllCoreComponentsReplaced() : void {
// Assemble a path to core components.
$components_path = $this->root . '/core/lib/Drupal/Component';
// Grab the 'replace' section of the core composer.json file.
$json = json_decode(file_get_contents($this->root . '/core/composer.json'), FALSE);
$composer_replace_packages = (array) $json->replace;
// Get a list of all the composer.json files in the components path.
$components_composer_json_files = [];
$composer_json_finder = new Finder();
$composer_json_finder
->name('composer.json')
->in($components_path)
->ignoreUnreadableDirs();
foreach ($composer_json_finder
->getIterator() as $composer_json) {
$components_composer_json_files[$composer_json
->getPathname()] = [
$composer_json
->getPathname(),
];
}
$this
->assertNotEmpty($components_composer_json_files);
$this
->assertCount(count($composer_replace_packages), $components_composer_json_files);
// Assert that each core components has a corresponding 'replace' in
// composer.json.
foreach ($components_composer_json_files as $components_composer_json_file) {
$json = json_decode(file_get_contents(reset($components_composer_json_file)), FALSE);
$component_name = $json->name;
$this
->assertArrayHasKey($component_name, $composer_replace_packages, 'Unable to find ' . $component_name . ' in replace list of composer.json');
}
}
/**
* Data provider for the scaffold files test for Drupal core.
*
* @return array
*/
public function providerTestExpectedScaffoldFiles() {
return [
[
'.editorconfig',
'assets/scaffold/files/editorconfig',
'[project-root]',
],
[
'.gitattributes',
'assets/scaffold/files/gitattributes',
'[project-root]',
],
[
'.csslintrc',
'assets/scaffold/files/csslintrc',
],
[
'.eslintignore',
'assets/scaffold/files/eslintignore',
],
[
'.eslintrc.json',
'assets/scaffold/files/eslintrc.json',
],
[
'.ht.router.php',
'assets/scaffold/files/ht.router.php',
],
[
'.htaccess',
'assets/scaffold/files/htaccess',
],
[
'example.gitignore',
'assets/scaffold/files/example.gitignore',
],
[
'index.php',
'assets/scaffold/files/index.php',
],
[
'INSTALL.txt',
'assets/scaffold/files/drupal.INSTALL.txt',
],
[
'README.md',
'assets/scaffold/files/drupal.README.md',
],
[
'robots.txt',
'assets/scaffold/files/robots.txt',
],
[
'update.php',
'assets/scaffold/files/update.php',
],
[
'web.config',
'assets/scaffold/files/web.config',
],
[
'sites/README.txt',
'assets/scaffold/files/sites.README.txt',
],
[
'sites/development.services.yml',
'assets/scaffold/files/development.services.yml',
],
[
'sites/example.settings.local.php',
'assets/scaffold/files/example.settings.local.php',
],
[
'sites/example.sites.php',
'assets/scaffold/files/example.sites.php',
],
[
'sites/default/default.services.yml',
'assets/scaffold/files/default.services.yml',
],
[
'sites/default/default.settings.php',
'assets/scaffold/files/default.settings.php',
],
[
'modules/README.txt',
'assets/scaffold/files/modules.README.txt',
],
[
'profiles/README.txt',
'assets/scaffold/files/profiles.README.txt',
],
[
'themes/README.txt',
'assets/scaffold/files/themes.README.txt',
],
];
}
/**
* Tests core's composer.json extra drupal-scaffold file-mappings section.
*
* Verify that every file listed in file-mappings exists in its destination
* path (mapping key) and also at its source path (mapping value), and that
* both of these files have exactly the same content.
*
* In Drupal 9, the files at the destination path will be removed. For the
* remainder of the Drupal 8 development cycle, these files will remain in
* order to maintain backwards compatibility with sites based on the template
* project drupal-composer/drupal-project.
*
* See https://www.drupal.org/project/drupal/issues/3075954
*
* @param string $destRelPath
* Path to scaffold file destination location
* @param string $sourceRelPath
* Path to scaffold file source location
* @param string $expectedDestination
* Named location to the destination path of the scaffold file
*
* @dataProvider providerTestExpectedScaffoldFiles
*/
public function testExpectedScaffoldFiles($destRelPath, $sourceRelPath, $expectedDestination = '[web-root]') {
// Grab the 'file-mapping' section of the core composer.json file.
$json = json_decode(file_get_contents($this->root . '/core/composer.json'));
$scaffold_file_mapping = (array) $json->extra->{'drupal-scaffold'}->{'file-mapping'};
// Assert that the 'file-mapping' section has the expected entry.
$this
->assertArrayHasKey("{$expectedDestination}/{$destRelPath}", $scaffold_file_mapping);
$this
->assertEquals($sourceRelPath, $scaffold_file_mapping["{$expectedDestination}/{$destRelPath}"]);
// Assert that the source file exists.
$this
->assertFileExists($this->root . '/core/' . $sourceRelPath);
// Assert that the destination file exists and has the same contents as
// the source file. Note that in Drupal 9, the destination file will be
// removed.
$this
->assertFileExists($this->root . '/' . $destRelPath);
$this
->assertFileEquals($this->root . '/core/' . $sourceRelPath, $this->root . '/' . $destRelPath, 'Scaffold source and destination files must have the same contents.');
}
// phpcs:disable
/**
* The following method is copied from \Composer\Package\Locker.
*
* @see https://github.com/composer/composer
*/
/**
* Returns the md5 hash of the sorted content of the composer file.
*
* @param string $composerFileContents The contents of the composer file.
*
* @return string
*/
protected static function getContentHash($composerFileContents) {
$content = json_decode($composerFileContents, true);
$relevantKeys = array(
'name',
'version',
'require',
'require-dev',
'conflict',
'replace',
'provide',
'minimum-stability',
'prefer-stable',
'repositories',
'extra',
);
$relevantContent = array();
foreach (array_intersect($relevantKeys, array_keys($content)) as $key) {
$relevantContent[$key] = $content[$key];
}
if (isset($content['config']['platform'])) {
$relevantContent['config']['platform'] = $content['config']['platform'];
}
ksort($relevantContent);
return md5(json_encode($relevantContent));
}
// phpcs:enable
/**
* Tests the vendor cleanup utilities do not have obsolete packages listed.
*/
public function testVendorCleanup() : void {
$lock = json_decode(file_get_contents($this->root . '/composer.lock'), TRUE);
$packages = [];
foreach (array_merge($lock['packages'], $lock['packages-dev']) as $package) {
$packages[] = $package['name'];
}
$reflection = new \ReflectionProperty(Config::class, 'defaultConfig');
$reflection
->setAccessible(TRUE);
$config = $reflection
->getValue();
foreach (array_keys($config) as $package) {
$this
->assertContains(strtolower($package), $packages);
}
}
}
Members
Name | Modifiers | Type | Description | Overrides |
---|---|---|---|---|
ComposerIntegrationTest:: |
protected static | function | Returns the md5 hash of the sorted content of the composer file. | |
ComposerIntegrationTest:: |
public | function | Data provider for all the composer.json provided by Drupal core. | |
ComposerIntegrationTest:: |
public | function | Data provider for the scaffold files test for Drupal core. | |
ComposerIntegrationTest:: |
public | function | Tests core's composer.json replace section. | |
ComposerIntegrationTest:: |
public | function | Tests composer.lock content-hash. | |
ComposerIntegrationTest:: |
public | function | Tests composer.json versions. | |
ComposerIntegrationTest:: |
public | function | Tests core's composer.json extra drupal-scaffold file-mappings section. | |
ComposerIntegrationTest:: |
public | function | Tests the vendor cleanup utilities do not have obsolete packages listed. | |
ComposerIntegrationTrait:: |
public | function | Get a Finder object to traverse all of the composer.json files in core. | |
PhpUnitWarnings:: |
private static | property | Deprecation warnings from PHPUnit to raise with @trigger_error(). | |
PhpUnitWarnings:: |
public | function | Converts PHPUnit deprecation warnings to E_USER_DEPRECATED. | |
UnitTestCase:: |
protected | property | The random generator. | |
UnitTestCase:: |
protected | property | The app root. | 1 |
UnitTestCase:: |
protected | function | Returns a stub class resolver. | |
UnitTestCase:: |
public | function | Returns a stub config factory that behaves according to the passed array. | |
UnitTestCase:: |
public | function | Returns a stub config storage that returns the supplied configuration. | |
UnitTestCase:: |
protected | function | Sets up a container with a cache tags invalidator. | |
UnitTestCase:: |
protected | function | Gets the random generator for the utility methods. | |
UnitTestCase:: |
public | function | Returns a stub translation manager that just returns the passed string. | |
UnitTestCase:: |
public | function | Generates a unique random string containing letters and numbers. | |
UnitTestCase:: |
protected | function | 206 | |
UnitTestCase:: |
public static | function |