Proyectos de Subversion Moodle

Rev

Autoría | Ultima modificación | Ver Log |

<?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_ai;

use core_ai\aiactions\generate_image;
use core_ai\aiactions\generate_text;
use core_ai\aiactions\summarise_text;
use core_ai\aiactions\explain_text;
use core_ai\aiactions\responses\response_generate_image;

/**
 * Test ai subsystem manager methods.
 *
 * @package    core_ai
 * @copyright  2024 Matt Porritt <matt.porritt@moodle.com>
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 * @covers     \core_ai\manager
 */
final class manager_test extends \advanced_testcase {
    /**
     * Test get_ai_plugin_classname.
     */
    public function test_get_ai_plugin_classname(): void {
        $manager = \core\di::get(manager::class);

        // We're working with a private method here, so we need to use reflection.
        $method = new \ReflectionMethod($manager, 'get_ai_plugin_classname');

        // Test a provider plugin.
        $classname = $method->invoke($manager, 'aiprovider_fooai');
        $this->assertEquals('aiprovider_fooai\\provider', $classname);

        // Test a placement plugin.
        $classname = $method->invoke($manager, 'aiplacement_fooplacement');
        $this->assertEquals('aiplacement_fooplacement\\placement', $classname);

        // Test an invalid plugin.
        $this->expectException(\coding_exception::class);
        $this->expectExceptionMessage('Plugin name does not start with \'aiprovider_\' or \'aiplacement_\': bar');
        $method->invoke($manager, 'bar');
    }

    /**
     * Test get_supported_actions.
     */
    public function test_get_supported_actions(): void {
        $manager = \core\di::get(manager::class);
        $actions = $manager->get_supported_actions('aiprovider_openai');

        // Assert array keys match the expected actions.
        $this->assertEquals([
            generate_text::class,
            generate_image::class,
            summarise_text::class,
            explain_text::class,
        ], $actions);
    }

    /**
     * Test create_provider_instance method.
     */
    public function test_create_provider_instance(): void {
        $this->resetAfterTest();
        global $DB;

        // Create the provider instance.
        $manager = \core\di::get(\core_ai\manager::class);
        $config = ['data' => 'goeshere'];
        $provider = $manager->create_provider_instance(
            classname: '\aiprovider_openai\provider',
            name: 'dummy',
            config: $config,
        );

        $this->assertIsInt($provider->id);
        $this->assertFalse($provider->enabled);
        $this->assertEquals('goeshere', $provider->config['data']);

        // Check the record was written to the DB.
        $record = $DB->get_record('ai_providers', ['id' => $provider->id], '*', MUST_EXIST);
        $this->assertEquals($provider->id, $record->id);
    }

    /**
     * Test create_provider_instance non provider class.
     */
    public function test_create_provider_instance_non_provider_class(): void {
        $this->resetAfterTest();

        // Should throw an exception as the class is not a provider.
        $this->expectException(\coding_exception::class);
        $this->expectExceptionMessage(' Provider class not valid: ' . $this::class);

        // Create the provider instance.
        $manager = \core\di::get(\core_ai\manager::class);
        $manager->create_provider_instance(
            classname: $this::class,
            name: 'dummy',
        );
    }

    /**
     * Test get_provider_record method
     */
    public function test_get_provider_record(): void {
        global $DB;

        $this->resetAfterTest();

        // Create a dummy provider record directly in the database.
        $config = ['data' => 'goeshere'];
        $record = new \stdClass();
        $record->name = 'dummy1';
        $record->provider = 'dummy';
        $record->enabled = 1;
        $record->config = json_encode($config);
        $record->actionconfig = json_encode(['generate_text' => 1]);
        $record->id = $DB->insert_record('ai_providers', $record);

        $manager = \core\di::get(\core_ai\manager::class);
        $provider = $manager->get_provider_record(['provider' => 'dummy']);

        $this->assertEquals($record->id, $provider->id);
    }

