Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 3.11.x will end 14 Nov 2022 (12 months plus 6 months extension).
  • Bug fixes for security issues in 3.11.x will end 13 Nov 2023 (18 months plus 12 months extension).
  • PHP version: minimum PHP 7.3.0 Note: minimum PHP version has increased since Moodle 3.10. PHP 7.4.x is supported too.
<?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/>.

/**
 * Airnotifier manager class
 *
 * @package    message_airnotifier
 * @category   external
 * @copyright  2012 Jerome Mouneyrac <jerome@moodle.com>
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 * @since Moodle 2.7
 */

defined('MOODLE_INTERNAL') || die;

/**
 * Airnotifier helper manager class
 *
 * @copyright  2012 Jerome Mouneyrac <jerome@moodle.com>
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */
class message_airnotifier_manager {

    /** @var string The Airnotifier public instance URL */
    const AIRNOTIFIER_PUBLICURL = 'https://messages.moodle.net';

    /**
     * Include the relevant javascript and language strings for the device
     * toolbox YUI module
     *
     * @return bool
     */
    public function include_device_ajax() {
        global $PAGE, $CFG;

        $config = new stdClass();
        $config->resturl = '/message/output/airnotifier/rest.php';
        $config->pageparams = array();

        // Include toolboxes.
        $PAGE->requires->yui_module('moodle-message_airnotifier-toolboxes', 'M.message.init_device_toolbox', array(array(
                'ajaxurl' => $config->resturl,
                'config' => $config,
                ))
        );

        // Required strings for the javascript.
        $PAGE->requires->strings_for_js(array('deletecheckdevicename'), 'message_airnotifier');
        $PAGE->requires->strings_for_js(array('show', 'hide'), 'moodle');

        return true;
    }

    /**
     * Return the user devices for a specific app.
     *
     * @param string $appname the app name .
     * @param int $userid if empty take the current user.
     * @return array all the devices
     */
    public function get_user_devices($appname, $userid = null) {
        global $USER, $DB;

        if (empty($userid)) {
            $userid = $USER->id;
        }

        $devices = array();

        $params = array('appid' => $appname, 'userid' => $userid);

        // First, we look all the devices registered for this user in the Moodle core.
        // We are going to allow only ios devices (since these are the ones that supports PUSH notifications).
        $userdevices = $DB->get_records('user_devices', $params);
        foreach ($userdevices as $device) {
            if (core_text::strtolower($device->platform)) {
                // Check if the device is known by airnotifier.
                if (!$airnotifierdev = $DB->get_record('message_airnotifier_devices',
                        array('userdeviceid' => $device->id))) {

                    // We have to create the device token in airnotifier.
                    if (! $this->create_token($device->pushid, $device->platform)) {
                        continue;
                    }

                    $airnotifierdev = new stdClass;
                    $airnotifierdev->userdeviceid = $device->id;
                    $airnotifierdev->enable = 1;
                    $airnotifierdev->id = $DB->insert_record('message_airnotifier_devices', $airnotifierdev);
                }
                $device->id = $airnotifierdev->id;
                $device->enable = $airnotifierdev->enable;
                $devices[] = $device;
            }
        }

        return $devices;
    }

    /**
     * Request and access key to Airnotifier
     *
     * @return mixed The access key or false in case of error
     */
    public function request_accesskey() {
        global $CFG, $USER;

        require_once($CFG->libdir . '/filelib.php');

        // Sending the request access key request to Airnotifier.
        $serverurl = $CFG->airnotifierurl . ':' . $CFG->airnotifierport . '/accesskeys/';
        // We use an APP Key "none", it can be anything.
        $header = array('Accept: application/json', 'X-AN-APP-NAME: ' . $CFG->airnotifierappname,
            'X-AN-APP-KEY: none');
        $curl = new curl();
        $curl->setHeader($header);

        // Site ids are stored as secrets in md5 in the Moodle public hub.
        $params = array(
            'url' => $CFG->wwwroot,
            'siteid' => md5($CFG->siteidentifier),
            'contact' => $USER->email,
            'description' => $CFG->wwwroot
            );
        $resp = $curl->post($serverurl, $params);

        if ($key = json_decode($resp, true)) {
            if (!empty($key['accesskey'])) {
                return $key['accesskey'];
            }
        }
        debugging("Unexpected response from the Airnotifier server: $resp");
        return false;
    }

