__  __    __   __  _____      _            _          _____ _          _ _ 
 |  \/  |   \ \ / / |  __ \    (_)          | |        / ____| |        | | |
 | \  / |_ __\ V /  | |__) | __ ___   ____ _| |_ ___  | (___ | |__   ___| | |
 | |\/| | '__|> <   |  ___/ '__| \ \ / / _` | __/ _ \  \___ \| '_ \ / _ \ | |
 | |  | | |_ / . \  | |   | |  | |\ V / (_| | ||  __/  ____) | | | |  __/ | |
 |_|  |_|_(_)_/ \_\ |_|   |_|  |_| \_/ \__,_|\__\___| |_____/|_| |_|\___V 2.1
 if you need WebShell for Seo everyday contact me on Telegram
 Telegram Address : @jackleet
        
        
For_More_Tools: Telegram: @jackleet | Bulk Smtp support mail sender | Business Mail Collector | Mail Bouncer All Mail | Bulk Office Mail Validator | Html Letter private



Upload:

Command:

www-data@216.73.216.148: ~ $
<?php
// This file is part of Moodle - http://moodle.org/
//
// Moodle is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Moodle is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Moodle.  If not, see <http://www.gnu.org/licenses/>.

namespace core;

use core\exception\coding_exception;
use core\tests\fake_plugins_test_trait;
use DirectoryIterator;
use ReflectionClass;
use ReflectionProperty;

/**
 * component related tests.
 *
 * @package    core
 * @category   test
 * @copyright  2013 Petr Skoda {@link http://skodak.org}
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 *
 * @covers \core\component
 */
final class component_test extends \advanced_testcase {
    use fake_plugins_test_trait;

    #[\Override]
    public function tearDown(): void {
        parent::tearDown();

        ini_set('error_log', null);
    }

    /**
     * To be changed if number of subsystems increases/decreases,
     * this is defined here to annoy devs that try to add more without any thinking,
     * always verify that it does not collide with any existing add-on modules and subplugins!!!
     */
    const SUBSYSTEMCOUNT = 80;

    public function test_get_core_subsystems(): void {
        global $CFG;

        $subsystems = component::get_core_subsystems();

        $this->assertCount(
            self::SUBSYSTEMCOUNT,
            $subsystems,
            'Oh, somebody added or removed a core subsystem, think twice before doing that!',
        );

        // Make sure all paths are full/null, exist and are inside dirroot.
        foreach ($subsystems as $subsystem => $fulldir) {
            $this->assertFalse(strpos($subsystem, '_'), 'Core subsystems must be one work without underscores');
            if ($fulldir === null) {
                if ($subsystem === 'filepicker' || $subsystem === 'help') { // phpcs:ignore
                    // Arrgghh, let's not introduce more subsystems for no real reason...
                } else {
                    // Lang strings.
                    $this->assertFileExists(
                        "$CFG->dirroot/lang/en/$subsystem.php",
                        'Core subsystems without fulldir are usually used for lang strings.',
                    );
                }
                continue;
            }
            $this->assertFileExists($fulldir);
            // Check that base uses realpath() separators and "/" in the subdirs.
            $this->assertStringStartsWith($CFG->dirroot . '/', $fulldir);
            $reldir = substr($fulldir, strlen($CFG->dirroot) + 1);
            $this->assertFalse(strpos($reldir, '\\'));
        }

        // Make sure all core language files are also subsystems!
        $items = new DirectoryIterator("$CFG->dirroot/lang/en");
        foreach ($items as $item) {
            if ($item->isDot() || $item->isDir()) {
                continue;
            }
            $file = $item->getFilename();
            if ($file === 'moodle.php') {
                // Do not add new lang strings unless really necessary!!!
                continue;
            }

            if (substr($file, -4) !== '.php') {
                continue;
            }
            $file = substr($file, 0, strlen($file) - 4);
            $this->assertArrayHasKey(
                $file,
                $subsystems,
                'All core lang files should be subsystems, think twice before adding anything!',
            );
        }
        unset($item);
        unset($items);
    }

    public function test_deprecated_get_core_subsystems(): void {
        global $CFG;

        $subsystems = component::get_core_subsystems();

        $this->assertSame($subsystems, get_core_subsystems(true));
        $this->assertDebuggingCalled();
        $this->resetDebugging();

        $realsubsystems = get_core_subsystems();
        $this->assertdebuggingcalledcount(2);
        $this->resetDebugging();

        $this->assertSame($realsubsystems, get_core_subsystems(false));
        $this->assertdebuggingcalledcount(2);
        $this->resetDebugging();

        $this->assertEquals(count($subsystems), count($realsubsystems));

        foreach ($subsystems as $subsystem => $fulldir) {
            $this->assertArrayHasKey($subsystem, $realsubsystems);
            if ($fulldir === null) {
                $this->assertNull($realsubsystems[$subsystem]);
                continue;
            }
            $this->assertSame($fulldir, $CFG->dirroot . '/' . $realsubsystems[$subsystem]);
        }
    }

    public function test_get_plugin_types(): void {
        global $CFG;

        $this->assertTrue(
            empty($CFG->themedir),
            'Non-empty $CFG->themedir is not covered by any tests yet, you need to disable it.',
        );

        $plugintypes = component::get_plugin_types();

        foreach ($plugintypes as $plugintype => $fulldir) {
            $this->assertStringStartsWith("$CFG->dirroot/", $fulldir);
        }
    }

    public function test_deprecated_get_plugin_types(): void {
        global $CFG;

        $plugintypes = component::get_plugin_types();

        $this->assertSame($plugintypes, get_plugin_types());
        $this->assertDebuggingCalled();
        $this->resetDebugging();

        $this->assertSame($plugintypes, get_plugin_types(true));
        $this->assertDebuggingCalled();
        $this->resetDebugging();

        $realplugintypes = get_plugin_types(false);
        $this->assertdebuggingcalledcount(2);
        $this->resetDebugging();

        foreach ($plugintypes as $plugintype => $fulldir) {
            $this->assertSame($fulldir, $CFG->dirroot . '/' . $realplugintypes[$plugintype]);
        }
    }

    public function test_get_plugin_list(): void {
        global $CFG;

        $plugintypes = component::get_plugin_types();

        foreach ($plugintypes as $plugintype => $fulldir) {
            $plugins = component::get_plugin_list($plugintype);
            foreach ($plugins as $pluginname => $plugindir) {
                $this->assertStringStartsWith("$CFG->dirroot/", $plugindir);
            }
            if ($plugintype !== 'auth') {
                // Let's crosscheck it with independent implementation (auth/db is an exception).
                $reldir = substr($fulldir, strlen($CFG->dirroot) + 1);
                $dirs = get_list_of_plugins($reldir);
                $dirs = array_values($dirs);
                $this->assertDebuggingCalled();
                $this->assertSame($dirs, array_keys($plugins));
            }
        }
    }

    public function test_deprecated_get_plugin_list(): void {
        $plugintypes = component::get_plugin_types();

        foreach ($plugintypes as $plugintype => $fulldir) {
            $plugins = component::get_plugin_list($plugintype);
            $this->assertSame($plugins, get_plugin_list($plugintype));
            $this->assertDebuggingCalled();
            $this->resetDebugging();
        }
    }

    public function test_get_plugin_directory(): void {
        $plugintypes = component::get_plugin_types();

        foreach ($plugintypes as $plugintype => $fulldir) {
            $plugins = component::get_plugin_list($plugintype);
            foreach ($plugins as $pluginname => $plugindir) {
                $this->assertSame($plugindir, component::get_plugin_directory($plugintype, $pluginname));
            }
        }
    }

    public function test_deprecated_get_plugin_directory(): void {
        $plugintypes = component::get_plugin_types();

        foreach ($plugintypes as $plugintype => $fulldir) {
            $plugins = component::get_plugin_list($plugintype);
            foreach ($plugins as $pluginname => $plugindir) {
                $this->assertSame(
                    component::get_plugin_directory($plugintype, $pluginname),
                    get_plugin_directory($plugintype, $pluginname),
                );
                $this->assertDebuggingCalled();
                $this->resetDebugging();
            }
        }
    }

    public function test_get_subsystem_directory(): void {
        $subsystems = component::get_core_subsystems();
        foreach ($subsystems as $subsystem => $fulldir) {
            $this->assertSame($fulldir, component::get_subsystem_directory($subsystem));
        }
    }

    /**
     * Test that the get_plugin_list_with_file() function returns the correct list of plugins.
     *
     * @covers \core\component::is_valid_plugin_name
     * @dataProvider is_valid_plugin_name_provider
     * @param array $arguments
     * @param bool $expected
     */
    public function test_is_valid_plugin_name(array $arguments, bool $expected): void {
        $this->assertEquals($expected, component::is_valid_plugin_name(...$arguments));
    }

    /**
     * Data provider for the is_valid_plugin_name function.
     *
     * @return array
     */
    public static function is_valid_plugin_name_provider(): array {
        return [
            [['mod', 'example1'], true],
            [['mod', 'feedback360'], true],
            [['mod', 'feedback_360'], false],
            [['mod', '2feedback'], false],
            [['mod', '1example'], false],
            [['mod', 'example.xx'], false],
            [['mod', '.example'], false],
            [['mod', '_example'], false],
            [['mod', 'example_'], false],
            [['mod', 'example_x1'], false],
            [['mod', 'example-x1'], false],
            [['mod', 'role'], false],

            [['tool', 'example1'], true],
            [['tool', 'example_x1'], true],
            [['tool', 'example_x1_xxx'], true],
            [['tool', 'feedback360'], true],
            [['tool', 'feed_back360'], true],
            [['tool', 'role'], true],
            [['tool', '1example'], false],
            [['tool', 'example.xx'], false],
            [['tool', 'example-xx'], false],
            [['tool', '.example'], false],
            [['tool', '_example'], false],
            [['tool', 'example_'], false],
            [['tool', 'example__x1'], false],

            // Some invalid cases.
            [['mod', null], false],
            [['mod', ''], false],
            [['tool', null], false],
            [['tool', ''], false],
        ];
    }

    public function test_normalize_componentname(): void {
        // Moodle core.
        $this->assertSame('core', component::normalize_componentname('core'));
        $this->assertSame('core', component::normalize_componentname('moodle'));
        $this->assertSame('core', component::normalize_componentname(''));

        // Moodle core subsystems.
        $this->assertSame('core_admin', component::normalize_componentname('admin'));
        $this->assertSame('core_admin', component::normalize_componentname('core_admin'));
        $this->assertSame('core_admin', component::normalize_componentname('moodle_admin'));

        // Activity modules and their subplugins.
        $this->assertSame('mod_workshop', component::normalize_componentname('workshop'));
        $this->assertSame('mod_workshop', component::normalize_componentname('mod_workshop'));
        $this->assertSame('workshopform_accumulative', component::normalize_componentname('workshopform_accumulative'));
        $this->assertSame('mod_quiz', component::normalize_componentname('quiz'));
        $this->assertSame('quiz_grading', component::normalize_componentname('quiz_grading'));
        $this->assertSame('mod_data', component::normalize_componentname('data'));
        $this->assertSame('datafield_checkbox', component::normalize_componentname('datafield_checkbox'));

        // Other plugin types.
        $this->assertSame('auth_ldap', component::normalize_componentname('auth_ldap'));
        $this->assertSame('enrol_self', component::normalize_componentname('enrol_self'));
        $this->assertSame('block_html', component::normalize_componentname('block_html'));
        $this->assertSame('auth_oauth2', component::normalize_componentname('auth_oauth2'));
        $this->assertSame('local_amos', component::normalize_componentname('local_amos'));
        $this->assertSame('local_admin', component::normalize_componentname('local_admin'));

        // Unknown words without underscore are supposed to be activity modules.
        $this->assertSame(
            'mod_whoonearthwouldcomewithsuchastupidnameofcomponent',
            component::normalize_componentname('whoonearthwouldcomewithsuchastupidnameofcomponent')
        );
        // Module names can not contain underscores, this must be a subplugin.
        $this->assertSame(
            'whoonearth_wouldcomewithsuchastupidnameofcomponent',
            component::normalize_componentname('whoonearth_wouldcomewithsuchastupidnameofcomponent')
        );
        $this->assertSame(
            'whoonearth_would_come_withsuchastupidnameofcomponent',
            component::normalize_componentname('whoonearth_would_come_withsuchastupidnameofcomponent')
        );
    }

    /**
     * Test \core_component::normalize_component function.
     *
     * @dataProvider normalise_component_provider
     * @param array $expected
     * @param string $args
     */
    public function test_normalize_component(array $expected, string $args): void {
        $this->assertSame(
            $expected,
            component::normalize_component($args),
        );
    }

    /**
     * Test the deprecated normalize_component function.
     *
     * @dataProvider normalise_component_provider
     * @param array $expected
     * @param string $args
     */
    public function test_deprecated_normalize_component(array $expected, string $args): void {
        $this->assertSame(
            $expected,
            normalize_component($args),
        );

        $this->assertDebuggingCalled();
    }

    /**
     * Data provider for the normalize_component function.
     */
    public static function normalise_component_provider(): array {
        return [
            // Moodle core.
            [['core', null], 'core'],
            [['core', null], ''],
            [['core', null], 'moodle'],

            // Moodle core subsystems.
            [['core', 'admin'], 'admin'],
            [['core', 'admin'], 'core_admin'],
            [['core', 'admin'], 'moodle_admin'],

            // Activity modules and their subplugins.
            [['mod', 'workshop'], 'workshop'],
            [['mod', 'workshop'], 'mod_workshop'],
            [['workshopform', 'accumulative'], 'workshopform_accumulative'],
            [['mod', 'quiz'], 'quiz'],
            [['quiz', 'grading'], 'quiz_grading'],
            [['mod', 'data'], 'data'],
            [['datafield', 'checkbox'], 'datafield_checkbox'],

            // Other plugin types.
            [['auth', 'ldap'], 'auth_ldap'],
            [['enrol', 'self'], 'enrol_self'],
            [['block', 'html'], 'block_html'],
            [['auth', 'oauth2'], 'auth_oauth2'],
            [['local', 'amos'], 'local_amos'],
            [['local', 'admin'], 'local_admin'],

            // Unknown words without underscore are supposed to be activity modules.
            [
                ['mod', 'whoonearthwouldcomewithsuchastupidnameofcomponent'],
                'whoonearthwouldcomewithsuchastupidnameofcomponent',
            ],
            // Module names can not contain underscores, this must be a subplugin.
            [
                ['whoonearth', 'wouldcomewithsuchastupidnameofcomponent'],
                'whoonearth_wouldcomewithsuchastupidnameofcomponent',
            ],
            [
                ['whoonearth', 'would_come_withsuchastupidnameofcomponent'],
                'whoonearth_would_come_withsuchastupidnameofcomponent',
            ],
        ];
    }

    public function test_get_component_directory(): void {
        $plugintypes = component::get_plugin_types();
        foreach ($plugintypes as $plugintype => $fulldir) {
            $plugins = component::get_plugin_list($plugintype);
            foreach ($plugins as $pluginname => $plugindir) {
                $this->assertSame($plugindir, component::get_component_directory(($plugintype . '_' . $pluginname)));
            }
        }

        $subsystems = component::get_core_subsystems();
        foreach ($subsystems as $subsystem => $fulldir) {
            $this->assertSame($fulldir, component::get_component_directory(('core_' . $subsystem)));
        }
    }

    /**
     * Unit tests for get_component_from_classname.
     *
     * @dataProvider get_component_from_classname_provider
     * @param string $classname The class name to test
     * @param string|null $expected The expected component
     * @covers \core\component::get_component_from_classname
     */
    public function test_get_component_from_classname(
        string $classname,
        string|null $expected,
    ): void {
        $this->assertEquals(
            $expected,
            component::get_component_from_classname($classname),
        );
    }

    /**
     * Data provider for get_component_from_classname tests.
     *
     * @return array
     */
    public static function get_component_from_classname_provider(): array {
        // Start off with testcases which have the leading \.
        $testcases = [
            // Core.
            [\core\example::class, 'core'],

            // A core subsystem.
            [\core_message\example::class, 'core_message'],

            // A fake core subsystem.
            [\core_fake\example::class, null],

            // A plugin.
            [\mod_forum\example::class, 'mod_forum'],

            // A plugin in the old style is not supported.
            [\mod_forum_example::class, null],

            // A fake plugin.
            [\mod_fake\example::class, null],

            // A subplugin.
            [\tiny_link\example::class, 'tiny_link'],
        ];

        // Duplicate the testcases, adding a nested namespace.
        $testcases = array_merge(
            $testcases,
            array_map(
                fn ($testcase) => [$testcase[0] . '\\in\\sub\\directory', $testcase[1]],
                $testcases,
            ),
        );

        // Duplicate the testcases, removing the leading \.
        return array_merge(
            $testcases,
            array_map(
                fn ($testcase) => [ltrim($testcase[0], '\\'), $testcase[1]],
                $testcases,
            ),
        );
    }

    public function test_deprecated_get_component_directory(): void {
        $plugintypes = component::get_plugin_types();
        foreach ($plugintypes as $plugintype => $fulldir) {
            $plugins = component::get_plugin_list($plugintype);
            foreach ($plugins as $pluginname => $plugindir) {
                $this->assertSame($plugindir, get_component_directory(($plugintype . '_' . $pluginname)));
                $this->assertDebuggingCalled();
                $this->resetDebugging();
            }
        }

        $subsystems = component::get_core_subsystems();
        foreach ($subsystems as $subsystem => $fulldir) {
            $this->assertSame($fulldir, get_component_directory(('core_' . $subsystem)));
            $this->assertDebuggingCalled();
            $this->resetDebugging();
        }
    }

    public function test_get_subtype_parent(): void {
        global $CFG;

        $this->assertNull(component::get_subtype_parent('mod'));

        // Any plugin with more subtypes is ok here.
        $this->assertFileExists("$CFG->dirroot/mod/assign/db/subplugins.json");
        $this->assertSame('mod_assign', component::get_subtype_parent('assignsubmission'));
        $this->assertSame('mod_assign', component::get_subtype_parent('assignfeedback'));
        $this->assertNull(component::get_subtype_parent('assignxxxxx'));
    }

    public function test_get_subplugins(): void {
        global $CFG;

        // Any plugin with more subtypes is ok here.
        $this->assertFileExists("$CFG->dirroot/mod/assign/db/subplugins.json");

        $subplugins = component::get_subplugins('mod_assign');
        $this->assertSame(['assignsubmission', 'assignfeedback'], array_keys($subplugins));

        $subs = component::get_plugin_list('assignsubmission');
        $feeds = component::get_plugin_list('assignfeedback');

        $this->assertSame(array_keys($subs), $subplugins['assignsubmission']);
        $this->assertSame(array_keys($feeds), $subplugins['assignfeedback']);

        // Any plugin without subtypes is ok here.
        $this->assertFileExists("$CFG->dirroot/mod/choice");
        $this->assertFileDoesNotExist("$CFG->dirroot/mod/choice/db/subplugins.json");

        $this->assertNull(component::get_subplugins('mod_choice'));

        $this->assertNull(component::get_subplugins('xxxx_yyyy'));
    }

    public function test_get_plugin_types_with_subplugins(): void {
        global $CFG;

        $types = component::get_plugin_types_with_subplugins();

        // Hardcode it here to detect if anybody hacks the code to include more subplugin types.
        $expected = [
            'mod' => "$CFG->dirroot/mod",
            'editor' => "$CFG->dirroot/lib/editor",
            'tool' => "$CFG->dirroot/$CFG->admin/tool",
            'local' => "$CFG->dirroot/local",
        ];

        $this->assertSame($expected, $types);
    }

    public function test_get_plugin_list_with_file(): void {
        $this->resetAfterTest(true);

        // No extra reset here because component reset automatically.

        $expected = [];
        $reports = component::get_plugin_list('report');
        foreach ($reports as $name => $fulldir) {
            if (file_exists("$fulldir/lib.php")) {
                $expected[] = $name;
            }
        }

        // Test cold.
        $list = component::get_plugin_list_with_file('report', 'lib.php', false);
        $this->assertEquals($expected, array_keys($list));

        // Test hot.
        $list = component::get_plugin_list_with_file('report', 'lib.php', false);
        $this->assertEquals($expected, array_keys($list));

        // Test with include.
        $list = component::get_plugin_list_with_file('report', 'lib.php', true);
        $this->assertEquals($expected, array_keys($list));

        // Test missing.
        $list = component::get_plugin_list_with_file('report', 'idontexist.php', true);
        $this->assertEquals([], array_keys($list));
    }

    /**
     * Tests for get_component_classes_in_namespace.
     */
    public function test_get_component_classes_in_namespace(): void {
        // Unexisting.
        $this->assertCount(0, component::get_component_classes_in_namespace('core_unexistingcomponent', 'something'));
        $this->assertCount(0, component::get_component_classes_in_namespace('auth_db', 'something'));

        // Matches the last namespace level name not partials.
        $this->assertCount(0, component::get_component_classes_in_namespace('auth_db', 'tas'));
        $this->assertCount(0, component::get_component_classes_in_namespace('core_user', 'course'));
        $this->assertCount(0, component::get_component_classes_in_namespace('mod_forum', 'output\\emaildigest'));
        $this->assertCount(0, component::get_component_classes_in_namespace('mod_forum', '\\output\\emaildigest'));

        // Without either a component or namespace it returns an empty array.
        $this->assertEmpty(component::get_component_classes_in_namespace());
        $this->assertEmpty(component::get_component_classes_in_namespace(null));
        $this->assertEmpty(component::get_component_classes_in_namespace(null, ''));
    }

    /**
     * Test that the get_component_classes_in_namespace() function returns classes in the correct namespace.
     *
     * @dataProvider get_component_classes_in_namespace_provider
     * @param array $methodargs
     * @param string $expectedclassnameformat
     */
    public function test_get_component_classes_in_namespace_provider(
        array $methodargs,
        string $expectedclassnameformat,
    ): void {
        $classlist = component::get_component_classes_in_namespace(...$methodargs);
        $this->assertGreaterThan(0, count($classlist));

        foreach (array_keys($classlist) as $classname) {
            $this->assertStringMatchesFormat($expectedclassnameformat, $classname);
        }
    }

    /**
     * Data provider for get_component_classes_in_namespace tests.
     *
     * @return array
     */
    public static function get_component_classes_in_namespace_provider(): array {
        return [
            // Matches the last namespace level name not partials.
            [
                ['mod_forum', 'output\\email'],
                'mod_forum\output\email\%s',
            ],
            [
                ['mod_forum', '\\output\\email'],
                'mod_forum\output\email\%s',
            ],
            [
                ['mod_forum', 'output\\email\\'],
                'mod_forum\output\email\%s',
            ],
            [
                ['mod_forum', '\\output\\email\\'],
                'mod_forum\output\email\%s',
            ],
            // Prefix with backslash if it doesn\'t come prefixed.
            [
                ['auth_db', 'task'],
                'auth_db\task\%s',
            ],
            [
                ['auth_db', '\\task'],
                'auth_db\task\%s',
            ],

            // Core as a component works, the function can normalise the component name.
            [
                ['core', 'update'],
                'core\update\%s',
            ],
            [
                ['', 'update'],
                'core\update\%s',
            ],
            [
                ['moodle', 'update'],
                'core\update\%s',
            ],

            // Multiple levels.
            [
                ['core_user', '\\output\\myprofile\\'],
                'core_user\output\myprofile\%s',
            ],
            [
                ['core_user', 'output\\myprofile\\'],
                'core_user\output\myprofile\%s',
            ],
            [
                ['core_user', '\\output\\myprofile'],
                'core_user\output\myprofile\%s',
            ],
            [
                ['core_user', 'output\\myprofile'],
                'core_user\output\myprofile\%s',
            ],

            // Without namespace it returns classes/ classes.
            [
                ['tool_mobile', ''],
                'tool_mobile\%s',
            ],
            [
                ['tool_filetypes'],
                'tool_filetypes\%s',
            ],

            // Multiple levels.
            [
                ['core_user', '\\output\\myprofile\\'],
                'core_user\output\myprofile\%s',
            ],

            // When no component is specified, classes are returned for the namespace in all components.
            // (We don't assert exact amounts here as the count of `output` classes will change depending on plugins installed).
            [
                ['core', 'output'],
                'core\%s',
            ],
            [
                [null, 'output'],
                '%s',
            ],
        ];
    }

    /**
     * Data provider for classloader test
     */
    public static function classloader_provider(): array {
        global $CFG;

        // As part of these tests, we Check that there are no unexpected problems with overlapping PSR namespaces.
        // This is not in the spec, but may come up in some libraries using both namespaces and PEAR-style class names.
        // If problems arise we can remove this test, but will need to add a warning.
        // Normalise to forward slash for testing purposes.
        $directory = str_replace('\\', '/', $CFG->dirroot) . "/lib/tests/fixtures/component/";

        $psr0 = [
          'psr0'      => 'lib/tests/fixtures/component/psr0',
          'overlap'   => 'lib/tests/fixtures/component/overlap',
        ];
        $psr4 = [
          'psr4'      => 'lib/tests/fixtures/component/psr4',
          'overlap'   => 'lib/tests/fixtures/component/overlap',
        ];
        return [
            'PSR-0 Classloading - Root' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr0_main',
                'includedfiles' => "{$directory}psr0/main.php",
            ],
            'PSR-0 Classloading - Sub namespace - underscores' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr0_subnamespace_example',
                'includedfiles' => "{$directory}psr0/subnamespace/example.php",
            ],
            'PSR-0 Classloading - Sub namespace - slashes' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr0\\subnamespace\\slashes',
                'includedfiles' => "{$directory}psr0/subnamespace/slashes.php",
            ],
            'PSR-4 Classloading - Root' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr4\\main',
                'includedfiles' => "{$directory}psr4/main.php",
            ],
            'PSR-4 Classloading - Sub namespace' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr4\\subnamespace\\example',
                'includedfiles' => "{$directory}psr4/subnamespace/example.php",
            ],
            'PSR-4 Classloading - Ensure underscores are not converted to paths' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr4\\subnamespace\\underscore_example',
                'includedfiles' => "{$directory}psr4/subnamespace/underscore_example.php",
            ],
            'Overlap - Ensure no unexpected problems with PSR-4 when overlapping namespaces.' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'overlap\\subnamespace\\example',
                'includedfiles' => "{$directory}overlap/subnamespace/example.php",
            ],
            'Overlap - Ensure no unexpected problems with PSR-0 overlapping namespaces.' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'overlap_subnamespace_example2',
                'includedfiles' => "{$directory}overlap/subnamespace/example2.php",
            ],
        ];
    }

    /**
     * Test the classloader.
     *
     * @dataProvider classloader_provider
     * @param array $psr0 The PSR-0 namespaces to be used in the test.
     * @param array $psr4 The PSR-4 namespaces to be used in the test.
     * @param string $classname The name of the class to attempt to load.
     * @param string $includedfiles The file expected to be loaded.
     * @runInSeparateProcess
     */
    public function test_classloader($psr0, $psr4, $classname, $includedfiles): void {
        $psr0namespaces = new ReflectionProperty(component::class, 'psr0namespaces');
        $psr0namespaces->setValue(null, $psr0);

        $psr4namespaces = new ReflectionProperty(component::class, 'psr4namespaces');
        $psr4namespaces->setValue(null, $psr4);

        component::classloader($classname);
        if (DIRECTORY_SEPARATOR != '/') {
            // Denormalise the expected path so that we can quickly compare with get_included_files.
            $includedfiles = str_replace('/', DIRECTORY_SEPARATOR, $includedfiles);
        }
        $this->assertContains($includedfiles, get_included_files());
        $this->assertTrue(class_exists($classname, false));
    }

    /**
     * Data provider for psr_classloader test
     */
    public static function psr_classloader_provider(): array {
        global $CFG;

        // As part of these tests, we Check that there are no unexpected problems with overlapping PSR namespaces.
        // This is not in the spec, but may come up in some libraries using both namespaces and PEAR-style class names.
        // If problems arise we can remove this test, but will need to add a warning.
        // Normalise to forward slash for testing purposes.
        $dirroot = str_replace('\\', '/', $CFG->dirroot);
        $directory = "{$dirroot}/lib/tests/fixtures/component/";

        $psr0 = [
          'psr0'      => 'lib/tests/fixtures/component/psr0',
          'overlap'   => 'lib/tests/fixtures/component/overlap',
        ];
        $psr4 = [
          'psr4'      => 'lib/tests/fixtures/component/psr4',
          'overlap'   => 'lib/tests/fixtures/component/overlap',
        ];
        return [
            'PSR-0 Classloading - Root' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr0_main',
                'file' => "{$directory}psr0/main.php",
            ],
            'PSR-0 Classloading - Sub namespace - underscores' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr0_subnamespace_example',
                'file' => "{$directory}psr0/subnamespace/example.php",
            ],
            'PSR-0 Classloading - Sub namespace - slashes' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr0\\subnamespace\\slashes',
                'file' => "{$directory}psr0/subnamespace/slashes.php",
            ],
            'PSR-0 Classloading - non-existent file' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr0_subnamespace_nonexistent_file',
                'file' => false,
            ],
            'PSR-4 Classloading - Root' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr4\\main',
                'file' => "{$directory}psr4/main.php",
            ],
            'PSR-4 Classloading - Sub namespace' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr4\\subnamespace\\example',
                'file' => "{$directory}psr4/subnamespace/example.php",
            ],
            'PSR-4 Classloading - Ensure underscores are not converted to paths' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr4\\subnamespace\\underscore_example',
                'file' => "{$directory}psr4/subnamespace/underscore_example.php",
            ],
            'PSR-4 Classloading - non-existent file' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'psr4\\subnamespace\\nonexistent',
                'file' => false,
            ],
            'Overlap - Ensure no unexpected problems with PSR-4 when overlapping namespaces.' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'overlap\\subnamespace\\example',
                'file' => "{$directory}overlap/subnamespace/example.php",
            ],
            'Overlap - Ensure no unexpected problems with PSR-0 overlapping namespaces.' => [
                'psr0' => $psr0,
                'psr4' => $psr4,
                'classname' => 'overlap_subnamespace_example2',
                'file' => "{$directory}overlap/subnamespace/example2.php",
            ],
            'PSR-4 namespaces can come from multiple sources - first source' => [
                'psr0' => $psr0,
                'psr4' => [
                    'Psr\\Http\\Message' => [
                        'lib/psr/http-message/src',
                        'lib/psr/http-factory/src',
                    ],
                ],
                'classname' => 'Psr\Http\Message\ServerRequestInterface',
                'file' => "{$dirroot}/lib/psr/http-message/src/ServerRequestInterface.php",
            ],
            'PSR-4 namespaces can come from multiple sources - second source' => [
                'psr0' => [],
                'psr4' => [
                    'Psr\\Http\\Message' => [
                        'lib/psr/http-message/src',
                        'lib/psr/http-factory/src',
                    ],
                ],
                'classname' => 'Psr\Http\Message\ServerRequestFactoryInterface',
                'file' => "{$dirroot}/lib/psr/http-factory/src/ServerRequestFactoryInterface.php",
            ],
        ];
    }

    /**
     * Test that the classloader can load from the test namespaces.
     */
    public function test_classloader_tests_namespace(): void {
        global $CFG;

        $this->resetAfterTest();

        $getclassfilecontent = function (string $classname, ?string $namespace): string {
            if ($namespace) {
                $content = "<?php\nnamespace $namespace;\nclass $classname {}";
            } else {
                $content = "<?php\nclass $classname {}";
            }
            return $content;
        };

        $vfileroot = \org\bovigo\vfs\vfsStream::setup('root', null, [
            'lib' => [
                'classes' => [
                    'example.php' => $getclassfilecontent('example', 'core'),
                ],
                'tests' => [
                    'classes' => [
                        'example_classname.php' => $getclassfilecontent('example_classname', \core\tests::class),
                    ],
                    'behat' => [
                        'example_classname.php' => $getclassfilecontent('example_classname', \core\behat::class),
                    ],
                ],
            ],
        ]);

        // Note: This is pretty hacky, but it's the only way to test the classloader.
        // We have to override the dirroot and libdir, and then reset the plugintypes property.
        $CFG->dirroot = $vfileroot->url();
        $CFG->libdir = $vfileroot->url() . '/lib';
        component::reset();

        // Existing classes do not break.
        $this->assertTrue(
            class_exists(\core\example::class),
        );

        // Test and behat classes work.
        $this->assertTrue(
            class_exists(\core\tests\example_classname::class),
        );
        $this->assertTrue(
            class_exists(\core\behat\example_classname::class),
        );

        // Non-existent classes do not do anything.
        $this->assertFalse(
            class_exists(\core\tests\example_classname_not_found::class),
        );
    }

    /**
     * Test the PSR classloader.
     *
     * @dataProvider psr_classloader_provider
     * @param array $psr0 The PSR-0 namespaces to be used in the test.
     * @param array $psr4 The PSR-4 namespaces to be used in the test.
     * @param string $classname The name of the class to attempt to load.
     * @param string|bool $file The expected file corresponding to the class or false for nonexistant.
     * @runInSeparateProcess
     */
    public function test_psr_classloader($psr0, $psr4, $classname, $file): void {
        return;
        $psr0namespaces = new ReflectionProperty(component::class, 'psr0namespaces');
        $psr0namespaces->setValue(null, $psr0);

        $psr4namespaces = new ReflectionProperty(component::class, 'psr4namespaces');
        $psr4namespaces->setValue(null, $psr4);

        $component = new ReflectionClass(component::class);
        $psrclassloader = $component->getMethod('psr_classloader');

        $returnvalue = $psrclassloader->invokeArgs(null, [$classname]);
        // Normalise to forward slashes for testing comparison.
        if ($returnvalue) {
            $returnvalue = str_replace('\\', '/', $returnvalue);
        }
        $this->assertEquals($file, $returnvalue);
    }

    /**
     * Data provider for get_class_file test
     */
    public static function get_class_file_provider(): array {
        global $CFG;

        return [
          'Getting a file with underscores' => [
              'classname' => 'Test_With_Underscores',
              'prefix' => "Test",
              'path' => 'test/src',
              'separators' => ['_'],
              'result' => $CFG->dirroot . "/test/src/With/Underscores.php",
          ],
          'Getting a file with slashes' => [
              'classname' => 'Test\\With\\Slashes',
              'prefix' => "Test",
              'path' => 'test/src',
              'separators' => ['\\'],
              'result' => $CFG->dirroot . "/test/src/With/Slashes.php",
          ],
          'Getting a file with multiple namespaces' => [
              'classname' => 'Test\\With\\Multiple\\Namespaces',
              'prefix' => "Test\\With",
              'path' => 'test/src',
              'separators' => ['\\'],
              'result' => $CFG->dirroot . "/test/src/Multiple/Namespaces.php",
          ],
          'Getting a file with multiple namespaces (non-existent)' => [
              'classname' => 'Nonexistent\\Namespace\\Test',
              'prefix' => "Test",
              'path' => 'test/src',
              'separators' => ['\\'],
              'result' => false,
          ],
        ];
    }

    /**
     * Test the PSR classloader.
     *
     * @dataProvider get_class_file_provider
     * @param string $classname the name of the class.
     * @param string $prefix The namespace prefix used to identify the base directory of the source files.
     * @param string $path The relative path to the base directory of the source files.
     * @param string[] $separators The characters that should be used for separating.
     * @param string|bool $result The expected result to be returned from get_class_file.
     */
    public function test_get_class_file($classname, $prefix, $path, $separators, $result): void {
        $component = new ReflectionClass(component::class);
        $psrclassloader = $component->getMethod('get_class_file');

        $file = $psrclassloader->invokeArgs(null, [$classname, $prefix, $path, $separators]);
        $this->assertEquals($result, $file);
    }

    /**
     * Confirm the get_component_list method contains an entry for every component.
     */
    public function test_get_component_list_contains_all_components(): void {
        global $CFG;
        $componentslist = component::get_component_list();

        // We should have an entry for each plugin type, and one additional for 'core'.
        $plugintypes = component::get_plugin_types();
        $numelementsexpected = count($plugintypes) + 1;
        $this->assertEquals($numelementsexpected, count($componentslist));

        // And an entry for each of the plugin types.
        foreach (array_keys($plugintypes) as $plugintype) {
            $this->assertArrayHasKey($plugintype, $componentslist);
        }

        // And one for 'core'.
        $this->assertArrayHasKey('core', $componentslist);

        // Check a few of the known plugin types to confirm their presence at their respective type index.
        $this->assertEquals($componentslist['core']['core_comment'], $CFG->dirroot . '/comment');
        $this->assertEquals($componentslist['mod']['mod_forum'], $CFG->dirroot . '/mod/forum');
        $this->assertEquals($componentslist['tool']['tool_usertours'], $CFG->dirroot . '/' . $CFG->admin . '/tool/usertours');
    }

    /**
     * Test the get_component_names() method.
     *
     * @dataProvider get_component_names_provider
     * @param bool $includecore Whether to include core in the list.
     * @param bool $coreexpected Whether core is expected to be in the list.
     */
    public function test_get_component_names(
        bool $includecore,
        bool $coreexpected,
    ): void {
        global $CFG;
        $componentnames = component::get_component_names($includecore);

        // We should have an entry for each plugin type.
        $plugintypes = component::get_plugin_types();
        $numplugintypes = 0;
        foreach (array_keys($plugintypes) as $type) {
            $numplugintypes += count(component::get_plugin_list($type));
        }
        // And an entry for each core subsystem.
        $numcomponents = $numplugintypes + count(component::get_core_subsystems());

        if ($coreexpected) {
            // Add one for core.
            $numcomponents++;
        }
        $this->assertEquals($numcomponents, count($componentnames));

        // Check a few of the known plugin types to confirm their presence at their respective type index.
        $this->assertContains('core_comment', $componentnames);
        $this->assertContains('mod_forum', $componentnames);
        $this->assertContains('tool_usertours', $componentnames);
        $this->assertContains('core_favourites', $componentnames);
        if ($coreexpected) {
            $this->assertContains('core', $componentnames);
        } else {
            $this->assertNotContains('core', $componentnames);
        }
    }

    /**
     * Data provider for get_component_names() test.
     *
     * @return array
     */
    public static function get_component_names_provider(): array {
        return [
            [false, false],
            [true, true],
        ];
    }

    /**
     * Basic tests for APIs related functions in the component class.
     */
    public function test_apis_methods(): void {
        $apis = component::get_core_apis();
        $this->assertIsArray($apis);

        $apinames = component::get_core_api_names();
        $this->assertIsArray($apis);

        // Both should return the very same APIs.
        $this->assertEquals($apinames, array_keys($apis));

        $this->assertFalse(component::is_core_api('lalala'));
        $this->assertTrue(component::is_core_api('privacy'));
    }

    /**
     * Test that the apis.json structure matches expectations
     *
     * While we include an apis.schema.json file in core, there isn't any PHP built-in allowing us
     * to validate it (3rd part libraries needed). Plus the schema doesn't allow to validate things
     * like uniqueness or sorting. We are going to do all that here.
     */
    public function test_apis_json_validation(): void {
        $apis = $sortedapis = component::get_core_apis();
        ksort($sortedapis); // We'll need this later.

        $subsystems = component::get_core_subsystems(); // To verify all apis are pointing to valid subsystems.
        $subsystems['core'] = 'anything'; // Let's add 'core' because it's a valid component for apis.

        // General structure validations.
        $this->assertIsArray($apis);
        $this->assertGreaterThan(25, count($apis));
        $this->assertArrayHasKey('privacy', $apis); // Verify a few.
        $this->assertArrayHasKey('external', $apis);
        $this->assertArrayHasKey('search', $apis);
        $this->assertEquals(array_keys($sortedapis), array_keys($apis)); // Verify json is sorted alphabetically.

        // Iterate over all apis and perform more validations.
        foreach ($apis as $apiname => $attributes) {
            // Message, to be used later and easier finding the problem.
            $message = "Validation problem found with API: {$apiname}";

            $this->assertIsObject($attributes, $message);
            $this->assertMatchesRegularExpression('/^[a-z][a-z0-9]+$/', $apiname, $message);
            $this->assertEquals(['component', 'allowedlevel2', 'allowedspread'], array_keys((array)$attributes), $message);

            // Verify attributes.
            if ($apiname !== 'core') { // Exception for core api, it doesn't have component.
                // Check that component attribute looks correct.
                $this->assertMatchesRegularExpression('/^(core|[a-z][a-z0-9_]+)$/', $attributes->component, $message);
                // Ensure that the api component (without the core_ prefix) is a correct subsystem.
                $this->assertArrayHasKey(str_replace('core_', '', $attributes->component), $subsystems, $message);
            } else {
                $this->assertNull($attributes->component, $message);
            }


            // Now check for the rest of attributes.
            $this->assertIsBool($attributes->allowedlevel2, $message);
            $this->assertIsBool($attributes->allowedspread, $message);

            // Cannot spread if level2 is not allowed.
            $this->assertLessThanOrEqual($attributes->allowedlevel2, $attributes->allowedspread, $message);
        }
    }

    /**
     * Test for monologo icons check in plugins.
     */
    public function test_has_monologo_icon(): void {
        // The Forum activity plugin has monologo icons.
        $this->assertTrue(component::has_monologo_icon('mod', 'forum'));
        // The core H5P subsystem doesn't have monologo icons.
        $this->assertFalse(component::has_monologo_icon('core', 'h5p'));
        // The function will return false for a non-existent component.
        $this->assertFalse(component::has_monologo_icon('randomcomponent', 'h5p'));
    }

    /*
     * Tests the getter for the db directory summary hash.
     *
     * @covers \core\component::get_all_directory_hashes
     */
    public function test_get_db_directories_hash(): void {
        $initial = component::get_all_component_hash();

        $dir = make_request_directory();
        $hashes = component::get_all_directory_hashes([$dir]);
        $emptydirhash = component::get_all_component_hash([$hashes]);

        // Confirm that a single empty directory is a different hash to the core hash.
        $this->assertNotEquals($initial, $emptydirhash);

        // Now lets add something to the dir, and check the hash is different.
        $file = fopen($dir . '/test.php', 'w');
        fwrite($file, 'sometestdata');
        fclose($file);

        $hashes = component::get_all_directory_hashes([$dir]);
        $onefiledirhash = component::get_all_component_hash([$hashes]);
        $this->assertNotEquals($emptydirhash, $onefiledirhash);

        // Now add a subdirectory inside the request dir. This should not affect the hash.
        mkdir($dir . '/subdir');
        $hashes = component::get_all_directory_hashes([$dir]);
        $finalhash = component::get_all_component_hash([$hashes]);
        $this->assertEquals($onefiledirhash, $finalhash);
    }

    /**
     * Data provider fetching all third-party lib directories.
     *
     * @return array
     */
    public static function core_thirdparty_libs_provider(): array {
        global $CFG;

        $libs = [];

        $xmlpath = $CFG->libdir . '/thirdpartylibs.xml';
        $xml = simplexml_load_file($xmlpath);
        foreach ($xml as $lib) {
            $base = realpath(dirname($xmlpath));
            $fullpath = "{$base}/{$lib->location}";
            $relativepath = substr($fullpath, strlen($CFG->dirroot));

            $libs[$relativepath] = [
                'name' => (string) $lib->name,
                'fullpath' => $fullpath,
                'relativepath' => $relativepath,
            ];
        }

        return $libs;
    }

    /**
     * Data provider fetching all third-party lib directories with a composer.json file.
     *
     * @return array
     */
    public static function core_thirdparty_libs_with_composer_provider(): array {
        return array_filter(self::core_thirdparty_libs_provider(), function ($lib) {
            return file_exists("{$lib['fullpath']}/composer.json");
        });
    }

    /**
     * Summary of test_composer_files
     *
     * @dataProvider core_thirdparty_libs_with_composer_provider
     * @param string $name
     * @param string $fullpath
     * @param string $relativepath
     */
    public function test_composer_files(
        string $name,
        string $fullpath,
        string $relativepath,
    ): void {
        $this->assertFileExists("{$fullpath}/composer.json");

        $composer = json_decode(file_get_contents("{$fullpath}/composer.json"), true);

        $rc = new ReflectionClass(\core\component::class);

        if (array_key_exists('autoload', $composer)) {
            // Check that the PSR-4 namespaces are present and correct.
            if (array_key_exists('psr-4', $composer['autoload'])) {
                $autoloadnamespaces = $rc->getProperty('psr4namespaces')->getValue(null);
                foreach ($composer['autoload']['psr-4'] as $namespace => $path) {
                    // Composer PSR-4 namespace autoloads may optionally have a trailing slash. Standardise the value.
                    $namespace = rtrim($namespace, '\\');

                    // If it exists in the composer.json the namespace must exist in our autoloader.
                    $this->assertArrayHasKey($namespace, $autoloadnamespaces);

                    // Ours should be standardised to not have a trailing slash.
                    $this->assertEquals(
                        rtrim($relativepath, '/'),
                        $relativepath,
                        "Moodle PSR-4 namespaces must have no trailing /",
                    );

                    // The composer.json can specify an array of possible values.
                    // Standardise the format to the array format.
                    $paths = is_array($path) ? $path : [$path];

                    foreach ($paths as $path) {
                        // The composer.json can specify any arbitrary directory within the folder.
                        // It always contains a leading slash (/) or backslash (\) on Windows.
                        // It may also have an optional trailing slash (/).
                        // Concatenate the parts and removes the slashes.
                        $relativenamespacepath = trim("{$relativepath}/{$path}", '/\\');

                        // The Moodle PSR-4 autoloader data has two formats:
                        // - a string, for a single source; or
                        // - an array, for multiple sources.
                        // Standardise the format to the latter format.
                        if (!is_array($autoloadnamespaces[$namespace])) {
                            $autoloadnamespaces[$namespace] = [$autoloadnamespaces[$namespace]];
                        }

                        // Ensure that the autoloader contains the normalised path.
                        $this->assertContains(
                            $relativenamespacepath,
                            $autoloadnamespaces[$namespace],
                            "Moodle PSR-4 namespace missing entry for library {$name}: {$namespace} => {$relativenamespacepath}",
                        );
                    }
                }
            }

            // Check that the composer autoload files are present.
            if (array_key_exists('files', $composer['autoload'])) {
                // The Moodle composer file autoloads are a simple string[].
                $autoloadnamefiles = $rc->getProperty('composerautoloadfiles')->getValue(null);
                foreach ($composer['autoload']['files'] as $file) {
                    $this->assertContains(trim($relativepath, '/\\') . "/{$file}", $autoloadnamefiles);
                }
            }
        }
    }

    /**
     * Test that fetching of subtype data throws an exception when a subplugins.php is present without a json equivalent.
     */
    public function test_fetch_subtypes_php_only(): void {
        $vfileroot = \org\bovigo\vfs\vfsStream::setup('root', null, [
            'plugintype' => [
                'exampleplugin' => [
                    'db' => [
                        'subplugins.php' => '',
                    ],
                ],
            ],
        ]);

        $this->expectException(coding_exception::class);
        $this->expectExceptionMessageMatches('/Use of subplugins.php has been deprecated and is no longer supported/');

        $pluginroot = $vfileroot->getChild('plugintype/exampleplugin');

        $rcm = new \ReflectionMethod(\core\component::class, 'fetch_subtypes');
        $rcm->invoke(null, $pluginroot->url());
    }

    /**
     * Test that fetching of subtype data does not throw an exception when a subplugins.php is present
     * with a json file equivalent.
     *
     * Note: The content of the php file is irrelevant and we no longer use it anyway.
     */
    public function test_fetch_subtypes_php_and_json(): void {
        global $CFG;

        $this->resetAfterTest();
        $vfileroot = \org\bovigo\vfs\vfsStream::setup('root', null, [
            'plugintype' => [
                'exampleplugin' => [
                    'db' => [
                        'subplugins.json' => json_encode([
                            'subplugintypes' => [
                                'exampleplugina' => 'apples',
                            ],
                        ]),
                        'subplugins.php' => '',
                    ],
                    'apples' => [],
                ],
            ],
        ]);

        $CFG->dirroot = $vfileroot->url();
        $pluginroot = $vfileroot->getChild('plugintype/exampleplugin');

        $rcm = new \ReflectionMethod(\core\component::class, 'fetch_subtypes');
        $subplugins = $rcm->invoke(null, $pluginroot->url());

        $this->assertEquals([
            'plugintypes' => [
                'exampleplugina' => $pluginroot->getChild('apples')->url(),
            ],
        ], $subplugins);
    }

    /**
     * Test that fetching of subtype data in a file which is missing the new subplugintypes key warns.
     */
    public function test_fetch_subtypes_plugintypes_only(): void {
        global $CFG;

        $this->resetAfterTest();
        $vfileroot = \org\bovigo\vfs\vfsStream::setup('root', null, [
            'plugintype' => [
                'exampleplugin' => [
                    'db' => [
                        'subplugins.json' => json_encode([
                            'plugintypes' => [
                                'exampleplugina' => 'plugintype/exampleplugin/apples',
                            ],
                        ]),
                        'subplugins.php' => '',
                    ],
                    'apples' => [],
                ],
            ],
        ]);

        $CFG->dirroot = $vfileroot->url();
        $pluginroot = $vfileroot->getChild('plugintype/exampleplugin');

        $logdir = make_request_directory();
        $logfile = "{$logdir}/error.log";
        ini_set('error_log', $logfile);

        $rcm = new \ReflectionMethod(\core\component::class, 'fetch_subtypes');
        $subplugins = $rcm->invoke(null, $pluginroot->url());

        $this->assertEquals([
            'plugintypes' => [
                'exampleplugina' => $pluginroot->getChild('apples')->url(),
            ],
        ], $subplugins);

        $warnings = file_get_contents($logfile);
        $this->assertMatchesRegularExpression('/No subplugintypes defined in .*subplugins.json/', $warnings);
    }

    /**
     * Ensure that invalid JSON in the subplugins.json file warns appropriately.
     *
     * @dataProvider invalid_subplugins_json_provider
     * @param string[] $expectedwarnings Errors to expect in the exception message
     * @param array[] $json The contents of the subplugins.json file
     */
    public function test_fetch_subtypes_json_invalid_values(
        array $expectedwarnings,
        array $json,
    ): void {
        global $CFG;

        $this->resetAfterTest();
        $vfileroot = \org\bovigo\vfs\vfsStream::setup('root', null, [
            'plugintype' => [
                'exampleplugin' => [
                    'db' => [
                        'subplugins.json' => json_encode($json),
                        'subplugins.php' => '',
                    ],
                    'apples' => [],
                    'pears' => [],
                ],
            ],
        ]);

        $CFG->dirroot = $vfileroot->url();
        $pluginroot = $vfileroot->getChild('plugintype/exampleplugin');

        $logdir = make_request_directory();
        $logfile = "{$logdir}/error.log";
        ini_set('error_log', $logfile);

        $rcm = new \ReflectionMethod(\core\component::class, 'fetch_subtypes');
        $rcm->invoke(null, $pluginroot->url());

        $warnings = file_get_contents($logfile);
        foreach ($expectedwarnings as $expectedwarning) {
            $this->assertMatchesRegularExpression($expectedwarning, $warnings);
        }
    }

    /**
     * Data provider for invalid subplugins.json files.
     *
     * @return array
     */
    public static function invalid_subplugins_json_provider(): array {
        return [
            'Invalid characters in subtype name' => [
                'expectedwarnings' => [
                    "/Invalid subtype .*APPLES.*detected.*invalid characters present/",
                ],
                'json' => [
                    'subplugintypes' => [
                        'APPLES' => 'plugintype/exampleplugin/apples',
                    ],
                ],
            ],

            'Subplugin which duplicates a core subsystem' => [
                'expectedwarnings' => [
                    "/Invalid subtype .*editor.*detected.*duplicates core subsystem/",
                ],
                'json' => [
                    'subplugintypes' => [
                        'editor' => 'apples',
                    ],
                ],
            ],

            'Subplugin directory does not exist' => [
                'expectedwarnings' => [
                    "/Invalid subtype directory/",
                ],
                'json' => [
                    'subplugintypes' => [
                        'exampleapples' => 'berries',
                    ],
                ],
            ],

            'More subplugintypes than plugintypes' => [
                'expectedwarnings' => [
                    "/Subplugintypes and plugintypes are not in sync/",
                ],
                'json' => [
                    'subplugintypes' => [
                        'apples' => 'pears',
                    ],
                    'plugintypes' => [],
                ],
            ],

            'More plugintypes than subplugintypes' => [
                'expectedwarnings' => [
                    "/Subplugintypes and plugintypes are not in sync /",
                ],
                'json' => [
                    'subplugintypes' => [
                        'apples' => 'apples',
                    ],
                    'plugintypes' => [
                        'apples' => 'plugintype/exampleplugin/apples',
                        'pears' => 'plugintype/exampleplugin/pears',
                    ],
                ],
            ],

            'subplugintype not defined in plugintype' => [
                'expectedwarnings' => [
                    "/Subplugintypes and plugintypes are not in sync for 'apples'/",
                ],
                'json' => [
                    'subplugintypes' => [
                        'apples' => 'apples',
                    ],
                    'plugintypes' => [
                        'pears' => 'plugintype/exampleplugin/pears',
                    ],
                ],
            ],
            'subplugintype does not match plugintype' => [
                'expectedwarnings' => [
                    "/Subplugintypes and plugintypes are not in sync for 'apples'/",
                ],
                'json' => [
                    'subplugintypes' => [
                        'apples' => 'apples',
                    ],
                    'plugintypes' => [
                        'apples' => 'plugintype/exampleplugin/pears',
                    ],
                ],
            ],
        ];
    }

    /**
     * Test various methods when a deprecated plugin type is introduced.
     *
     * @runInSeparateProcess
     * @return void
     */
    public function test_core_component_deprecated_plugintype(): void {
        $this->resetAfterTest();

        // Inject the 'fake' plugin type.
        $this->add_full_mocked_plugintype(
            plugintype: 'fake',
            path: 'lib/tests/fixtures/fakeplugins/fake'
        );

        $componenthashbefore = component::get_all_component_hash();
        $versionhashbefore = component::get_all_versions_hash();

        // Deprecation-specific APIs - pre-deprecation.
        $this->assertArrayHasKey('fake', component::get_plugin_types());
        $this->assertArrayHasKey('fullfeatured', component::get_plugin_list('fake'));
        $this->assertFalse(component::is_deprecated_plugin_type('fake'));
        $this->assertFalse(component::is_deleted_plugin_type('fake'));
        $this->assertFalse(component::is_plugintype_in_deprecation('fake'));
        $this->assertArrayNotHasKey('fake', component::get_deprecated_plugin_types());
        $this->assertArrayNotHasKey('fullfeatured', component::get_deprecated_plugin_list('fake'));
        $this->assertArrayNotHasKey('fake', component::get_deleted_plugin_types());
        $this->assertArrayNotHasKey('fullfeatured', component::get_deleted_plugin_list('fake'));
        $this->assertArrayHasKey('fake', component::get_all_plugin_types());
        $this->assertArrayHasKey('fullfeatured', component::get_all_plugins_list('fake'));

        // Deprecate the fake plugintype via mocking component sources.
        $this->deprecate_full_mocked_plugintype('fake');

        // Verify before/after hashes have changed, since the plugintype is no longer part of the hash calcs.
        $this->assertNotEquals(component::get_all_component_hash(), $componenthashbefore);
        $this->assertNotEquals(component::get_all_versions_hash(), $versionhashbefore);

        // Deprecation-specific APIs - post-deprecation.
        $this->assertTrue(component::is_deprecated_plugin_type('fake'));
        $this->assertFalse(component::is_deleted_plugin_type('fake'));
        $this->assertTrue(component::is_plugintype_in_deprecation('fake'));
        $this->assertArrayHasKey('fake', component::get_deprecated_plugin_types());
        $this->assertArrayHasKey('fullfeatured', component::get_deprecated_plugin_list('fake'));
        $this->assertArrayNotHasKey('fake', component::get_deleted_plugin_types());
        $this->assertArrayNotHasKey('fullfeatured', component::get_deleted_plugin_list('fake'));
        $this->assertArrayHasKey('fake', component::get_all_plugin_types());
        $this->assertArrayHasKey('fullfeatured', component::get_all_plugins_list('fake'));

        // Deprecated plugins excluded from the following for B/C.
        $this->assertArrayNotHasKey('fake', component::get_plugin_types());
        $this->assertArrayNotHasKey('fullfeatured', component::get_plugin_list('fake'));
        $this->assertArrayNotHasKey('fake', component::get_component_list());
        $this->assertEmpty(component::get_plugin_list_with_file('fake', 'classes/dummy.php'));
        $this->assertEmpty(component::get_plugin_list_with_class('fake', 'dummy'));

        // Deprecated plugins excluded by default for B/C, but can be included by request.
        $this->assertNotContains('fake_fullfeatured', component::get_component_names());
        $this->assertContains('fake_fullfeatured', component::get_component_names(false, true));

        // Deprecated plugins included in the following.
        $this->assertIsString(component::get_plugin_directory('fake', 'fullfeatured')); // Used by string manager.
        $this->assertIsString(component::get_component_directory('fake_fullfeatured')); // Uses get_plugin_directory().
        $this->assertTrue(component::has_monologo_icon('fake', 'fullfeatured'));  // Uses get_plugin_directory().
        $this->assertEquals('fake_fullfeatured', component::get_component_from_classname(\fake_fullfeatured\example::class));

        // Class autoloading of deprecated plugins is permitted, to facilitate plugin migration code.
        $this->assertArrayHasKey('fake_fullfeatured\dummy',
            component::get_component_classes_in_namespace('fake_fullfeatured'));
        $this->assertTrue(class_exists(\fake_fullfeatured\dummy::class));
    }

    /**
     * Test various core_component APIs when dealing with deleted plugin types.
     *
     * @runInSeparateProcess
     * @return void
     */
    public function test_core_component_deleted_plugintype(): void {
        $this->resetAfterTest();

        // Inject the 'fake' plugin type.
        $this->add_full_mocked_plugintype(
            plugintype: 'fake',
            path: 'lib/tests/fixtures/fakeplugins/fake',
        );

        // Delete the fake plugintype via mocking component sources.
        $this->delete_full_mocked_plugintype('fake');

        // Deprecation-specific methods.
        $this->assertFalse(component::is_deprecated_plugin_type('fake'));
        $this->assertTrue(component::is_deleted_plugin_type('fake'));
        $this->assertTrue(component::is_plugintype_in_deprecation('fake'));
        $this->assertArrayNotHasKey('fake', component::get_deprecated_plugin_types());
        $this->assertArrayNotHasKey('fullfeatured', component::get_deprecated_plugin_list('fake'));
        $this->assertArrayHasKey('fake', component::get_deleted_plugin_types());
        $this->assertArrayHasKey('fullfeatured', component::get_deleted_plugin_list('fake'));
        $this->assertArrayHasKey('fake', component::get_all_plugin_types());
        $this->assertArrayHasKey('fullfeatured', component::get_all_plugins_list('fake'));

        // Deleted plugintypes/plugins are not included in other methods.
        $this->assertArrayNotHasKey('fake', component::get_plugin_types());
        $this->assertArrayNotHasKey('fullfeatured', component::get_plugin_list('fake'));
        $this->assertNotContains('fake_fullfeatured', component::get_component_names());
        $this->assertNotContains('fake_fullfeatured', component::get_component_names(false, true));
        $this->assertArrayNotHasKey('fake', component::get_component_list());
        $this->assertEmpty(component::get_plugin_list_with_file('fake', 'classes/dummy.php'));
        $this->assertEmpty(component::get_plugin_list_with_class('fake', 'dummy'));
        $this->assertFalse(component::has_monologo_icon('fake', 'fullfeatured'));
        $this->assertNull(component::get_plugin_directory('fake', 'fullfeatured'));
        $this->assertNull(component::get_component_directory('fake_fullfeatured'));
        $this->assertNull(component::get_component_from_classname(\fake_fullfeatured\example::class));

        // Class autoloading of deleted plugins is not supported.
        $this->assertArrayNotHasKey('fake_fullfeatured\dummy',
            component::get_component_classes_in_namespace('fake_fullfeatured'));
        $this->assertFalse(class_exists(fake_fullfeatured\dummy::class));
    }

    /**
     * Test various core_component APIs when dealing with deprecated subplugins.
     *
     * @runInSeparateProcess
     * @return void
     */
    public function test_core_component_deprecated_subplugintype(): void {
        $this->resetAfterTest();

        // Inject the 'fake' plugin type. This includes three mock subplugins:
        // 1. fullsubtype_example: a regular plugin type, not deprecated, nor deleted.
        // 2. fulldeprecatedsubtype_test: a deprecated subplugin type.
        // 3. fulldeletedsubtype_demo: a deleted subplugin type.
        $this->add_full_mocked_plugintype(
            plugintype: 'fake',
            path: 'lib/tests/fixtures/fakeplugins/fake',
            subpluginsupport: true
        );
        $this->assert_deprecation_apis_subplugins();
    }

    /**
     * Verify that a plugin which supports subplugins cannot be deprecated.
     *
     * @runInSeparateProcess
     * @return void
     */
    public function test_core_component_deprecated_subplugintype_supporting_subplugins(): void {
        $this->resetAfterTest();

        // Inject the 'fake' plugin type. This includes three mock subplugins:
        // 1. fullsubtype_example: a regular plugin type, not deprecated, nor deleted.
        // 2. fulldeprecatedsubtype_test: a deprecated subplugin type.
        // 3. fulldeletedsubtype_demo: a deleted subplugin type.
        $this->add_full_mocked_plugintype(
            plugintype: 'fake',
            path: 'lib/tests/fixtures/fakeplugins/fake',
            subpluginsupport: true
        );

        // Try to deprecate the fake plugintype via mocking component sources.
        $this->deprecate_full_mocked_plugintype('fake');

        // Deprecation unsupported, so verify core_component treats all plugins the same as before the deprecation attempt.
        // Debugging is expected to be emitted during core_component::init().
        $this->assert_deprecation_apis_subplugins();
        $this->assertDebuggingCalled('Deprecation of a plugin type which supports subplugins is not supported. ' .
            'These plugin types will continue to be treated as active.', DEBUG_DEVELOPER);
    }

    /**
     * Verify that a plugin which supports subplugins cannot be deleted.
     *
     * @runInSeparateProcess
     * @return void
     */
    public function test_core_component_deleted_subplugintype_supporting_subplugins(): void {
        $this->resetAfterTest();

        // Inject the 'fake' plugin type. This includes three mock subplugins:
        // 1. fullsubtype_example: a regular plugin type, not deprecated, nor deleted.
        // 2. fulldeprecatedsubtype_test: a deprecated subplugin type.
        // 3. fulldeletedsubtype_demo: a deleted subplugin type.
        $this->add_full_mocked_plugintype(
            plugintype: 'fake',
            path: 'lib/tests/fixtures/fakeplugins/fake',
            subpluginsupport: true
        );

        // Try to delete the fake plugintype via mocking component sources.
        $this->delete_full_mocked_plugintype('fake');

        // Deletion unsupported, so verify core_component treats all plugins the same as before the deletion attempt.
        // Debugging is expected to be emitted during core_component::init().
        $this->assert_deprecation_apis_subplugins();
        $this->assertDebuggingCalled('Deprecation of a plugin type which supports subplugins is not supported. ' .
            'These plugin types will continue to be treated as active.', DEBUG_DEVELOPER);
    }

    /**
     * Helper asserting the returns for various core_component APIs when dealing with deprecated and deleted subplugins.
     *
     * @return void
     */
    protected function assert_deprecation_apis_subplugins(): void {
        // Deprecation-specific methods.
        $this->assertFalse(component::is_deprecated_plugin_type('fake'));
        $this->assertFalse(component::is_deprecated_plugin_type('fullsubtype'));
        $this->assertTrue(component::is_deprecated_plugin_type('fulldeprecatedsubtype'));
        $this->assertFalse(component::is_deprecated_plugin_type('fulldeletedsubtype'));

        $this->assertFalse(component::is_deleted_plugin_type('fake'));
        $this->assertFalse(component::is_deleted_plugin_type('fullsubtype'));
        $this->assertFalse(component::is_deleted_plugin_type('fulldeprecatedsubtype'));
        $this->assertTrue(component::is_deleted_plugin_type('fulldeletedsubtype'));

        $this->assertFalse(component::is_plugintype_in_deprecation('fake'));
        $this->assertFalse(component::is_plugintype_in_deprecation('fullsubtype'));
        $this->assertTrue(component::is_plugintype_in_deprecation('fulldeprecatedsubtype'));
        $this->assertTrue(component::is_plugintype_in_deprecation('fulldeletedsubtype'));

        $this->assertArrayNotHasKey('fake', component::get_deprecated_plugin_types());
        $this->assertArrayNotHasKey('fullsubtype', component::get_deprecated_plugin_types());
        $this->assertArrayHasKey('fulldeprecatedsubtype', component::get_deprecated_plugin_types());
        $this->assertArrayNotHasKey('fulldeletedsubtype', component::get_deprecated_plugin_types());

        $this->assertArrayNotHasKey('fullfeatured', component::get_deprecated_plugin_list('fake'));
        $this->assertArrayNotHasKey('example', component::get_deprecated_plugin_list('fullsubtype'));
        $this->assertArrayHasKey('test', component::get_deprecated_plugin_list('fulldeprecatedsubtype'));
        $this->assertArrayNotHasKey('demo', component::get_deprecated_plugin_list('fulldeletedsubtype'));

        $this->assertArrayNotHasKey('fake', component::get_deleted_plugin_types());
        $this->assertArrayNotHasKey('fullsubtype', component::get_deleted_plugin_types());
        $this->assertArrayNotHasKey('fulldeprecatedsubtype', component::get_deleted_plugin_types());
        $this->assertArrayHasKey('fulldeletedsubtype', component::get_deleted_plugin_types());

        $this->assertArrayNotHasKey('fullfeatured', component::get_deleted_plugin_list('fake'));
        $this->assertArrayNotHasKey('example', component::get_deleted_plugin_list('fullsubtype'));
        $this->assertArrayNotHasKey('test', component::get_deleted_plugin_list('fulldeprecatedsubtype'));
        $this->assertArrayHasKey('demo', component::get_deleted_plugin_list('fulldeletedsubtype'));

        $this->assertArrayHasKey('fake', component::get_all_plugin_types());
        $this->assertArrayHasKey('fullsubtype', component::get_all_plugin_types());
        $this->assertArrayHasKey('fulldeprecatedsubtype', component::get_all_plugin_types());
        $this->assertArrayHasKey('fulldeletedsubtype', component::get_all_plugin_types());

        $this->assertArrayHasKey('fullfeatured', component::get_all_plugins_list('fake'));
        $this->assertArrayHasKey('example', component::get_all_plugins_list('fullsubtype'));
        $this->assertArrayHasKey('test', component::get_all_plugins_list('fulldeprecatedsubtype'));
        $this->assertArrayHasKey('demo', component::get_all_plugins_list('fulldeletedsubtype'));

        // Deprecated and deleted plugins excluded from the following for B/C.
        $this->assertArrayHasKey('fake', component::get_plugin_types());
        $this->assertArrayHasKey('fullsubtype', component::get_plugin_types());
        $this->assertArrayNotHasKey('fulldeprecatedsubtype', component::get_plugin_types());
        $this->assertArrayNotHasKey('fulldeletedsubtype', component::get_plugin_types());

        $this->assertNotEmpty(component::get_plugin_list('fake'));
        $this->assertNotEmpty(component::get_plugin_list('fullsubtype'));
        $this->assertEmpty(component::get_plugin_list('fulldeprecatedsubtype'));
        $this->assertEmpty(component::get_plugin_list('fulldeletedsubtype'));

        $this->assertArrayHasKey('fake', component::get_component_list());
        $this->assertArrayHasKey('fullsubtype', component::get_component_list());
        $this->assertArrayNotHasKey('fulldeprecatedsubtype', component::get_component_list());
        $this->assertArrayNotHasKey('fulldeletedsubtype', component::get_component_list());

        $this->assertArrayHasKey('fullfeatured', component::get_plugin_list_with_file('fake', 'classes/dummy.php'));
        $this->assertArrayHasKey('example', component::get_plugin_list_with_file('fullsubtype', 'classes/dummy.php'));
        $this->assertEmpty(component::get_plugin_list_with_file('fulldeprecatedsubtype', 'classes/dummy.php'));
        $this->assertEmpty(component::get_plugin_list_with_file('fulldeletedsubtype', 'classes/dummy.php'));

        $this->assertArrayHasKey('fake_fullfeatured', component::get_plugin_list_with_class('fake', 'dummy'));
        $this->assertArrayHasKey('fullsubtype_example', component::get_plugin_list_with_class('fullsubtype', 'dummy'));
        $this->assertEmpty(component::get_plugin_list_with_class('fulldeprecatedsubtype', 'dummy'));
        $this->assertEmpty(component::get_plugin_list_with_class('fulldeletedsubtype', 'dummy'));

        $this->assertArrayHasKey('fullsubtype', component::get_subplugins('fake_fullfeatured'));
        $this->assertContains('example', component::get_subplugins('fake_fullfeatured')['fullsubtype']);
        $this->assertArrayNotHasKey('fulldeprecatedsubtype', component::get_subplugins('fake_fullfeatured'));
        $this->assertArrayNotHasKey('fulldeletedsubtype', component::get_subplugins('fake_fullfeatured'));

        $this->assertArrayHasKey('fullsubtype', component::get_all_subplugins('fake_fullfeatured'));
        $this->assertContains('example', component::get_all_subplugins('fake_fullfeatured')['fullsubtype']);
        $this->assertContains('test', component::get_all_subplugins('fake_fullfeatured')['fulldeprecatedsubtype']);
        $this->assertContains('demo', component::get_all_subplugins('fake_fullfeatured')['fulldeletedsubtype']);

        // Deprecated plugins excluded by default for B/C, but can be included by request.
        // Deleted plugins are always excluded.
        $this->assertContains('fake_fullfeatured', component::get_component_names());
        $this->assertContains('fullsubtype_example', component::get_component_names());
        $this->assertNotContains('fulldeprecatedsubtype_test', component::get_component_names());
        $this->assertNotContains('fulldeletedsubtype_demo', component::get_component_names());
        $this->assertContains('fulldeprecatedsubtype_test', component::get_component_names(false, true));
        $this->assertNotContains('fulldeletedsubtype_demo', component::get_component_names(false, true));

        // Deprecated plugins included in the following, but deleted plugins are excluded.
        $this->assertIsString(component::get_plugin_directory('fake', 'fullfeatured')); // Used by string manager.
        $this->assertIsString(component::get_plugin_directory('fullsubtype', 'example'));
        $this->assertIsString(component::get_plugin_directory('fulldeprecatedsubtype', 'test'));
        $this->assertNull(component::get_plugin_directory('fulldeletedsubtype', 'demo'));

        $this->assertIsString(component::get_component_directory('fake_fullfeatured')); // Uses get_plugin_directory().
        $this->assertIsString(component::get_component_directory('fullsubtype_example'));
        $this->assertIsString(component::get_component_directory('fulldeprecatedsubtype_test'));
        $this->assertNull(component::get_component_directory('fulldeletedsubtype_demo'));

        $this->assertTrue(component::has_monologo_icon('fullsubtype', 'example')); // Uses get_plugin_directory().
        $this->assertTrue(component::has_monologo_icon('fulldeprecatedsubtype', 'test'));
        $this->assertFalse(component::has_monologo_icon('fulldeletedsubtype', 'demo'));

        $this->assertEquals('fake_fullfeatured', component::get_component_from_classname(\fake_fullfeatured\example::class));
        $this->assertEquals('fullsubtype_example',
            component::get_component_from_classname(\fullsubtype_example\example::class));
        $this->assertEquals('fulldeprecatedsubtype_test',
            component::get_component_from_classname(\fulldeprecatedsubtype_test\example::class));
        $this->assertNull(component::get_component_from_classname(\fulldeletedsubtype_demo\example::class));

        // Deprecated and deleted plugins included in the following.
        $this->assertEquals('fake_fullfeatured', component::get_subtype_parent('fullsubtype'));
        $this->assertEquals('fake_fullfeatured', component::get_subtype_parent('fulldeprecatedsubtype'));
        $this->assertEquals('fake_fullfeatured', component::get_subtype_parent('fulldeletedsubtype'));

        // Class autoloading of deprecated plugins is permitted, to facilitate plugin migration code, but not for deleted plugins.
        $this->assertArrayHasKey('fake_fullfeatured\dummy',
            component::get_component_classes_in_namespace('fake_fullfeatured'));
        $this->assertArrayHasKey('fullsubtype_example\dummy',
            component::get_component_classes_in_namespace('fullsubtype_example'));
        $this->assertArrayHasKey('fulldeprecatedsubtype_test\dummy',
            component::get_component_classes_in_namespace('fulldeprecatedsubtype_test'));
        $this->assertEquals([], component::get_component_classes_in_namespace('fulldeletedsubtype_demo'));

        $this->assertTrue(class_exists(\fake_fullfeatured\dummy::class));
        $this->assertTrue(class_exists(\fullsubtype_example\dummy::class));
        $this->assertTrue(class_exists(\fulldeprecatedsubtype_test\dummy::class));
        $this->assertFalse(class_exists(\fulldeletedsubtype_demo\dummy::class));
    }
}

Filemanager

Name Type Size Permission Actions
analytics Folder 0755
behat Folder 0755
classes Folder 0755
content Folder 0755
context Folder 0755
db Folder 0755
event Folder 0755
external Folder 0755
fixtures Folder 0755
hook Folder 0755
hub Folder 0755
lock Folder 0755
moodlenet Folder 0755
navigation Folder 0755
oauth2 Folder 0755
other Folder 0755
output Folder 0755
performance Folder 0755
plugininfo Folder 0755
privacy Folder 0755
route Folder 0755
router Folder 0755
session Folder 0755
task Folder 0755
accesslib_has_capability_test.php File 29.77 KB 0644
accesslib_test.php File 245.34 KB 0644
adminlib_test.php File 7.42 KB 0644
admintree_test.php File 18.08 KB 0644
ajaxlib_test.php File 4.43 KB 0644
analysers_test.php File 12.71 KB 0644
antivirus_test.php File 11.98 KB 0644
attribute_helper_test.php File 8.41 KB 0644
authlib_test.php File 22.97 KB 0644
behat_lib_test.php File 3.3 KB 0644
blocklib_test.php File 36.31 KB 0644
check_test.php File 2.31 KB 0644
client_test.php File 4.32 KB 0644
collator_test.php File 12.1 KB 0644
completionlib_test.php File 93.4 KB 0644
component_test.php File 82.81 KB 0644
componentlib_test.php File 6.93 KB 0644
configonlylib_test.php File 8.95 KB 0644
content_test.php File 4.79 KB 0644
context_block_test.php File 4.17 KB 0644
context_helper_test.php File 22.28 KB 0644
context_test.php File 3.42 KB 0644
core_media_player_native_test.php File 6.44 KB 0644
core_renderer_template_exploit_test.php File 16.54 KB 0644
core_renderer_test.php File 7.57 KB 0644
core_userfeedback_test.php File 2.3 KB 0644
coverage.php File 3.25 KB 0644
cron_test.php File 6.82 KB 0644
csvclass_test.php File 5.66 KB 0644
curl_security_helper_test.php File 14.88 KB 0644
customcontext_test.php File 4.67 KB 0644
dataformat_test.php File 4.18 KB 0644
datalib_test.php File 48.93 KB 0644
datalib_update_with_unique_index_test.php File 6.12 KB 0644
date_legacy_test.php File 13.67 KB 0644
date_test.php File 32.57 KB 0644
deprecation_test.php File 16.17 KB 0644
di_test.php File 5.33 KB 0644
editorlib_test.php File 1.96 KB 0644
emoticon_manager_test.php File 4.2 KB 0644
encryption_test.php File 6.82 KB 0644
environment_test.php File 8.86 KB 0644
exporter_test.php File 16.83 KB 0644
externallib_test.php File 2.03 KB 0644
filelib_test.php File 86.06 KB 0644
filestorage_zip_archive_test.php File 2.54 KB 0644
filetypes_test.php File 10.09 KB 0644
filter_manager_test.php File 3.33 KB 0644
filterlib_test.php File 37.09 KB 0644
formatting_test.php File 26.09 KB 0644
formslib_test.php File 40.1 KB 0644
gdlib_test.php File 5.73 KB 0644
googlelib_test.php File 1.62 KB 0644
gradelib_test.php File 12.01 KB 0644
grades_external_test.php File 11.22 KB 0644
grading_external_test.php File 26.55 KB 0644
graphlib_test.php File 7.14 KB 0644
grouplib_test.php File 110.82 KB 0644
h5p_clean_orphaned_records_task_test.php File 3.17 KB 0644
html2text_test.php File 8.82 KB 0644
htmlpurifier_test.php File 23.11 KB 0644
http_client_test.php File 14.67 KB 0644
ip_utils_test.php File 19.55 KB 0644
jquery_test.php File 1.59 KB 0644
ldaplib_test.php File 17.77 KB 0644
licenselib_test.php File 11.84 KB 0644
locale_test.php File 4.96 KB 0644
lock_config_test.php File 3.48 KB 0644
lock_test.php File 6.39 KB 0644
markdown_test.php File 2.27 KB 0644
mathslib_test.php File 13.51 KB 0644
medialib_test.php File 19.68 KB 0644
message_test.php File 16.43 KB 0644
messagelib_test.php File 56.44 KB 0644
minify_test.php File 3.15 KB 0644
modinfolib_test.php File 104.59 KB 0644
moodle_page_test.php File 34.65 KB 0644
moodlelib_current_language_test.php File 7.68 KB 0644
moodlelib_partial_test.php File 4.62 KB 0644
moodlelib_test.php File 244.13 KB 0644
myprofilelib_test.php File 12.2 KB 0644
navigationlib_test.php File 32.65 KB 0644
notification_test.php File 4.37 KB 0644
oauth2_test.php File 23.46 KB 0644
outputcomponents_test.php File 32.07 KB 0644
outputfactories_test.php File 6.83 KB 0644
outputrenderers_test.php File 1.63 KB 0644
outputrequirementslib_test.php File 15.72 KB 0644
param_test.php File 4.11 KB 0644
pdflib_test.php File 3.21 KB 0644
persistent_test.php File 30.68 KB 0644
phpxmlrpc_test.php File 2.05 KB 0644
plugin_manager_test.php File 46.99 KB 0644
portfoliolib_test.php File 8.41 KB 0644
progress_display_test.php File 3.76 KB 0644
progress_test.php File 14.53 KB 0644
qrcode_test.php File 1.69 KB 0644
questionlib_test.php File 75.67 KB 0644
regex_test.php File 1.56 KB 0644
report_helper_test.php File 12.78 KB 0644
requirejs_test.php File 2.23 KB 0644
router_test.php File 7.66 KB 0644
rsslib_test.php File 7.41 KB 0644
rtlcss_test.php File 57.03 KB 0644
sample_questions.ser File 141.76 KB 0644
sample_questions.xml File 102.62 KB 0644
sample_questions_with_old_image_tag.ser File 4.85 KB 0644
sample_questions_with_old_image_tag.xml File 4.08 KB 0644
sample_questions_wrong.xml File 102.57 KB 0644
scss_test.php File 4.3 KB 0644
session_redis_cluster_test.php File 4.17 KB 0644
sessionlib_test.php File 6.06 KB 0644
setuplib_test.php File 20.21 KB 0644
statslib_test.php File 26.36 KB 0644
stored_progress_bar_test.php File 7.17 KB 0644
string_manager_standard_test.php File 10.35 KB 0644
system_clock_test.php File 2.42 KB 0644
text_test.php File 26.68 KB 0644
theme_config_test.php File 7.25 KB 0644
update_api_test.php File 6.65 KB 0644
update_checker_test.php File 10.91 KB 0644
update_code_manager_test.php File 9.12 KB 0644
update_validator_test.php File 18.32 KB 0644
upgrade_util_test.php File 5.36 KB 0644
upgradelib_test.php File 75.46 KB 0644
url_test.php File 30.74 KB 0644
user_menu_test.php File 4.45 KB 0644
user_test.php File 43.35 KB 0644
useragent_test.php File 67.06 KB 0644
weblib_format_text_test.php File 14.44 KB 0644
weblib_test.php File 42.09 KB 0644
xhprof_test.php File 10.05 KB 0644
xmlize_test.php File 2.57 KB 0644
xsendfilelib_test.php File 5 KB 0644
Filemanager