    /**
     * Test get_provider_records method.
     */
    public function test_get_provider_records(): void {
        $this->resetAfterTest();
        global $DB;

        // Create some dummy provider records directly in the database.
        $config = ['data' => 'goeshere'];
        $record1 = new \stdClass();
        $record1->name = 'dummy1';
        $record1->provider = 'dummy';
        $record1->enabled = 1;
        $record1->config = json_encode($config);
        $record1->actionconfig = json_encode(['generate_text' => 1]);

        $record2 = new \stdClass();
        $record2->name = 'dummy2';
        $record2->provider = 'dummy';
        $record2->enabled = 1;
        $record2->config = json_encode($config);
        $record2->actionconfig = json_encode(['generate_text' => 1]);

        $DB->insert_records('ai_providers', [
                $record1,
                $record2,
        ]);

        // Get the provider records.
        $manager = \core\di::get(\core_ai\manager::class);
        $providers = $manager->get_provider_records(['provider' => 'dummy']);

        // Assert that the records were returned.
        $this->assertCount(2, $providers);
    }

    /**
     * Test get_provider_instances method.
     */
    public function test_get_provider_instances(): void {
        $this->resetAfterTest();
        global $DB;

        $manager = \core\di::get(\core_ai\manager::class);
        $config = ['data' => 'goeshere'];

        $provider = $manager->create_provider_instance(
                classname: '\aiprovider_openai\provider',
                name: 'dummy',
                config: $config,
        );

        // Create an instance record for a non provider class.
        $record = new \stdClass();
        $record->name = 'dummy';
        $record->provider = 'dummy';
        $record->enabled = 1;
        $record->config = json_encode($config);

        $DB->insert_record('ai_providers', $record);

        // Get the provider instances.
        $instances = $manager->get_provider_instances();
        $this->assertDebuggingCalled('Unable to find a provider class for dummy');
        $this->assertCount(1, $instances);
        $this->assertEquals($provider->id, $instances[$provider->id]->id);
    }

    /**
     * Test update_provider_instance method.
     */
    public function test_update_provider_instance(): void {
        $this->resetAfterTest();
        global $DB;

        // Create the provider instance.
        $manager = \core\di::get(\core_ai\manager::class);
        $config = ['data' => 'goeshere'];
        $provider = $manager->create_provider_instance(
                classname: '\aiprovider_openai\provider',
                name: 'dummy',
                config: $config,
        );

        // Update the provider instance.
        $config['data'] = 'updateddata';
        $manager->update_provider_instance($provider, $config);

        // Check the record was updated in the DB.
        $record = $DB->get_record('ai_providers', ['id' => $provider->id], '*', MUST_EXIST);
        $this->assertEquals($provider->id, $record->id);
        $this->assertEquals('updateddata', json_decode($record->config)->data);
    }

    /**
     * Test delete_provider_instance method.
     */
    public function test_delete_provider_instance(): void {
        $this->resetAfterTest();
        global $DB;

        // Create the provider instance.
        $manager = \core\di::get(\core_ai\manager::class);
        $config = ['data' => 'goeshere'];
        $provider = $manager->create_provider_instance(
                classname: '\aiprovider_openai\provider',
                name: 'dummy',
                config: $config,
        );

        // Delete the provider instance.
        $manager->delete_provider_instance($provider);

        // Check the record was deleted from the DB.
        $record = $DB->record_exists('ai_providers', ['id' => $provider->id]);
        $this->assertFalse($record);
    }