    /**
     * Create a device token in the Airnotifier instance
     * @param string $token The token to be created
     * @param string $deviceplatform The device platform (Android, iOS, iOS-fcm, etc...)
     * @return bool True if all was right
     */
    private function create_token($token, $deviceplatform = '') {
        global $CFG;

        if (!$this->is_system_configured()) {
            return false;
        }

        require_once($CFG->libdir . '/filelib.php');

        $serverurl = $CFG->airnotifierurl . ':' . $CFG->airnotifierport . '/tokens/' . $token;
        $header = array('Accept: application/json', 'X-AN-APP-NAME: ' . $CFG->airnotifierappname,
            'X-AN-APP-KEY: ' . $CFG->airnotifieraccesskey);
        $curl = new curl;
        $curl->setHeader($header);
        $params = [];
        if (!empty($deviceplatform)) {
            $params["device"] = $deviceplatform;
        }
        $resp = $curl->post($serverurl, $params);

        if ($token = json_decode($resp, true)) {
            if (!empty($token['status'])) {
                return $token['status'] == 'ok' || $token['status'] == 'token exists';
            }
        }
        debugging("Unexpected response from the Airnotifier server: $resp");
        return false;
    }

    /**
     * Tests whether the airnotifier settings have been configured
     * @return boolean true if airnotifier is configured
     */
    public function is_system_configured() {
        global $CFG;

        return (!empty($CFG->airnotifierurl) && !empty($CFG->airnotifierport) &&
                !empty($CFG->airnotifieraccesskey)  && !empty($CFG->airnotifierappname) &&
                !empty($CFG->airnotifiermobileappname));
    }

    /**
     * Enables or disables a registered user device so it can receive Push notifications
     *
     * @param  int $deviceid the device id
     * @param  bool $enable  true to enable it, false to disable it
     * @return bool true if the device was enabled, false in case of error
     * @since  Moodle 3.2
     */
    public static function enable_device($deviceid, $enable) {
        global $DB, $USER;

        if (!$device = $DB->get_record('message_airnotifier_devices', array('id' => $deviceid), '*')) {
            return false;
        }

        // Check that the device belongs to the current user.
        if (!$userdevice = $DB->get_record('user_devices', array('id' => $device->userdeviceid, 'userid' => $USER->id), '*')) {
            return false;
        }

        $device->enable = $enable;
        return $DB->update_record('message_airnotifier_devices', $device);
    }

