This commit is contained in:
Ilya Tregubov 2024-03-28 09:05:53 +08:00
commit 5180272bd5
4 changed files with 195 additions and 9 deletions

View File

@ -69,9 +69,14 @@ abstract class base_send_notification extends adhoc_task {
/**
* Get the bigbluebutton instance that this notification is for.
*
* @return instance
* @return instance|null null if the instance could not be loaded.
*/
protected function get_instance(): instance {
protected function get_instance(): ?instance {
// This means the customdata is broken, and needs to be fixed.
if (empty($this->get_custom_data()->instanceid)) {
throw new \coding_exception("Task custom data was missing instanceid");
}
if ($this->instance === null) {
$this->instance = instance::get_from_instanceid($this->get_custom_data()->instanceid);
}
@ -180,6 +185,13 @@ abstract class base_send_notification extends adhoc_task {
*/
protected function send_all_notifications(): void {
$instance = $this->get_instance();
// Cannot do anything without a valid instance.
if (empty($instance)) {
mtrace("Instance was empty, skipping");
return;
}
foreach ($this->get_recipients() as $recipient) {
try {
\core_user::require_active_user($recipient, true, true);

View File

@ -17,6 +17,7 @@
namespace mod_bigbluebuttonbn\task;
use core\task\adhoc_task;
use core_user;
use mod_bigbluebuttonbn\local\proxy\bigbluebutton_proxy;
/**
@ -33,6 +34,24 @@ class completion_update_state extends adhoc_task {
public function execute() {
// Get the custom data.
$data = $this->get_custom_data();
// Ensure the customdata structure is corect.
if (empty($data->bigbluebuttonbn->id) || empty($data->userid)) {
throw new \coding_exception("Task customdata was missing bigbluebuttonbn->id or userid");
}
// If coursemodule does not exist, ignore (likely has been deleted).
if (get_coursemodule_from_instance('bigbluebuttonbn', $data->bigbluebuttonbn->id) === false) {
mtrace("Course module does not exist, ignoring.");
return;
}
// If user does not exist, ignore (likely has been deleted).
if (core_user::get_user($data->userid) === false) {
mtrace("User does not exist, ignoring.");
return;
}
mtrace("Task completion_update_state running for user {$data->userid}");
// Process the completion.

View File

@ -28,13 +28,14 @@ use advanced_testcase;
* @coversDefaultClass \mod_bigbluebuttonbn\task\base_send_notification
*/
class base_send_notification_test extends advanced_testcase {
/**
* Check if set instance ID works correctly
* Returns mock base_send_notification class
*
* @return base_send_notification
*/
public function test_set_instance_id(): void {
$this->resetAfterTest();
$stub = $this->getMockForAbstractClass(
private function get_mock(): base_send_notification {
return $this->getMockForAbstractClass(
base_send_notification::class,
[],
'',
@ -43,7 +44,26 @@ class base_send_notification_test extends advanced_testcase {
true,
[]
);
}
/**
* Returns reflection method for base_send_notification->get_instance
*
* @return \ReflectionMethod
*/
private function get_instance_reflection(): \ReflectionMethod {
$rc = new \ReflectionClass(base_send_notification::class);
$rcm = $rc->getMethod('get_instance');
$rcm->setAccessible(true);
return $rcm;
}
/**
* Check if set instance ID works correctly
*/
public function test_set_instance_id(): void {
$this->resetAfterTest();
$stub = $this->get_mock();
$generator = $this->getDataGenerator();
$course = $generator->create_course();
$instancedata = $generator->create_module('bigbluebuttonbn', [
@ -52,10 +72,23 @@ class base_send_notification_test extends advanced_testcase {
$stub->set_instance_id($instancedata->id);
$rc = new \ReflectionClass(base_send_notification::class);
$rcm = $rc->getMethod('get_instance');
$rcm = $this->get_instance_reflection();
$instance = $rcm->invoke($stub);
$this->assertNotNull($instance);
$this->assertEquals($instancedata->id, $instance->get_instance_id());
}
/**
* Check if instanceid missing is checked and handled.
*/
public function test_set_instanceid_missing(): void {
$this->resetAfterTest();
$stub = $this->get_mock();
$rcm = $this->get_instance_reflection();
// This should throw a coding exception since there is no instanceid set.
$this->expectException(\coding_exception::class);
$this->expectExceptionMessage("Task custom data was missing instanceid");
$rcm->invoke($stub);
}
}

View File

@ -0,0 +1,122 @@
<?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 mod_bigbluebuttonbn\task;
use advanced_testcase;
use mod_bigbluebuttonbn\task\completion_update_state;
/**
* Completion_update_state task tests.
*
* @package mod_bigbluebuttonbn
* @copyright 2024 Catalyst IT
* @author Matthew Hilton <matthewhilton@catalyst-au.net>
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
* @covers \mod_bigbluebuttonbn\task\completion_update_state
*/
final class completion_update_state_task_test extends advanced_testcase {
/**
* Providers data to test_invalid_customdata test
* @return array
*/
public static function invalid_customdata_provider(): array {
return [
'empty' => [
'customdata' => [],
'expectoutput' => "",
'expectexceptionmessage' => "Task customdata was missing bigbluebuttonbn->id or userid",
],
'bbb id set but is invalid' => [
'customdata' => [
'bigbluebuttonbn' => -1,
],
'expectoutput' => "",
'expectexceptionmessage' => "Task customdata was missing bigbluebuttonbn->id or userid",
],
'bbb id is valid, but there is no user' => [
'customdata' => [
'bigbluebuttonbn' => ':bbb',
],
'expectoutput' => "",
'expectexceptionmessage' => "Task customdata was missing bigbluebuttonbn->id or userid",
],
'bbb id is valid, but the user is not given' => [
'customdata' => [
'bigbluebuttonbn' => ':bbb',
],
'expectoutput' => "",
'expectexceptionmessage' => "Task customdata was missing bigbluebuttonbn->id or userid",
],
'bbb id is valid, but the user given is invalid' => [
'customdata' => [
'bigbluebuttonbn' => ':bbb',
'userid' => -1,
],
'expectoutput' => "User does not exist, ignoring.\n",
'expectexceptionmessage' => "",
],
'bbb and userid is valid' => [
'customdata' => [
'bigbluebuttonbn' => ':bbb',
'userid' => ':userid',
],
// Expects this output, since all the necessary data is there.
'expectoutput' => "Task completion_update_state running for user :userid\nCompletion not enabled\n",
'expectexceptionmessage' => "",
],
];
}
/**
* Tests the task handles an invalid cmid gracefully.
* @param array $customdata customdata to set (with placeholders to replace with real data).
* @param string $expectoutput any output expected from the test, or empty to not expect output.
* @param string $expectexceptionmessage exception message expected from test, or empty to expect nothing.
* @dataProvider invalid_customdata_provider
*/
public function test_invalid_customdata(array $customdata, string $expectoutput, string $expectexceptionmessage): void {
$this->resetAfterTest();
$customdata = (object) $customdata;
// Replace any placeholders in the customdata.
if (!empty($customdata->bigbluebuttonbn) && $customdata->bigbluebuttonbn == ':bbb') {
$course = $this->getDataGenerator()->create_course();
$module = $this->getDataGenerator()->create_module('bigbluebuttonbn', ['course' => $course->id]);
$customdata->bigbluebuttonbn = $module;
}
$user = $this->getDataGenerator()->create_user();
// Replace userid placeholders.
if (!empty($customdata->userid) && $customdata->userid == ':userid') {
$customdata->userid = $user->id;
}
$task = new completion_update_state();
$task->set_custom_data($customdata);
if (!empty($expectoutput)) {
$expectoutput = str_replace(':userid', $user->id, $expectoutput);
$this->expectOutputString($expectoutput);
}
if (!empty($expectexceptionmessage)) {
$this->expectExceptionMessage($expectexceptionmessage);
}
$task->execute();
}
}