Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 4.2.x will end 22 April 2024 (12 months).
  • Bug fixes for security issues in 4.2.x will end 7 October 2024 (18 months).
  • PHP version: minimum PHP 8.0.0 Note: minimum PHP version has increased since Moodle 4.1. PHP 8.1.x is supported too.

Differences Between: [Versions 310 and 402] [Versions 311 and 402] [Versions 39 and 402] [Versions 400 and 402] [Versions 401 and 402]

   1  <?php
   2  // This file is part of Moodle - http://moodle.org/
   3  //
   4  // Moodle is free software: you can redistribute it and/or modify
   5  // it under the terms of the GNU General Public License as published by
   6  // the Free Software Foundation, either version 3 of the License, or
   7  // (at your option) any later version.
   8  //
   9  // Moodle is distributed in the hope that it will be useful,
  10  // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12  // GNU General Public License for more details.
  13  //
  14  // You should have received a copy of the GNU General Public License
  15  // along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
  16  
  17  /**
  18   * Contains class used to return information to display for the message area.
  19   *
  20   * @package    core_message
  21   * @copyright  2016 Mark Nelson <markn@moodle.com>
  22   * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  23   */
  24  
  25  namespace core_message;
  26  
  27  use core_favourites\local\entity\favourite;
  28  
  29  defined('MOODLE_INTERNAL') || die();
  30  
  31  require_once($CFG->dirroot . '/lib/messagelib.php');
  32  
  33  /**
  34   * Class used to return information to display for the message area.
  35   *
  36   * @copyright  2016 Mark Nelson <markn@moodle.com>
  37   * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
  38   */
  39  class api {
  40  
  41      /**
  42       * The action for reading a message.
  43       */
  44      const MESSAGE_ACTION_READ = 1;
  45  
  46      /**
  47       * The action for deleting a message.
  48       */
  49      const MESSAGE_ACTION_DELETED = 2;
  50  
  51      /**
  52       * The action for reading a message.
  53       */
  54      const CONVERSATION_ACTION_MUTED = 1;
  55  
  56      /**
  57       * The privacy setting for being messaged by anyone within courses user is member of.
  58       */
  59      const MESSAGE_PRIVACY_COURSEMEMBER = 0;
  60  
  61      /**
  62       * The privacy setting for being messaged only by contacts.
  63       */
  64      const MESSAGE_PRIVACY_ONLYCONTACTS = 1;
  65  
  66      /**
  67       * The privacy setting for being messaged by anyone on the site.
  68       */
  69      const MESSAGE_PRIVACY_SITE = 2;
  70  
  71      /**
  72       * An individual conversation.
  73       */
  74      const MESSAGE_CONVERSATION_TYPE_INDIVIDUAL = 1;
  75  
  76      /**
  77       * A group conversation.
  78       */
  79      const MESSAGE_CONVERSATION_TYPE_GROUP = 2;
  80  
  81      /**
  82       * A self conversation.
  83       */
  84      const MESSAGE_CONVERSATION_TYPE_SELF = 3;
  85  
  86      /**
  87       * The state for an enabled conversation area.
  88       */
  89      const MESSAGE_CONVERSATION_ENABLED = 1;
  90  
  91      /**
  92       * The state for a disabled conversation area.
  93       */
  94      const MESSAGE_CONVERSATION_DISABLED = 0;
  95  
  96      /**
  97       * The max message length.
  98       */
  99      const MESSAGE_MAX_LENGTH = 4096;
 100  
 101      /**
 102       * Handles searching for messages in the message area.
 103       *
 104       * @param int $userid The user id doing the searching
 105       * @param string $search The string the user is searching
 106       * @param int $limitfrom
 107       * @param int $limitnum
 108       * @return array
 109       */
 110      public static function search_messages($userid, $search, $limitfrom = 0, $limitnum = 0) {
 111          global $DB;
 112  
 113          // Get the user fields we want.
 114          $userfieldsapi = \core_user\fields::for_userpic()->including('lastaccess');
 115          $ufields = $userfieldsapi->get_sql('u', false, 'userfrom_', '', false)->selects;
 116          $ufields2 = $userfieldsapi->get_sql('u2', false, 'userto_', '', false)->selects;
 117          // Add the uniqueid column to make each row unique and avoid SQL errors.
 118          $uniqueidsql = $DB->sql_concat('m.id', "'_'", 'm.useridfrom', "'_'", 'mcm.userid');
 119  
 120          $sql = "SELECT $uniqueidsql AS uniqueid, m.id, m.useridfrom, mcm.userid as useridto, m.subject, m.fullmessage,
 121                         m.fullmessagehtml, m.fullmessageformat, m.smallmessage, m.conversationid, m.timecreated, 0 as isread,
 122                         $ufields, mub.id as userfrom_blocked, $ufields2, mub2.id as userto_blocked
 123                    FROM (
 124                          SELECT m2.id AS id
 125                            FROM {messages} m2
 126                           WHERE m2.useridfrom = ?
 127                           UNION
 128                          SELECT m3.id AS id
 129                            FROM {message_conversation_members} mcm3
 130                      INNER JOIN {messages} m3 ON mcm3.conversationid = m3.conversationid
 131                           WHERE mcm3.userid = ?
 132                         ) der
 133              INNER JOIN {messages} m
 134                      ON der.id = m.id
 135              INNER JOIN {user} u
 136                      ON u.id = m.useridfrom
 137              INNER JOIN {message_conversations} mc
 138                      ON mc.id = m.conversationid
 139              INNER JOIN {message_conversation_members} mcm
 140                      ON mcm.conversationid = m.conversationid
 141              INNER JOIN {user} u2
 142                      ON u2.id = mcm.userid
 143               LEFT JOIN {message_users_blocked} mub
 144                      ON (mub.blockeduserid = u.id AND mub.userid = ?)
 145               LEFT JOIN {message_users_blocked} mub2
 146                      ON (mub2.blockeduserid = u2.id AND mub2.userid = ?)
 147               LEFT JOIN {message_user_actions} mua
 148                      ON (mua.messageid = m.id AND mua.userid = ? AND mua.action = ?)
 149                   WHERE (m.useridfrom = ? OR mcm.userid = ?)
 150                     AND (m.useridfrom != mcm.userid OR mc.type = ?)
 151                     AND u.deleted = 0
 152                     AND u2.deleted = 0
 153                     AND mua.id is NULL
 154                     AND " . $DB->sql_like('smallmessage', '?', false) . "
 155                ORDER BY timecreated DESC";
 156  
 157          $params = array($userid, $userid, $userid, $userid, $userid, self::MESSAGE_ACTION_DELETED, $userid, $userid,
 158              self::MESSAGE_CONVERSATION_TYPE_SELF, '%' . $search . '%');
 159  
 160          // Convert the messages into searchable contacts with their last message being the message that was searched.
 161          $conversations = array();
 162          if ($messages = $DB->get_records_sql($sql, $params, $limitfrom, $limitnum)) {
 163              foreach ($messages as $message) {
 164                  $prefix = 'userfrom_';
 165                  if ($userid == $message->useridfrom) {
 166                      $prefix = 'userto_';
 167                      // If it from the user, then mark it as read, even if it wasn't by the receiver.
 168                      $message->isread = true;
 169                  }
 170                  $blockedcol = $prefix . 'blocked';
 171                  $message->blocked = $message->$blockedcol ? 1 : 0;
 172  
 173                  $message->messageid = $message->id;
 174                  // To avoid duplicate messages, only add the message if it hasn't been added previously.
 175                  if (!array_key_exists($message->messageid, $conversations)) {
 176                      $conversations[$message->messageid] = helper::create_contact($message, $prefix);
 177                  }
 178              }
 179              // Remove the messageid keys (to preserve the expected type).
 180              $conversations = array_values($conversations);
 181          }
 182  
 183          return $conversations;
 184      }
 185  
 186      /**
 187       * @deprecated since 3.6
 188       */
 189      public static function search_users_in_course() {
 190          throw new \coding_exception('\core_message\api::search_users_in_course has been removed.');
 191      }
 192  
 193      /**
 194       * @deprecated since 3.6
 195       */
 196      public static function search_users() {
 197          throw new \coding_exception('\core_message\api::search_users has been removed.');
 198      }
 199  
 200      /**
 201       * Handles searching for user.
 202       *
 203       * @param int $userid The user id doing the searching
 204       * @param string $search The string the user is searching
 205       * @param int $limitfrom
 206       * @param int $limitnum
 207       * @return array
 208       */
 209      public static function message_search_users(int $userid, string $search, int $limitfrom = 0, int $limitnum = 20) : array {
 210          global $CFG, $DB;
 211  
 212          // Check if messaging is enabled.
 213          if (empty($CFG->messaging)) {
 214              throw new \moodle_exception('disabled', 'message');
 215          }
 216  
 217          require_once($CFG->dirroot . '/user/lib.php');
 218  
 219          // Used to search for contacts.
 220          $fullname = $DB->sql_fullname();
 221  
 222          // Users not to include.
 223          $excludeusers = array($CFG->siteguest);
 224          if (!$selfconversation = self::get_self_conversation($userid)) {
 225              // Userid should only be excluded when she hasn't a self-conversation.
 226              $excludeusers[] = $userid;
 227          }
 228          list($exclude, $excludeparams) = $DB->get_in_or_equal($excludeusers, SQL_PARAMS_NAMED, 'param', false);
 229  
 230          $params = array('search' => '%' . $DB->sql_like_escape($search) . '%', 'userid1' => $userid, 'userid2' => $userid);
 231  
 232          // Ok, let's search for contacts first.
 233          $sql = "SELECT u.id
 234                    FROM {user} u
 235                    JOIN {message_contacts} mc
 236                      ON (u.id = mc.contactid AND mc.userid = :userid1) OR (u.id = mc.userid AND mc.contactid = :userid2)
 237                   WHERE u.deleted = 0
 238                     AND u.confirmed = 1
 239                     AND " . $DB->sql_like($fullname, ':search', false) . "
 240                     AND u.id $exclude
 241                ORDER BY " . $DB->sql_fullname();
 242          $foundusers = $DB->get_records_sql_menu($sql, $params + $excludeparams, $limitfrom, $limitnum);
 243  
 244          $contacts = [];
 245          if (!empty($foundusers)) {
 246              $contacts = helper::get_member_info($userid, array_keys($foundusers));
 247              foreach ($contacts as $memberuserid => $memberinfo) {
 248                  $contacts[$memberuserid]->conversations = self::get_conversations_between_users($userid, $memberuserid, 0, 1000);
 249              }
 250          }
 251  
 252          // We need to get all the user details for a fullname in the visibility checks.
 253          $namefields = \core_user\fields::for_name()
 254              // Required by the visibility checks.
 255              ->including('deleted');
 256  
 257          // Let's get those non-contacts.
 258          // Because we can't achieve all the required visibility checks in SQL, we'll iterate through the non-contact records
 259          // and stop once we have enough matching the 'visible' criteria.
 260  
 261          // Use a local generator to achieve this iteration.
 262          $getnoncontactusers = function ($limitfrom = 0, $limitnum = 0) use (
 263              $fullname,
 264              $exclude,
 265              $params,
 266              $excludeparams,
 267              $userid,
 268              $selfconversation,
 269              $namefields
 270          ) {
 271              global $DB, $CFG;
 272  
 273              $joinenrolled = '';
 274              $enrolled = '';
 275              $unionself = '';
 276              $enrolledparams = [];
 277  
 278              // Since we want to order a UNION we need to list out all the user fields individually this will
 279              // allow us to reference the fullname correctly.
 280              $userfields = $namefields->get_sql('u')->selects;
 281  
 282              $select = "u.id, " . $DB->sql_fullname() . " AS sortingname" . $userfields;
 283  
 284              // When messageallusers is false valid non-contacts must be enrolled on one of the users courses.
 285              if (empty($CFG->messagingallusers)) {
 286                  $joinenrolled = "JOIN {user_enrolments} ue ON ue.userid = u.id
 287                                   JOIN {enrol} e ON e.id = ue.enrolid";
 288                  $enrolled = "AND e.courseid IN (
 289                                  SELECT e.courseid
 290                                    FROM {user_enrolments} ue
 291                                    JOIN {enrol} e ON e.id = ue.enrolid
 292                                   WHERE ue.userid = :enroluserid
 293                                  )";
 294  
 295                  if ($selfconversation !== false) {
 296                      // We must include the user themselves, when they have a self conversation, even if they are not
 297                      // enrolled on any courses.
 298                      $unionself = "UNION SELECT u.id FROM {user} u
 299                                           WHERE u.id = :self AND ". $DB->sql_like($fullname, ':selfsearch', false);
 300                  }
 301                  $enrolledparams = ['enroluserid' => $userid, 'self' => $userid, 'selfsearch' => $params['search']];
 302              }
 303  
 304              $sql = "SELECT $select
 305                        FROM (
 306                          SELECT DISTINCT u.id
 307                            FROM {user} u $joinenrolled
 308                           WHERE u.deleted = 0
 309                             AND u.confirmed = 1
 310                             AND " . $DB->sql_like($fullname, ':search', false) . "
 311                             AND u.id $exclude $enrolled
 312                             AND NOT EXISTS (SELECT mc.id
 313                                               FROM {message_contacts} mc
 314                                              WHERE (mc.userid = u.id AND mc.contactid = :userid1)
 315                                                 OR (mc.userid = :userid2 AND mc.contactid = u.id)) $unionself
 316                           ) targetedusers
 317                        JOIN {user} u ON u.id = targetedusers.id
 318                    ORDER BY 2";
 319              while ($records = $DB->get_records_sql($sql, $params + $excludeparams + $enrolledparams, $limitfrom, $limitnum)) {
 320                  yield $records;
 321                  $limitfrom += $limitnum;
 322              }
 323          };
 324  
 325          // Fetch in batches of $limitnum * 2 to improve the chances of matching a user without going back to the DB.
 326          // The generator cannot function without a sensible limiter, so set one if this is not set.
 327          $batchlimit = ($limitnum == 0) ? 20 : $limitnum;
 328  
 329          // We need to make the offset param work with the generator.
 330          // Basically, if we want to get say 10 records starting at the 40th record, we need to see 50 records and return only
 331          // those after the 40th record. We can never pass the method's offset param to the generator as we need to manage the
 332          // position within those valid records ourselves.
 333          // See MDL-63983 dealing with performance improvements to this area of code.
 334          $noofvalidseenrecords = 0;
 335          $returnedusers = [];
 336  
 337          // Only fields that are also part of user_get_default_fields() are valid when passed into user_get_user_details().
 338          $fields = array_intersect($namefields->get_required_fields(), user_get_default_fields());
 339  
 340          foreach ($getnoncontactusers(0, $batchlimit) as $users) {
 341              foreach ($users as $id => $user) {
 342                  // User visibility checks: only return users who are visible to the user performing the search.
 343                  // Which visibility check to use depends on the 'messagingallusers' (site wide messaging) setting:
 344                  // - If enabled, return matched users whose profiles are visible to the current user anywhere (site or course).
 345                  // - If disabled, only return matched users whose course profiles are visible to the current user.
 346                  $userdetails = \core_message\helper::search_get_user_details($user, $fields);
 347  
 348                  // Return the user only if the searched field is returned.
 349                  // Otherwise it means that the $USER was not allowed to search the returned user.
 350                  if (!empty($userdetails) and !empty($userdetails['fullname'])) {
 351                      // We know we've matched, but only save the record if it's within the offset area we need.
 352                      if ($limitfrom == 0) {
 353                          // No offset specified, so just save.
 354                          $returnedusers[$id] = $user;
 355                      } else {
 356                          // There is an offset in play.
 357                          // If we've passed enough records already (> offset value), then we can save this one.
 358                          if ($noofvalidseenrecords >= $limitfrom) {
 359                              $returnedusers[$id] = $user;
 360                          }
 361                      }
 362                      if (count($returnedusers) == $limitnum) {
 363                          break 2;
 364                      }
 365                      $noofvalidseenrecords++;
 366                  }
 367              }
 368          }
 369          $foundusers = $returnedusers;
 370  
 371          $noncontacts = [];
 372          if (!empty($foundusers)) {
 373              $noncontacts = helper::get_member_info($userid, array_keys($foundusers));
 374              foreach ($noncontacts as $memberuserid => $memberinfo) {
 375                  if ($memberuserid !== $userid) {
 376                      $noncontacts[$memberuserid]->conversations = self::get_conversations_between_users($userid, $memberuserid, 0,
 377                          1000);
 378                  } else {
 379                      $noncontacts[$memberuserid]->conversations[$selfconversation->id] = $selfconversation;
 380                  }
 381              }
 382          }
 383  
 384          return array(array_values($contacts), array_values($noncontacts));
 385      }
 386  
 387      /**
 388       * Gets extra fields, like image url and subname for any conversations linked to components.
 389       *
 390       * The subname is like a subtitle for the conversation, to compliment it's name.
 391       * The imageurl is the location of the image for the conversation, as might be seen on a listing of conversations for a user.
 392       *
 393       * @param array $conversations a list of conversations records.
 394       * @return array the array of subnames, index by conversation id.
 395       * @throws \coding_exception
 396       * @throws \dml_exception
 397       */
 398      protected static function get_linked_conversation_extra_fields(array $conversations) : array {
 399          global $DB, $PAGE;
 400  
 401          $renderer = $PAGE->get_renderer('core');
 402  
 403          $linkedconversations = [];
 404          foreach ($conversations as $conversation) {
 405              if (!is_null($conversation->component) && !is_null($conversation->itemtype)) {
 406                  $linkedconversations[$conversation->component][$conversation->itemtype][$conversation->id]
 407                      = $conversation->itemid;
 408              }
 409          }
 410          if (empty($linkedconversations)) {
 411              return [];
 412          }
 413  
 414          // TODO: MDL-63814: Working out the subname for linked conversations should be done in a generic way.
 415          // Get the itemid, but only for course group linked conversation for now.
 416          $extrafields = [];
 417          if (!empty($linkeditems = $linkedconversations['core_group']['groups'])) { // Format: [conversationid => itemid].
 418              // Get the name of the course to which the group belongs.
 419              list ($groupidsql, $groupidparams) = $DB->get_in_or_equal(array_values($linkeditems), SQL_PARAMS_NAMED, 'groupid');
 420              $sql = "SELECT g.*, c.shortname as courseshortname
 421                        FROM {groups} g
 422                        JOIN {course} c
 423                          ON g.courseid = c.id
 424                       WHERE g.id $groupidsql";
 425              $courseinfo = $DB->get_records_sql($sql, $groupidparams);
 426              foreach ($linkeditems as $convid => $groupid) {
 427                  if (array_key_exists($groupid, $courseinfo)) {
 428                      $group = $courseinfo[$groupid];
 429                      // Subname.
 430                      $extrafields[$convid]['subname'] = format_string($courseinfo[$groupid]->courseshortname);
 431  
 432                      // Imageurl.
 433                      $extrafields[$convid]['imageurl'] = $renderer->image_url('g/g1')->out(false); // default image.
 434                      if ($url = get_group_picture_url($group, $group->courseid, true)) {
 435                          $extrafields[$convid]['imageurl'] = $url->out(false);
 436                      }
 437                  }
 438              }
 439          }
 440          return $extrafields;
 441      }
 442  
 443  
 444      /**
 445       * Returns the contacts and their conversation to display in the contacts area.
 446       *
 447       * ** WARNING **
 448       * It is HIGHLY recommended to use a sensible limit when calling this function. Trying
 449       * to retrieve too much information in a single call will cause performance problems.
 450       * ** WARNING **
 451       *
 452       * This function has specifically been altered to break each of the data sets it
 453       * requires into separate database calls. This is to avoid the performance problems
 454       * observed when attempting to join large data sets (e.g. the message tables and
 455       * the user table).
 456       *
 457       * While it is possible to gather the data in a single query, and it may even be
 458       * more efficient with a correctly tuned database, we have opted to trade off some of
 459       * the benefits of a single query in order to ensure this function will work on
 460       * most databases with default tunings and with large data sets.
 461       *
 462       * @param int $userid The user id
 463       * @param int $limitfrom
 464       * @param int $limitnum
 465       * @param int $type the type of the conversation, if you wish to filter to a certain type (see api constants).
 466       * @param bool $favourites whether to include NO favourites (false) or ONLY favourites (true), or null to ignore this setting.
 467       * @param bool $mergeself whether to include self-conversations (true) or ONLY private conversations (false)
 468       *             when private conversations are requested.
 469       * @return array the array of conversations
 470       * @throws \moodle_exception
 471       */
 472      public static function get_conversations($userid, $limitfrom = 0, $limitnum = 20, int $type = null,
 473              bool $favourites = null, bool $mergeself = false) {
 474          global $DB;
 475  
 476          if (!is_null($type) && !in_array($type, [self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL,
 477                  self::MESSAGE_CONVERSATION_TYPE_GROUP, self::MESSAGE_CONVERSATION_TYPE_SELF])) {
 478              throw new \moodle_exception("Invalid value ($type) for type param, please see api constants.");
 479          }
 480  
 481          self::lazy_create_self_conversation($userid);
 482  
 483          // We need to know which conversations are favourites, so we can either:
 484          // 1) Include the 'isfavourite' attribute on conversations (when $favourite = null and we're including all conversations)
 485          // 2) Restrict the results to ONLY those conversations which are favourites (when $favourite = true)
 486          // 3) Restrict the results to ONLY those conversations which are NOT favourites (when $favourite = false).
 487          $service = \core_favourites\service_factory::get_service_for_user_context(\context_user::instance($userid));
 488          $favouriteconversations = $service->find_favourites_by_type('core_message', 'message_conversations');
 489          $favouriteconversationids = array_column($favouriteconversations, 'itemid');
 490          if ($favourites && empty($favouriteconversationids)) {
 491              return []; // If we are aiming to return ONLY favourites, and we have none, there's nothing more to do.
 492          }
 493  
 494          // CONVERSATIONS AND MOST RECENT MESSAGE.
 495          // Include those conversations with messages first (ordered by most recent message, desc), then add any conversations which
 496          // don't have messages, such as newly created group conversations.
 497          // Because we're sorting by message 'timecreated', those conversations without messages could be at either the start or the
 498          // end of the results (behaviour for sorting of nulls differs between DB vendors), so we use the case to presort these.
 499  
 500          // If we need to return ONLY favourites, or NO favourites, generate the SQL snippet.
 501          $favouritesql = "";
 502          $favouriteparams = [];
 503          if (null !== $favourites && !empty($favouriteconversationids)) {
 504              list ($insql, $favouriteparams) =
 505                      $DB->get_in_or_equal($favouriteconversationids, SQL_PARAMS_NAMED, 'favouriteids', $favourites);
 506              $favouritesql = " AND mc.id {$insql} ";
 507          }
 508  
 509          // If we need to restrict type, generate the SQL snippet.
 510          $typesql = "";
 511          $typeparams = [];
 512          if (!is_null($type)) {
 513              if ($mergeself && $type == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL) {
 514                  // When $megerself is set to true, the self-conversations are returned also with the private conversations.
 515                  $typesql = " AND (mc.type = :convtype1 OR mc.type = :convtype2) ";
 516                  $typeparams = ['convtype1' => $type, 'convtype2' => self::MESSAGE_CONVERSATION_TYPE_SELF];
 517              } else {
 518                  $typesql = " AND mc.type = :convtype ";
 519                  $typeparams = ['convtype' => $type];
 520              }
 521          }
 522  
 523          $sql = "SELECT m.id as messageid, mc.id as id, mc.name as conversationname, mc.type as conversationtype, m.useridfrom,
 524                         m.smallmessage, m.fullmessage, m.fullmessageformat, m.fullmessagetrust, m.fullmessagehtml, m.timecreated,
 525                         mc.component, mc.itemtype, mc.itemid, mc.contextid, mca.action as ismuted
 526                    FROM {message_conversations} mc
 527              INNER JOIN {message_conversation_members} mcm
 528                      ON (mcm.conversationid = mc.id AND mcm.userid = :userid3)
 529              LEFT JOIN (
 530                            SELECT m.conversationid, MAX(m.id) AS messageid
 531                              FROM {messages} m
 532                        INNER JOIN (
 533                                        SELECT m.conversationid, MAX(m.timecreated) as maxtime
 534                                          FROM {messages} m
 535                                    INNER JOIN {message_conversation_members} mcm
 536                                            ON mcm.conversationid = m.conversationid
 537                                     LEFT JOIN {message_user_actions} mua
 538                                            ON (mua.messageid = m.id AND mua.userid = :userid AND mua.action = :action)
 539                                         WHERE mua.id is NULL
 540                                           AND mcm.userid = :userid2
 541                                      GROUP BY m.conversationid
 542                                   ) maxmessage
 543                                 ON maxmessage.maxtime = m.timecreated AND maxmessage.conversationid = m.conversationid
 544                           GROUP BY m.conversationid
 545                         ) lastmessage
 546                      ON lastmessage.conversationid = mc.id
 547              LEFT JOIN {messages} m
 548                     ON m.id = lastmessage.messageid
 549              LEFT JOIN {message_conversation_actions} mca
 550                     ON (mca.conversationid = mc.id AND mca.userid = :userid4 AND mca.action = :convaction)
 551                  WHERE mc.id IS NOT NULL
 552                    AND mc.enabled = 1 $typesql $favouritesql
 553                ORDER BY (CASE WHEN m.timecreated IS NULL THEN 0 ELSE 1 END) DESC, m.timecreated DESC, id DESC";
 554  
 555          $params = array_merge($favouriteparams, $typeparams, ['userid' => $userid, 'action' => self::MESSAGE_ACTION_DELETED,
 556              'userid2' => $userid, 'userid3' => $userid, 'userid4' => $userid, 'convaction' => self::CONVERSATION_ACTION_MUTED]);
 557          $conversationset = $DB->get_recordset_sql($sql, $params, $limitfrom, $limitnum);
 558  
 559          $conversations = [];
 560          $selfconversations = []; // Used to track conversations with one's self.
 561          $members = [];
 562          $individualmembers = [];
 563          $groupmembers = [];
 564          $selfmembers = [];
 565          foreach ($conversationset as $conversation) {
 566              $conversations[$conversation->id] = $conversation;
 567              $members[$conversation->id] = [];
 568          }
 569          $conversationset->close();
 570  
 571          // If there are no conversations found, then return early.
 572          if (empty($conversations)) {
 573              return [];
 574          }
 575  
 576          // COMPONENT-LINKED CONVERSATION FIELDS.
 577          // Conversations linked to components may have extra information, such as:
 578          // - subname: Essentially a subtitle for the conversation. So you'd have "name: subname".
 579          // - imageurl: A URL to the image for the linked conversation.
 580          // For now, this is ONLY course groups.
 581          $convextrafields = self::get_linked_conversation_extra_fields($conversations);
 582  
 583          // MEMBERS.
 584          // Ideally, we want to get 1 member for each conversation, but this depends on the type and whether there is a recent
 585          // message or not.
 586          //
 587          // For 'individual' type conversations between 2 users, regardless of who sent the last message,
 588          // we want the details of the other member in the conversation (i.e. not the current user).
 589          //
 590          // For 'group' type conversations, we want the details of the member who sent the last message, if there is one.
 591          // This can be the current user or another group member, but for groups without messages, this will be empty.
 592          //
 593          // For 'self' type conversations, we want the details of the current user.
 594          //
 595          // This also means that if type filtering is specified and only group conversations are returned, we don't need this extra
 596          // query to get the 'other' user as we already have that information.
 597  
 598          // Work out which members we have already, and which ones we might need to fetch.
 599          // If all the last messages were from another user, then we don't need to fetch anything further.
 600          foreach ($conversations as $conversation) {
 601              if ($conversation->conversationtype == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL) {
 602                  if (!is_null($conversation->useridfrom) && $conversation->useridfrom != $userid) {
 603                      $members[$conversation->id][$conversation->useridfrom] = $conversation->useridfrom;
 604                      $individualmembers[$conversation->useridfrom] = $conversation->useridfrom;
 605                  } else {
 606                      $individualconversations[] = $conversation->id;
 607                  }
 608              } else if ($conversation->conversationtype == self::MESSAGE_CONVERSATION_TYPE_GROUP) {
 609                  // If we have a recent message, the sender is our member.
 610                  if (!is_null($conversation->useridfrom)) {
 611                      $members[$conversation->id][$conversation->useridfrom] = $conversation->useridfrom;
 612                      $groupmembers[$conversation->useridfrom] = $conversation->useridfrom;
 613                  }
 614              } else if ($conversation->conversationtype == self::MESSAGE_CONVERSATION_TYPE_SELF) {
 615                  $selfconversations[$conversation->id] = $conversation->id;
 616                  $members[$conversation->id][$userid] = $userid;
 617                  $selfmembers[$userid] = $userid;
 618              }
 619          }
 620          // If we need to fetch any member information for any of the individual conversations.
 621          // This is the case if any of the individual conversations have a recent message sent by the current user.
 622          if (!empty($individualconversations)) {
 623              list ($icidinsql, $icidinparams) = $DB->get_in_or_equal($individualconversations, SQL_PARAMS_NAMED, 'convid');
 624              $indmembersql = "SELECT mcm.id, mcm.conversationid, mcm.userid
 625                          FROM {message_conversation_members} mcm
 626                         WHERE mcm.conversationid $icidinsql
 627                         AND mcm.userid != :userid
 628                         ORDER BY mcm.id";
 629              $indmemberparams = array_merge($icidinparams, ['userid' => $userid]);
 630              $conversationmembers = $DB->get_records_sql($indmembersql, $indmemberparams);
 631  
 632              foreach ($conversationmembers as $mid => $member) {
 633                  $members[$member->conversationid][$member->userid] = $member->userid;
 634                  $individualmembers[$member->userid] = $member->userid;
 635              }
 636          }
 637  
 638          // We could fail early here if we're sure that:
 639          // a) we have no otherusers for all the conversations (users may have been deleted)
 640          // b) we're sure that all conversations are individual (1:1).
 641  
 642          // We need to pull out the list of users info corresponding to the memberids in the conversations.This
 643          // needs to be done in a separate query to avoid doing a join on the messages tables and the user
 644          // tables because on large sites these tables are massive which results in extremely slow
 645          // performance (typically due to join buffer exhaustion).
 646          if (!empty($individualmembers) || !empty($groupmembers) || !empty($selfmembers)) {
 647              // Now, we want to remove any duplicates from the group members array. For individual members we will
 648              // be doing a more extensive call as we want their contact requests as well as privacy information,
 649              // which is not necessary for group conversations.
 650              $diffgroupmembers = array_diff($groupmembers, $individualmembers);
 651  
 652              $individualmemberinfo = helper::get_member_info($userid, $individualmembers, true, true);
 653              $groupmemberinfo = helper::get_member_info($userid, $diffgroupmembers);
 654              $selfmemberinfo = helper::get_member_info($userid, $selfmembers);
 655  
 656              // Don't use array_merge, as we lose array keys.
 657              $memberinfo = $individualmemberinfo + $groupmemberinfo + $selfmemberinfo;
 658  
 659              if (empty($memberinfo)) {
 660                  return [];
 661              }
 662  
 663              // Update the members array with the member information.
 664              $deletedmembers = [];
 665              foreach ($members as $convid => $memberarr) {
 666                  foreach ($memberarr as $key => $memberid) {
 667                      if (array_key_exists($memberid, $memberinfo)) {
 668                          // If the user is deleted, remember that.
 669                          if ($memberinfo[$memberid]->isdeleted) {
 670                              $deletedmembers[$convid][] = $memberid;
 671                          }
 672  
 673                          $members[$convid][$key] = clone $memberinfo[$memberid];
 674  
 675                          if ($conversations[$convid]->conversationtype == self::MESSAGE_CONVERSATION_TYPE_GROUP) {
 676                              // Remove data we don't need for group.
 677                              $members[$convid][$key]->requirescontact = null;
 678                              $members[$convid][$key]->canmessage = null;
 679                              $members[$convid][$key]->contactrequests = [];
 680                          }
 681                      } else { // Remove all members and individual conversations where we could not get the member's information.
 682                          unset($members[$convid][$key]);
 683  
 684                          // If the conversation is an individual conversation, then we should remove it from the list.
 685                          if ($conversations[$convid]->conversationtype == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL) {
 686                              unset($conversations[$convid]);
 687                          }
 688                      }
 689                  }
 690              }
 691          }
 692  
 693          // MEMBER COUNT.
 694          $cids = array_column($conversations, 'id');
 695          list ($cidinsql, $cidinparams) = $DB->get_in_or_equal($cids, SQL_PARAMS_NAMED, 'convid');
 696          $membercountsql = "SELECT conversationid, count(DISTINCT userid) AS membercount
 697                               FROM {message_conversation_members} mcm
 698                              WHERE mcm.conversationid $cidinsql
 699                           GROUP BY mcm.conversationid";
 700          $membercounts = $DB->get_records_sql($membercountsql, $cidinparams);
 701  
 702          // UNREAD MESSAGE COUNT.
 703          // Finally, let's get the unread messages count for this user so that we can add it
 704          // to the conversation. Remember we need to ignore the messages the user sent.
 705          $unreadcountssql = 'SELECT m.conversationid, count(m.id) as unreadcount
 706                                FROM {messages} m
 707                          INNER JOIN {message_conversations} mc
 708                                  ON mc.id = m.conversationid
 709                          INNER JOIN {message_conversation_members} mcm
 710                                  ON m.conversationid = mcm.conversationid
 711                           LEFT JOIN {message_user_actions} mua
 712                                  ON (mua.messageid = m.id AND mua.userid = ? AND
 713                                     (mua.action = ? OR mua.action = ?))
 714                               WHERE mcm.userid = ?
 715                                 AND m.useridfrom != ?
 716                                 AND mua.id is NULL
 717                            GROUP BY m.conversationid';
 718          $unreadcounts = $DB->get_records_sql($unreadcountssql, [$userid, self::MESSAGE_ACTION_READ, self::MESSAGE_ACTION_DELETED,
 719              $userid, $userid]);
 720  
 721          // For the self-conversations, get the total number of messages (to know if the conversation is new or it has been emptied).
 722          $selfmessagessql = "SELECT COUNT(m.id)
 723                                FROM {messages} m
 724                          INNER JOIN {message_conversations} mc
 725                                  ON mc.id = m.conversationid
 726                               WHERE mc.type = ? AND convhash = ?";
 727          $selfmessagestotal = $DB->count_records_sql(
 728              $selfmessagessql,
 729              [self::MESSAGE_CONVERSATION_TYPE_SELF, helper::get_conversation_hash([$userid])]
 730          );
 731  
 732          // Because we'll be calling format_string on each conversation name and passing contexts, we preload them here.
 733          // This warms the cache and saves potentially hitting the DB once for each context fetch below.
 734          \context_helper::preload_contexts_by_id(array_column($conversations, 'contextid'));
 735  
 736          // Now, create the final return structure.
 737          $arrconversations = [];
 738          foreach ($conversations as $conversation) {
 739              // Do not include any individual which do not contain a recent message for the user.
 740              // This happens if the user has deleted all messages.
 741              // Exclude the self-conversations with messages but without a recent message because the user has deleted all them.
 742              // Self-conversations without any message should be included, to display them first time they are created.
 743              // Group conversations with deleted users or no messages are always returned.
 744              if ($conversation->conversationtype == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL && empty($conversation->messageid) ||
 745                     ($conversation->conversationtype == self::MESSAGE_CONVERSATION_TYPE_SELF && empty($conversation->messageid)
 746                      && $selfmessagestotal > 0)) {
 747                  continue;
 748              }
 749  
 750              $conv = new \stdClass();
 751              $conv->id = $conversation->id;
 752  
 753              // Name should be formatted and depends on the context the conversation resides in.
 754              // If not set, the context is always context_user.
 755              if (is_null($conversation->contextid)) {
 756                  $convcontext = \context_user::instance($userid);
 757                  // We'll need to check the capability to delete messages for all users in context system when contextid is null.
 758                  $contexttodeletemessageforall = \context_system::instance();
 759              } else {
 760                  $convcontext = \context::instance_by_id($conversation->contextid);
 761                  $contexttodeletemessageforall = $convcontext;
 762              }
 763              $conv->name = format_string($conversation->conversationname, true, ['context' => $convcontext]);
 764  
 765              $conv->subname = $convextrafields[$conv->id]['subname'] ?? null;
 766              $conv->imageurl = $convextrafields[$conv->id]['imageurl'] ?? null;
 767              $conv->type = $conversation->conversationtype;
 768              $conv->membercount = $membercounts[$conv->id]->membercount;
 769              $conv->isfavourite = in_array($conv->id, $favouriteconversationids);
 770              $conv->isread = isset($unreadcounts[$conv->id]) ? false : true;
 771              $conv->unreadcount = isset($unreadcounts[$conv->id]) ? $unreadcounts[$conv->id]->unreadcount : null;
 772              $conv->ismuted = $conversation->ismuted ? true : false;
 773              $conv->members = $members[$conv->id];
 774  
 775              // Add the most recent message information.
 776              $conv->messages = [];
 777              // Add if the user has to allow delete messages for all users in the conversation.
 778              $conv->candeletemessagesforallusers = has_capability('moodle/site:deleteanymessage',  $contexttodeletemessageforall);
 779              if ($conversation->smallmessage) {
 780                  $msg = new \stdClass();
 781                  $msg->id = $conversation->messageid;
 782                  $msg->text = message_format_message_text($conversation);
 783                  $msg->useridfrom = $conversation->useridfrom;
 784                  $msg->timecreated = $conversation->timecreated;
 785                  $conv->messages[] = $msg;
 786              }
 787  
 788              $arrconversations[] = $conv;
 789          }
 790          return $arrconversations;
 791      }
 792  
 793      /**
 794       * Returns all conversations between two users
 795       *
 796       * @param int $userid1 One of the user's id
 797       * @param int $userid2 The other user's id
 798       * @param int $limitfrom
 799       * @param int $limitnum
 800       * @return array
 801       * @throws \dml_exception
 802       */
 803      public static function get_conversations_between_users(int $userid1, int $userid2,
 804                                                             int $limitfrom = 0, int $limitnum = 20) : array {
 805  
 806          global $DB;
 807  
 808          if ($userid1 == $userid2) {
 809              return array();
 810          }
 811  
 812          // Get all conversation where both user1 and user2 are members.
 813          // TODO: Add subname value. Waiting for definite table structure.
 814          $sql = "SELECT mc.id, mc.type, mc.name, mc.timecreated
 815                    FROM {message_conversations} mc
 816              INNER JOIN {message_conversation_members} mcm1
 817                      ON mc.id = mcm1.conversationid
 818              INNER JOIN {message_conversation_members} mcm2
 819                      ON mc.id = mcm2.conversationid
 820                   WHERE mcm1.userid = :userid1
 821                     AND mcm2.userid = :userid2
 822                     AND mc.enabled != 0
 823                ORDER BY mc.timecreated DESC";
 824  
 825          return $DB->get_records_sql($sql, array('userid1' => $userid1, 'userid2' => $userid2), $limitfrom, $limitnum);
 826      }
 827  
 828      /**
 829       * Return a conversation.
 830       *
 831       * @param int $userid The user id to get the conversation for
 832       * @param int $conversationid The id of the conversation to fetch
 833       * @param bool $includecontactrequests Should contact requests be included between members
 834       * @param bool $includeprivacyinfo Should privacy info be included between members
 835       * @param int $memberlimit Limit number of members to load
 836       * @param int $memberoffset Offset members by this amount
 837       * @param int $messagelimit Limit number of messages to load
 838       * @param int $messageoffset Offset the messages
 839       * @param bool $newestmessagesfirst Order messages by newest first
 840       * @return \stdClass
 841       */
 842      public static function get_conversation(
 843          int $userid,
 844          int $conversationid,
 845          bool $includecontactrequests = false,
 846          bool $includeprivacyinfo = false,
 847          int $memberlimit = 0,
 848          int $memberoffset = 0,
 849          int $messagelimit = 0,
 850          int $messageoffset = 0,
 851          bool $newestmessagesfirst = true
 852      ) {
 853          global $USER, $DB;
 854  
 855          $systemcontext = \context_system::instance();
 856          $canreadallmessages = has_capability('moodle/site:readallmessages', $systemcontext);
 857          if (($USER->id != $userid) && !$canreadallmessages) {
 858              throw new \moodle_exception('You do not have permission to perform this action.');
 859          }
 860  
 861          $conversation = $DB->get_record('message_conversations', ['id' => $conversationid]);
 862          if (!$conversation) {
 863              return null;
 864          }
 865  
 866          // Get the context of the conversation. This will be used to check whether the conversation is a favourite.
 867          // This will be either 'user' (for individual conversations) or, in the case of linked conversations,
 868          // the context stored in the record.
 869          $userctx = \context_user::instance($userid);
 870          $conversationctx = empty($conversation->contextid) ? $userctx : \context::instance_by_id($conversation->contextid);
 871  
 872          $isconversationmember = $DB->record_exists(
 873              'message_conversation_members',
 874              [
 875                  'conversationid' => $conversationid,
 876                  'userid' => $userid
 877              ]
 878          );
 879  
 880          if (!$isconversationmember && !$canreadallmessages) {
 881              throw new \moodle_exception('You do not have permission to view this conversation.');
 882          }
 883  
 884          $members = self::get_conversation_members(
 885              $userid,
 886              $conversationid,
 887              $includecontactrequests,
 888              $includeprivacyinfo,
 889              $memberoffset,
 890              $memberlimit
 891          );
 892          if ($conversation->type != self::MESSAGE_CONVERSATION_TYPE_SELF) {
 893              // Strip out the requesting user to match what get_conversations does, except for self-conversations.
 894              $members = array_filter($members, function($member) use ($userid) {
 895                  return $member->id != $userid;
 896              });
 897          }
 898  
 899          $messages = self::get_conversation_messages(
 900              $userid,
 901              $conversationid,
 902              $messageoffset,
 903              $messagelimit,
 904              $newestmessagesfirst ? 'timecreated DESC' : 'timecreated ASC'
 905          );
 906  
 907          $service = \core_favourites\service_factory::get_service_for_user_context(\context_user::instance($userid));
 908          $isfavourite = $service->favourite_exists('core_message', 'message_conversations', $conversationid, $conversationctx);
 909  
 910          $convextrafields = self::get_linked_conversation_extra_fields([$conversation]);
 911          $subname = isset($convextrafields[$conversationid]) ? $convextrafields[$conversationid]['subname'] : null;
 912          $imageurl = isset($convextrafields[$conversationid]) ? $convextrafields[$conversationid]['imageurl'] : null;
 913  
 914          $unreadcountssql = 'SELECT count(m.id)
 915                                FROM {messages} m
 916                          INNER JOIN {message_conversations} mc
 917                                  ON mc.id = m.conversationid
 918                           LEFT JOIN {message_user_actions} mua
 919                                  ON (mua.messageid = m.id AND mua.userid = ? AND
 920                                     (mua.action = ? OR mua.action = ?))
 921                               WHERE m.conversationid = ?
 922                                 AND m.useridfrom != ?
 923                                 AND mua.id is NULL';
 924          $unreadcount = $DB->count_records_sql(
 925              $unreadcountssql,
 926              [
 927                  $userid,
 928                  self::MESSAGE_ACTION_READ,
 929                  self::MESSAGE_ACTION_DELETED,
 930                  $conversationid,
 931                  $userid
 932              ]
 933          );
 934  
 935          $membercount = $DB->count_records('message_conversation_members', ['conversationid' => $conversationid]);
 936  
 937          $ismuted = false;
 938          if ($DB->record_exists('message_conversation_actions', ['userid' => $userid,
 939                  'conversationid' => $conversationid, 'action' => self::CONVERSATION_ACTION_MUTED])) {
 940              $ismuted = true;
 941          }
 942  
 943          // Get the context of the conversation. This will be used to check if the user can delete all messages in the conversation.
 944          $deleteallcontext = empty($conversation->contextid) ? $systemcontext : \context::instance_by_id($conversation->contextid);
 945  
 946          return (object) [
 947              'id' => $conversation->id,
 948              'name' => $conversation->name,
 949              'subname' => $subname,
 950              'imageurl' => $imageurl,
 951              'type' => $conversation->type,
 952              'membercount' => $membercount,
 953              'isfavourite' => $isfavourite,
 954              'isread' => empty($unreadcount),
 955              'unreadcount' => $unreadcount,
 956              'ismuted' => $ismuted,
 957              'members' => $members,
 958              'messages' => $messages['messages'],
 959              'candeletemessagesforallusers' => has_capability('moodle/site:deleteanymessage', $deleteallcontext)
 960          ];
 961      }
 962  
 963      /**
 964       * Mark a conversation as a favourite for the given user.
 965       *
 966       * @param int $conversationid the id of the conversation to mark as a favourite.
 967       * @param int $userid the id of the user to whom the favourite belongs.
 968       * @return favourite the favourite object.
 969       * @throws \moodle_exception if the user or conversation don't exist.
 970       */
 971      public static function set_favourite_conversation(int $conversationid, int $userid) : favourite {
 972          global $DB;
 973  
 974          if (!self::is_user_in_conversation($userid, $conversationid)) {
 975              throw new \moodle_exception("Conversation doesn't exist or user is not a member");
 976          }
 977          // Get the context for this conversation.
 978          $conversation = $DB->get_record('message_conversations', ['id' => $conversationid]);
 979          $userctx = \context_user::instance($userid);
 980          if (empty($conversation->contextid)) {
 981              // When the conversation hasn't any contextid value defined, the favourite will be added to the user context.
 982              $conversationctx = $userctx;
 983          } else {
 984              // If the contextid is defined, the favourite will be added there.
 985              $conversationctx = \context::instance_by_id($conversation->contextid);
 986          }
 987  
 988          $ufservice = \core_favourites\service_factory::get_service_for_user_context($userctx);
 989  
 990          if ($favourite = $ufservice->get_favourite('core_message', 'message_conversations', $conversationid, $conversationctx)) {
 991              return $favourite;
 992          } else {
 993              return $ufservice->create_favourite('core_message', 'message_conversations', $conversationid, $conversationctx);
 994          }
 995      }
 996  
 997      /**
 998       * Unset a conversation as a favourite for the given user.
 999       *
1000       * @param int $conversationid the id of the conversation to unset as a favourite.
1001       * @param int $userid the id to whom the favourite belongs.
1002       * @throws \moodle_exception if the favourite does not exist for the user.
1003       */
1004      public static function unset_favourite_conversation(int $conversationid, int $userid) {
1005          global $DB;
1006  
1007          // Get the context for this conversation.
1008          $conversation = $DB->get_record('message_conversations', ['id' => $conversationid]);
1009          $userctx = \context_user::instance($userid);
1010          if (empty($conversation->contextid)) {
1011              // When the conversation hasn't any contextid value defined, the favourite will be added to the user context.
1012              $conversationctx = $userctx;
1013          } else {
1014              // If the contextid is defined, the favourite will be added there.
1015              $conversationctx = \context::instance_by_id($conversation->contextid);
1016          }
1017  
1018          $ufservice = \core_favourites\service_factory::get_service_for_user_context($userctx);
1019          $ufservice->delete_favourite('core_message', 'message_conversations', $conversationid, $conversationctx);
1020      }
1021  
1022      /**
1023       * @deprecated since 3.6
1024       */
1025      public static function get_contacts() {
1026          throw new \coding_exception('\core_message\api::get_contacts has been removed.');
1027      }
1028  
1029      /**
1030       * Get the contacts for a given user.
1031       *
1032       * @param int $userid
1033       * @param int $limitfrom
1034       * @param int $limitnum
1035       * @return array An array of contacts
1036       */
1037      public static function get_user_contacts(int $userid, int $limitfrom = 0, int $limitnum = 0) {
1038          global $DB;
1039  
1040          $sql = "SELECT *
1041                    FROM {message_contacts} mc
1042                   WHERE mc.userid = ? OR mc.contactid = ?
1043                ORDER BY timecreated DESC, id ASC";
1044          if ($contacts = $DB->get_records_sql($sql, [$userid, $userid], $limitfrom, $limitnum)) {
1045              $userids = [];
1046              foreach ($contacts as $contact) {
1047                  if ($contact->userid == $userid) {
1048                      $userids[] = $contact->contactid;
1049                  } else {
1050                      $userids[] = $contact->userid;
1051                  }
1052              }
1053              return helper::get_member_info($userid, $userids);
1054          }
1055  
1056          return [];
1057      }
1058  
1059      /**
1060       * Returns the contacts count.
1061       *
1062       * @param int $userid The user id
1063       * @return array
1064       */
1065      public static function count_contacts(int $userid) : int {
1066          global $DB;
1067  
1068          $sql = "SELECT COUNT(id)
1069                    FROM {message_contacts}
1070                   WHERE userid = ? OR contactid = ?";
1071          return $DB->count_records_sql($sql, [$userid, $userid]);
1072      }
1073  
1074      /**
1075       * @deprecated since 3.10
1076       */
1077      public static function get_contacts_with_unread_message_count() {
1078          throw new \coding_exception('\core_message\api::get_contacts_with_unread_message_count has been removed.');
1079      }
1080  
1081      /**
1082       * @deprecated since 3.10
1083       */
1084      public static function get_non_contacts_with_unread_message_count() {
1085          throw new \coding_exception('\core_message\api::get_non_contacts_with_unread_message_count has been removed.');
1086      }
1087  
1088      /**
1089       * @deprecated since 3.6
1090       */
1091      public static function get_messages() {
1092          throw new \coding_exception('\core_message\api::get_messages has been removed.');
1093      }
1094  
1095      /**
1096       * Returns the messages for the defined conversation.
1097       *
1098       * @param  int $userid The current user.
1099       * @param  int $convid The conversation where the messages belong. Could be an object or just the id.
1100       * @param  int $limitfrom Return a subset of records, starting at this point (optional).
1101       * @param  int $limitnum Return a subset comprising this many records in total (optional, required if $limitfrom is set).
1102       * @param  string $sort The column name to order by including optionally direction.
1103       * @param  int $timefrom The time from the message being sent.
1104       * @param  int $timeto The time up until the message being sent.
1105       * @return array of messages
1106       */
1107      public static function get_conversation_messages(int $userid, int $convid, int $limitfrom = 0, int $limitnum = 0,
1108          string $sort = 'timecreated ASC', int $timefrom = 0, int $timeto = 0) : array {
1109  
1110          if (!empty($timefrom)) {
1111              // Check the cache to see if we even need to do a DB query.
1112              $cache = \cache::make('core', 'message_time_last_message_between_users');
1113              $key = helper::get_last_message_time_created_cache_key($convid);
1114              $lastcreated = $cache->get($key);
1115  
1116              // The last known message time is earlier than the one being requested so we can
1117              // just return an empty result set rather than having to query the DB.
1118              if ($lastcreated && $lastcreated < $timefrom) {
1119                  return helper::format_conversation_messages($userid, $convid, []);
1120              }
1121          }
1122  
1123          $messages = helper::get_conversation_messages($userid, $convid, 0, $limitfrom, $limitnum, $sort, $timefrom, $timeto);
1124          return helper::format_conversation_messages($userid, $convid, $messages);
1125      }
1126  
1127      /**
1128       * @deprecated since 3.6
1129       */
1130      public static function get_most_recent_message() {
1131          throw new \coding_exception('\core_message\api::get_most_recent_message has been removed.');
1132      }
1133  
1134      /**
1135       * Returns the most recent message in a conversation.
1136       *
1137       * @param int $convid The conversation identifier.
1138       * @param int $currentuserid The current user identifier.
1139       * @return \stdClass|null The most recent message.
1140       */
1141      public static function get_most_recent_conversation_message(int $convid, int $currentuserid = 0) {
1142          global $USER;
1143  
1144          if (empty($currentuserid)) {
1145              $currentuserid = $USER->id;
1146          }
1147  
1148          if ($messages = helper::get_conversation_messages($currentuserid, $convid, 0, 0, 1, 'timecreated DESC')) {
1149              $convmessages = helper::format_conversation_messages($currentuserid, $convid, $messages);
1150              return array_pop($convmessages['messages']);
1151          }
1152  
1153          return null;
1154      }
1155  
1156      /**
1157       * @deprecated since 3.6
1158       */
1159      public static function get_profile() {
1160          throw new \coding_exception('\core_message\api::get_profile has been removed.');
1161      }
1162  
1163      /**
1164       * Checks if a user can delete messages they have either received or sent.
1165       *
1166       * @param int $userid The user id of who we want to delete the messages for (this may be done by the admin
1167       *  but will still seem as if it was by the user)
1168       * @param int $conversationid The id of the conversation
1169       * @return bool Returns true if a user can delete the conversation, false otherwise.
1170       */
1171      public static function can_delete_conversation(int $userid, int $conversationid = null) : bool {
1172          global $USER;
1173  
1174          if (is_null($conversationid)) {
1175              debugging('\core_message\api::can_delete_conversation() now expects a \'conversationid\' to be passed.',
1176                  DEBUG_DEVELOPER);
1177              return false;
1178          }
1179  
1180          $systemcontext = \context_system::instance();
1181  
1182          if (has_capability('moodle/site:deleteanymessage', $systemcontext)) {
1183              return true;
1184          }
1185  
1186          if (!self::is_user_in_conversation($userid, $conversationid)) {
1187              return false;
1188          }
1189  
1190          if (has_capability('moodle/site:deleteownmessage', $systemcontext) &&
1191                  $USER->id == $userid) {
1192              return true;
1193          }
1194  
1195          return false;
1196      }
1197  
1198      /**
1199       * @deprecated since 3.6
1200       */
1201      public static function delete_conversation() {
1202          throw new \coding_exception('\core_message\api::delete_conversation() is deprecated, please use ' .
1203              '\core_message\api::delete_conversation_by_id() instead.');
1204      }
1205  
1206      /**
1207       * Deletes a conversation for a specified user.
1208       *
1209       * This function does not verify any permissions.
1210       *
1211       * @param int $userid The user id of who we want to delete the messages for (this may be done by the admin
1212       *  but will still seem as if it was by the user)
1213       * @param int $conversationid The id of the other user in the conversation
1214       */
1215      public static function delete_conversation_by_id(int $userid, int $conversationid) {
1216          global $DB, $USER;
1217  
1218          // Get all messages belonging to this conversation that have not already been deleted by this user.
1219          $sql = "SELECT m.*
1220                   FROM {messages} m
1221             INNER JOIN {message_conversations} mc
1222                     ON m.conversationid = mc.id
1223              LEFT JOIN {message_user_actions} mua
1224                     ON (mua.messageid = m.id AND mua.userid = ? AND mua.action = ?)
1225                  WHERE mua.id is NULL
1226                    AND mc.id = ?
1227               ORDER BY m.timecreated ASC";
1228          $messages = $DB->get_records_sql($sql, [$userid, self::MESSAGE_ACTION_DELETED, $conversationid]);
1229  
1230          // Ok, mark these as deleted.
1231          foreach ($messages as $message) {
1232              $mua = new \stdClass();
1233              $mua->userid = $userid;
1234              $mua->messageid = $message->id;
1235              $mua->action = self::MESSAGE_ACTION_DELETED;
1236              $mua->timecreated = time();
1237              $mua->id = $DB->insert_record('message_user_actions', $mua);
1238  
1239              \core\event\message_deleted::create_from_ids($userid, $USER->id,
1240                  $message->id, $mua->id)->trigger();
1241          }
1242      }
1243  
1244      /**
1245       * Returns the count of unread conversations (collection of messages from a single user) for
1246       * the given user.
1247       *
1248       * @param \stdClass $user the user who's conversations should be counted
1249       * @return int the count of the user's unread conversations
1250       */
1251      public static function count_unread_conversations($user = null) {
1252          global $USER, $DB;
1253  
1254          if (empty($user)) {
1255              $user = $USER;
1256          }
1257  
1258          $sql = "SELECT COUNT(DISTINCT(m.conversationid))
1259                    FROM {messages} m
1260              INNER JOIN {message_conversations} mc
1261                      ON m.conversationid = mc.id
1262              INNER JOIN {message_conversation_members} mcm
1263                      ON mc.id = mcm.conversationid
1264               LEFT JOIN {message_user_actions} mua
1265                      ON (mua.messageid = m.id AND mua.userid = ? AND mua.action = ?)
1266                   WHERE mcm.userid = ?
1267                     AND mc.enabled = ?
1268                     AND mcm.userid != m.useridfrom
1269                     AND mua.id is NULL";
1270  
1271          return $DB->count_records_sql($sql, [$user->id, self::MESSAGE_ACTION_READ, $user->id,
1272              self::MESSAGE_CONVERSATION_ENABLED]);
1273      }
1274  
1275      /**
1276       * Checks if a user can mark all messages as read.
1277       *
1278       * @param int $userid The user id of who we want to mark the messages for
1279       * @param int $conversationid The id of the conversation
1280       * @return bool true if user is permitted, false otherwise
1281       * @since 3.6
1282       */
1283      public static function can_mark_all_messages_as_read(int $userid, int $conversationid) : bool {
1284          global $USER;
1285  
1286          $systemcontext = \context_system::instance();
1287  
1288          if (has_capability('moodle/site:readallmessages', $systemcontext)) {
1289              return true;
1290          }
1291  
1292          if (!self::is_user_in_conversation($userid, $conversationid)) {
1293              return false;
1294          }
1295  
1296          if ($USER->id == $userid) {
1297              return true;
1298          }
1299  
1300          return false;
1301      }
1302  
1303      /**
1304       * Returns the count of conversations (collection of messages from a single user) for
1305       * the given user.
1306       *
1307       * @param int $userid The user whose conversations should be counted.
1308       * @return array the array of conversations counts, indexed by type.
1309       */
1310      public static function get_conversation_counts(int $userid) : array {
1311          global $DB;
1312          self::lazy_create_self_conversation($userid);
1313  
1314          // Some restrictions we need to be aware of:
1315          // - Individual conversations containing soft-deleted user must be counted.
1316          // - Individual conversations containing only deleted messages must NOT be counted.
1317          // - Self-conversations with 0 messages must be counted.
1318          // - Self-conversations containing only deleted messages must NOT be counted.
1319          // - Group conversations with 0 messages must be counted.
1320          // - Linked conversations which are disabled (enabled = 0) must NOT be counted.
1321          // - Any type of conversation can be included in the favourites count, however, the type counts and the favourites count
1322          // are mutually exclusive; any conversations which are counted in favourites cannot be counted elsewhere.
1323  
1324          // First, ask the favourites service to give us the join SQL for favourited conversations,
1325          // so we can include favourite information in the query.
1326          $usercontext = \context_user::instance($userid);
1327          $favservice = \core_favourites\service_factory::get_service_for_user_context($usercontext);
1328          list($favsql, $favparams) = $favservice->get_join_sql_by_type('core_message', 'message_conversations', 'fav', 'mc.id');
1329  
1330          $sql = "SELECT mc.type, fav.itemtype, COUNT(DISTINCT mc.id) as count, MAX(maxvisibleconvmessage.convid) as maxconvidmessage
1331                    FROM {message_conversations} mc
1332              INNER JOIN {message_conversation_members} mcm
1333                      ON mcm.conversationid = mc.id
1334               LEFT JOIN (
1335                                SELECT m.conversationid as convid, MAX(m.timecreated) as maxtime
1336                                  FROM {messages} m
1337                            INNER JOIN {message_conversation_members} mcm
1338                                    ON mcm.conversationid = m.conversationid
1339                             LEFT JOIN {message_user_actions} mua
1340                                    ON (mua.messageid = m.id AND mua.userid = :userid AND mua.action = :action)
1341                                 WHERE mua.id is NULL
1342                                   AND mcm.userid = :userid2
1343                              GROUP BY m.conversationid
1344                         ) maxvisibleconvmessage
1345                      ON maxvisibleconvmessage.convid = mc.id
1346                 $favsql
1347                   WHERE mcm.userid = :userid3
1348                     AND mc.enabled = :enabled
1349                     AND (
1350                            (mc.type = :individualtype AND maxvisibleconvmessage.convid IS NOT NULL) OR
1351                            (mc.type = :grouptype) OR
1352                            (mc.type = :selftype)
1353                         )
1354                GROUP BY mc.type, fav.itemtype
1355                ORDER BY mc.type ASC";
1356  
1357          $params = [
1358              'userid' => $userid,
1359              'userid2' => $userid,
1360              'userid3' => $userid,
1361              'userid4' => $userid,
1362              'userid5' => $userid,
1363              'action' => self::MESSAGE_ACTION_DELETED,
1364              'enabled' => self::MESSAGE_CONVERSATION_ENABLED,
1365              'individualtype' => self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL,
1366              'grouptype' => self::MESSAGE_CONVERSATION_TYPE_GROUP,
1367              'selftype' => self::MESSAGE_CONVERSATION_TYPE_SELF,
1368          ] + $favparams;
1369  
1370          // Assemble the return array.
1371          $counts = [
1372              'favourites' => 0,
1373              'types' => [
1374                  self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL => 0,
1375                  self::MESSAGE_CONVERSATION_TYPE_GROUP => 0,
1376                  self::MESSAGE_CONVERSATION_TYPE_SELF => 0
1377              ]
1378          ];
1379  
1380          // For the self-conversations, get the total number of messages (to know if the conversation is new or it has been emptied).
1381          $selfmessagessql = "SELECT COUNT(m.id)
1382                                FROM {messages} m
1383                          INNER JOIN {message_conversations} mc
1384                                  ON mc.id = m.conversationid
1385                               WHERE mc.type = ? AND convhash = ?";
1386          $selfmessagestotal = $DB->count_records_sql(
1387              $selfmessagessql,
1388              [self::MESSAGE_CONVERSATION_TYPE_SELF, helper::get_conversation_hash([$userid])]
1389          );
1390  
1391          $countsrs = $DB->get_recordset_sql($sql, $params);
1392          foreach ($countsrs as $key => $val) {
1393              // Empty self-conversations with deleted messages should be excluded.
1394              if ($val->type == self::MESSAGE_CONVERSATION_TYPE_SELF && empty($val->maxconvidmessage) && $selfmessagestotal > 0) {
1395                  continue;
1396              }
1397              if (!empty($val->itemtype)) {
1398                  $counts['favourites'] += $val->count;
1399                  continue;
1400              }
1401              $counts['types'][$val->type] = $val->count;
1402          }
1403          $countsrs->close();
1404  
1405          return $counts;
1406      }
1407  
1408      /**
1409       * Marks all messages being sent to a user in a particular conversation.
1410       *
1411       * If $conversationdid is null then it marks all messages as read sent to $userid.
1412       *
1413       * @param int $userid
1414       * @param int|null $conversationid The conversation the messages belong to mark as read, if null mark all
1415       */
1416      public static function mark_all_messages_as_read($userid, $conversationid = null) {
1417          global $DB;
1418  
1419          $messagesql = "SELECT m.*
1420                           FROM {messages} m
1421                     INNER JOIN {message_conversations} mc
1422                             ON mc.id = m.conversationid
1423                     INNER JOIN {message_conversation_members} mcm
1424                             ON mcm.conversationid = mc.id
1425                      LEFT JOIN {message_user_actions} mua
1426                             ON (mua.messageid = m.id AND mua.userid = ? AND mua.action = ?)
1427                          WHERE mua.id is NULL
1428                            AND mcm.userid = ?
1429                            AND m.useridfrom != ?";
1430          $messageparams = [];
1431          $messageparams[] = $userid;
1432          $messageparams[] = self::MESSAGE_ACTION_READ;
1433          $messageparams[] = $userid;
1434          $messageparams[] = $userid;
1435          if (!is_null($conversationid)) {
1436              $messagesql .= " AND mc.id = ?";
1437              $messageparams[] = $conversationid;
1438          }
1439  
1440          $messages = $DB->get_recordset_sql($messagesql, $messageparams);
1441          foreach ($messages as $message) {
1442              self::mark_message_as_read($userid, $message);
1443          }
1444          $messages->close();
1445      }
1446  
1447      /**
1448       * Marks all notifications being sent from one user to another user as read.
1449       *
1450       * If the from user is null then it marks all notifications as read sent to the to user.
1451       *
1452       * @param int $touserid the id of the message recipient
1453       * @param int|null $fromuserid the id of the message sender, null if all messages
1454       * @param int|null $timecreatedto mark notifications created before this time as read
1455       * @return void
1456       */
1457      public static function mark_all_notifications_as_read($touserid, $fromuserid = null, $timecreatedto = null) {
1458          global $DB;
1459  
1460          $notificationsql = "SELECT n.*
1461                                FROM {notifications} n
1462                               WHERE useridto = ?
1463                                 AND timeread is NULL";
1464          $notificationsparams = [$touserid];
1465          if (!empty($fromuserid)) {
1466              $notificationsql .= " AND useridfrom = ?";
1467              $notificationsparams[] = $fromuserid;
1468          }
1469          if (!empty($timecreatedto)) {
1470              $notificationsql .= " AND timecreated <= ?";
1471              $notificationsparams[] = $timecreatedto;
1472          }
1473  
1474          $notifications = $DB->get_recordset_sql($notificationsql, $notificationsparams);
1475          foreach ($notifications as $notification) {
1476              self::mark_notification_as_read($notification);
1477          }
1478          $notifications->close();
1479      }
1480  
1481      /**
1482       * @deprecated since 3.5
1483       */
1484      public static function mark_all_read_for_user() {
1485          throw new \coding_exception('\core_message\api::mark_all_read_for_user has been removed. Please either use ' .
1486              '\core_message\api::mark_all_notifications_as_read or \core_message\api::mark_all_messages_as_read');
1487      }
1488  
1489      /**
1490       * Returns message preferences.
1491       *
1492       * @param array $processors
1493       * @param array $providers
1494       * @param \stdClass $user
1495       * @return \stdClass
1496       * @since 3.2
1497       */
1498      public static function get_all_message_preferences($processors, $providers, $user) {
1499          $preferences = helper::get_providers_preferences($providers, $user->id);
1500          $preferences->userdefaultemail = $user->email; // May be displayed by the email processor.
1501  
1502          // For every processors put its options on the form (need to get function from processor's lib.php).
1503          foreach ($processors as $processor) {
1504              $processor->object->load_data($preferences, $user->id);
1505          }
1506  
1507          // Load general messaging preferences.
1508          $preferences->blocknoncontacts = self::get_user_privacy_messaging_preference($user->id);
1509          $preferences->mailformat = $user->mailformat;
1510          $preferences->mailcharset = get_user_preferences('mailcharset', '', $user->id);
1511  
1512          return $preferences;
1513      }
1514  
1515      /**
1516       * Count the number of users blocked by a user.
1517       *
1518       * @param \stdClass $user The user object
1519       * @return int the number of blocked users
1520       */
1521      public static function count_blocked_users($user = null) {
1522          global $USER, $DB;
1523  
1524          if (empty($user)) {
1525              $user = $USER;
1526          }
1527  
1528          $sql = "SELECT count(mub.id)
1529                    FROM {message_users_blocked} mub
1530                   WHERE mub.userid = :userid";
1531          return $DB->count_records_sql($sql, array('userid' => $user->id));
1532      }
1533  
1534      /**
1535       * @deprecated since 3.8
1536       */
1537      public static function can_post_message() {
1538          throw new \coding_exception(
1539              '\core_message\api::can_post_message is deprecated and no longer used, ' .
1540              'please use \core_message\api::can_send_message instead.'
1541          );
1542      }
1543  
1544      /**
1545       * Determines if a user is permitted to send another user a private message.
1546       *
1547       * @param int $recipientid The recipient user id.
1548       * @param int $senderid The sender user id.
1549       * @param bool $evenifblocked This lets the user know, that even if the recipient has blocked the user
1550       *        the user is still able to send a message.
1551       * @return bool true if user is permitted, false otherwise.
1552       */
1553      public static function can_send_message(int $recipientid, int $senderid, bool $evenifblocked = false) : bool {
1554          $systemcontext = \context_system::instance();
1555  
1556          if (!has_capability('moodle/site:sendmessage', $systemcontext, $senderid)) {
1557              return false;
1558          }
1559  
1560          if (has_capability('moodle/site:readallmessages', $systemcontext, $senderid)) {
1561              return true;
1562          }
1563  
1564          // Check if the recipient can be messaged by the sender.
1565          return self::can_contact_user($recipientid, $senderid, $evenifblocked);
1566      }
1567  
1568      /**
1569       * Determines if a user is permitted to send a message to a given conversation.
1570       * If no sender is provided then it defaults to the logged in user.
1571       *
1572       * @param int $userid the id of the user on which the checks will be applied.
1573       * @param int $conversationid the id of the conversation we wish to check.
1574       * @return bool true if the user can send a message to the conversation, false otherwise.
1575       * @throws \moodle_exception
1576       */
1577      public static function can_send_message_to_conversation(int $userid, int $conversationid) : bool {
1578          global $DB;
1579  
1580          $systemcontext = \context_system::instance();
1581          if (!has_capability('moodle/site:sendmessage', $systemcontext, $userid)) {
1582              return false;
1583          }
1584  
1585          if (!self::is_user_in_conversation($userid, $conversationid)) {
1586              return false;
1587          }
1588  
1589          // User can post messages and is in the conversation, but we need to check the conversation type to
1590          // know whether or not to check the user privacy settings via can_contact_user().
1591          $conversation = $DB->get_record('message_conversations', ['id' => $conversationid], '*', MUST_EXIST);
1592          if ($conversation->type == self::MESSAGE_CONVERSATION_TYPE_GROUP ||
1593              $conversation->type == self::MESSAGE_CONVERSATION_TYPE_SELF) {
1594              return true;
1595          } else if ($conversation->type == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL) {
1596              // Get the other user in the conversation.
1597              $members = self::get_conversation_members($userid, $conversationid);
1598              $otheruser = array_filter($members, function($member) use($userid) {
1599                  return $member->id != $userid;
1600              });
1601              $otheruser = reset($otheruser);
1602  
1603              return self::can_contact_user($otheruser->id, $userid);
1604          } else {
1605              throw new \moodle_exception("Invalid conversation type '$conversation->type'.");
1606          }
1607      }
1608  
1609      /**
1610       * Send a message from a user to a conversation.
1611       *
1612       * This method will create the basic eventdata and delegate to message creation to message_send.
1613       * The message_send() method is responsible for event data that is specific to each recipient.
1614       *
1615       * @param int $userid the sender id.
1616       * @param int $conversationid the conversation id.
1617       * @param string $message the message to send.
1618       * @param int $format the format of the message to send.
1619       * @return \stdClass the message created.
1620       * @throws \coding_exception
1621       * @throws \moodle_exception if the user is not permitted to send a message to the conversation.
1622       */
1623      public static function send_message_to_conversation(int $userid, int $conversationid, string $message,
1624                                                          int $format) : \stdClass {
1625          global $DB, $PAGE;
1626  
1627          if (!self::can_send_message_to_conversation($userid, $conversationid)) {
1628              throw new \moodle_exception("User $userid cannot send a message to conversation $conversationid");
1629          }
1630  
1631          $eventdata = new \core\message\message();
1632          $eventdata->courseid         = 1;
1633          $eventdata->component        = 'moodle';
1634          $eventdata->name             = 'instantmessage';
1635          $eventdata->userfrom         = \core_user::get_user($userid);
1636          $eventdata->convid           = $conversationid;
1637  
1638          if ($format == FORMAT_HTML) {
1639              $eventdata->fullmessagehtml  = $message;
1640              // Some message processors may revert to sending plain text even if html is supplied,
1641              // so we keep both plain and html versions if we're intending to send html.
1642              $eventdata->fullmessage = html_to_text($eventdata->fullmessagehtml);
1643          } else {
1644              $eventdata->fullmessage      = $message;
1645              $eventdata->fullmessagehtml  = '';
1646          }
1647  
1648          $eventdata->fullmessageformat = $format;
1649          $eventdata->smallmessage = $message; // Store the message unfiltered. Clean up on output.
1650  
1651          $eventdata->timecreated     = time();
1652          $eventdata->notification    = 0;
1653          // Custom data for event.
1654          $customdata = [
1655              'actionbuttons' => [
1656                  'send' => get_string('send', 'message'),
1657              ],
1658              'placeholders' => [
1659                  'send' => get_string('writeamessage', 'message'),
1660              ],
1661          ];
1662  
1663          $userpicture = new \user_picture($eventdata->userfrom);
1664          $userpicture->size = 1; // Use f1 size.
1665          $userpicture = $userpicture->get_url($PAGE)->out(false);
1666  
1667          $conv = $DB->get_record('message_conversations', ['id' => $conversationid]);
1668          if ($conv->type == self::MESSAGE_CONVERSATION_TYPE_GROUP) {
1669              $convextrafields = self::get_linked_conversation_extra_fields([$conv]);
1670              // Conversation images.
1671              $customdata['notificationsendericonurl'] = $userpicture;
1672              $imageurl = isset($convextrafields[$conv->id]) ? $convextrafields[$conv->id]['imageurl'] : null;
1673              if ($imageurl) {
1674                  $customdata['notificationiconurl'] = $imageurl;
1675              }
1676              // Conversation name.
1677              if (is_null($conv->contextid)) {
1678                  $convcontext = \context_user::instance($userid);
1679              } else {
1680                  $convcontext = \context::instance_by_id($conv->contextid);
1681              }
1682              $customdata['conversationname'] = format_string($conv->name, true, ['context' => $convcontext]);
1683          } else if ($conv->type == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL) {
1684              $customdata['notificationiconurl'] = $userpicture;
1685          }
1686          $eventdata->customdata = $customdata;
1687  
1688          $messageid = message_send($eventdata);
1689  
1690          if (!$messageid) {
1691              throw new \moodle_exception('messageundeliveredbynotificationsettings', 'moodle');
1692          }
1693  
1694          $messagerecord = $DB->get_record('messages', ['id' => $messageid], 'id, useridfrom, fullmessage,
1695                  timecreated, fullmessagetrust');
1696          $message = (object) [
1697              'id' => $messagerecord->id,
1698              'useridfrom' => $messagerecord->useridfrom,
1699              'text' => $messagerecord->fullmessage,
1700              'timecreated' => $messagerecord->timecreated,
1701              'fullmessagetrust' => $messagerecord->fullmessagetrust
1702          ];
1703          return $message;
1704      }
1705  
1706      /**
1707       * Get the messaging preference for a user.
1708       * If the user has not any messaging privacy preference:
1709       * - When $CFG->messagingallusers = false the default user preference is MESSAGE_PRIVACY_COURSEMEMBER.
1710       * - When $CFG->messagingallusers = true the default user preference is MESSAGE_PRIVACY_SITE.
1711       *
1712       * @param  int    $userid The user identifier.
1713       * @return int    The default messaging preference.
1714       */
1715      public static function get_user_privacy_messaging_preference(int $userid) : int {
1716          global $CFG, $USER;
1717  
1718          // When $CFG->messagingallusers is enabled, default value for the messaging preference will be "Anyone on the site";
1719          // otherwise, the default value will be "My contacts and anyone in my courses".
1720          if (empty($CFG->messagingallusers)) {
1721              $defaultprefvalue = self::MESSAGE_PRIVACY_COURSEMEMBER;
1722          } else {
1723              $defaultprefvalue = self::MESSAGE_PRIVACY_SITE;
1724          }
1725          if ($userid == $USER->id) {
1726              $user = $USER;
1727          } else {
1728              $user = $userid;
1729          }
1730          $privacypreference = get_user_preferences('message_blocknoncontacts', $defaultprefvalue, $user);
1731  
1732          // When the $CFG->messagingallusers privacy setting is disabled, MESSAGE_PRIVACY_SITE is
1733          // also disabled, so it has to be replaced to MESSAGE_PRIVACY_COURSEMEMBER.
1734          if (empty($CFG->messagingallusers) && $privacypreference == self::MESSAGE_PRIVACY_SITE) {
1735              $privacypreference = self::MESSAGE_PRIVACY_COURSEMEMBER;
1736          }
1737  
1738          return $privacypreference;
1739      }
1740  
1741      /**
1742       * @deprecated since 3.6
1743       */
1744      public static function is_user_non_contact_blocked() {
1745          throw new \coding_exception('\core_message\api::is_user_non_contact_blocked() is deprecated');
1746      }
1747  
1748      /**
1749       * @deprecated since 3.6
1750       */
1751      public static function is_user_blocked() {
1752          throw new \coding_exception('\core_message\api::is_user_blocked is deprecated and should not be used.');
1753      }
1754  
1755      /**
1756       * Get specified message processor, validate corresponding plugin existence and
1757       * system configuration.
1758       *
1759       * @param string $name  Name of the processor.
1760       * @param bool $ready only return ready-to-use processors.
1761       * @return mixed $processor if processor present else empty array.
1762       * @since Moodle 3.2
1763       */
1764      public static function get_message_processor($name, $ready = false) {
1765          global $DB, $CFG;
1766  
1767          $processor = $DB->get_record('message_processors', array('name' => $name));
1768          if (empty($processor)) {
1769              // Processor not found, return.
1770              return array();
1771          }
1772  
1773          $processor = self::get_processed_processor_object($processor);
1774          if ($ready) {
1775              if ($processor->enabled && $processor->configured) {
1776                  return $processor;
1777              } else {
1778                  return array();
1779              }
1780          } else {
1781              return $processor;
1782          }
1783      }
1784  
1785      /**
1786       * Returns weather a given processor is enabled or not.
1787       * Note:- This doesn't check if the processor is configured or not.
1788       *
1789       * @param string $name Name of the processor
1790       * @return bool
1791       */
1792      public static function is_processor_enabled($name) {
1793  
1794          $cache = \cache::make('core', 'message_processors_enabled');
1795          $status = $cache->get($name);
1796  
1797          if ($status === false) {
1798              $processor = self::get_message_processor($name);
1799              if (!empty($processor)) {
1800                  $cache->set($name, $processor->enabled);
1801                  return $processor->enabled;
1802              } else {
1803                  return false;
1804              }
1805          }
1806  
1807          return $status;
1808      }
1809  
1810      /**
1811       * Set status of a processor.
1812       *
1813       * @param \stdClass $processor processor record.
1814       * @param 0|1 $enabled 0 or 1 to set the processor status.
1815       * @return bool
1816       * @since Moodle 3.2
1817       */
1818      public static function update_processor_status($processor, $enabled) {
1819          global $DB;
1820          $cache = \cache::make('core', 'message_processors_enabled');
1821          $cache->delete($processor->name);
1822          return $DB->set_field('message_processors', 'enabled', $enabled, array('id' => $processor->id));
1823      }
1824  
1825      /**
1826       * Given a processor object, loads information about it's settings and configurations.
1827       * This is not a public api, instead use @see \core_message\api::get_message_processor()
1828       * or @see \get_message_processors()
1829       *
1830       * @param \stdClass $processor processor object
1831       * @return \stdClass processed processor object
1832       * @since Moodle 3.2
1833       */
1834      public static function get_processed_processor_object(\stdClass $processor) {
1835          global $CFG;
1836  
1837          $processorfile = $CFG->dirroot. '/message/output/'.$processor->name.'/message_output_'.$processor->name.'.php';
1838          if (is_readable($processorfile)) {
1839              include_once($processorfile);
1840              $processclass = 'message_output_' . $processor->name;
1841              if (class_exists($processclass)) {
1842                  $pclass = new $processclass();
1843                  $processor->object = $pclass;
1844                  $processor->configured = 0;
1845                  if ($pclass->is_system_configured()) {
1846                      $processor->configured = 1;
1847                  }
1848                  $processor->hassettings = 0;
1849                  if (is_readable($CFG->dirroot.'/message/output/'.$processor->name.'/settings.php')) {
1850                      $processor->hassettings = 1;
1851                  }
1852                  $processor->available = 1;
1853              } else {
1854                  throw new \moodle_exception('errorcallingprocessor', 'message');
1855              }
1856          } else {
1857              $processor->available = 0;
1858          }
1859          return $processor;
1860      }
1861  
1862      /**
1863       * Retrieve users blocked by $user1
1864       *
1865       * @param int $userid The user id of the user whos blocked users we are returning
1866       * @return array the users blocked
1867       */
1868      public static function get_blocked_users($userid) {
1869          global $DB;
1870  
1871          $userfieldsapi = \core_user\fields::for_userpic()->including('lastaccess');
1872          $userfields = $userfieldsapi->get_sql('u', false, '', '', false)->selects;
1873          $blockeduserssql = "SELECT $userfields
1874                                FROM {message_users_blocked} mub
1875                          INNER JOIN {user} u
1876                                  ON u.id = mub.blockeduserid
1877                               WHERE u.deleted = 0
1878                                 AND mub.userid = ?
1879                            GROUP BY $userfields
1880                            ORDER BY u.firstname ASC";
1881          return $DB->get_records_sql($blockeduserssql, [$userid]);
1882      }
1883  
1884      /**
1885       * Mark a single message as read.
1886       *
1887       * @param int $userid The user id who marked the message as read
1888       * @param \stdClass $message The message
1889       * @param int|null $timeread The time the message was marked as read, if null will default to time()
1890       */
1891      public static function mark_message_as_read($userid, $message, $timeread = null) {
1892          global $DB;
1893  
1894          if (is_null($timeread)) {
1895              $timeread = time();
1896          }
1897  
1898          $mua = new \stdClass();
1899          $mua->userid = $userid;
1900          $mua->messageid = $message->id;
1901          $mua->action = self::MESSAGE_ACTION_READ;
1902          $mua->timecreated = $timeread;
1903          $mua->id = $DB->insert_record('message_user_actions', $mua);
1904  
1905          // Get the context for the user who received the message.
1906          $context = \context_user::instance($userid, IGNORE_MISSING);
1907          // If the user no longer exists the context value will be false, in this case use the system context.
1908          if ($context === false) {
1909              $context = \context_system::instance();
1910          }
1911  
1912          // Trigger event for reading a message.
1913          $event = \core\event\message_viewed::create(array(
1914              'objectid' => $mua->id,
1915              'userid' => $userid, // Using the user who read the message as they are the ones performing the action.
1916              'context' => $context,
1917              'relateduserid' => $message->useridfrom,
1918              'other' => array(
1919                  'messageid' => $message->id
1920              )
1921          ));
1922          $event->trigger();
1923      }
1924  
1925      /**
1926       * Mark a single notification as read.
1927       *
1928       * @param \stdClass $notification The notification
1929       * @param int|null $timeread The time the message was marked as read, if null will default to time()
1930       */
1931      public static function mark_notification_as_read($notification, $timeread = null) {
1932          global $DB;
1933  
1934          if (is_null($timeread)) {
1935              $timeread = time();
1936          }
1937  
1938          if (is_null($notification->timeread)) {
1939              $updatenotification = new \stdClass();
1940              $updatenotification->id = $notification->id;
1941              $updatenotification->timeread = $timeread;
1942  
1943              $DB->update_record('notifications', $updatenotification);
1944  
1945              // Trigger event for reading a notification.
1946              \core\event\notification_viewed::create_from_ids(
1947                  $notification->useridfrom,
1948                  $notification->useridto,
1949                  $notification->id
1950              )->trigger();
1951          }
1952      }
1953  
1954      /**
1955       * Checks if a user can delete a message.
1956       *
1957       * @param int $userid the user id of who we want to delete the message for (this may be done by the admin
1958       *  but will still seem as if it was by the user)
1959       * @param int $messageid The message id
1960       * @return bool Returns true if a user can delete the message, false otherwise.
1961       */
1962      public static function can_delete_message($userid, $messageid) {
1963          global $DB, $USER;
1964  
1965          $systemcontext = \context_system::instance();
1966  
1967          $conversationid = $DB->get_field('messages', 'conversationid', ['id' => $messageid], MUST_EXIST);
1968  
1969          if (has_capability('moodle/site:deleteanymessage', $systemcontext)) {
1970              return true;
1971          }
1972  
1973          if (!self::is_user_in_conversation($userid, $conversationid)) {
1974              return false;
1975          }
1976  
1977          if (has_capability('moodle/site:deleteownmessage', $systemcontext) &&
1978                  $USER->id == $userid) {
1979              return true;
1980          }
1981  
1982          return false;
1983      }
1984  
1985      /**
1986       * Deletes a message.
1987       *
1988       * This function does not verify any permissions.
1989       *
1990       * @param int $userid the user id of who we want to delete the message for (this may be done by the admin
1991       *  but will still seem as if it was by the user)
1992       * @param int $messageid The message id
1993       * @return bool
1994       */
1995      public static function delete_message($userid, $messageid) {
1996          global $DB, $USER;
1997  
1998          if (!$DB->record_exists('messages', ['id' => $messageid])) {
1999              return false;
2000          }
2001  
2002          // Check if the user has already deleted this message.
2003          if (!$DB->record_exists('message_user_actions', ['userid' => $userid,
2004                  'messageid' => $messageid, 'action' => self::MESSAGE_ACTION_DELETED])) {
2005              $mua = new \stdClass();
2006              $mua->userid = $userid;
2007              $mua->messageid = $messageid;
2008              $mua->action = self::MESSAGE_ACTION_DELETED;
2009              $mua->timecreated = time();
2010              $mua->id = $DB->insert_record('message_user_actions', $mua);
2011  
2012              // Trigger event for deleting a message.
2013              \core\event\message_deleted::create_from_ids($userid, $USER->id,
2014                  $messageid, $mua->id)->trigger();
2015  
2016              return true;
2017          }
2018  
2019          return false;
2020      }
2021  
2022      /**
2023       * Returns the conversation between two users.
2024       *
2025       * @param array $userids
2026       * @return int|bool The id of the conversation, false if not found
2027       */
2028      public static function get_conversation_between_users(array $userids) {
2029          global $DB;
2030  
2031          if (empty($userids)) {
2032              return false;
2033          }
2034  
2035          $hash = helper::get_conversation_hash($userids);
2036  
2037          if ($conversation = $DB->get_record('message_conversations', ['type' => self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL,
2038                  'convhash' => $hash])) {
2039              return $conversation->id;
2040          }
2041  
2042          return false;
2043      }
2044  
2045      /**
2046       * @deprecated since 3.8
2047       */
2048      public static function get_individual_conversations_between_users() {
2049          throw new \coding_exception('\core_message\api::get_individual_conversations_between_users ' .
2050              ' is deprecated and no longer used.');
2051      }
2052  
2053      /**
2054       * Returns the self conversation for a user.
2055       *
2056       * @param int $userid The user id to get the self-conversations
2057       * @return \stdClass|false The self-conversation object or false if it doesn't exist
2058       * @since Moodle 3.7
2059       */
2060      public static function get_self_conversation(int $userid) {
2061          global $DB;
2062          self::lazy_create_self_conversation($userid);
2063  
2064          $conditions = [
2065              'type' => self::MESSAGE_CONVERSATION_TYPE_SELF,
2066              'convhash' => helper::get_conversation_hash([$userid])
2067          ];
2068          return $DB->get_record('message_conversations', $conditions);
2069      }
2070  
2071      /**
2072       * @deprecated since 3.6
2073       */
2074      public static function create_conversation_between_users() {
2075          throw new \coding_exception('\core_message\api::create_conversation_between_users is deprecated, please use ' .
2076              '\core_message\api::create_conversation instead.');
2077      }
2078  
2079      /**
2080       * Creates a conversation with selected users and messages.
2081       *
2082       * @param int $type The type of conversation
2083       * @param int[] $userids The array of users to add to the conversation
2084       * @param string|null $name The name of the conversation
2085       * @param int $enabled Determines if the conversation is created enabled or disabled
2086       * @param string|null $component Defines the Moodle component which the conversation belongs to, if any
2087       * @param string|null $itemtype Defines the type of the component
2088       * @param int|null $itemid The id of the component
2089       * @param int|null $contextid The id of the context
2090       * @return \stdClass
2091       */
2092      public static function create_conversation(int $type, array $userids, string $name = null,
2093              int $enabled = self::MESSAGE_CONVERSATION_ENABLED, string $component = null,
2094              string $itemtype = null, int $itemid = null, int $contextid = null) {
2095  
2096          global $DB;
2097  
2098          $validtypes = [
2099              self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL,
2100              self::MESSAGE_CONVERSATION_TYPE_GROUP,
2101              self::MESSAGE_CONVERSATION_TYPE_SELF
2102          ];
2103  
2104          if (!in_array($type, $validtypes)) {
2105              throw new \moodle_exception('An invalid conversation type was specified.');
2106          }
2107  
2108          // Sanity check.
2109          if ($type == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL) {
2110              if (count($userids) > 2) {
2111                  throw new \moodle_exception('An individual conversation can not have more than two users.');
2112              }
2113              if ($userids[0] == $userids[1]) {
2114                  throw new \moodle_exception('Trying to create an individual conversation instead of a self conversation.');
2115              }
2116          } else if ($type == self::MESSAGE_CONVERSATION_TYPE_SELF) {
2117              if (count($userids) != 1) {
2118                  throw new \moodle_exception('A self conversation can not have more than one user.');
2119              }
2120          }
2121  
2122          $conversation = new \stdClass();
2123          $conversation->type = $type;
2124          $conversation->name = $name;
2125          $conversation->convhash = null;
2126          if ($type == self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL || $type == self::MESSAGE_CONVERSATION_TYPE_SELF) {
2127              $conversation->convhash = helper::get_conversation_hash($userids);
2128  
2129              // Don't blindly create a conversation between 2 users if there is already one present - return that.
2130              // This stops us making duplicate self and individual conversations, which is invalid.
2131              if ($record = $DB->get_record('message_conversations', ['convhash' => $conversation->convhash])) {
2132                  return $record;
2133              }
2134          }
2135          $conversation->component = $component;
2136          $conversation->itemtype = $itemtype;
2137          $conversation->itemid = $itemid;
2138          $conversation->contextid = $contextid;
2139          $conversation->enabled = $enabled;
2140          $conversation->timecreated = time();
2141          $conversation->timemodified = $conversation->timecreated;
2142          $conversation->id = $DB->insert_record('message_conversations', $conversation);
2143  
2144          // Add users to this conversation.
2145          $arrmembers = [];
2146          foreach ($userids as $userid) {
2147              $member = new \stdClass();
2148              $member->conversationid = $conversation->id;
2149              $member->userid = $userid;
2150              $member->timecreated = time();
2151              $member->id = $DB->insert_record('message_conversation_members', $member);
2152  
2153              $arrmembers[] = $member;
2154          }
2155  
2156          $conversation->members = $arrmembers;
2157  
2158          return $conversation;
2159      }
2160  
2161      /**
2162       * Checks if a user can create a group conversation.
2163       *
2164       * @param int $userid The id of the user attempting to create the conversation
2165       * @param \context $context The context they are creating the conversation from, most likely course context
2166       * @return bool
2167       */
2168      public static function can_create_group_conversation(int $userid, \context $context) : bool {
2169          global $CFG;
2170  
2171          // If we can't message at all, then we can't create a conversation.
2172          if (empty($CFG->messaging)) {
2173              return false;
2174          }
2175  
2176          // We need to check they have the capability to create the conversation.
2177          return has_capability('moodle/course:creategroupconversations', $context, $userid);
2178      }
2179  
2180      /**
2181       * Checks if a user can create a contact request.
2182       *
2183       * @param int $userid The id of the user who is creating the contact request
2184       * @param int $requesteduserid The id of the user being requested
2185       * @return bool
2186       */
2187      public static function can_create_contact(int $userid, int $requesteduserid) : bool {
2188          global $CFG;
2189  
2190          // If we can't message at all, then we can't create a contact.
2191          if (empty($CFG->messaging)) {
2192              return false;
2193          }
2194  
2195          // If we can message anyone on the site then we can create a contact.
2196          if ($CFG->messagingallusers) {
2197              return true;
2198          }
2199  
2200          // We need to check if they are in the same course.
2201          return enrol_sharing_course($userid, $requesteduserid);
2202      }
2203  
2204      /**
2205       * Handles creating a contact request.
2206       *
2207       * @param int $userid The id of the user who is creating the contact request
2208       * @param int $requesteduserid The id of the user being requested
2209       * @return \stdClass the request
2210       */
2211      public static function create_contact_request(int $userid, int $requesteduserid) : \stdClass {
2212          global $DB, $PAGE, $SITE;
2213  
2214          $request = new \stdClass();
2215          $request->userid = $userid;
2216          $request->requesteduserid = $requesteduserid;
2217          $request->timecreated = time();
2218  
2219          $request->id = $DB->insert_record('message_contact_requests', $request);
2220  
2221          // Send a notification.
2222          $userfrom = \core_user::get_user($userid);
2223          $userfromfullname = fullname($userfrom);
2224          $userto = \core_user::get_user($requesteduserid);
2225          $url = new \moodle_url('/message/index.php', ['view' => 'contactrequests']);
2226  
2227          $subject = get_string_manager()->get_string('messagecontactrequestsubject', 'core_message', (object) [
2228              'sitename' => format_string($SITE->fullname, true, ['context' => \context_system::instance()]),
2229              'user' => $userfromfullname,
2230          ], $userto->lang);
2231  
2232          $fullmessage = get_string_manager()->get_string('messagecontactrequest', 'core_message', (object) [
2233              'url' => $url->out(),
2234              'user' => $userfromfullname,
2235          ], $userto->lang);
2236  
2237          $message = new \core\message\message();
2238          $message->courseid = SITEID;
2239          $message->component = 'moodle';
2240          $message->name = 'messagecontactrequests';
2241          $message->notification = 1;
2242          $message->userfrom = $userfrom;
2243          $message->userto = $userto;
2244          $message->subject = $subject;
2245          $message->fullmessage = text_to_html($fullmessage);
2246          $message->fullmessageformat = FORMAT_HTML;
2247          $message->fullmessagehtml = $fullmessage;
2248          $message->smallmessage = '';
2249          $message->contexturl = $url->out(false);
2250          $userpicture = new \user_picture($userfrom);
2251          $userpicture->size = 1; // Use f1 size.
2252          $userpicture->includetoken = $userto->id; // Generate an out-of-session token for the user receiving the message.
2253          $message->customdata = [
2254              'notificationiconurl' => $userpicture->get_url($PAGE)->out(false),
2255              'actionbuttons' => [
2256                  'accept' => get_string_manager()->get_string('accept', 'moodle', null, $userto->lang),
2257                  'reject' => get_string_manager()->get_string('reject', 'moodle', null, $userto->lang),
2258              ],
2259          ];
2260  
2261          message_send($message);
2262  
2263          return $request;
2264      }
2265  
2266  
2267      /**
2268       * Handles confirming a contact request.
2269       *
2270       * @param int $userid The id of the user who created the contact request
2271       * @param int $requesteduserid The id of the user confirming the request
2272       */
2273      public static function confirm_contact_request(int $userid, int $requesteduserid) {
2274          global $DB;
2275  
2276          if ($request = $DB->get_record('message_contact_requests', ['userid' => $userid,
2277                  'requesteduserid' => $requesteduserid])) {
2278              self::add_contact($userid, $requesteduserid);
2279  
2280              $DB->delete_records('message_contact_requests', ['id' => $request->id]);
2281          }
2282      }
2283  
2284      /**
2285       * Handles declining a contact request.
2286       *
2287       * @param int $userid The id of the user who created the contact request
2288       * @param int $requesteduserid The id of the user declining the request
2289       */
2290      public static function decline_contact_request(int $userid, int $requesteduserid) {
2291          global $DB;
2292  
2293          if ($request = $DB->get_record('message_contact_requests', ['userid' => $userid,
2294                  'requesteduserid' => $requesteduserid])) {
2295              $DB->delete_records('message_contact_requests', ['id' => $request->id]);
2296          }
2297      }
2298  
2299      /**
2300       * Handles returning the contact requests for a user.
2301       *
2302       * This also includes the user data necessary to display information
2303       * about the user.
2304       *
2305       * It will not include blocked users.
2306       *
2307       * @param int $userid
2308       * @param int $limitfrom
2309       * @param int $limitnum
2310       * @return array The list of contact requests
2311       */
2312      public static function get_contact_requests(int $userid, int $limitfrom = 0, int $limitnum = 0) : array {
2313          global $DB;
2314  
2315          $sql = "SELECT mcr.userid
2316                    FROM {message_contact_requests} mcr
2317               LEFT JOIN {message_users_blocked} mub
2318                      ON (mub.userid = ? AND mub.blockeduserid = mcr.userid)
2319                   WHERE mcr.requesteduserid = ?
2320                     AND mub.id is NULL
2321                ORDER BY mcr.timecreated ASC";
2322          if ($contactrequests = $DB->get_records_sql($sql, [$userid, $userid], $limitfrom, $limitnum)) {
2323              $userids = array_keys($contactrequests);
2324              return helper::get_member_info($userid, $userids);
2325          }
2326  
2327          return [];
2328      }
2329  
2330      /**
2331       * Returns the number of contact requests the user has received.
2332       *
2333       * @param int $userid The ID of the user we want to return the number of received contact requests for
2334       * @return int The count
2335       */
2336      public static function get_received_contact_requests_count(int $userid) : int {
2337          global $DB;
2338          $sql = "SELECT COUNT(mcr.id)
2339                    FROM {message_contact_requests} mcr
2340               LEFT JOIN {message_users_blocked} mub
2341                      ON mub.userid = mcr.requesteduserid AND mub.blockeduserid = mcr.userid
2342                   WHERE mcr.requesteduserid = :requesteduserid
2343                     AND mub.id IS NULL";
2344          $params = ['requesteduserid' => $userid];
2345          return $DB->count_records_sql($sql, $params);
2346      }
2347  
2348      /**
2349       * Handles adding a contact.
2350       *
2351       * @param int $userid The id of the user who requested to be a contact
2352       * @param int $contactid The id of the contact
2353       */
2354      public static function add_contact(int $userid, int $contactid) {
2355          global $DB;
2356  
2357          $messagecontact = new \stdClass();
2358          $messagecontact->userid = $userid;
2359          $messagecontact->contactid = $contactid;
2360          $messagecontact->timecreated = time();
2361          $messagecontact->id = $DB->insert_record('message_contacts', $messagecontact);
2362  
2363          $eventparams = [
2364              'objectid' => $messagecontact->id,
2365              'userid' => $userid,
2366              'relateduserid' => $contactid,
2367              'context' => \context_user::instance($userid)
2368          ];
2369          $event = \core\event\message_contact_added::create($eventparams);
2370          $event->add_record_snapshot('message_contacts', $messagecontact);
2371          $event->trigger();
2372      }
2373  
2374      /**
2375       * Handles removing a contact.
2376       *
2377       * @param int $userid The id of the user who is removing a user as a contact
2378       * @param int $contactid The id of the user to be removed as a contact
2379       */
2380      public static function remove_contact(int $userid, int $contactid) {
2381          global $DB;
2382  
2383          if ($contact = self::get_contact($userid, $contactid)) {
2384              $DB->delete_records('message_contacts', ['id' => $contact->id]);
2385  
2386              $event = \core\event\message_contact_removed::create(array(
2387                  'objectid' => $contact->id,
2388                  'userid' => $userid,
2389                  'relateduserid' => $contactid,
2390                  'context' => \context_user::instance($userid)
2391              ));
2392              $event->add_record_snapshot('message_contacts', $contact);
2393              $event->trigger();
2394          }
2395      }
2396  
2397      /**
2398       * Handles blocking a user.
2399       *
2400       * @param int $userid The id of the user who is blocking
2401       * @param int $usertoblockid The id of the user being blocked
2402       */
2403      public static function block_user(int $userid, int $usertoblockid) {
2404          global $DB;
2405  
2406          $blocked = new \stdClass();
2407          $blocked->userid = $userid;
2408          $blocked->blockeduserid = $usertoblockid;
2409          $blocked->timecreated = time();
2410          $blocked->id = $DB->insert_record('message_users_blocked', $blocked);
2411  
2412          // Trigger event for blocking a contact.
2413          $event = \core\event\message_user_blocked::create(array(
2414              'objectid' => $blocked->id,
2415              'userid' => $userid,
2416              'relateduserid' => $usertoblockid,
2417              'context' => \context_user::instance($userid)
2418          ));
2419          $event->add_record_snapshot('message_users_blocked', $blocked);
2420          $event->trigger();
2421      }
2422  
2423      /**
2424       * Handles unblocking a user.
2425       *
2426       * @param int $userid The id of the user who is unblocking
2427       * @param int $usertounblockid The id of the user being unblocked
2428       */
2429      public static function unblock_user(int $userid, int $usertounblockid) {
2430          global $DB;
2431  
2432          if ($blockeduser = $DB->get_record('message_users_blocked',
2433                  ['userid' => $userid, 'blockeduserid' => $usertounblockid])) {
2434              $DB->delete_records('message_users_blocked', ['id' => $blockeduser->id]);
2435  
2436              // Trigger event for unblocking a contact.
2437              $event = \core\event\message_user_unblocked::create(array(
2438                  'objectid' => $blockeduser->id,
2439                  'userid' => $userid,
2440                  'relateduserid' => $usertounblockid,
2441                  'context' => \context_user::instance($userid)
2442              ));
2443              $event->add_record_snapshot('message_users_blocked', $blockeduser);
2444              $event->trigger();
2445          }
2446      }
2447  
2448      /**
2449       * Checks if users are already contacts.
2450       *
2451       * @param int $userid The id of one of the users
2452       * @param int $contactid The id of the other user
2453       * @return bool Returns true if they are a contact, false otherwise
2454       */
2455      public static function is_contact(int $userid, int $contactid) : bool {
2456          global $DB;
2457  
2458          $sql = "SELECT id
2459                    FROM {message_contacts} mc
2460                   WHERE (mc.userid = ? AND mc.contactid = ?)
2461                      OR (mc.userid = ? AND mc.contactid = ?)";
2462          return $DB->record_exists_sql($sql, [$userid, $contactid, $contactid, $userid]);
2463      }
2464  
2465      /**
2466       * Returns the row in the database table message_contacts that represents the contact between two people.
2467       *
2468       * @param int $userid The id of one of the users
2469       * @param int $contactid The id of the other user
2470       * @return mixed A fieldset object containing the record, false otherwise
2471       */
2472      public static function get_contact(int $userid, int $contactid) {
2473          global $DB;
2474  
2475          $sql = "SELECT mc.*
2476                    FROM {message_contacts} mc
2477                   WHERE (mc.userid = ? AND mc.contactid = ?)
2478                      OR (mc.userid = ? AND mc.contactid = ?)";
2479          return $DB->get_record_sql($sql, [$userid, $contactid, $contactid, $userid]);
2480      }
2481  
2482      /**
2483       * Checks if a user is already blocked.
2484       *
2485       * @param int $userid
2486       * @param int $blockeduserid
2487       * @return bool Returns true if they are a blocked, false otherwise
2488       */
2489      public static function is_blocked(int $userid, int $blockeduserid) : bool {
2490          global $DB;
2491  
2492          return $DB->record_exists('message_users_blocked', ['userid' => $userid, 'blockeduserid' => $blockeduserid]);
2493      }
2494  
2495      /**
2496       * Get contact requests between users.
2497       *
2498       * @param int $userid The id of the user who is creating the contact request
2499       * @param int $requesteduserid The id of the user being requested
2500       * @return \stdClass[]
2501       */
2502      public static function get_contact_requests_between_users(int $userid, int $requesteduserid) : array {
2503          global $DB;
2504  
2505          $sql = "SELECT *
2506                    FROM {message_contact_requests} mcr
2507                   WHERE (mcr.userid = ? AND mcr.requesteduserid = ?)
2508                      OR (mcr.userid = ? AND mcr.requesteduserid = ?)";
2509          return $DB->get_records_sql($sql, [$userid, $requesteduserid, $requesteduserid, $userid]);
2510      }
2511  
2512      /**
2513       * Checks if a contact request already exists between users.
2514       *
2515       * @param int $userid The id of the user who is creating the contact request
2516       * @param int $requesteduserid The id of the user being requested
2517       * @return bool Returns true if a contact request exists, false otherwise
2518       */
2519      public static function does_contact_request_exist(int $userid, int $requesteduserid) : bool {
2520          global $DB;
2521  
2522          $sql = "SELECT id
2523                    FROM {message_contact_requests} mcr
2524                   WHERE (mcr.userid = ? AND mcr.requesteduserid = ?)
2525                      OR (mcr.userid = ? AND mcr.requesteduserid = ?)";
2526          return $DB->record_exists_sql($sql, [$userid, $requesteduserid, $requesteduserid, $userid]);
2527      }
2528  
2529      /**
2530       * Checks if a user is already in a conversation.
2531       *
2532       * @param int $userid The id of the user we want to check if they are in a group
2533       * @param int $conversationid The id of the conversation
2534       * @return bool Returns true if a contact request exists, false otherwise
2535       */
2536      public static function is_user_in_conversation(int $userid, int $conversationid) : bool {
2537          global $DB;
2538  
2539          return $DB->record_exists('message_conversation_members', ['conversationid' => $conversationid,
2540              'userid' => $userid]);
2541      }
2542  
2543      /**
2544       * Checks if the sender can message the recipient.
2545       *
2546       * @param int $recipientid
2547       * @param int $senderid
2548       * @param bool $evenifblocked This lets the user know, that even if the recipient has blocked the user
2549       *        the user is still able to send a message.
2550       * @return bool true if recipient hasn't blocked sender and sender can contact to recipient, false otherwise.
2551       */
2552      protected static function can_contact_user(int $recipientid, int $senderid, bool $evenifblocked = false) : bool {
2553          if (has_capability('moodle/site:messageanyuser', \context_system::instance(), $senderid) ||
2554              $recipientid == $senderid) {
2555              // The sender has the ability to contact any user across the entire site or themselves.
2556              return true;
2557          }
2558  
2559          // The initial value of $cancontact is null to indicate that a value has not been determined.
2560          $cancontact = null;
2561  
2562          if (self::is_blocked($recipientid, $senderid) || $evenifblocked) {
2563              // The recipient has specifically blocked this sender.
2564              $cancontact = false;
2565          }
2566  
2567          $sharedcourses = null;
2568          if (null === $cancontact) {
2569              // There are three user preference options:
2570              // - Site: Allow anyone not explicitly blocked to contact me;
2571              // - Course members: Allow anyone I am in a course with to contact me; and
2572              // - Contacts: Only allow my contacts to contact me.
2573              //
2574              // The Site option is only possible when the messagingallusers site setting is also enabled.
2575  
2576              $privacypreference = self::get_user_privacy_messaging_preference($recipientid);
2577              if (self::MESSAGE_PRIVACY_SITE === $privacypreference) {
2578                  // The user preference is to allow any user to contact them.
2579                  // No need to check anything else.
2580                  $cancontact = true;
2581              } else {
2582                  // This user only allows their own contacts, and possibly course peers, to contact them.
2583                  // If the users are contacts then we can avoid the more expensive shared courses check.
2584                  $cancontact = self::is_contact($senderid, $recipientid);
2585  
2586                  if (!$cancontact && self::MESSAGE_PRIVACY_COURSEMEMBER === $privacypreference) {
2587                      // The users are not contacts and the user allows course member messaging.
2588                      // Check whether these two users share any course together.
2589                      $sharedcourses = enrol_get_shared_courses($recipientid, $senderid, true);
2590                      $cancontact = (!empty($sharedcourses));
2591                  }
2592              }
2593          }
2594  
2595          if (false === $cancontact) {
2596              // At the moment the users cannot contact one another.
2597              // Check whether the messageanyuser capability applies in any of the shared courses.
2598              // This is intended to allow teachers to message students regardless of message settings.
2599  
2600              // Note: You cannot use empty($sharedcourses) here because this may be an empty array.
2601              if (null === $sharedcourses) {
2602                  $sharedcourses = enrol_get_shared_courses($recipientid, $senderid, true);
2603              }
2604  
2605              foreach ($sharedcourses as $course) {
2606                  // Note: enrol_get_shared_courses will preload any shared context.
2607                  if (has_capability('moodle/site:messageanyuser', \context_course::instance($course->id), $senderid)) {
2608                      $cancontact = true;
2609                      break;
2610                  }
2611              }
2612          }
2613  
2614          return $cancontact;
2615      }
2616  
2617      /**
2618       * Add some new members to an existing conversation.
2619       *
2620       * @param array $userids User ids array to add as members.
2621       * @param int $convid The conversation id. Must exists.
2622       * @throws \dml_missing_record_exception If convid conversation doesn't exist
2623       * @throws \dml_exception If there is a database error
2624       * @throws \moodle_exception If trying to add a member(s) to a non-group conversation
2625       */
2626      public static function add_members_to_conversation(array $userids, int $convid) {
2627          global $DB;
2628  
2629          $conversation = $DB->get_record('message_conversations', ['id' => $convid], '*', MUST_EXIST);
2630  
2631          // We can only add members to a group conversation.
2632          if ($conversation->type != self::MESSAGE_CONVERSATION_TYPE_GROUP) {
2633              throw new \moodle_exception('You can not add members to a non-group conversation.');
2634          }
2635  
2636          // Be sure we are not trying to add a non existing user to the conversation. Work only with existing users.
2637          list($useridcondition, $params) = $DB->get_in_or_equal($userids, SQL_PARAMS_NAMED);
2638          $existingusers = $DB->get_fieldset_select('user', 'id', "id $useridcondition", $params);
2639  
2640          // Be sure we are not adding a user is already member of the conversation. Take all the members.
2641          $memberuserids = array_values($DB->get_records_menu(
2642              'message_conversation_members', ['conversationid' => $convid], 'id', 'id, userid')
2643          );
2644  
2645          // Work with existing new members.
2646          $members = array();
2647          $newuserids = array_diff($existingusers, $memberuserids);
2648          foreach ($newuserids as $userid) {
2649              $member = new \stdClass();
2650              $member->conversationid = $convid;
2651              $member->userid = $userid;
2652              $member->timecreated = time();
2653              $members[] = $member;
2654          }
2655  
2656          $DB->insert_records('message_conversation_members', $members);
2657      }
2658  
2659      /**
2660       * Remove some members from an existing conversation.
2661       *
2662       * @param array $userids The user ids to remove from conversation members.
2663       * @param int $convid The conversation id. Must exists.
2664       * @throws \dml_exception
2665       * @throws \moodle_exception If trying to remove a member(s) from a non-group conversation
2666       */
2667      public static function remove_members_from_conversation(array $userids, int $convid) {
2668          global $DB;
2669  
2670          $conversation = $DB->get_record('message_conversations', ['id' => $convid], '*', MUST_EXIST);
2671  
2672          if ($conversation->type != self::MESSAGE_CONVERSATION_TYPE_GROUP) {
2673              throw new \moodle_exception('You can not remove members from a non-group conversation.');
2674          }
2675  
2676          list($useridcondition, $params) = $DB->get_in_or_equal($userids, SQL_PARAMS_NAMED);
2677          $params['convid'] = $convid;
2678  
2679          $DB->delete_records_select('message_conversation_members',
2680              "conversationid = :convid AND userid $useridcondition", $params);
2681      }
2682  
2683      /**
2684       * Count conversation members.
2685       *
2686       * @param int $convid The conversation id.
2687       * @return int Number of conversation members.
2688       * @throws \dml_exception
2689       */
2690      public static function count_conversation_members(int $convid) : int {
2691          global $DB;
2692  
2693          return $DB->count_records('message_conversation_members', ['conversationid' => $convid]);
2694      }
2695  
2696      /**
2697       * Checks whether or not a conversation area is enabled.
2698       *
2699       * @param string $component Defines the Moodle component which the area was added to.
2700       * @param string $itemtype Defines the type of the component.
2701       * @param int $itemid The id of the component.
2702       * @param int $contextid The id of the context.
2703       * @return bool Returns if a conversation area exists and is enabled, false otherwise
2704       */
2705      public static function is_conversation_area_enabled(string $component, string $itemtype, int $itemid, int $contextid) : bool {
2706          global $DB;
2707  
2708          return $DB->record_exists('message_conversations',
2709              [
2710                  'itemid' => $itemid,
2711                  'contextid' => $contextid,
2712                  'component' => $component,
2713                  'itemtype' => $itemtype,
2714                  'enabled' => self::MESSAGE_CONVERSATION_ENABLED
2715              ]
2716          );
2717      }
2718  
2719      /**
2720       * Get conversation by area.
2721       *
2722       * @param string $component Defines the Moodle component which the area was added to.
2723       * @param string $itemtype Defines the type of the component.
2724       * @param int $itemid The id of the component.
2725       * @param int $contextid The id of the context.
2726       * @return \stdClass
2727       */
2728      public static function get_conversation_by_area(string $component, string $itemtype, int $itemid, int $contextid) {
2729          global $DB;
2730  
2731          return $DB->get_record('message_conversations',
2732              [
2733                  'itemid' => $itemid,
2734                  'contextid' => $contextid,
2735                  'component' => $component,
2736                  'itemtype'  => $itemtype
2737              ]
2738          );
2739      }
2740  
2741      /**
2742       * Enable a conversation.
2743       *
2744       * @param int $conversationid The id of the conversation.
2745       * @return void
2746       */
2747      public static function enable_conversation(int $conversationid) {
2748          global $DB;
2749  
2750          $conversation = new \stdClass();
2751          $conversation->id = $conversationid;
2752          $conversation->enabled = self::MESSAGE_CONVERSATION_ENABLED;
2753          $conversation->timemodified = time();
2754          $DB->update_record('message_conversations', $conversation);
2755      }
2756  
2757      /**
2758       * Disable a conversation.
2759       *
2760       * @param int $conversationid The id of the conversation.
2761       * @return void
2762       */
2763      public static function disable_conversation(int $conversationid) {
2764          global $DB;
2765  
2766          $conversation = new \stdClass();
2767          $conversation->id = $conversationid;
2768          $conversation->enabled = self::MESSAGE_CONVERSATION_DISABLED;
2769          $conversation->timemodified = time();
2770          $DB->update_record('message_conversations', $conversation);
2771      }
2772  
2773      /**
2774       * Update the name of a conversation.
2775       *
2776       * @param int $conversationid The id of a conversation.
2777       * @param string $name The main name of the area
2778       * @return void
2779       */
2780      public static function update_conversation_name(int $conversationid, string $name) {
2781          global $DB;
2782  
2783          if ($conversation = $DB->get_record('message_conversations', array('id' => $conversationid))) {
2784              if ($name <> $conversation->name) {
2785                  $conversation->name = $name;
2786                  $conversation->timemodified = time();
2787                  $DB->update_record('message_conversations', $conversation);
2788              }
2789          }
2790      }
2791  
2792      /**
2793       * Returns a list of conversation members.
2794       *
2795       * @param int $userid The user we are returning the conversation members for, used by helper::get_member_info.
2796       * @param int $conversationid The id of the conversation
2797       * @param bool $includecontactrequests Do we want to include contact requests with this data?
2798       * @param bool $includeprivacyinfo Do we want to include privacy requests with this data?
2799       * @param int $limitfrom
2800       * @param int $limitnum
2801       * @return array
2802       */
2803      public static function get_conversation_members(int $userid, int $conversationid, bool $includecontactrequests = false,
2804                                                      bool $includeprivacyinfo = false, int $limitfrom = 0,
2805                                                      int $limitnum = 0) : array {
2806          global $DB;
2807  
2808          if ($members = $DB->get_records('message_conversation_members', ['conversationid' => $conversationid],
2809                  'timecreated ASC, id ASC', 'userid', $limitfrom, $limitnum)) {
2810              $userids = array_keys($members);
2811              $members = helper::get_member_info($userid, $userids, $includecontactrequests, $includeprivacyinfo);
2812  
2813              return $members;
2814          }
2815  
2816          return [];
2817      }
2818  
2819      /**
2820       * Get the unread counts for all conversations for the user, sorted by type, and including favourites.
2821       *
2822       * @param int $userid the id of the user whose conversations we'll check.
2823       * @return array the unread counts for each conversation, indexed by type.
2824       */
2825      public static function get_unread_conversation_counts(int $userid) : array {
2826          global $DB;
2827  
2828          // Get all conversations the user is in, and check unread.
2829          $unreadcountssql = 'SELECT conv.id, conv.type, indcounts.unreadcount
2830                                FROM {message_conversations} conv
2831                          INNER JOIN (
2832                                        SELECT m.conversationid, count(m.id) as unreadcount
2833                                          FROM {messages} m
2834                                    INNER JOIN {message_conversations} mc
2835                                            ON mc.id = m.conversationid
2836                                    INNER JOIN {message_conversation_members} mcm
2837                                            ON m.conversationid = mcm.conversationid
2838                                     LEFT JOIN {message_user_actions} mua
2839                                            ON (mua.messageid = m.id AND mua.userid = ? AND
2840                                               (mua.action = ? OR mua.action = ?))
2841                                         WHERE mcm.userid = ?
2842                                           AND m.useridfrom != ?
2843                                           AND mua.id is NULL
2844                                      GROUP BY m.conversationid
2845                                     ) indcounts
2846                                  ON indcounts.conversationid = conv.id
2847                               WHERE conv.enabled = 1';
2848  
2849          $unreadcounts = $DB->get_records_sql($unreadcountssql, [$userid, self::MESSAGE_ACTION_READ, self::MESSAGE_ACTION_DELETED,
2850              $userid, $userid]);
2851  
2852          // Get favourites, so we can track these separately.
2853          $service = \core_favourites\service_factory::get_service_for_user_context(\context_user::instance($userid));
2854          $favouriteconversations = $service->find_favourites_by_type('core_message', 'message_conversations');
2855          $favouriteconvids = array_flip(array_column($favouriteconversations, 'itemid'));
2856  
2857          // Assemble the return array.
2858          $counts = ['favourites' => 0, 'types' => [
2859              self::MESSAGE_CONVERSATION_TYPE_INDIVIDUAL => 0,
2860              self::MESSAGE_CONVERSATION_TYPE_GROUP => 0,
2861              self::MESSAGE_CONVERSATION_TYPE_SELF => 0
2862          ]];
2863          foreach ($unreadcounts as $convid => $info) {
2864              if (isset($favouriteconvids[$convid])) {
2865                  $counts['favourites']++;
2866                  continue;
2867              }
2868              $counts['types'][$info->type]++;
2869          }
2870  
2871          return $counts;
2872      }
2873  
2874      /**
2875       * Handles muting a conversation.
2876       *
2877       * @param int $userid The id of the user
2878       * @param int $conversationid The id of the conversation
2879       */
2880      public static function mute_conversation(int $userid, int $conversationid) : void {
2881          global $DB;
2882  
2883          $mutedconversation = new \stdClass();
2884          $mutedconversation->userid = $userid;
2885          $mutedconversation->conversationid = $conversationid;
2886          $mutedconversation->action = self::CONVERSATION_ACTION_MUTED;
2887          $mutedconversation->timecreated = time();
2888  
2889          $DB->insert_record('message_conversation_actions', $mutedconversation);
2890      }
2891  
2892      /**
2893       * Handles unmuting a conversation.
2894       *
2895       * @param int $userid The id of the user
2896       * @param int $conversationid The id of the conversation
2897       */
2898      public static function unmute_conversation(int $userid, int $conversationid) : void {
2899          global $DB;
2900  
2901          $DB->delete_records('message_conversation_actions',
2902              [
2903                  'userid' => $userid,
2904                  'conversationid' => $conversationid,
2905                  'action' => self::CONVERSATION_ACTION_MUTED
2906              ]
2907          );
2908      }
2909  
2910      /**
2911       * Checks whether a conversation is muted or not.
2912       *
2913       * @param int $userid The id of the user
2914       * @param int $conversationid The id of the conversation
2915       * @return bool Whether or not the conversation is muted or not
2916       */
2917      public static function is_conversation_muted(int $userid, int $conversationid) : bool {
2918          global $DB;
2919  
2920          return $DB->record_exists('message_conversation_actions',
2921              [
2922                  'userid' => $userid,
2923                  'conversationid' => $conversationid,
2924                  'action' => self::CONVERSATION_ACTION_MUTED
2925              ]
2926          );
2927      }
2928  
2929      /**
2930       * Completely removes all related data in the DB for a given conversation.
2931       *
2932       * @param int $conversationid The id of the conversation
2933       */
2934      public static function delete_all_conversation_data(int $conversationid) {
2935          global $DB;
2936  
2937          $conv = $DB->get_record('message_conversations', ['id' => $conversationid], 'id, contextid');
2938          $convcontext = !empty($conv->contextid) ? \context::instance_by_id($conv->contextid) : null;
2939  
2940          $DB->delete_records('message_conversations', ['id' => $conversationid]);
2941          $DB->delete_records('message_conversation_members', ['conversationid' => $conversationid]);
2942          $DB->delete_records('message_conversation_actions', ['conversationid' => $conversationid]);
2943  
2944          // Now, go through and delete any messages and related message actions for the conversation.
2945          if ($messages = $DB->get_records('messages', ['conversationid' => $conversationid])) {
2946              $messageids = array_keys($messages);
2947  
2948              list($insql, $inparams) = $DB->get_in_or_equal($messageids);
2949              $DB->delete_records_select('message_user_actions', "messageid $insql", $inparams);
2950  
2951              // Delete the messages now.
2952              $DB->delete_records('messages', ['conversationid' => $conversationid]);
2953          }
2954  
2955          // Delete all favourite records for all users relating to this conversation.
2956          $service = \core_favourites\service_factory::get_service_for_component('core_message');
2957          $service->delete_favourites_by_type_and_item('message_conversations', $conversationid, $convcontext);
2958      }
2959  
2960      /**
2961       * Checks if a user can delete a message for all users.
2962       *
2963       * @param int $userid the user id of who we want to delete the message for all users
2964       * @param int $messageid The message id
2965       * @return bool Returns true if a user can delete the message for all users, false otherwise.
2966       */
2967      public static function can_delete_message_for_all_users(int $userid, int $messageid) : bool {
2968          global $DB;
2969  
2970          $sql = "SELECT mc.id, mc.contextid
2971                    FROM {message_conversations} mc
2972              INNER JOIN {messages} m
2973                      ON mc.id = m.conversationid
2974                   WHERE m.id = :messageid";
2975          $conversation = $DB->get_record_sql($sql, ['messageid' => $messageid]);
2976  
2977          if (!empty($conversation->contextid)) {
2978              return has_capability('moodle/site:deleteanymessage',
2979                  \context::instance_by_id($conversation->contextid), $userid);
2980          }
2981  
2982          return has_capability('moodle/site:deleteanymessage', \context_system::instance(), $userid);
2983      }
2984      /**
2985       * Delete a message for all users.
2986       *
2987       * This function does not verify any permissions.
2988       *
2989       * @param int $messageid The message id
2990       * @return void
2991       */
2992      public static function delete_message_for_all_users(int $messageid) {
2993          global $DB, $USER;
2994  
2995          if (!$DB->record_exists('messages', ['id' => $messageid])) {
2996              return false;
2997          }
2998  
2999          // Get all members in the conversation where the message belongs.
3000          $membersql = "SELECT mcm.id, mcm.userid
3001                          FROM {message_conversation_members} mcm
3002                    INNER JOIN {messages} m
3003                            ON mcm.conversationid = m.conversationid
3004                         WHERE m.id = :messageid";
3005          $params = [
3006              'messageid' => $messageid
3007          ];
3008          $members = $DB->get_records_sql($membersql, $params);
3009          if ($members) {
3010              foreach ($members as $member) {
3011                  self::delete_message($member->userid, $messageid);
3012              }
3013          }
3014      }
3015  
3016      /**
3017       * Create a self conversation for a user, only if one doesn't already exist.
3018       *
3019       * @param int $userid the user to whom the conversation belongs.
3020       */
3021      protected static function lazy_create_self_conversation(int $userid) : void {
3022          global $DB;
3023          // Check if the self-conversation for this user exists.
3024          // If not, create and star it for the user.
3025          // Don't use the API methods here, as they in turn may rely on
3026          // lazy creation and we'll end up with recursive loops of doom.
3027          $conditions = [
3028              'type' => self::MESSAGE_CONVERSATION_TYPE_SELF,
3029              'convhash' => helper::get_conversation_hash([$userid])
3030          ];
3031          if (empty($DB->get_record('message_conversations', $conditions))) {
3032              $selfconversation = self::create_conversation(self::MESSAGE_CONVERSATION_TYPE_SELF, [$userid]);
3033              self::set_favourite_conversation($selfconversation->id, $userid);
3034          }
3035      }
3036  }