    /**
     * Check the system configuration to detect possible issues.
     *
     * @return array result checks
     */
    public function check_configuration(): array {
        global $CFG, $DB;

        $results = [];
        // Check Mobile services enabled.
        $summary = html_writer::link((new moodle_url('/admin/settings.php', ['section' => 'mobilesettings'])),
                get_string('enablemobilewebservice', 'admin'));
        if (empty($CFG->enablewebservices) || empty($CFG->enablemobilewebservice)) {
            $results[] = new core\check\result(core\check\result::CRITICAL, $summary, get_string('enablewsdescription', 'webservice'));
        } else {
            $results[] = new core\check\result(core\check\result::OK, $summary, get_string('enabled', 'admin'));
        }

> // Check message outputs are not disabled in config.php. // Check Mobile notifications enabled. > $summary = get_string('noemailevernotset', 'message_airnotifier'); require_once($CFG->dirroot . '/message/lib.php'); > if (!empty($CFG->noemailever)) { $processors = get_message_processors(); > $results[] = new core\check\result(core\check\result::CRITICAL, $summary, $enabled = false; > get_string('noemaileverset', 'message_airnotifier')); foreach ($processors as $processor => $status) { > } else { if ($processor == 'airnotifier' && $status->enabled) { > $results[] = new core\check\result(core\check\result::OK, $summary, get_string('disabled', 'admin')); $enabled = true; > } } >
} $summary = html_writer::link((new moodle_url('/admin/message.php')), get_string('enableprocessor', 'message_airnotifier')); if ($enabled) { $results[] = new core\check\result(core\check\result::OK, $summary, get_string('enabled', 'admin')); } else { $results[] = new core\check\result(core\check\result::CRITICAL, $summary, get_string('mobilenotificationsdisabledwarning', 'tool_mobile')); } // Check Mobile notifications configuration is ok. $summary = html_writer::link((new moodle_url('/admin/settings.php', ['section' => 'messagesettingairnotifier'])), get_string('notificationsserverconfiguration', 'message_airnotifier')); if ($this->is_system_configured()) { $results[] = new core\check\result(core\check\result::OK, $summary, get_string('configured', 'message_airnotifier')); } else { $results[] = new core\check\result(core\check\result::ERROR, $summary, get_string('notconfigured', 'message_airnotifier')); } // Check settings properly formatted. Only display in case of errors. $settingstocheck = ['airnotifierappname', 'airnotifiermobileappname']; if ($this->is_system_configured()) { foreach ($settingstocheck as $setting) { if ($CFG->$setting != trim($CFG->$setting)) { $summary = html_writer::link((new moodle_url('/admin/settings.php', ['section' => 'messagesettingairnotifier'])), get_string('notificationsserverconfiguration', 'message_airnotifier')); $results[] = new core\check\result(core\check\result::ERROR, $summary, get_string('airnotifierfielderror', 'message_airnotifier', get_string($setting, 'message_airnotifier'))); } } } // Check connectivity with Airnotifier. $url = $CFG->airnotifierurl . ':' . $CFG->airnotifierport; $curl = new \curl(); $curl->setopt(['CURLOPT_TIMEOUT' => 5, 'CURLOPT_CONNECTTIMEOUT' => 5]); $curl->get($url); $info = $curl->get_info(); $summary = html_writer::link($url, get_string('airnotifierurl', 'message_airnotifier')); if (!empty($info['http_code']) && ($info['http_code'] == 200 || $info['http_code'] == 302)) { $results[] = new core\check\result(core\check\result::OK, $summary, get_string('online', 'message')); } else { $details = get_string('serverconnectivityerror', 'message_airnotifier', $url); $curlerrno = $curl->get_errno(); if (!empty($curlerrno)) { $details .= ' CURL ERROR: ' . $curlerrno . ' - ' . $curl->error; } $results[] = new core\check\result(core\check\result::ERROR, $summary, $details); } // Check access key by trying to create an invalid token. $settingsurl = new moodle_url('/admin/settings.php', ['section' => 'messagesettingairnotifier']); $summary = html_writer::link($settingsurl, get_string('airnotifieraccesskey', 'message_airnotifier')); if (!empty($CFG->airnotifieraccesskey)) { $url = $CFG->airnotifierurl . ':' . $CFG->airnotifierport . '/tokens/testtoken'; $header = ['Accept: application/json', 'X-AN-APP-NAME: ' . $CFG->airnotifierappname, 'X-AN-APP-KEY: ' . $CFG->airnotifieraccesskey]; $curl->setHeader($header); $response = $curl->post($url); $info = $curl->get_info(); if ($curlerrno = $curl->get_errno()) { $details = get_string('serverconnectivityerror', 'message_airnotifier', $url); $details .= ' CURL ERROR: ' . $curlerrno . ' - ' . $curl->error; $results[] = new core\check\result(core\check\result::ERROR, $summary, $details); } else if (!empty($info['http_code']) && $info['http_code'] == 400 && $key = json_decode($response, true)) { if ($key['error'] == 'Invalid access key') { $results[] = new core\check\result(core\check\result::ERROR, $summary, $key['error']); } else { $results[] = new core\check\result(core\check\result::OK, $summary, get_string('enabled', 'admin')); } } } else { $results[] = new core\check\result(core\check\result::ERROR, $summary, get_string('requestaccesskey', 'message_airnotifier')); } // Check default preferences. $preferences = (array) get_message_output_default_preferences(); $providerscount = 0; $providersconfigured = 0; foreach ($preferences as $prefname => $prefval) { if (strpos($prefname, 'message_provider') === 0) { $providerscount++; if (strpos($prefval, 'airnotifier') !== false) { $providersconfigured++; } } } $summary = html_writer::link((new moodle_url('/admin/message.php')), get_string('managemessageoutputs', 'message')); if ($providersconfigured == 0) { $results[] = new core\check\result(core\check\result::ERROR, $summary, get_string('messageprovidersempty', 'message_airnotifier')); } else if ($providersconfigured / $providerscount < 0.25) { // Less than a 25% of the providers are enabled by default for users. $results[] = new core\check\result(core\check\result::WARNING, $summary, get_string('messageproviderslow', 'message_airnotifier')); } else { $results[] = new core\check\result(core\check\result::OK, $summary, get_string('configured', 'message_airnotifier')); } // Check user devices from last month. $recentdevicescount = $DB->count_records_select('user_devices', 'appid = ? AND timemodified > ?', [$CFG->airnotifiermobileappname, time() - (WEEKSECS * 4)]); $summary = get_string('userdevices', 'message_airnotifier'); if (!empty($recentdevicescount)) { $results[] = new core\check\result(core\check\result::OK, $summary, get_string('configured', 'message_airnotifier')); } else { $results[] = new core\check\result(core\check\result::ERROR, $summary, get_string('nodevices', 'message_airnotifier')); } return $results; } /** * Send a test notification to the given user. * * @param stdClass $user user object */ public function send_test_notification(stdClass $user): void { global $CFG; require_once($CFG->dirroot . '/message/output/airnotifier/message_output_airnotifier.php'); $data = new stdClass; $data->userto = clone $user; $data->subject = 'Push Notification Test'; $data->fullmessage = 'This is a test message send at: ' . userdate(time()); $data->notification = 1; // The send_message method always return true, so it does not make sense to return anything. $airnotifier = new message_output_airnotifier(); $airnotifier->send_message($data); } /** * Check whether the given user has enabled devices or not for the given app. * * @param string $appname the app to check * @param int $userid the user to check the devices for (empty for current user) * @return bool true when the user has enabled devices, false otherwise */ public function has_enabled_devices(string $appname, int $userid = null): bool { $enableddevices = false; $devices = $this->get_user_devices($appname, $userid); foreach ($devices as $device) { if (!$device->enable) { continue; } $enableddevices = true; break; } return $enableddevices; } }