Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

API Selectively register test folder in manifest #10150

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
"require": {
"bramus/monolog-colored-line-formatter": "^2",
"composer/installers": "^1 || ^2",
"composer/semver": "^1 || ^3",
maxime-rainville marked this conversation as resolved.
Show resolved Hide resolved
"embed/embed": "^3",
"league/csv": "^8 || ^9",
"m1/env": "^2.1",
Expand Down
30 changes: 27 additions & 3 deletions src/Core/CoreKernel.php
Original file line number Diff line number Diff line change
Expand Up @@ -509,6 +509,18 @@ protected function getIncludeTests()
return false;
}

/**
* When manifests are discovering files, tests files in modules using the following CI library type will be ignored.
*
* The purpose of this method is to avoid loading PHPUnit test files with incompatible definitions.
*
* @return string[] List of CI types to ignore as defined by `Module`.
*/
protected function getIgnoredCIConfigs(): array
{
return [];
}

/**
* Boot all manifests
*
Expand All @@ -517,10 +529,18 @@ protected function getIncludeTests()
protected function bootManifests($flush)
{
// Setup autoloader
$this->getClassLoader()->init($this->getIncludeTests(), $flush);
$this->getClassLoader()->init(
$this->getIncludeTests(),
$flush,
$this->getIgnoredCIConfigs()
);

// Find modules
$this->getModuleLoader()->init($this->getIncludeTests(), $flush);
$this->getModuleLoader()->init(
$this->getIncludeTests(),
$flush,
$this->getIgnoredCIConfigs()
);

// Flush config
if ($flush) {
Expand All @@ -538,7 +558,11 @@ protected function bootManifests($flush)
$defaultSet->setProject(
ModuleManifest::config()->get('project')
);
$defaultSet->init($this->getIncludeTests(), $flush);
$defaultSet->init(
$this->getIncludeTests(),
$flush,
$this->getIgnoredCIConfigs()
);
}
}

Expand Down
5 changes: 3 additions & 2 deletions src/Core/Manifest/ClassLoader.php
Original file line number Diff line number Diff line change
Expand Up @@ -120,13 +120,14 @@ public function getItemPath($class)
*
* @param bool $includeTests
* @param bool $forceRegen
* @param string[] $ignoredCIConfigs
*/
public function init($includeTests = false, $forceRegen = false)
public function init($includeTests = false, $forceRegen = false, array $ignoredCIConfigs = [])
{
foreach ($this->manifests as $manifest) {
/** @var ClassManifest $instance */
$instance = $manifest['instance'];
$instance->init($includeTests, $forceRegen);
$instance->init($includeTests, $forceRegen, $ignoredCIConfigs);
}

$this->registerAutoloader();
Expand Down
9 changes: 6 additions & 3 deletions src/Core/Manifest/ClassManifest.php
Original file line number Diff line number Diff line change
Expand Up @@ -260,8 +260,9 @@ public function isFlushed()
*
maxime-rainville marked this conversation as resolved.
Show resolved Hide resolved
* @param bool $includeTests
* @param bool $forceRegen
* @param string[] $ignoredCIConfigs
*/
public function init($includeTests = false, $forceRegen = false)
public function init($includeTests = false, $forceRegen = false, array $ignoredCIConfigs = [])
{
$this->cache = $this->buildCache($includeTests);

Expand All @@ -275,7 +276,7 @@ public function init($includeTests = false, $forceRegen = false)
}

// Build
$this->regenerate($includeTests);
$this->regenerate($includeTests, $ignoredCIConfigs);
}