    /**
     * Test get_providers_for_actions.
     */
    public function test_get_providers_for_actions(): void {
        $this->resetAfterTest();
        $manager = \core\di::get(manager::class);
        $actions = [
            generate_text::class,
            summarise_text::class,
            explain_text::class,
        ];

        // Create two provider instances.
        $config = [
            'apikey' => 'goeshere',
            'endpoint' => 'https://example.com',
        ];
        $provider1 = $manager->create_provider_instance(
            classname: '\aiprovider_openai\provider',
            name: 'dummy1',
            enabled: true,
            config: $config,
        );

        $config['apiendpoint'] = 'https://example.com';
        $provider2 = $manager->create_provider_instance(
            classname: '\aiprovider_azureai\provider',
            name: 'dummy2',
            enabled: true,
            config: $config,
        );

        // Get the providers for the actions.
        $providers = $manager->get_providers_for_actions($actions);

        // Assert that the providers array is indexed by action name.
        $this->assertEquals($actions, array_keys($providers));

        // Assert that there is only one provider for each action.
        $this->assertCount(2, $providers[generate_text::class]);
        $this->assertCount(2, $providers[summarise_text::class]);
        $this->assertCount(2, $providers[explain_text::class]);

        // Disable the generate text action for the Open AI provider.
        $setresult = $manager->set_action_state(
                plugin: $provider1->provider,
                actionbasename: generate_text::class::get_basename(),
                enabled: 0,
                instanceid: $provider1->id);
        // Assert that the action was disabled.
        $this->assertFalse($setresult);

        $providers = $manager->get_providers_for_actions($actions, true);

        // Assert that there is no provider for the generate text action.
        $this->assertCount(1, $providers[generate_text::class]);
        $this->assertCount(2, $providers[summarise_text::class]);

        // Ordering the provider instances.
        // Re-enable the generate text action for the Openai provider.
        $manager->set_action_state(
            plugin: $provider1->provider,
            actionbasename: generate_text::class::get_basename(),
            enabled: 1,
            instanceid: $provider1->id,
        );

        // Move the $provider2 to the first provider for the generate text action.
        $manager->change_provider_order($provider2->id, \core\plugininfo\aiprovider::MOVE_UP);
        // Get the new providers for the actions.
        $providers = $manager->get_providers_for_actions($actions);
        // Assert whether provider2 is the first provider and provider1 is the last provider for the generate text action.
        $this->assertEquals($providers[generate_text::class][0], $provider2);
        $this->assertEquals($providers[generate_text::class][1], $provider1);

        // Move the $provider2 to the last provider for the generate text action.
        $manager->change_provider_order($provider2->id, \core\plugininfo\aiprovider::MOVE_DOWN);
        // Get the new providers for the actions.
        $providers = $manager->get_providers_for_actions($actions);
        // Assert whether provider1 is the first provider and provider2 is the last provider for the generate text action.
        $this->assertEquals($providers[generate_text::class][0], $provider1);
        $this->assertEquals($providers[generate_text::class][1], $provider2);
    }

    /**
     * Test process_action fail.
     */
    public function test_process_action_fail(): void {
        $this->resetAfterTest();
        global $DB;
        $managermock = $this->getMockBuilder(manager::class)
            ->setConstructorArgs([$DB])
            ->onlyMethods(['call_action_provider'])
            ->getMock();

        $expectedresult = new aiactions\responses\response_generate_image(
            success: true,
        );

        // Set up the expectation for call_action_provider to return the defined result.
        $managermock->expects($this->any())
            ->method('call_action_provider')
            ->willReturn($expectedresult);

        $action = new generate_image(
            contextid: 1,
            userid: 1,
            prompttext: 'This is a test prompt',
            quality: 'hd',
            aspectratio: 'square',
            numimages: 1,
            style: 'vivid',
        );

        // Success should be false as there are no enabled providers.
        $result = $managermock->process_action($action);
        $this->assertFalse($result->get_success());
    }

    /**
     * Test process_action.
     */
    public function test_process_action(): void {
        $this->resetAfterTest();
        global $DB;

        // Create two provider instances.
        $manager = \core\di::get(manager::class);
        $config = ['apikey' => 'goeshere'];
        $provider1 = $manager->create_provider_instance(
                classname: '\aiprovider_openai\provider',
                name: 'dummy1',
                enabled: true,
                config: $config,
        );

        $config['apiendpoint'] = 'https://example.com';
        $provider2 = $manager->create_provider_instance(
                classname: '\aiprovider_azureai\provider',
                name: 'dummy2',
                enabled: true,
                config: $config,
        );

        $managermock = $this->getMockBuilder(manager::class)
            ->setConstructorArgs([$DB])
            ->onlyMethods(['call_action_provider'])
            ->getMock();

        $expectedresult = new aiactions\responses\response_generate_image(
            success: true,
        );

        // Set up the expectation for call_action_provider to return the defined result.
        $managermock->expects($this->any())
            ->method('call_action_provider')
            ->willReturn($expectedresult);

        $action = new generate_image(
            contextid: 1,
            userid: 1,
            prompttext: 'This is a test prompt',
            quality: 'hd',
            aspectratio: 'square',
            numimages: 1,
            style: 'vivid',
        );

        // Should now return the expected result.
        $result = $managermock->process_action($action);
        $this->assertEquals($expectedresult, $result);
    }

    /**
     * Test set_user_policy.
     */
    public function test_set_user_policy(): void {
        $this->resetAfterTest();
        global $DB;

        $result = manager::user_policy_accepted(1, 1);
        $this->assertTrue($result);

        // Check record exists.
        $record = $DB->record_exists('ai_policy_register', ['userid' => 1, 'contextid' => 1]);
        $this->assertTrue($record);
    }

