Skip to content
Snippets Groups Projects
Verified Commit bef045d9 authored by Alex Pott's avatar Alex Pott
Browse files

Issue #3450616 by catch, quietone, smustgrave: Optimize test order when --directory is used

(cherry picked from commit af883167)
parent 83ff161c
No related branches found
No related tags found
34 merge requests!10663Issue #3495778: Update phpdoc in FileSaveHtaccessLoggingTest,!10451Issue #3472458 by watergate, smustgrave: CKEditor 5 show blocks label is not translated,!103032838547 Fix punctuation rules for inline label suffix colon with CSS only,!10150Issue #3467294 by quietone, nod_, smustgrave, catch, longwave: Change string...,!10130Resolve #3480321 "Second level menu",!9936Issue #3483087: Check the module:// prefix in the translation server path and replace it with the actual module path,!9933Issue #3394728 by ankondrat4: Undefined array key "#prefix" and deprecated function: explode() in Drupal\file\Element\ManagedFile::uploadAjaxCallback(),!9914Issue #3451136 by quietone, gapple, ghost of drupal past: Improve...,!9882Draft: Issue #3481777 In bulk_form ensure the triggering element is the bulk_form button,!9839Issue #3445469 by pooja_sharma, smustgrave: Add additional test coverage for...,!9815Issue #3480025: There is no way to remove entity cache items,!9757Issue #3478869 Add "All" or overview links to parent links,!9752Issue #3439910 by pooja_sharma, vensires: Fix Toolbar tests that rely on UID1's super user behavior,!9749Issue #3439910 by pooja_sharma, vensires: Fix Toolbar tests that rely on UID1's super user behavior,!9678Issue #3465132 by catch, Spokje, nod_: Show test run time by class in run-tests.sh output,!9578Issue #3304746 by scott_euser, casey, smustgrave: BigPipe cannot handle (GET)...,!9449Issue #3344041: Allow textarea widgets to be used for text (formatted) fields,!8945🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥🔥...,!8893Resolve #3444391 "Navigation center sm logo",!8772Issue #3445909 by seanB, smustgrave, alexpott, catch: Add static caching to...,!8723Make dblog entities,!8708Filter out disabled display extenders on save,!8691#3459116 - Update Manager Settings: Validate same email address added multiple times,!8665Issue #3449743 by catch: Try to optimize test ordering when run-tests.sh is...,!8598Draft: Issue #3458215: Migrate Toolbar button to SDC,!8572Reorder checkboxes on "Development settings" page,!8538Issue #3457009: Fixing xxception message thrown.,!8516Update file EntityReferenceItem.php,!8505Issue #3456528: _node_mass_update_batch_process fails during user cancel when revision is deleted,!8494Issue #3452511: Convert ProviderRepositoryTest to a kernel test,!8482Resolve #3456202 "Phpunit tests support",!8463Prevent re-install if site already exists,!8392Issue #3454196: Filter placeholders without arguments are not replaced when HTML corrector filter applied afterwards,!8384Issue #3446403 by edutrul, jnicola, mradcliffe: [name]: This field is missing for example recipe
Pipeline #196352 passed with warnings
Pipeline: drupal

