2009-05-26 02:46:09 +00:00
< ? php
// This file is part of Moodle - http://moodle.org/
//
// Moodle is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Moodle is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
2008-07-24 08:38:03 +00:00
/**
2012-01-05 12:05:02 +07:00
* Functions for interacting with the message system
2008-07-24 08:38:03 +00:00
*
2012-01-05 12:05:02 +07:00
* @ package core_message
* @ copyright 2008 Luis Rodrigues and Martin Dougiamas
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
2008-07-24 08:38:03 +00:00
*/
2010-07-25 13:35:05 +00:00
defined ( 'MOODLE_INTERNAL' ) || die ();
2016-02-26 17:47:58 +11:00
require_once ( __DIR__ . '/../message/lib.php' );
2011-05-19 17:25:39 +01:00
2008-07-24 08:38:03 +00:00
/**
2009-11-07 10:27:57 +00:00
* Called when a message provider wants to send a message .
2012-02-22 11:15:42 +13:00
* This functions checks the message recipient ' s message processor configuration then
2012-01-05 12:05:02 +07:00
* sends the message to the configured processors
2009-11-07 10:27:57 +00:00
*
2012-03-22 17:20:59 +01:00
* Required parameters of the $eventdata object :
2010-11-09 06:47:53 +00:00
* component string component name . must exist in message_providers
* name string message type name . must exist in message_providers
2011-05-27 10:07:26 +01:00
* userfrom object | int the user sending the message
* userto object | int the message recipient
2010-10-27 08:47:33 +00:00
* subject string the message subject
2012-03-22 17:20:59 +01:00
* fullmessage string the full message in a given format
* fullmessageformat int the format if the full message ( FORMAT_MOODLE , FORMAT_HTML , .. )
* fullmessagehtml string the full version ( the message processor will choose with one to use )
* smallmessage string the small version of the message
*
* Optional parameters of the $eventdata object :
* notification bool should the message be considered as a notification rather than a personal message
* contexturl string if this is a notification then you can specify a url to view the event . For example the forum post the user is being notified of .
* contexturlname string the display text for contexturl
2009-11-07 10:27:57 +00:00
*
2021-07-23 16:38:40 +01:00
* Note : processor failure will not reported as false return value in all scenarios ,
* for example when it is called while a database transaction is open ,
2014-06-10 17:21:40 +12:00
* earlier versions did not do it consistently either .
*
2012-01-05 12:05:02 +07:00
* @ category message
2016-07-20 12:40:34 +01:00
* @ param \core\message\message $eventdata information about the message ( component , userfrom , userto , ... )
2021-07-23 16:38:40 +01:00
* @ return mixed the integer ID of the new message or false if there was a problem ( with submitted data or sending the message to the message processor )
2008-07-24 08:38:03 +00:00
*/
2018-07-24 09:49:53 +08:00
function message_send ( \core\message\message $eventdata ) {
2018-11-08 18:07:00 +08:00
global $CFG , $DB , $SITE ;
2008-07-24 08:38:03 +00:00
2011-02-10 13:25:38 +08:00
//new message ID to return
$messageid = false ;
2013-11-20 15:25:17 +08:00
// Fetch default (site) preferences
$defaultpreferences = get_message_output_default_preferences ();
$preferencebase = $eventdata -> component . '_' . $eventdata -> name ;
2018-02-15 20:34:50 +01:00
// If the message provider is disabled via preferences, then don't send the message.
2013-11-20 15:25:17 +08:00
if ( ! empty ( $defaultpreferences -> { $preferencebase . '_disable' })) {
return $messageid ;
}
2014-04-04 17:53:25 +02:00
// By default a message is a notification. Only personal/private messages aren't notifications.
if ( ! isset ( $eventdata -> notification )) {
$eventdata -> notification = 1 ;
}
2018-11-12 10:05:17 +08:00
if ( ! is_object ( $eventdata -> userfrom )) {
$eventdata -> userfrom = core_user :: get_user ( $eventdata -> userfrom );
}
if ( ! $eventdata -> userfrom ) {
debugging ( 'Attempt to send msg from unknown user' , DEBUG_NORMAL );
return false ;
}
// Legacy messages (FROM a single user TO a single user) must be converted into conversation messages.
// Then, these will be passed through the conversation messages code below.
if ( ! $eventdata -> notification && ! $eventdata -> convid ) {
// If messaging is disabled at the site level, then the 'instantmessage' provider is always disabled.
// Given this is the only 'message' type message provider, we can exit now if this is the case.
// Don't waste processing time trying to work out the other conversation member, if it's an individual
// conversation, just throw a generic debugging notice and return.
if ( empty ( $CFG -> messaging ) || $eventdata -> component !== 'moodle' || $eventdata -> name !== 'instantmessage' ) {
debugging ( 'Attempt to send msg from a provider ' . $eventdata -> component . '/' . $eventdata -> name .
' that is inactive or not allowed for the user id=' . $eventdata -> userto -> id , DEBUG_NORMAL );
return false ;
2018-11-08 18:07:00 +08:00
}
2018-11-12 10:05:17 +08:00
if ( ! is_object ( $eventdata -> userto )) {
$eventdata -> userto = core_user :: get_user ( $eventdata -> userto );
}
if ( ! $eventdata -> userto ) {
debugging ( 'Attempt to send msg to unknown user' , DEBUG_NORMAL );
2018-11-08 18:07:00 +08:00
return false ;
}
2018-11-12 10:05:17 +08:00
// Verify all necessary data fields are present.
if ( ! isset ( $eventdata -> userto -> auth ) or ! isset ( $eventdata -> userto -> suspended )
or ! isset ( $eventdata -> userto -> deleted ) or ! isset ( $eventdata -> userto -> emailstop )) {
debugging ( 'Necessary properties missing in userto object, fetching full record' , DEBUG_DEVELOPER );
$eventdata -> userto = core_user :: get_user ( $eventdata -> userto -> id );
}
$usertoisrealuser = ( core_user :: is_real_user ( $eventdata -> userto -> id ) != false );
// If recipient is internal user (noreply user), and emailstop is set then don't send any msg.
if ( ! $usertoisrealuser && ! empty ( $eventdata -> userto -> emailstop )) {
debugging ( 'Attempt to send msg to internal (noreply) user' , DEBUG_NORMAL );
return false ;
}
2019-03-06 21:26:29 +01:00
if ( $eventdata -> userfrom -> id == $eventdata -> userto -> id ) {
// It's a self conversation.
$conversation = \core_message\api :: get_self_conversation ( $eventdata -> userfrom -> id );
if ( empty ( $conversation )) {
$conversation = \core_message\api :: create_conversation (
\core_message\api :: MESSAGE_CONVERSATION_TYPE_SELF ,
[ $eventdata -> userfrom -> id ]
);
}
} else {
if ( ! $conversationid = \core_message\api :: get_conversation_between_users ([ $eventdata -> userfrom -> id ,
$eventdata -> userto -> id ])) {
// It's a private conversation between users.
$conversation = \core_message\api :: create_conversation (
\core_message\api :: MESSAGE_CONVERSATION_TYPE_INDIVIDUAL ,
[
$eventdata -> userfrom -> id ,
$eventdata -> userto -> id
]
);
}
2018-11-12 10:05:17 +08:00
}
// We either have found a conversation, or created one.
2019-03-06 21:26:29 +01:00
$conversationid = ! empty ( $conversationid ) ? $conversationid : $conversation -> id ;
2018-11-12 10:05:17 +08:00
$eventdata -> convid = $conversationid ;
}
// This is a message directed to a conversation, not a specific user as was the way in legacy messaging.
// The above code has adapted the legacy messages into conversation messages.
// We must call send_message_to_conversation(), which handles per-member processor iteration and triggers
// a per-conversation event.
// All eventdata for messages should now have a convid, as we fixed this above.
if ( ! $eventdata -> notification ) {
2018-11-08 18:07:00 +08:00
// Only one message will be saved to the DB.
$conversationid = $eventdata -> convid ;
$table = 'messages' ;
$tabledata = new stdClass ();
$tabledata -> courseid = $eventdata -> courseid ;
$tabledata -> useridfrom = $eventdata -> userfrom -> id ;
$tabledata -> conversationid = $conversationid ;
$tabledata -> subject = $eventdata -> subject ;
$tabledata -> fullmessage = $eventdata -> fullmessage ;
$tabledata -> fullmessageformat = $eventdata -> fullmessageformat ;
$tabledata -> fullmessagehtml = $eventdata -> fullmessagehtml ;
$tabledata -> smallmessage = $eventdata -> smallmessage ;
$tabledata -> timecreated = time ();
2019-02-05 14:59:28 +01:00
$tabledata -> customdata = $eventdata -> customdata ;
2018-11-08 18:07:00 +08:00
2019-04-07 10:56:53 +02:00
// The Trusted Content system.
// Texts created or uploaded by such users will be marked as trusted and will not be cleaned before display.
if ( trusttext_active ()) {
// Individual conversations are always in system context.
$messagecontext = \context_system :: instance ();
// We need to know the type of conversation and the contextid if it is a group conversation.
if ( $conv = $DB -> get_record ( 'message_conversations' , [ 'id' => $conversationid ], 'id, type, contextid' )) {
if ( $conv -> type == \core_message\api :: MESSAGE_CONVERSATION_TYPE_GROUP && $conv -> contextid ) {
$messagecontext = \context :: instance_by_id ( $conv -> contextid );
}
}
$tabledata -> fullmessagetrust = trusttext_trusted ( $messagecontext );
} else {
$tabledata -> fullmessagetrust = false ;
}
2018-11-12 09:11:48 +08:00
if ( $messageid = message_handle_phpunit_redirection ( $eventdata , $table , $tabledata )) {
return $messageid ;
2018-11-08 18:07:00 +08:00
}
// Cache messages.
if ( ! empty ( $eventdata -> convid )) {
// Cache the timecreated value of the last message in this conversation.
$cache = cache :: make ( 'core' , 'message_time_last_message_between_users' );
$key = \core_message\helper :: get_last_message_time_created_cache_key ( $eventdata -> convid );
$cache -> set ( $key , $tabledata -> timecreated );
}
// Store unread message just in case we get a fatal error any time later.
$tabledata -> id = $DB -> insert_record ( $table , $tabledata );
$eventdata -> savedmessageid = $tabledata -> id ;
return \core\message\manager :: send_message_to_conversation ( $eventdata , $tabledata );
}
2018-11-12 10:05:17 +08:00
// Else the message is a notification.
2014-06-10 17:21:40 +12:00
if ( ! is_object ( $eventdata -> userto )) {
2013-08-30 14:28:17 +08:00
$eventdata -> userto = core_user :: get_user ( $eventdata -> userto );
2010-10-27 08:47:33 +00:00
}
2014-06-10 17:21:40 +12:00
if ( ! $eventdata -> userto ) {
debugging ( 'Attempt to send msg to unknown user' , DEBUG_NORMAL );
return false ;
}
2018-02-15 20:34:50 +01:00
// If the provider's component is disabled or the user can't receive messages from it, don't send the message.
$isproviderallowed = false ;
foreach ( message_get_providers_for_user ( $eventdata -> userto -> id ) as $provider ) {
if ( $provider -> component === $eventdata -> component && $provider -> name === $eventdata -> name ) {
$isproviderallowed = true ;
break ;
}
}
if ( ! $isproviderallowed ) {
debugging ( 'Attempt to send msg from a provider ' . $eventdata -> component . '/' . $eventdata -> name .
' that is inactive or not allowed for the user id=' . $eventdata -> userto -> id , DEBUG_NORMAL );
return false ;
}
2014-06-10 17:21:40 +12:00
// Verify all necessary data fields are present.
if ( ! isset ( $eventdata -> userto -> auth ) or ! isset ( $eventdata -> userto -> suspended )
or ! isset ( $eventdata -> userto -> deleted ) or ! isset ( $eventdata -> userto -> emailstop )) {
debugging ( 'Necessary properties missing in userto object, fetching full record' , DEBUG_DEVELOPER );
$eventdata -> userto = core_user :: get_user ( $eventdata -> userto -> id );
}
2013-08-30 14:28:17 +08:00
$usertoisrealuser = ( core_user :: is_real_user ( $eventdata -> userto -> id ) != false );
// If recipient is internal user (noreply user), and emailstop is set then don't send any msg.
if ( ! $usertoisrealuser && ! empty ( $eventdata -> userto -> emailstop )) {
debugging ( 'Attempt to send msg to internal (noreply) user' , DEBUG_NORMAL );
return false ;
}
2018-01-04 15:01:37 +08:00
// Check if we are creating a notification or message.
2018-11-12 10:05:17 +08:00
$table = 'notifications' ;
$tabledata = new stdClass ();
$tabledata -> useridfrom = $eventdata -> userfrom -> id ;
$tabledata -> useridto = $eventdata -> userto -> id ;
$tabledata -> subject = $eventdata -> subject ;
$tabledata -> fullmessage = $eventdata -> fullmessage ;
$tabledata -> fullmessageformat = $eventdata -> fullmessageformat ;
$tabledata -> fullmessagehtml = $eventdata -> fullmessagehtml ;
$tabledata -> smallmessage = $eventdata -> smallmessage ;
$tabledata -> eventtype = $eventdata -> name ;
$tabledata -> component = $eventdata -> component ;
$tabledata -> timecreated = time ();
2019-02-05 14:59:28 +01:00
$tabledata -> customdata = $eventdata -> customdata ;
2018-11-12 10:05:17 +08:00
if ( ! empty ( $eventdata -> contexturl )) {
$tabledata -> contexturl = ( string ) $eventdata -> contexturl ;
2010-10-26 08:00:29 +00:00
} else {
2018-11-12 10:05:17 +08:00
$tabledata -> contexturl = null ;
2010-10-26 08:00:29 +00:00
}
2018-11-12 10:05:17 +08:00
if ( ! empty ( $eventdata -> contexturlname )) {
$tabledata -> contexturlname = ( string ) $eventdata -> contexturlname ;
} else {
$tabledata -> contexturlname = null ;
}
2008-07-24 08:38:03 +00:00
2018-11-12 09:11:48 +08:00
if ( $messageid = message_handle_phpunit_redirection ( $eventdata , $table , $tabledata )) {
return $messageid ;
2012-10-13 17:48:06 +02:00
}
2017-05-01 13:12:47 +08:00
// Fetch enabled processors.
2018-11-12 10:05:17 +08:00
$processors = get_message_processors ( true );
2011-05-19 17:25:39 +01:00
// Preset variables
$processorlist = array ();
// Fill in the array of processors to be used based on default and user preferences
foreach ( $processors as $processor ) {
2013-08-30 14:28:17 +08:00
// Skip adding processors for internal user, if processor doesn't support sending message to internal user.
if ( ! $usertoisrealuser && ! $processor -> object -> can_send_to_any_users ()) {
continue ;
}
2011-05-19 17:25:39 +01:00
// First find out permissions
2020-01-29 16:45:32 +01:00
$defaultlockedpreference = $processor -> name . '_provider_' . $preferencebase . '_locked' ;
$locked = false ;
if ( isset ( $defaultpreferences -> { $defaultlockedpreference })) {
$locked = $defaultpreferences -> { $defaultlockedpreference };
2011-05-19 17:25:39 +01:00
} else {
2012-10-30 11:03:12 +08:00
// MDL-25114 They supplied an $eventdata->component $eventdata->name combination which doesn't
// exist in the message_provider table (thus there is no default settings for them).
2020-01-29 16:45:32 +01:00
$preferrormsg = " Could not load preference $defaultlockedpreference . Make sure the component and name you supplied
2012-10-30 11:03:12 +08:00
to message_send () are valid . " ;
throw new coding_exception ( $preferrormsg );
2010-11-09 06:01:20 +00:00
}
2011-05-19 17:25:39 +01:00
2020-01-29 16:45:32 +01:00
$preferencename = 'message_provider_' . $preferencebase . '_enabled' ;
$forced = false ;
if ( $locked && isset ( $defaultpreferences -> { $preferencename })) {
$userpreference = $defaultpreferences -> { $preferencename };
$forced = in_array ( $processor -> name , explode ( ',' , $userpreference ));
}
2011-05-19 17:25:39 +01:00
// Find out if user has configured this output
2011-08-16 14:06:11 +08:00
// Some processors cannot function without settings from the user
2011-05-31 09:07:38 +01:00
$userisconfigured = $processor -> object -> is_user_configured ( $eventdata -> userto );
2011-05-19 17:25:39 +01:00
2011-06-24 14:19:40 +08:00
// DEBUG: notify if we are forcing unconfigured output
2020-01-29 16:45:32 +01:00
if ( $forced && ! $userisconfigured ) {
2011-05-19 17:25:39 +01:00
debugging ( 'Attempt to force message delivery to user who has "' . $processor -> name . '" output unconfigured' , DEBUG_NORMAL );
}
// Populate the list of processors we will be using
2020-01-29 16:45:32 +01:00
if ( $forced && $userisconfigured ) {
2011-11-16 13:01:55 +08:00
// An admin is forcing users to use this message processor. Use this processor unconditionally.
2011-05-19 17:25:39 +01:00
$processorlist [] = $processor -> name ;
2020-01-29 16:45:32 +01:00
} else if ( ! $forced && ! $locked && $userisconfigured && ! $eventdata -> userto -> emailstop ) {
2011-11-16 13:01:55 +08:00
// User has not disabled notifications
// See if user set any notification preferences, otherwise use site default ones
2020-01-29 16:45:32 +01:00
if ( $userpreference = get_user_preferences ( $preferencename , null , $eventdata -> userto )) {
2011-05-19 17:25:39 +01:00
if ( in_array ( $processor -> name , explode ( ',' , $userpreference ))) {
$processorlist [] = $processor -> name ;
}
2020-01-29 16:45:32 +01:00
} else if ( isset ( $defaultpreferences -> { $preferencename })) {
if ( in_array ( $processor -> name , explode ( ',' , $defaultpreferences -> { $preferencename }))) {
2011-05-19 17:25:39 +01:00
$processorlist [] = $processor -> name ;
}
}
}
2008-08-02 00:04:29 +00:00
}
2008-07-24 08:38:03 +00:00
2014-06-10 17:21:40 +12:00
// Store unread message just in case we get a fatal error any time later.
2018-01-04 15:01:37 +08:00
$tabledata -> id = $DB -> insert_record ( $table , $tabledata );
$eventdata -> savedmessageid = $tabledata -> id ;
2014-02-19 21:25:51 -08:00
2014-06-10 17:21:40 +12:00
// Let the manager do the sending or buffering when db transaction in progress.
2021-07-23 16:38:40 +01:00
try {
return \core\message\manager :: send_message ( $eventdata , $tabledata , $processorlist );
} catch ( \moodle_exception $exception ) {
return false ;
}
2008-07-24 08:38:03 +00:00
}
2018-11-12 09:11:48 +08:00
/**
* Helper method containing the PHPUnit specific code , used to redirect and capture messages / notifications .
*
* @ param \core\message\message $eventdata the message object
* @ param string $table the table to store the tabledata in , either messages or notifications .
* @ param stdClass $tabledata the data to be stored when creating the message / notification .
* @ return int the id of the stored message .
*/
function message_handle_phpunit_redirection ( \core\message\message $eventdata , string $table , \stdClass $tabledata ) {
global $DB ;
if ( PHPUNIT_TEST and class_exists ( 'phpunit_util' )) {
// Add some more tests to make sure the normal code can actually work.
$componentdir = core_component :: get_component_directory ( $eventdata -> component );
if ( ! $componentdir or ! is_dir ( $componentdir )) {
throw new coding_exception ( 'Invalid component specified in message-send(): ' . $eventdata -> component );
}
if ( ! file_exists ( " $componentdir /db/messages.php " )) {
throw new coding_exception ( " $eventdata->component does not contain db/messages.php necessary for message_send() " );
}
$messageproviders = null ;
include ( " $componentdir /db/messages.php " );
if ( ! isset ( $messageproviders [ $eventdata -> name ])) {
throw new coding_exception ( " Missing messaging defaults for event ' $eventdata->name ' in ' $eventdata->component ' " .
" messages.php file " );
}
unset ( $componentdir );
unset ( $messageproviders );
// Now ask phpunit if it wants to catch this message.
if ( phpunit_util :: is_redirecting_messages ()) {
$messageid = $DB -> insert_record ( $table , $tabledata );
$message = $DB -> get_record ( $table , array ( 'id' => $messageid ));
if ( $eventdata -> notification ) {
// Add the useridto attribute for BC.
$message -> useridto = $eventdata -> userto -> id ;
// Mark the notification as read.
\core_message\api :: mark_notification_as_read ( $message );
} else {
// Add the useridto attribute for BC.
if ( isset ( $eventdata -> userto )) {
$message -> useridto = $eventdata -> userto -> id ;
}
// Mark the message as read for each of the other users.
$sql = " SELECT u.*
FROM { message_conversation_members } mcm
JOIN { user } u
ON ( mcm . conversationid = : convid AND u . id = mcm . userid AND u . id != : userid ) " ;
$otherusers = $DB -> get_records_sql ( $sql , [ 'convid' => $eventdata -> convid , 'userid' => $eventdata -> userfrom -> id ]);
foreach ( $otherusers as $othermember ) {
\core_message\api :: mark_message_as_read ( $othermember -> id , $message );
}
}
// Unit tests need this detail.
$message -> notification = $eventdata -> notification ;
phpunit_util :: message_sent ( $message );
return $messageid ;
}
}
}
2008-07-31 08:01:46 +00:00
/**
2012-01-05 12:05:02 +07:00
* Updates the message_providers table with the current set of message providers
2011-05-27 09:28:09 +01:00
*
2021-09-30 07:42:14 +02:00
* @ param string $component For example 'moodle' , 'mod_forum' or 'block_activity_results'
2012-01-05 12:05:02 +07:00
* @ return boolean True on success
2008-07-31 08:01:46 +00:00
*/
function message_update_providers ( $component = 'moodle' ) {
global $DB ;
// load message providers from files
$fileproviders = message_get_providers_from_file ( $component );
// load message providers from the database
$dbproviders = message_get_providers_from_db ( $component );
foreach ( $fileproviders as $messagename => $fileprovider ) {
if ( ! empty ( $dbproviders [ $messagename ])) { // Already exists in the database
2011-05-31 10:55:30 +01:00
// check if capability has changed
2008-07-31 08:01:46 +00:00
if ( $dbproviders [ $messagename ] -> capability == $fileprovider [ 'capability' ]) { // Same, so ignore
// exact same message provider already present in db, ignore this entry
unset ( $dbproviders [ $messagename ]);
continue ;
} else { // Update existing one
2010-09-21 08:07:44 +00:00
$provider = new stdClass ();
2009-11-07 10:27:57 +00:00
$provider -> id = $dbproviders [ $messagename ] -> id ;
$provider -> capability = $fileprovider [ 'capability' ];
2008-07-31 08:01:46 +00:00
$DB -> update_record ( 'message_providers' , $provider );
unset ( $dbproviders [ $messagename ]);
continue ;
}
} else { // New message provider, add it
2010-09-21 08:07:44 +00:00
$provider = new stdClass ();
2008-07-31 08:01:46 +00:00
$provider -> name = $messagename ;
$provider -> component = $component ;
$provider -> capability = $fileprovider [ 'capability' ];
2011-05-20 15:10:27 +01:00
$transaction = $DB -> start_delegated_transaction ();
2008-07-31 08:01:46 +00:00
$DB -> insert_record ( 'message_providers' , $provider );
2011-05-20 15:10:27 +01:00
message_set_default_message_preference ( $component , $messagename , $fileprovider );
$transaction -> allow_commit ();
2008-07-31 08:01:46 +00:00
}
}
foreach ( $dbproviders as $dbprovider ) { // Delete old ones
$DB -> delete_records ( 'message_providers' , array ( 'id' => $dbprovider -> id ));
2011-05-31 10:55:30 +01:00
$DB -> delete_records_select ( 'config_plugins' , " plugin = 'message' AND " . $DB -> sql_like ( 'name' , '?' , false ), array ( " %_provider_ { $component } _ { $dbprovider -> name } _% " ));
$DB -> delete_records_select ( 'user_preferences' , $DB -> sql_like ( 'name' , '?' , false ), array ( " message_provider_ { $component } _ { $dbprovider -> name } _% " ));
2012-08-14 11:40:28 +12:00
cache_helper :: invalidate_by_definition ( 'core' , 'config' , array (), 'message' );
2008-07-31 08:01:46 +00:00
}
2011-05-31 10:05:39 +01:00
return true ;
2008-07-31 08:01:46 +00:00
}
2011-06-01 11:44:36 +01:00
/**
* This function populates default message preferences for all existing providers
* when the new message processor is added .
*
* @ param string $processorname The name of message processor plugin ( e . g . 'email' , 'jabber' )
2012-01-05 12:05:02 +07:00
* @ throws invalid_parameter_exception if $processorname does not exist in the database
2011-06-01 11:44:36 +01:00
*/
function message_update_processors ( $processorname ) {
global $DB ;
// validate if our processor exists
$processor = $DB -> get_records ( 'message_processors' , array ( 'name' => $processorname ));
if ( empty ( $processor )) {
throw new invalid_parameter_exception ();
}
$providers = $DB -> get_records_sql ( 'SELECT DISTINCT component FROM {message_providers}' );
$transaction = $DB -> start_delegated_transaction ();
foreach ( $providers as $provider ) {
// load message providers from files
$fileproviders = message_get_providers_from_file ( $provider -> component );
foreach ( $fileproviders as $messagename => $fileprovider ) {
message_set_default_message_preference ( $provider -> component , $messagename , $fileprovider , $processorname );
}
}
$transaction -> allow_commit ();
}
2011-05-20 15:10:27 +01:00
/**
2012-01-05 12:05:02 +07:00
* Setting default messaging preferences for particular message provider
2011-05-27 09:28:09 +01:00
*
2011-05-20 15:10:27 +01:00
* @ param string $component The name of component ( e . g . moodle , mod_forum , etc . )
* @ param string $messagename The name of message provider
* @ param array $fileprovider The value of $messagename key in the array defined in plugin messages . php
2012-01-05 12:05:02 +07:00
* @ param string $processorname The optional name of message processor
2011-05-20 15:10:27 +01:00
*/
2011-06-01 11:44:36 +01:00
function message_set_default_message_preference ( $component , $messagename , $fileprovider , $processorname = '' ) {
2011-05-20 15:10:27 +01:00
global $DB ;
// Fetch message processors
2011-06-01 11:44:36 +01:00
$condition = null ;
// If we need to process a particular processor, set the select condition
if ( ! empty ( $processorname )) {
$condition = array ( 'name' => $processorname );
}
$processors = $DB -> get_records ( 'message_processors' , $condition );
2011-05-20 15:10:27 +01:00
// load default messaging preferences
$defaultpreferences = get_message_output_default_preferences ();
2011-05-27 09:28:09 +01:00
// Setting default preference
2011-05-20 15:10:27 +01:00
$componentproviderbase = $component . '_' . $messagename ;
2020-01-29 16:45:32 +01:00
$enabledpref = [];
// Set 'locked' preference first for each messaging processor.
2011-05-20 15:10:27 +01:00
foreach ( $processors as $processor ) {
2020-01-29 16:45:32 +01:00
$preferencename = $processor -> name . '_provider_' . $componentproviderbase . '_locked' ;
// If we do not have this setting yet, set it.
2011-05-31 09:46:22 +01:00
if ( ! isset ( $defaultpreferences -> { $preferencename })) {
2020-01-29 16:45:32 +01:00
// Determine plugin default settings.
2011-05-20 15:10:27 +01:00
$plugindefault = 0 ;
if ( isset ( $fileprovider [ 'defaults' ][ $processor -> name ])) {
$plugindefault = $fileprovider [ 'defaults' ][ $processor -> name ];
}
2020-01-29 16:45:32 +01:00
// Get string values of the settings.
list ( $locked , $enabled ) = translate_message_default_setting ( $plugindefault , $processor -> name );
// Store default preferences for current processor.
set_config ( $preferencename , $locked , 'message' );
// Save enabled settings.
if ( $enabled ) {
$enabledpref [] = $processor -> name ;
2011-05-20 15:10:27 +01:00
}
2011-06-01 11:44:36 +01:00
}
2011-05-20 15:10:27 +01:00
}
2020-01-29 16:45:32 +01:00
// Now set enabled preferences.
if ( ! empty ( $enabledpref )) {
$preferencename = 'message_provider_' . $componentproviderbase . '_enabled' ;
2011-06-01 11:44:36 +01:00
if ( isset ( $defaultpreferences -> { $preferencename })) {
// We have the default preferences for this message provider, which
// likely means that we have been adding a new processor. Add defaults
// to exisitng preferences.
2020-01-29 16:45:32 +01:00
$enabledpref = array_merge ( $enabledpref , explode ( ',' , $defaultpreferences -> { $preferencename }));
2011-06-01 11:44:36 +01:00
}
2020-01-29 16:45:32 +01:00
set_config ( $preferencename , join ( ',' , $enabledpref ), 'message' );
2011-05-20 15:10:27 +01:00
}
}
2008-07-31 08:01:46 +00:00
/**
2011-05-28 08:46:47 +01:00
* Returns the active providers for the user specified , based on capability
2011-05-27 09:28:09 +01:00
*
2011-05-28 08:46:47 +01:00
* @ param int $userid id of user
2012-01-05 12:05:02 +07:00
* @ return array An array of message providers
2008-07-31 08:01:46 +00:00
*/
2011-05-28 08:46:47 +01:00
function message_get_providers_for_user ( $userid ) {
2011-07-05 15:23:37 +01:00
global $DB , $CFG ;
2008-07-31 08:01:46 +00:00
2012-05-15 16:55:21 +01:00
$providers = get_message_providers ();
2008-07-31 08:01:46 +00:00
2012-10-23 10:26:33 +08:00
// Ensure user is not allowed to configure instantmessage if it is globally disabled.
if ( ! $CFG -> messaging ) {
foreach ( $providers as $providerid => $provider ) {
if ( $provider -> name == 'instantmessage' ) {
unset ( $providers [ $providerid ]);
break ;
2008-07-31 08:01:46 +00:00
}
}
2012-10-23 10:26:33 +08:00
}
2012-04-19 18:03:18 +07:00
2012-10-23 10:26:33 +08:00
// If the component is an enrolment plugin, check it is enabled
foreach ( $providers as $providerid => $provider ) {
2013-07-16 22:41:00 +02:00
list ( $type , $name ) = core_component :: normalize_component ( $provider -> component );
2012-10-23 10:26:33 +08:00
if ( $type == 'enrol' && ! enrol_is_enabled ( $name )) {
2011-07-05 15:23:37 +01:00
unset ( $providers [ $providerid ]);
2012-10-23 10:26:33 +08:00
}
}
// Now we need to check capabilities. We need to eliminate the providers
// where the user does not have the corresponding capability anywhere.
// Here we deal with the common simple case of the user having the
// capability in the system context. That handles $CFG->defaultuserroleid.
// For the remaining providers/capabilities, we need to do a more complex
// query involving all overrides everywhere.
$unsureproviders = array ();
$unsurecapabilities = array ();
$systemcontext = context_system :: instance ();
foreach ( $providers as $providerid => $provider ) {
if ( empty ( $provider -> capability ) || has_capability ( $provider -> capability , $systemcontext , $userid )) {
// The provider is relevant to this user.
2012-04-19 18:03:18 +07:00
continue ;
}
2012-10-23 10:26:33 +08:00
$unsureproviders [ $providerid ] = $provider ;
$unsurecapabilities [ $provider -> capability ] = 1 ;
unset ( $providers [ $providerid ]);
}
if ( empty ( $unsureproviders )) {
// More complex checks are not required.
return $providers ;
}
// Now check the unsure capabilities.
list ( $capcondition , $params ) = $DB -> get_in_or_equal (
array_keys ( $unsurecapabilities ), SQL_PARAMS_NAMED );
$params [ 'userid' ] = $userid ;
$sql = " SELECT DISTINCT rc.capability, 1
FROM { role_assignments } ra
JOIN { context } actx ON actx . id = ra . contextid
JOIN { role_capabilities } rc ON rc . roleid = ra . roleid
JOIN { context } cctx ON cctx . id = rc . contextid
WHERE ra . userid = : userid
AND rc . capability $capcondition
AND rc . permission > 0
2012-11-13 15:09:14 +08:00
AND ( " . $DB->sql_concat ('actx.path', " '/' " ). " LIKE " . $DB->sql_concat ('cctx.path', " '/%' " ).
" OR " . $DB -> sql_concat ( 'cctx.path' , " '/' " ) . " LIKE " . $DB -> sql_concat ( 'actx.path' , " '/%' " ) . " ) " ;
2012-10-23 10:26:33 +08:00
if ( ! empty ( $CFG -> defaultfrontpageroleid )) {
$frontpagecontext = context_course :: instance ( SITEID );
list ( $capcondition2 , $params2 ) = $DB -> get_in_or_equal (
array_keys ( $unsurecapabilities ), SQL_PARAMS_NAMED );
$params = array_merge ( $params , $params2 );
$params [ 'frontpageroleid' ] = $CFG -> defaultfrontpageroleid ;
$params [ 'frontpagepathpattern' ] = $frontpagecontext -> path . '/' ;
$sql .= "
2012-11-13 22:34:36 +08:00
UNION
2012-10-23 10:26:33 +08:00
SELECT DISTINCT rc . capability , 1
FROM { role_capabilities } rc
JOIN { context } cctx ON cctx . id = rc . contextid
WHERE rc . roleid = : frontpageroleid
AND rc . capability $capcondition2
AND rc . permission > 0
2012-11-13 15:09:14 +08:00
AND " . $DB->sql_concat ('cctx.path', " '/' " ). " LIKE : frontpagepathpattern " ;
2012-10-23 10:26:33 +08:00
}
$relevantcapabilities = $DB -> get_records_sql_menu ( $sql , $params );
// Add back any providers based on the detailed capability check.
foreach ( $unsureproviders as $providerid => $provider ) {
if ( array_key_exists ( $provider -> capability , $relevantcapabilities )) {
$providers [ $providerid ] = $provider ;
2011-07-05 15:23:37 +01:00
}
2008-07-31 08:01:46 +00:00
}
return $providers ;
}
/**
* Gets the message providers that are in the database for this component .
2011-05-27 09:28:09 +01:00
*
2012-01-05 12:05:02 +07:00
* This is an internal function used within messagelib . php
2008-07-31 08:01:46 +00:00
*
2012-02-22 11:15:42 +13:00
* @ see message_update_providers ()
2021-09-30 07:42:14 +02:00
* @ param string $component A moodle component like 'moodle' , 'mod_forum' , 'block_activity_results'
2012-01-05 12:05:02 +07:00
* @ return array An array of message providers
2008-07-31 08:01:46 +00:00
*/
function message_get_providers_from_db ( $component ) {
global $DB ;
2009-11-07 10:27:57 +00:00
return $DB -> get_records ( 'message_providers' , array ( 'component' => $component ), '' , 'name, id, component, capability' ); // Name is unique per component
2008-07-31 08:01:46 +00:00
}
/**
2012-01-05 12:05:02 +07:00
* Loads the messages definitions for a component from file
2011-05-27 09:28:09 +01:00
*
2012-01-05 12:05:02 +07:00
* If no messages are defined for the component , return an empty array .
* This is an internal function used within messagelib . php
2008-07-31 08:01:46 +00:00
*
2012-01-05 12:05:02 +07:00
* @ see message_update_providers ()
* @ see message_update_processors ()
2021-09-30 07:42:14 +02:00
* @ param string $component A moodle component like 'moodle' , 'mod_forum' , 'block_activity_results'
2012-01-05 12:05:02 +07:00
* @ return array An array of message providers or empty array if not exists
2008-07-31 08:01:46 +00:00
*/
function message_get_providers_from_file ( $component ) {
2013-07-16 22:42:37 +02:00
$defpath = core_component :: get_component_directory ( $component ) . '/db/messages.php' ;
2008-07-31 08:01:46 +00:00
$messageproviders = array ();
if ( file_exists ( $defpath )) {
require ( $defpath );
}
foreach ( $messageproviders as $name => $messageprovider ) { // Fix up missing values if required
if ( empty ( $messageprovider [ 'capability' ])) {
$messageproviders [ $name ][ 'capability' ] = NULL ;
}
2011-05-20 15:10:27 +01:00
if ( empty ( $messageprovider [ 'defaults' ])) {
$messageproviders [ $name ][ 'defaults' ] = array ();
}
2008-07-31 08:01:46 +00:00
}
return $messageproviders ;
}
/**
2012-01-05 12:05:02 +07:00
* Remove all message providers for particular component and corresponding settings
2011-05-27 09:28:09 +01:00
*
2021-09-30 07:42:14 +02:00
* @ param string $component A moodle component like 'moodle' , 'mod_forum' , 'block_activity_results'
2011-05-27 09:28:09 +01:00
* @ return void
2008-07-31 08:01:46 +00:00
*/
2011-06-02 09:27:05 +01:00
function message_provider_uninstall ( $component ) {
2010-09-17 07:58:04 +00:00
global $DB ;
2010-10-25 09:29:34 +00:00
2011-05-20 15:10:27 +01:00
$transaction = $DB -> start_delegated_transaction ();
$DB -> delete_records ( 'message_providers' , array ( 'component' => $component ));
$DB -> delete_records_select ( 'config_plugins' , " plugin = 'message' AND " . $DB -> sql_like ( 'name' , '?' , false ), array ( " %_provider_ { $component } _% " ));
$DB -> delete_records_select ( 'user_preferences' , $DB -> sql_like ( 'name' , '?' , false ), array ( " message_provider_ { $component } _% " ));
$transaction -> allow_commit ();
2012-08-14 11:40:28 +12:00
// Purge all messaging settings from the caches. They are stored by plugin so we have to clear all message settings.
cache_helper :: invalidate_by_definition ( 'core' , 'config' , array (), 'message' );
2008-08-02 00:04:29 +00:00
}
2011-06-02 09:27:05 +01:00
/**
2012-01-05 12:05:02 +07:00
* Uninstall a message processor
2011-06-02 09:27:05 +01:00
*
2012-01-05 12:05:02 +07:00
* @ param string $name A message processor name like 'email' , 'jabber'
2011-06-02 09:27:05 +01:00
*/
function message_processor_uninstall ( $name ) {
global $DB ;
$transaction = $DB -> start_delegated_transaction ();
$DB -> delete_records ( 'message_processors' , array ( 'name' => $name ));
2012-04-03 14:48:54 +01:00
$DB -> delete_records_select ( 'config_plugins' , " plugin = ? " , array ( " message_ { $name } " ));
2020-01-29 16:45:32 +01:00
// Delete permission preferences only, we do not care about enabled defaults,
// they will be removed on the next attempt to update the preferences.
2011-06-02 09:27:05 +01:00
$DB -> delete_records_select ( 'config_plugins' , " plugin = 'message' AND " . $DB -> sql_like ( 'name' , '?' , false ), array ( " { $name } _provider_% " ));
$transaction -> allow_commit ();
2012-08-14 11:40:28 +12:00
// Purge all messaging settings from the caches. They are stored by plugin so we have to clear all message settings.
cache_helper :: invalidate_by_definition ( 'core' , 'config' , array (), array ( 'message' , " message_ { $name } " ));
2011-06-02 09:27:05 +01:00
}