/**
Expand Down Expand Up @@ -500,8 +501,9 @@ public function getOwnerModule($class)
* Completely regenerates the manifest file.
*
* @param bool $includeTests
* @param string[] $ignoredCIConfigs
*/
public function regenerate($includeTests)
public function regenerate($includeTests, array $ignoredCIConfigs = [])
{
// Reset the manifest so stale info doesn't cause errors.
$this->loadState([]);
Expand All @@ -513,6 +515,7 @@ public function regenerate($includeTests)
'name_regex' => '/^[^_].*\\.php$/',
'ignore_files' => ['index.php', 'cli-script.php'],
'ignore_tests' => !$includeTests,
'ignored_ci_configs' => $ignoredCIConfigs,
'file_callback' => function ($basename, $pathname, $depth) use ($includeTests, $finder) {
$this->handleFile($basename, $pathname, $includeTests);
},
Expand Down
55 changes: 51 additions & 4 deletions src/Core/Manifest/ManifestFileFinder.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,18 @@

namespace SilverStripe\Core\Manifest;

use RuntimeException;
use SilverStripe\Assets\FileFinder;

/**
* An extension to the default file finder with some extra filters to faciliate
* An extension to the default file finder with some extra filters to facilitate
* autoload and template manifest generation:
* - Only modules with _config.php files are scanned.
* - If a _manifest_exclude file is present inside a directory it is ignored.
* - Assets and module language directories are ignored.
* - Module tests directories are skipped if the ignore_tests option is not
* set to false.
* - Module tests directories are skipped if either of these conditions is meant:
* - the `ignore_tests` option is not set to false.
* - the module PHP CI configuration matches one of the `ignored_ci_configs`
*/
class ManifestFileFinder extends FileFinder
{
Expand All @@ -32,7 +34,8 @@ class ManifestFileFinder extends FileFinder
'include_themes' => false,
'ignore_tests' => true,
'min_depth' => 1,
'ignore_dirs' => ['node_modules']
'ignore_dirs' => ['node_modules'],
'ignored_ci_configs' => []
];

public function acceptDir($basename, $pathname, $depth)
Expand Down Expand Up @@ -73,6 +76,14 @@ public function acceptDir($basename, $pathname, $depth)
return false;
}

// Skip if test dir inside vendor module with unexpected CI Configuration
if ($depth > 3 && $basename === self::TESTS_DIR && $ignoredCIConfig = $this->getOption('ignored_ci_configs')) {
$ciLib = $this->findModuleCIPhpConfiguration($basename, $pathname, $depth);
if (in_array($ciLib, $ignoredCIConfig)) {
return false;
}
}

return parent::acceptDir($basename, $pathname, $depth);
}

Expand Down Expand Up @@ -251,4 +262,40 @@ public function isDirectoryIgnored($basename, $pathname, $depth)

return false;
}

/**
* Find out the root of the current module and read the PHP CI configuration from tho composer file
*
* @param string $basename Name of the current folder
* @param string $pathname Full path the parent folder
* @param string $depth Depth of the current folder
*/
private function findModuleCIPhpConfiguration(string $basename, string $pathname, int $depth): string
{
if ($depth < 1) {
// We went all the way back to the root of the project
return Module::CI_UNKNOWN;
}

// We pop the current folder and use the next entry the pathname
$newBasename = basename($pathname);
$newPathname = dirname($pathname);
$newDepth = $depth - 1;

if ($this->isDirectoryModule($newBasename, $newPathname, $newDepth)) {
// We've reached the root of the module folder, we can read the PHP CI config now
$module = new Module($newPathname, $this->upLevels($newPathname, $newDepth));
$config = $module->getCIConfig();

if (empty($config['PHP'])) {
// This should never happen
throw new RuntimeException('Module::getCIConfig() did not return a PHP CI value');
}

return $config['PHP'];
}

// We haven't reach our module root yet ... let's look up one more level
return $this->findModuleCIPhpConfiguration($newBasename, $newPathname, $newDepth);
}
}
141 changes: 137 additions & 4 deletions src/Core/Manifest/Module.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,16 @@

namespace SilverStripe\Core\Manifest;

use Composer\Semver\Semver;
use Exception;
use InvalidArgumentException;
use RuntimeException;
use Serializable;
use SilverStripe\Core\Path;
use SilverStripe\Dev\Deprecation;

/**
* Abstraction of a PHP Package. Can be used to retrieve information about SilverStripe modules, and other packages
* Abstraction of a PHP Package. Can be used to retrieve information about Silverstripe CMS modules, and other packages
* managed via composer, by reading their `composer.json` file.
*/
class Module implements Serializable
Expand All @@ -19,6 +21,23 @@ class Module implements Serializable
*/
const TRIM_CHARS = ' /\\';

/**
* Return value of getCIConfig() when module uses PHPUNit 9
*/
const CI_PHPUNIT_NINE = 'CI_PHPUNIT_NINE';

/**
* Return value of getCIConfig() when module uses PHPUNit 5
*/
const CI_PHPUNIT_FIVE = 'CI_PHPUNIT_FIVE';

/**
* Return value of getCIConfig() when module does not use any CI
*/
const CI_UNKNOWN = 'CI_UNKNOWN';



