. /** * A scheduled task for forum cron. * * @package mod_forum * @copyright 2014 Dan Poltawski * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace mod_forum\task; defined('MOODLE_INTERNAL') || die(); require_once($CFG->dirroot . '/mod/forum/lib.php'); /** * The main scheduled task for the forum. * * @package mod_forum * @copyright 2018 Andrew Nicols * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class cron_task extends \core\task\scheduled_task { // Use the logging trait to get some nice, juicy, logging. use \core\task\logging_trait; /** * @var The list of courses which contain posts to be sent. */ protected $courses = []; /** * @var The list of forums which contain posts to be sent. */ protected $forums = []; /** * @var The list of discussions which contain posts to be sent. */ protected $discussions = []; /** * @var The list of posts to be sent. */ protected $posts = []; /** * @var The list of post authors. */ protected $users = []; /** * @var The list of subscribed users. */ protected $subscribedusers = []; /** * @var The list of digest users. */ protected $digestusers = []; /** * @var The list of adhoc data for sending. */ protected $adhocdata = []; /** * Get a descriptive name for this task (shown to admins). * * @return string */ public function get_name() { return get_string('crontask', 'mod_forum'); } /** * Execute the scheduled task. */ public function execute() { global $CFG, $DB; $timenow = time(); // Delete any really old posts in the digest queue. $weekago = $timenow - (7 * 24 * 3600); $this->log_start("Removing old digest records from 7 days ago."); $DB->delete_records_select('forum_queue', "timemodified < ?", array($weekago)); $this->log_finish("Removed all old digest records."); $endtime = $timenow - $CFG->maxeditingtime; $starttime = $endtime - (2 * DAYSECS); $this->log_start("Fetching unmailed posts."); if (!$posts = $this->get_unmailed_posts($starttime, $endtime, $timenow)) { $this->log_finish("No posts found.", 1); return false; } $this->log_finish("Done"); // Process post data and turn into adhoc tasks. $this->process_post_data($posts); // Mark posts as read. list($in, $params) = $DB->get_in_or_equal(array_keys($posts)); $DB->set_field_select('forum_posts', 'mailed', 1, "id {$in}", $params); } /** * Process all posts and convert to appropriated hoc tasks. * * @param \stdClass[] $posts */ protected function process_post_data($posts) { $discussionids = []; $forumids = []; $courseids = []; $this->log_start("Processing post information"); $start = microtime(true); foreach ($posts as $id => $post) { $discussionids[$post->discussion] = true; $forumids[$post->forum] = true; $courseids[$post->course] = true; $this->add_data_for_post($post); $this->posts[$id] = $post; } $this->log_finish(sprintf("Processed %s posts", count($this->posts))); if (empty($this->posts)) { $this->log("No posts found. Returning early."); return; } // Please note, this order is intentional. // The forum cache makes use of the course. $this->log_start("Filling caches"); $start = microtime(true); $this->log_start("Filling course cache", 1); $this->fill_course_cache(array_keys($courseids)); $this->log_finish("Done", 1); $this->log_start("Filling forum cache", 1); $this->fill_forum_cache(array_keys($forumids)); $this->log_finish("Done", 1); $this->log_start("Filling discussion cache", 1); $this->fill_discussion_cache(array_keys($discussionids)); $this->log_finish("Done", 1); $this->log_start("Filling user subscription cache", 1); $this->fill_user_subscription_cache(); $this->log_finish("Done", 1); $this->log_start("Filling digest cache", 1); $this->fill_digest_cache(); $this->log_finish("Done", 1); $this->log_finish("All caches filled"); $this->log_start("Queueing user tasks."); $this->queue_user_tasks(); $this->log_finish("All tasks queued."); } /** * Fill the course cache. * * @param int[] $courseids */ protected function fill_course_cache($courseids) { global $DB; list($in, $params) = $DB->get_in_or_equal($courseids); $this->courses = $DB->get_records_select('course', "id $in", $params); } /** * Fill the forum cache. * * @param int[] $forumids */ protected function fill_forum_cache($forumids) { global $DB; $requiredfields = [ 'id', 'course', 'forcesubscribe', 'type', ]; list($in, $params) = $DB->get_in_or_equal($forumids); $this->forums = $DB->get_records_select('forum', "id $in", $params, '', implode(', ', $requiredfields)); foreach ($this->forums as $id => $forum) { \mod_forum\subscriptions::fill_subscription_cache($id); \mod_forum\subscriptions::fill_discussion_subscription_cache($id); } } /** * Fill the discussion cache. * * @param int[] $discussionids */ protected function fill_discussion_cache($discussionids) { global $DB; if (empty($discussionids)) { $this->discussion = []; } else { $requiredfields = [ 'id', 'groupid', 'firstpost', 'timestart', 'timeend', ]; list($in, $params) = $DB->get_in_or_equal($discussionids); $this->discussions = $DB->get_records_select( 'forum_discussions', "id $in", $params, '', implode(', ', $requiredfields)); } } /** * Fill the cache of user digest preferences. */ protected function fill_digest_cache() { global $DB; if (empty($this->users)) { return; } // Get the list of forum subscriptions for per-user per-forum maildigest settings. list($in, $params) = $DB->get_in_or_equal(array_keys($this->users)); $digestspreferences = $DB->get_recordset_select( 'forum_digests', "userid $in", $params, '', 'id, userid, forum, maildigest'); foreach ($digestspreferences as $digestpreference) { if (!isset($this->digestusers[$digestpreference->forum])) { $this->digestusers[$digestpreference->forum] = []; } $this->digestusers[$digestpreference->forum][$digestpreference->userid] = $digestpreference->maildigest; } $digestspreferences->close(); } /** * Add dsta for the current forum post to the structure of adhoc data. * * @param \stdClass $post */ protected function add_data_for_post($post) { if (!isset($this->adhocdata[$post->course])) { $this->adhocdata[$post->course] = []; } if (!isset($this->adhocdata[$post->course][$post->forum])) { $this->adhocdata[$post->course][$post->forum] = []; } if (!isset($this->adhocdata[$post->course][$post->forum][$post->discussion])) { $this->adhocdata[$post->course][$post->forum][$post->discussion] = []; } $this->adhocdata[$post->course][$post->forum][$post->discussion][$post->id] = $post->id; } /** * Fill the cache of user subscriptions. */ protected function fill_user_subscription_cache() { foreach ($this->forums as $forum) { $cm = get_fast_modinfo($this->courses[$forum->course])->instances['forum'][$forum->id]; $modcontext = \context_module::instance($cm->id); $this->subscribedusers[$forum->id] = []; if ($users = \mod_forum\subscriptions::fetch_subscribed_users($forum, 0, $modcontext, 'u.id, u.maildigest', true)) { foreach ($users as $user) { // This user is subscribed to this forum. $this->subscribedusers[$forum->id][$user->id] = $user->id; if (!isset($this->users[$user->id])) { // Store minimal user info. $this->users[$user->id] = $user; } } // Release memory. unset($users); } } } /** * Queue the user tasks. */ protected function queue_user_tasks() { global $CFG, $DB; $timenow = time(); $sitetimezone = \core_date::get_server_timezone(); $counts = [ 'digests' => 0, 'individuals' => 0, 'users' => 0, 'ignored' => 0, 'messages' => 0, ]; $this->log("Processing " . count($this->users) . " users", 1); foreach ($this->users as $user) { $usercounts = [ 'digests' => 0, 'messages' => 0, ]; $send = false; // Setup this user so that the capabilities are cached, and environment matches receiving user. cron_setup_user($user); list($individualpostdata, $digestpostdata) = $this->fetch_posts_for_user($user); if (!empty($digestpostdata)) { // Insert all of the records for the digest. $DB->insert_records('forum_queue', $digestpostdata); $digesttime = usergetmidnight($timenow, $sitetimezone) + ($CFG->digestmailtime * 3600); $task = new \mod_forum\task\send_user_digests(); $task->set_userid($user->id); $task->set_component('mod_forum'); $task->set_next_run_time($digesttime); \core\task\manager::reschedule_or_queue_adhoc_task($task); $usercounts['digests']++; $send = true; } if (!empty($individualpostdata)) { $usercounts['messages'] += count($individualpostdata); $task = new \mod_forum\task\send_user_notifications(); $task->set_userid($user->id); $task->set_custom_data($individualpostdata); $task->set_component('mod_forum'); \core\task\manager::queue_adhoc_task($task); $counts['individuals']++; $send = true; } if ($send) { $counts['users']++; $counts['messages'] += $usercounts['messages']; $counts['digests'] += $usercounts['digests']; } else { $counts['ignored']++; } $this->log(sprintf("Queued %d digests and %d messages for %s", $usercounts['digests'], $usercounts['messages'], $user->id ), 2); } $this->log( sprintf( "Queued %d digests, and %d individual tasks for %d post mails. " . "Unique users: %d (%d ignored)", $counts['digests'], $counts['individuals'], $counts['messages'], $counts['users'], $counts['ignored'] ), 1); } /** * Fetch posts for this user. * * @param \stdClass $user The user to fetch posts for. */ protected function fetch_posts_for_user($user) { // We maintain a mapping of user groups for each forum. $usergroups = []; $digeststructure = []; $poststructure = $this->adhocdata; $poststosend = []; foreach ($poststructure as $courseid => $forumids) { $course = $this->courses[$courseid]; foreach ($forumids as $forumid => $discussionids) { $forum = $this->forums[$forumid]; $maildigest = forum_get_user_maildigest_bulk($this->digestusers, $user, $forumid); if (!isset($this->subscribedusers[$forumid][$user->id])) { // This user has no subscription of any kind to this forum. // Do not send them any posts at all. unset($poststructure[$courseid][$forumid]); continue; } $subscriptiontime = \mod_forum\subscriptions::fetch_discussion_subscription($forum->id, $user->id); $cm = get_fast_modinfo($course)->instances['forum'][$forumid]; foreach ($discussionids as $discussionid => $postids) { $discussion = $this->discussions[$discussionid]; if (!\mod_forum\subscriptions::is_subscribed($user->id, $forum, $discussionid, $cm)) { // The user does not subscribe to this forum as a whole, or to this specific discussion. unset($poststructure[$courseid][$forumid][$discussionid]); continue; } if ($discussion->groupid > 0 and $groupmode = groups_get_activity_groupmode($cm, $course)) { // This discussion has a groupmode set (SEPARATEGROUPS or VISIBLEGROUPS). // Check whether the user can view it based on their groups. if (!isset($usergroups[$forum->id])) { $usergroups[$forum->id] = groups_get_all_groups($courseid, $user->id, $cm->groupingid); } if (!isset($usergroups[$forum->id][$discussion->groupid])) { // This user is not a member of this group, or the group no longer exists. $modcontext = \context_module::instance($cm->id); if (!has_capability('moodle/site:accessallgroups', $modcontext, $user)) { // This user does not have the accessallgroups and is not a member of the group. // Do not send posts from other groups when in SEPARATEGROUPS or VISIBLEGROUPS. unset($poststructure[$courseid][$forumid][$discussionid]); continue; } } } foreach ($postids as $postid) { $post = $this->posts[$postid]; if ($subscriptiontime) { // Skip posts if the user subscribed to the discussion after it was created. $subscribedafter = isset($subscriptiontime[$post->discussion]); $subscribedafter = $subscribedafter && ($subscriptiontime[$post->discussion] > $post->created); if ($subscribedafter) { // The user subscribed to the discussion/forum after this post was created. unset($poststructure[$courseid][$forumid][$discussionid][$postid]); continue; } } if ($maildigest > 0) { // This user wants the mails to be in digest form. $digeststructure[] = (object) [ 'userid' => $user->id, 'discussionid' => $discussion->id, 'postid' => $post->id, 'timemodified' => $post->created, ]; unset($poststructure[$courseid][$forumid][$discussionid][$postid]); continue; } else { // Add this post to the list of postids to be sent. $poststosend[] = $postid; } } } if (empty($poststructure[$courseid][$forumid])) { // This user is not subscribed to any discussions in this forum at all. unset($poststructure[$courseid][$forumid]); continue; } } if (empty($poststructure[$courseid])) { // This user is not subscribed to any forums in this course. unset($poststructure[$courseid]); } } return [$poststosend, $digeststructure]; } /** * Returns a list of all new posts that have not been mailed yet * * @param int $starttime posts created after this time * @param int $endtime posts created before this * @param int $now used for timed discussions only * @return array */ protected function get_unmailed_posts($starttime, $endtime, $now = null) { global $CFG, $DB; $params = array(); $params['mailed'] = FORUM_MAILED_PENDING; $params['ptimestart'] = $starttime; $params['ptimeend'] = $endtime; $params['mailnow'] = 1; if (!empty($CFG->forum_enabletimedposts)) { if (empty($now)) { $now = time(); } $selectsql = "AND (p.created >= :ptimestart OR d.timestart >= :pptimestart)"; $params['pptimestart'] = $starttime; $timedsql = "AND (d.timestart < :dtimestart AND (d.timeend = 0 OR d.timeend > :dtimeend))"; $params['dtimestart'] = $now; $params['dtimeend'] = $now; } else { $timedsql = ""; $selectsql = "AND p.created >= :ptimestart"; } return $DB->get_records_sql( "SELECT p.id, p.discussion, d.forum, d.course, p.created, p.parent, p.userid FROM {forum_posts} p JOIN {forum_discussions} d ON d.id = p.discussion WHERE p.mailed = :mailed $selectsql AND (p.created < :ptimeend OR p.mailnow = :mailnow) $timedsql ORDER BY p.modified ASC", $params); } }