    /**
     * Test get_user_policy.
     */
    public function test_get_user_policy(): void {
        $this->resetAfterTest();
        global $DB;

        // Should be false for user initially.
        $result = manager::get_user_policy_status(1);
        $this->assertFalse($result);

        // Manually add record to the database.
        $record = new \stdClass();
        $record->userid = 1;
        $record->contextid = 1;
        $record->timeaccepted = time();

        $DB->insert_record('ai_policy_register', $record);

        // Should be true for user now.
        $result = manager::get_user_policy_status(1);
        $this->assertTrue($result);
    }

    /**
     * Test policy cache data source.
     */
    public function test_user_policy_caching(): void {
        global $DB;
        $this->resetAfterTest();

        $user1 = $this->getDataGenerator()->create_user();
        $user2 = $this->getDataGenerator()->create_user();
        $user3 = $this->getDataGenerator()->create_user();

        // Manually add records to the database.
        $record1 = new \stdClass();
        $record1->userid = $user1->id;
        $record1->contextid = 1;
        $record1->timeaccepted = time();

        $record2 = new \stdClass();
        $record2->userid = $user2->id;
        $record2->contextid = 1;
        $record2->timeaccepted = time();

        $DB->insert_records('ai_policy_register', [
            $record1,
            $record2,
        ]);

        $policycache = \cache::make('core', 'ai_policy');

        // Test single user.
        $this->assertFalse($policycache->has($user1->id));
        $this->assertFalse($policycache->has($user2->id));
        $this->assertFalse($policycache->has($user3->id));

        $result = $policycache->get($user1->id);
        $this->assertTrue($result);
        $result = $policycache->get($user2->id);
        $this->assertTrue($result);
        $result = $policycache->get($user3->id);
        $this->assertFalse($result);

        $this->assertTrue($policycache->has($user1->id));
        $this->assertTrue($policycache->has($user2->id));

        // Purge the cache.
        $policycache->purge();

        // Test multiple users.
        $this->assertFalse($policycache->has($user1->id));
        $this->assertFalse($policycache->has($user2->id));
        $this->assertFalse($policycache->has($user3->id));

        $result = $policycache->get_many([$user1->id, $user2->id, $user3->id]);
        $this->assertNotEmpty($policycache->get_many($result));
        $this->assertTrue($result[$user1->id]);
        $this->assertTrue($result[$user2->id]);
        $this->assertFalse($result[$user3->id]);

        $this->assertTrue($policycache->has($user1->id));
        $this->assertTrue($policycache->has($user2->id));
    }

    /**
     * Test store_action_result.
     */
    public function test_store_action_result(): void {
        $this->resetAfterTest();
        global $DB;

        $contextid = 1;
        $userid = 1;
        $prompttext = 'This is a test prompt';
        $aspectratio = 'square';
        $quality = 'hd';
        $numimages = 1;
        $style = 'vivid';

        $action = new generate_image(
            contextid: 1,
            userid: $userid,
            prompttext: $prompttext,
            quality: $quality,
            aspectratio: $aspectratio,
            numimages: $numimages,
            style: $style
        );

        $body = [
            'revisedprompt' => 'This is a revised prompt',
            'imageurl' => 'https://example.com/image.png',
            'model' => 'dall-e-3',
        ];
        $actionresponse = new response_generate_image(
            success: true,
        );
        $actionresponse->set_response_data($body);

        $manager = \core\di::get(manager::class);
        $config = ['data' => 'goeshere'];
        $provider = $manager->create_provider_instance(
                classname: '\aiprovider_openai\provider',
                name: 'dummy',
                config: $config,
        );

        // We're working with a private method here, so we need to use reflection.
        $method = new \ReflectionMethod($manager, 'store_action_result');
        $storeresult = $method->invoke($manager, $provider, $action, $actionresponse);

        // Check the record was written to the DB with expected values.
        $record = $DB->get_record('ai_action_register', ['id' => $storeresult], '*', MUST_EXIST);
        $this->assertEquals($action->get_basename(), $record->actionname);
        $this->assertEquals($userid, $record->userid);
        $this->assertEquals($contextid, $record->contextid);
        $this->assertEquals($provider->get_name(), $record->provider);
        $this->assertEquals($actionresponse->get_errorcode(), $record->errorcode);
        $this->assertEquals($actionresponse->get_errormessage(), $record->errormessage);
        $this->assertEquals($action->get_configuration('timecreated'), $record->timecreated);
        $this->assertEquals($actionresponse->get_timecreated(), $record->timecompleted);
        $this->assertEquals($actionresponse->get_model_used(), $record->model);
    }