#196372

    Pipeline: drupal

    #196367

      Pipeline: drupal

      #196358

        ......@@ -119,7 +119,9 @@ public function registerTestNamespaces() {
        * @param string $extension
        * (optional) The name of an extension to limit discovery to; e.g., 'node'.
        * @param string[] $types
        * An array of included test types.
        * (optional) An array of included test types.
        * @param string|null $directory
        * (optional) Limit discovered tests to a specific directory.
        *
        * @return array
        * An array of tests keyed by the group name. If a test is annotated to
        ......@@ -140,7 +142,7 @@ public function registerTestNamespaces() {
        * @todo Remove singular grouping; retain list of groups in 'group' key.
        * @see https://www.drupal.org/node/2296615
        */
        public function getTestClasses($extension = NULL, array $types = []) {
        public function getTestClasses($extension = NULL, array $types = [], ?string $directory = NULL) {
        if (!isset($extension) && empty($types)) {
        if (!empty($this->testClasses)) {
        return $this->testClasses;
        ......@@ -148,7 +150,7 @@ public function getTestClasses($extension = NULL, array $types = []) {
        }
        $list = [];
        $classmap = $this->findAllClassFiles($extension);
        $classmap = $this->findAllClassFiles($extension, $directory);
        // Prevent expensive class loader lookups for each reflected test class by
        // registering the complete classmap of test classes to the class loader.
        ......@@ -200,12 +202,14 @@ public function getTestClasses($extension = NULL, array $types = []) {
        *
        * @param string $extension
        * (optional) The name of an extension to limit discovery to; e.g., 'node'.
        * @param string|null $directory
        * (optional) Limit discovered tests to a specific directory.
        *
        * @return array
        * A classmap containing all discovered class files; i.e., a map of
        * fully-qualified classnames to path names.
        */
        public function findAllClassFiles($extension = NULL) {
        public function findAllClassFiles($extension = NULL, ?string $directory = NULL) {
        $classmap = [];
        $namespaces = $this->registerTestNamespaces();
        if (isset($extension)) {
        ......@@ -215,7 +219,7 @@ public function findAllClassFiles($extension = NULL) {
        }
        foreach ($namespaces as $namespace => $paths) {
        foreach ($paths as $path) {
        if (!is_dir($path)) {
        if (!is_dir($path) || (!is_null($directory) && !str_contains($path, $directory))) {
        continue;
        }
        $classmap += static::scanDirectory($namespace, $path);
        ......
        ......@@ -900,15 +900,17 @@ function simpletest_script_get_test_list() {
        $types_processed = empty($args['types']);
        $test_list = [];
        $slow_tests = [];
        if ($args['all'] || $args['module']) {
        if ($args['all'] || $args['module'] || $args['directory']) {
        try {
        $groups = $test_discovery->getTestClasses($args['module'], $args['types']);
        $groups = $test_discovery->getTestClasses($args['module'], $args['types'], $args['directory']);
        $types_processed = TRUE;
        }
        catch (Exception $e) {
        echo (string) $e;
        exit(SIMPLETEST_SCRIPT_EXIT_EXCEPTION);
        }
        // If the tests are run in parallel jobs, ensure that slow tests are
        // distributed between each job.
        if ((int) $args['ci-parallel-node-total'] > 1) {
        if (key($groups) === '#slow') {
        $slow_tests = array_keys(array_shift($groups));
        ......@@ -916,7 +918,28 @@ function simpletest_script_get_test_list() {
        }
        $all_tests = [];
        foreach ($groups as $group => $tests) {
        $all_tests = array_merge($all_tests, array_keys($tests));
        if ($group === '#slow') {
        $slow_group = $tests;
        }
        else {
        $all_tests = array_merge($all_tests, array_keys($tests));
        }
        }
        // If no type has been set, order the tests alphabetically by test namespace
        // so that unit tests run last. This takes advantage of the fact that Build,
        // Functional, Functional JavaScript, Kernel, Unit roughly corresponds to
        // test time.
        usort($all_tests, function ($a, $b) {
        $slice = function ($class) {
        $parts = explode('\\', $class);
        return implode('\\', array_slice($parts, 3));
        };
        return $slice($a) > $slice($b) ? 1 : -1;
        });
        // If the tests are not being run in parallel, then ensure slow tests run all
        // together first.
        if ((int) $args['ci-parallel-node-total'] <= 1 && !empty($slow_group)) {
        $all_tests = array_merge(array_keys($slow_group), $all_tests);
        }
        $test_list = array_unique($all_tests);
        $test_list = array_diff($test_list, $slow_tests);
        ......@@ -958,42 +981,6 @@ function simpletest_script_get_test_list() {
        $test_list = array_merge($test_list, $parser->getTestListFromFile($file));
        }
        }
        elseif ($args['directory']) {
        // Extract test case class names from specified directory.
        // Find all tests in the PSR-X structure; Drupal\$extension\Tests\*.php
        // Since we do not want to hard-code too many structural file/directory
        // assumptions about PSR-4 files and directories, we check for the
        // minimal conditions only; i.e., a '*.php' file that has '/Tests/' in
        // its path.
        // Ignore anything from third party vendors.
        $ignore = ['.', '..', 'vendor'];
        $files = [];
        if ($args['directory'][0] === '/') {
        $directory = $args['directory'];
        }
        else {
        $directory = DRUPAL_ROOT . "/" . $args['directory'];
        }
        foreach (\Drupal::service('file_system')->scanDirectory($directory, '/\.php$/', $ignore) as $file) {
        // '/Tests/' can be contained anywhere in the file's path (there can be
        // sub-directories below /Tests), but must be contained literally.
        // Case-insensitive to match all Simpletest and PHPUnit tests:
        // ./lib/Drupal/foo/Tests/Bar/Baz.php
        // ./foo/src/Tests/Bar/Baz.php
        // ./foo/tests/Drupal/foo/Tests/FooTest.php
        // ./foo/tests/src/FooTest.php
        // $file->filename doesn't give us a directory, so we use $file->uri
        // Strip the drupal root directory and trailing slash off the URI.
        $filename = substr($file->uri, strlen(DRUPAL_ROOT) + 1);
        if (stripos($filename, '/Tests/')) {
        $files[$filename] = $filename;
        }
        }
        $parser = new TestFileParser();
        foreach ($files as $file) {
        $test_list = array_merge($test_list, $parser->getTestListFromFile($file));
        }
        }
        else {
        try {
        $groups = $test_discovery->getTestClasses(NULL, $args['types']);
        ......
        0% Loading or .
        You are about to add 0 people to the discussion. Proceed with caution.
        Finish editing this message first!
        Please register or to comment