/**
* Full directory path to this module with no trailing slash
*
Expand Down Expand Up @@ -64,7 +83,7 @@ public function __construct($path, $basePath)
* Gets name of this module. Used as unique key and identifier for this module.
*
* If installed by composer, this will be the full composer name (vendor/name).
* If not insalled by composer this will default to the basedir()
* If not installed by composer this will default to the `basedir()`
*
* @return string
*/
Expand All @@ -74,7 +93,7 @@ public function getName()
}

/**
* Get full composer name. Will be null if no composer.json is available
* Get full composer name. Will be `null` if no composer.json is available
*
* @return string|null
*/
Expand Down Expand Up @@ -130,7 +149,7 @@ public function getShortName()

/**
* Name of the resource directory where vendor resources should be exposed as defined by the `extra.resources-dir`
* key in the composer file. A blank string will will be returned if the key is undefined.
* key in the composer file. A blank string will be returned if the key is undefined.
*
* Only applicable when reading the composer file for the main project.
* @return string
Expand Down Expand Up @@ -275,6 +294,120 @@ public function hasResource($path)
->getResource($path)
->exists();
}

/**
* Determine what configurations the module is using to run various aspects of its CI. THe only aspect
* that is observed is `PHP`
* @return array List of configuration aspects e.g.: `['PHP' => 'CI_PHPUNIT_NINE']`
* @internal
maxime-rainville marked this conversation as resolved.
Show resolved Hide resolved
*/
public function getCIConfig(): array
{
return [
'PHP' => $this->getPhpCiConfig()
];
}

/**
* Determine what CI Configuration the module uses to test its PHP code.
*/
private function getPhpCiConfig(): string
{
// We don't have any composer data at all
if (empty($this->composerData)) {
return self::CI_UNKNOWN;
}

// We don't have any dev dependencies
if (empty($this->composerData['require-dev']) || !is_array($this->composerData['require-dev'])) {
return self::CI_UNKNOWN;
}

// We are assuming a typical setup where the CI lib is defined in require-dev rather than require
$requireDev = $this->composerData['require-dev'];

// Try to pick which CI we are using based on phpunit constraint
$phpUnitConstraint = $this->requireDevConstraint(['sminnee/phpunit', 'phpunit/phpunit']);
if ($phpUnitConstraint) {
if ($this->constraintSatisfies(
$phpUnitConstraint,
['5.7.0', '5.0.0', '5.x-dev', '5.7.x-dev'],
5
)) {
return self::CI_PHPUNIT_FIVE;
}
if ($this->constraintSatisfies(
$phpUnitConstraint,
['9.0.0', '9.5.0', '9.x-dev', '9.5.x-dev'],
9
)) {
return self::CI_PHPUNIT_NINE;
}
}

// Try to pick which CI we are using based on recipe-testing constraint
$recipeTestingConstraint = $this->requireDevConstraint(['silverstripe/recipe-testing']);
if ($recipeTestingConstraint) {
if ($this->constraintSatisfies(
$recipeTestingConstraint,
['1.0.0', '1.1.0', '1.2.0', '1.1.x-dev', '1.2.x-dev', '1.x-dev'],
1
)) {
return self::CI_PHPUNIT_FIVE;
}
if ($this->constraintSatisfies(
$recipeTestingConstraint,
['2.0.0', '2.0.x-dev', '2.x-dev'],
2
)) {
return self::CI_PHPUNIT_NINE;
}
}

return self::CI_UNKNOWN;
}

/**
* Retrieve the constraint for the first module that is found in the require-dev section
* @param string[] $modules
* @return false|string
*/
private function requireDevConstraint(array $modules)
{
if (empty($this->composerData['require-dev']) || !is_array($this->composerData['require-dev'])) {
return false;
}

$requireDev = $this->composerData['require-dev'];
foreach ($modules as $module) {
if (isset($requireDev[$module])) {
return $requireDev[$module];
}
}

return false;
}

/**
* Determines if the provided constraint allows at least one of the version provided
*/
private function constraintSatisfies(
string $constraint,
array $possibleVersions,
int $majorVersionFallback
): bool {
// Let's see of any of our possible versions is allowed by the constraint
if (!empty(Semver::satisfiedBy($possibleVersions, $constraint))) {
return true;
}

// Let's see if we are using an exact version constraint. e.g. ~1.2.3 or 1.2.3 or ~1.2 or 1.2.*
if (preg_match("/^~?$majorVersionFallback(\.(\d+)|\*){0,2}/", $constraint)) {
return true;
}

return false;
}
}

/**
Expand Down
Loading