    /**
     * Test call_action_provider.
     */
    public function test_call_action_provider(): void {
        $this->resetAfterTest();
        $contextid = 1;
        $userid = 1;
        $prompttext = 'This is a test prompt';
        $aspectratio = 'square';
        $quality = 'hd';
        $numimages = 1;
        $style = 'vivid';

        $action = new generate_image(
            contextid: $contextid,
            userid: $userid,
            prompttext: $prompttext,
            quality: $quality,
            aspectratio: $aspectratio,
            numimages: $numimages,
            style: $style
        );

        $manager = \core\di::get(manager::class);
        $config = ['apikey' => 'goeshere'];
        $provider = $manager->create_provider_instance(
                classname: '\aiprovider_openai\provider',
                name: 'dummy',
                config: $config,
        );

        // We're working with a private method here, so we need to use reflection.
        $method = new \ReflectionMethod($manager, 'call_action_provider');
        $actionresult = $method->invoke($manager, $provider, $action);

        // Assert the result was of the correct type.
        $this->assertInstanceOf(response_generate_image::class, $actionresult);
    }

    /**
     * Test is_action enabled.
     */
    public function test_is_action_enabled(): void {
        $this->resetAfterTest();
        $action = generate_image::class;
        $manager = \core\di::get(manager::class);

        $config = ['apikey' => 'goeshere'];
        $provider = $manager->create_provider_instance(
            classname: '\aiprovider_openai\provider',
            name: 'dummy',
            config: $config,
        );

        // Should be enabled by default.
        $result = $manager->is_action_enabled(
            plugin: $provider->provider,
            actionclass: $action,
            instanceid: $provider->id
        );
        $this->assertTrue($result);

        // Disable the action.
        $manager->set_action_state(
            plugin: $provider->provider,
            actionbasename: $action::get_basename(),
            enabled: 0,
            instanceid: $provider->id
        );

        // Should now be disabled.
        $result = $manager->is_action_enabled(
            plugin: $provider->provider,
            actionclass: $action,
            instanceid: $provider->id
        );
        $this->assertFalse($result);
    }

    /**
     * Test enable_action.
     */
    public function test_enable_action(): void {
        $this->resetAfterTest();
        $action = generate_image::class;

        $manager = \core\di::get(manager::class);

        $config = ['apikey' => 'goeshere'];
        $provider = $manager->create_provider_instance(
                classname: '\aiprovider_openai\provider',
                name: 'dummy',
                config: $config,
        );

        // Disable the action.
        $setresult = $manager->set_action_state(
                plugin: $provider->provider,
                actionbasename: $action::get_basename(),
                enabled: 0,
                instanceid: $provider->id);

        // Should now be disabled.
        $this->assertFalse($setresult);
        $result = $manager->is_action_enabled(
                plugin: $provider->provider,
                actionclass: $action,
                instanceid: $provider->id
        );
        $this->assertFalse($result);

        // Enable the action.
        $manager = \core\di::get(manager::class);
        $result = $manager->set_action_state(
                plugin: $provider->provider,
                actionbasename: generate_text::class::get_basename(),
                enabled: 1,
                instanceid: $provider->id);
        $this->assertTrue($result);
    }

    /**
     * Test is_action_available method.
     */
    public function test_is_action_available(): void {
        $this->resetAfterTest();
        $action = generate_image::class;

        // Plugin is disabled by default, action state should not matter. Everything should be false.
        $manager = \core\di::get(manager::class);
        $result = $manager->is_action_available($action);
        $this->assertFalse($result);

        // Create the provider instance, actions will be enabled by default when the plugin is enabled.
        $config = ['apikey' => 'goeshere'];
        $provider = $manager->create_provider_instance(
            classname: '\aiprovider_openai\provider',
            name: 'dummy',
            enabled: true,
            config: $config,
        );

        // Should now be available.
        $result = $manager->is_action_available($action);
        $this->assertTrue($result);

        // Disable the action.
        $manager->set_action_state(
                plugin: $provider->provider,
                actionbasename: $action::get_basename(),
                enabled: 0,
                instanceid: $provider->id);

        // Should now be unavailable.
        $result = $manager->is_action_available($action);
        $this->assertFalse($result);
    }
}