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 mod_assign;
use DateTime;
use core\output\html_writer;
defined('MOODLE_INTERNAL') || die();
require_once($CFG->dirroot . '/mod/assign/locallib.php');
/**
* Helper for sending assignment related notifications.
*
* @package mod_assign
* @copyright 2024 David Woloszyn <david.woloszyn@moodle.com>
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
class notification_helper {
/**
* @var int Due soon time interval of 48 hours.
*/
private const INTERVAL_DUE_SOON = (DAYSECS * 2);
/**
* @var int Overdue time interval of 2 hours.
*/
private const INTERVAL_OVERDUE = (HOURSECS * 2);
/**
* @var int Due digest time interval of 7 days.
*/
private const INTERVAL_DUE_DIGEST = WEEKSECS;
/**
* @var string Due soon notification type.
*/
public const TYPE_DUE_SOON = 'assign_due_soon';
/**
* @var string Overdue notification type.
*/
public const TYPE_OVERDUE = 'assign_overdue';
/**
* @var string Due digest notification type.
*/
public const TYPE_DUE_DIGEST = 'assign_due_digest';
/**
* Get all assignments that have an approaching due date (includes users and groups with due date overrides).
*
* @return \moodle_recordset Returns the matching assignment records.
*/
public static function get_due_soon_assignments(): \moodle_recordset {
global $DB;
$timenow = self::get_time_now();
$futuretime = self::get_future_time(self::INTERVAL_DUE_SOON);
$sql = "SELECT DISTINCT a.id
FROM {assign} a
JOIN {course_modules} cm ON a.id = cm.instance
JOIN {course} c ON a.course = c.id
JOIN {modules} m ON cm.module = m.id AND m.name = :modulename
LEFT JOIN {assign_overrides} ao ON a.id = ao.assignid
WHERE (a.duedate < :futuretime OR ao.duedate < :ao_futuretime)
AND (a.duedate > :timenow OR ao.duedate > :ao_timenow)
AND cm.visible = 1
AND c.visible = 1";
$params = [
'timenow' => $timenow,
'futuretime' => $futuretime,
'ao_timenow' => $timenow,
'ao_futuretime' => $futuretime,
'modulename' => 'assign',
];
return $DB->get_recordset_sql($sql, $params);
}
/**
* Get all assignments that are overdue, but not exceeding the cut-off date (includes users and groups with due date overrides).
*
* We don't want to get every single overdue assignment ever.
* We just want the ones within the specified window.
*
* @return \moodle_recordset Returns the matching assignment records.
*/
public static function get_overdue_assignments(): \moodle_recordset {
global $DB;
$timenow = self::get_time_now();
$timewindow = self::get_time_now() - self::INTERVAL_OVERDUE;
// Get all assignments that:
// - Are overdue.
// - Do not exceed the window of time in the past.
// - Are still within the cut-off (if it is set).
$sql = "SELECT DISTINCT a.id
FROM {assign} a
JOIN {course_modules} cm ON a.id = cm.instance
JOIN {course} c ON a.course = c.id
JOIN {modules} m ON cm.module = m.id AND m.name = :modulename
LEFT JOIN {assign_overrides} ao ON a.id = ao.assignid
WHERE (a.duedate < :dd_timenow OR ao.duedate < :dd_ao_timenow)
AND (a.duedate > :dd_timewindow OR ao.duedate > :dd_ao_timewindow)
AND ((a.cutoffdate > :co_timenow OR a.cutoffdate = 0) OR
(ao.cutoffdate > :co_ao_timenow OR ao.cutoffdate = 0))
AND cm.visible = 1
AND c.visible = 1";
$params = [
'dd_timenow' => $timenow,
'dd_ao_timenow' => $timenow,
'dd_timewindow' => $timewindow,
'dd_ao_timewindow' => $timewindow,
'co_timenow' => $timenow,
'co_ao_timenow' => $timenow,
'modulename' => 'assign',
];
return $DB->get_recordset_sql($sql, $params);
}
/**
* Get all assignments that are due in 7 days (includes users and groups with due date overrides).
*
* @return \moodle_recordset Returns the matching assignment records.
*/
public static function get_due_digest_assignments(): \moodle_recordset {
global $DB;
$futuretime = self::get_future_time(self::INTERVAL_DUE_DIGEST);
$day = self::get_day_start_and_end($futuretime);
$sql = "SELECT DISTINCT a.id
FROM {assign} a
JOIN {course_modules} cm ON a.id = cm.instance
JOIN {course} c ON a.course = c.id
JOIN {modules} m ON cm.module = m.id AND m.name = :modulename
LEFT JOIN {assign_overrides} ao ON a.id = ao.assignid
WHERE (a.duedate <= :endofday OR ao.duedate <= :ao_endofday)
AND (a.duedate >= :startofday OR ao.duedate >= :ao_startofday)
AND cm.visible = 1
AND c.visible = 1";
$params = [
'startofday' => $day['start'],
'endofday' => $day['end'],
'ao_startofday' => $day['start'],
'ao_endofday' => $day['end'],
'modulename' => 'assign',
];
return $DB->get_recordset_sql($sql, $params);
}
/**
* Get all assignments for a user that are due in 7 days (includes users and groups with due date overrides).
*
* @param int $userid The user id.
* @return \moodle_recordset Returns the matching assignment records.
*/
public static function get_due_digest_assignments_for_user(int $userid): \moodle_recordset {
global $DB;
$futuretime = self::get_future_time(self::INTERVAL_DUE_DIGEST);
$day = self::get_day_start_and_end($futuretime);
$sql = "SELECT DISTINCT a.id,
a.duedate,
a.name AS assignmentname,
c.fullname AS coursename,
cm.id AS cmid
FROM {assign} a
JOIN {course} c ON a.course = c.id
JOIN {course_modules} cm ON a.id = cm.instance
JOIN {modules} m ON cm.module = m.id AND m.name = :modulename
JOIN {enrol} e ON c.id = e.courseid
JOIN {user_enrolments} ue ON e.id = ue.enrolid
LEFT JOIN {assign_overrides} ao ON a.id = ao.assignid
WHERE (a.duedate <= :endofday OR ao.duedate <= :ao_endofday)
AND (a.duedate >= :startofday OR ao.duedate >= :ao_startofday)
AND ue.userid = :userid
AND cm.visible = 1
AND c.visible = 1
ORDER BY a.duedate ASC";
$params = [
'startofday' => $day['start'],
'endofday' => $day['end'],
'ao_startofday' => $day['start'],
'ao_endofday' => $day['end'],
'modulename' => 'assign',
'userid' => $userid,
];
return $DB->get_recordset_sql($sql, $params);
}
/**
* Get all assignment users that we should send the notification to.
*
* @param int $assignmentid The assignment id.
* @param string $type The notification type.
* @return array The users after all filtering has been applied.
*/
public static function get_users_within_assignment(int $assignmentid, string $type): array {
// Get assignment data.
$assignmentobj = self::get_assignment_data($assignmentid);
// Get our assignment users.
$users = $assignmentobj->list_participants(0, true);
foreach ($users as $key => $user) {
// Check if the user has submitted already.
$submission = $assignmentobj->get_user_submission($user->id, false);
if ($submission && $submission->status === ASSIGN_SUBMISSION_STATUS_SUBMITTED) {
unset($users[$key]);
continue;
}
// Determine key dates with respect to any overrides.
$duedate = $assignmentobj->override_exists($user->id)->duedate ?? $assignmentobj->get_instance()->duedate;
$cutoffdate = $assignmentobj->override_exists($user->id)->cutoffdate ?? $assignmentobj->get_instance()->cutoffdate;
// If the due date has no value, unset this user.
if (empty($duedate)) {
unset($users[$key]);
continue;
}
// Perform some checks depending on the notification type.
$match = [];
$checksent = true;
switch ($type) {
case self::TYPE_DUE_SOON:
$range = [
'lower' => self::get_time_now(),
'upper' => self::get_future_time(self::INTERVAL_DUE_SOON),
];
if (!self::is_time_within_range($duedate, $range)) {
unset($users[$key]);
break;
}
$match = [
'assignmentid' => $assignmentid,
'duedate' => $duedate,
];
break;
case self::TYPE_OVERDUE:
if ($duedate > self::get_time_now()) {
unset($users[$key]);
break;
}
// Check if the cut-off date is set and passed already.
if (!empty($cutoffdate) && self::get_time_now() > $cutoffdate) {
unset($users[$key]);
break;
}
$match = [
'assignmentid' => $assignmentid,
'duedate' => $duedate,
'cutoffdate' => $cutoffdate,
];
break;
case self::TYPE_DUE_DIGEST:
$checksent = false;
$futuretime = self::get_future_time(self::INTERVAL_DUE_DIGEST);
$day = self::get_day_start_and_end($futuretime);
$range = [
'lower' => $day['start'],
'upper' => $day['end'],
];
if (!self::is_time_within_range($duedate, $range)) {
unset($users[$key]);
break;
}
break;
default:
break;
}
// Check if the user has already received this notification.
if ($checksent && self::has_user_been_sent_a_notification_already($user->id, json_encode($match), $type)) {
unset($users[$key]);
}
}
return $users;
}
/**
* Send the due soon notification to the user.
*
* @param int $assignmentid The assignment id.
* @param int $userid The user id.
*/
public static function send_due_soon_notification_to_user(int $assignmentid, int $userid): void {
try {
// Get assignment data.
$assignmentobj = self::get_assignment_data($assignmentid);
} catch (\dml_missing_record_exception) {
// The assignment has vanished, nothing to do.
mtrace("No notification send as the assignment $assignmentid can no longer be found in the database.");
return;
}
// Check if the due date still within range.
$assignmentobj->update_effective_access($userid);
$duedate = $assignmentobj->get_instance($userid)->duedate;
$range = [
'lower' => self::get_time_now(),
'upper' => self::get_future_time(self::INTERVAL_DUE_SOON),
];
if (!self::is_time_within_range($duedate, $range)) {
return;
}
// Check if the user has submitted already.
$submission = $assignmentobj->get_user_submission($userid, false);
if ($submission && $submission->status === ASSIGN_SUBMISSION_STATUS_SUBMITTED) {
return;
}
// Build the user's notification message.
$user = $assignmentobj->get_participant($userid);
$urlparams = [
'id' => $assignmentobj->get_course_module()->id,
'action' => 'view',
];
$url = new \moodle_url('/mod/assign/view.php', $urlparams);
$stringparams = [
'firstname' => $user->firstname,
'assignmentname' => $assignmentobj->get_instance()->name,
'coursename' => $assignmentobj->get_course()->fullname,
'duedate' => userdate($duedate),
'url' => $url,
];
$messagedata = [
'user' => \core_user::get_user($user->id),
'url' => $url->out(false),
'subject' => get_string('assignmentduesoonsubject', 'mod_assign', $stringparams),
'assignmentname' => $assignmentobj->get_instance()->name,
'html' => get_string('assignmentduesoonhtml', 'mod_assign', $stringparams),
'sms' => get_string('assignmentduesoonsms', 'mod_assign', $stringparams),
];
$message = new \core\message\message();
$message->component = 'mod_assign';
$message->name = self::TYPE_DUE_SOON;
$message->userfrom = \core_user::get_noreply_user();
$message->userto = $messagedata['user'];
$message->subject = $messagedata['subject'];
$message->fullmessageformat = FORMAT_HTML;
$message->fullmessage = html_to_text($messagedata['html']);
$message->fullmessagehtml = $messagedata['html'];
$message->fullmessagesms = $messagedata['sms'];
$message->smallmessage = $messagedata['subject'];
$message->notification = 1;
$message->contexturl = $messagedata['url'];
$message->contexturlname = $messagedata['assignmentname'];
// Use custom data to avoid future notifications being sent again.
$message->customdata = [
'assignmentid' => $assignmentid,
'duedate' => $duedate,
];
message_send($message);
}
/**
* Send the overdue notification to the user.
*
* @param int $assignmentid The assignment id.
* @param int $userid The user id.
*/
public static function send_overdue_notification_to_user(int $assignmentid, int $userid): void {
try {
// Get assignment data.
$assignmentobj = self::get_assignment_data($assignmentid);
} catch (\dml_missing_record_exception) {
// The assignment has vanished, nothing to do.
mtrace("No notification send as the assignment $assignmentid can no longer be found in the database.");
return;
}
// Get the user and check they are a still a valid participant.
$user = $assignmentobj->get_participant($userid);
if (empty($user)) {
return;
}
// Check if the due date still considered overdue.
$assignmentobj->update_effective_access($userid);
$duedate = $assignmentobj->get_instance($userid)->duedate;
if ($duedate > self::get_time_now()) {
return;
}
// Check if the cut-off date is set and passed already.
$cutoffdate = $assignmentobj->get_instance($userid)->cutoffdate;
if (!empty($cutoffdate) && self::get_time_now() > $cutoffdate) {
return;
}
// Check if the user has submitted already.
$submission = $assignmentobj->get_user_submission($userid, false);
if ($submission && $submission->status === ASSIGN_SUBMISSION_STATUS_SUBMITTED) {
return;
}
// Build the user's notification message.
$urlparams = [
'id' => $assignmentobj->get_course_module()->id,
'action' => 'view',
];
$url = new \moodle_url('/mod/assign/view.php', $urlparams);
// Prepare the cut-off date html string.
$snippet = '';
if (!empty($cutoffdate)) {
$snippet = get_string('assignmentoverduehtmlcutoffsnippet', 'mod_assign', ['cutoffdate' => userdate($cutoffdate)]);
}
$stringparams = [
'firstname' => $user->firstname,
'assignmentname' => $assignmentobj->get_instance()->name,
'coursename' => $assignmentobj->get_course()->fullname,
'duedate' => userdate($duedate),
'url' => $url,
'cutoffsnippet' => $snippet,
];
$messagedata = [
'user' => \core_user::get_user($user->id),
'url' => $url->out(false),
'subject' => get_string('assignmentoverduesubject', 'mod_assign', $stringparams),
'assignmentname' => $assignmentobj->get_instance()->name,
'html' => get_string('assignmentoverduehtml', 'mod_assign', $stringparams),
'sms' => get_string('assignmentoverduesms', 'mod_assign', $stringparams),
];
$message = new \core\message\message();
$message->component = 'mod_assign';
$message->name = self::TYPE_OVERDUE;
$message->userfrom = \core_user::get_noreply_user();
$message->userto = $messagedata['user'];
$message->subject = $messagedata['subject'];
$message->fullmessageformat = FORMAT_HTML;
$message->fullmessage = html_to_text($messagedata['html']);
$message->fullmessagehtml = $messagedata['html'];
$message->fullmessagesms = $messagedata['sms'];
$message->smallmessage = $messagedata['subject'];
$message->notification = 1;
$message->contexturl = $messagedata['url'];
$message->contexturlname = $messagedata['assignmentname'];
// Use custom data to avoid future notifications being sent again.
$message->customdata = [
'assignmentid' => $assignmentid,
'duedate' => $duedate,
'cutoffdate' => $cutoffdate,
];
message_send($message);
}
/**
* Get all the assignments and send the due digest notification to the user.
*
* @param int $userid The user id.
*/
public static function send_due_digest_notification_to_user(int $userid): void {
// Get all the user's assignments due in 7 days.
$assignments = self::get_due_digest_assignments_for_user($userid);
$assignmentsfordigest = [];
foreach ($assignments as $assignment) {
$assignmentobj = self::get_assignment_data($assignment->id);
// Check if the user has submitted already.
$submission = $assignmentobj->get_user_submission($userid, false);
if ($submission && $submission->status === ASSIGN_SUBMISSION_STATUS_SUBMITTED) {
continue;
}
// Check if the module is visible to the user.
$cm = $assignmentobj->get_course_module();
if (!\core_availability\info_module::is_user_visible($cm, $userid)) {
continue;
}
// Check if the due date is still within range.
$assignmentobj->update_effective_access($userid);
$duedate = $assignmentobj->get_instance($userid)->duedate;
$futuretime = self::get_future_time(self::INTERVAL_DUE_DIGEST);
$day = self::get_day_start_and_end($futuretime);
$range = [
'lower' => $day['start'],
'upper' => $day['end'],
];
if (!self::is_time_within_range($duedate, $range)) {
continue;
}
// Record the assignment data to help us build the digest.
$urlparams = [
'id' => $assignmentobj->get_course_module()->id,
'action' => 'view',
];
$assignmentsfordigest[$assignment->id] = [
'assignmentname' => $assignmentobj->get_instance()->name,
'coursename' => $assignmentobj->get_course()->fullname,
'duetime' => userdate($duedate, get_string('strftimetime12', 'langconfig')),
'url' => new \moodle_url('/mod/assign/view.php', $urlparams),
];
}
$assignments->close();
// If there are no assignments in the digest, don't send anything.
if (empty($assignmentsfordigest)) {
return;
}
// Build the digest.
$digestarray = [];
foreach ($assignmentsfordigest as $digestitem) {
$digestarray[] = get_string('assignmentduedigestitem', 'mod_assign', $digestitem);
}
// Put the digest into list.
$digest = html_writer::alist($digestarray);
// Get user's object.
$userobject = \core_user::get_user($userid);
$stringparams = [
'firstname' => $userobject->firstname,
'duedate' => userdate(self::get_future_time(self::INTERVAL_DUE_DIGEST), get_string('strftimedaydate', 'langconfig')),
'digest' => $digest,
];
$messagedata = [
'user' => $userobject,
'subject' => get_string('assignmentduedigestsubject', 'mod_assign'),
'html' => get_string('assignmentduedigesthtml', 'mod_assign', $stringparams),
'sms' => get_string('assignmentduedigestsms', 'mod_assign', $stringparams),
];
$message = new \core\message\message();
$message->component = 'mod_assign';
$message->name = self::TYPE_DUE_DIGEST;
$message->userfrom = \core_user::get_noreply_user();
$message->userto = $messagedata['user'];
$message->subject = $messagedata['subject'];
$message->fullmessageformat = FORMAT_HTML;
$message->fullmessage = html_to_text($messagedata['html']);
$message->fullmessagehtml = $messagedata['html'];
$message->fullmessagesms = $messagedata['sms'];
$message->smallmessage = $messagedata['subject'];
$message->notification = 1;
message_send($message);
}
/**
* Get the time now.
*
* @return int The time now as a timestamp.
*/
protected static function get_time_now(): int {
return \core\di::get(\core\clock::class)->time();
}
/**
* Get a future time.
*
* @param int $interval Amount of seconds added to the now time.
* @return int The time now value plus the interval.
*/
protected static function get_future_time(int $interval): int {
return self::get_time_now() + $interval;
}
/**
* Get the timestamps for the start (00:00:00) and end (23:59:59) of the provided day.
*
* @param int $timestamp The timestamp to base the calculation on.
* @return array Day start and end timestamps.
*/
protected static function get_day_start_and_end(int $timestamp): array {
$day = [];
$date = new DateTime();
$date->setTimestamp($timestamp);
$date->setTime(0, 0, 0);
$day['start'] = $date->getTimestamp();
$date->setTime(23, 59, 59);
$day['end'] = $date->getTimestamp();
return $day;
}
/**
* Check if a time is within the current time now and the future time values (inclusive).
*
* @param int $time The timestamp to check.
* @param array $range Lower and upper times to check.
* @return boolean
*/
protected static function is_time_within_range(int $time, array $range): bool {
return ($time >= $range['lower'] && $time <= $range['upper']);
}
/**
* Check if a user has been sent a notification already.
*
* @param int $userid The user id.
* @param string $match The custom data string to match on.
* @param string $type The notification/event type to match.
* @return bool Returns true if already sent.
*/
protected static function has_user_been_sent_a_notification_already(int $userid, string $match, string $type): bool {
global $DB;
$sql = $DB->sql_compare_text('customdata', 255) . " = " . $DB->sql_compare_text(':match', 255) . "
AND useridto = :userid
AND component = :component
AND eventtype = :eventtype";
return $DB->record_exists_select('notifications', $sql, [
'userid' => $userid,
'match' => $match,
'component' => 'mod_assign',
'eventtype' => $type,
]);
}
/**
* Get the assignment object, including the course and course module.
*
* @param int $assignmentid The assignment id.
* @return \assign Returns the assign object.
*/
protected static function get_assignment_data(int $assignmentid): \assign {
[$course, $assigncm] = get_course_and_cm_from_instance($assignmentid, 'assign');
$cmcontext = \context_module::instance($assigncm->id);
return new \assign($cmcontext, $assigncm, $course);
}
}