905 lines
34 KiB
PHP
905 lines
34 KiB
PHP
<?php
|
|
// This file is part of the Studyplan plugin for Moodle
|
|
//
|
|
// 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 <https://www.gnu.org/licenses/>.
|
|
|
|
/**
|
|
* Webservice class for handling associations of cohorts and users to a studyplan
|
|
* @package local_treestudyplan
|
|
* @copyright 2023 P.M. Kuipers
|
|
* @license https://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
|
|
namespace local_treestudyplan;
|
|
defined('MOODLE_INTERNAL') || die();
|
|
|
|
use local_treestudyplan\local\helpers\webservicehelper;
|
|
use local_treestudyplan\task\autocohortsync;
|
|
|
|
require_once($CFG->libdir.'/externallib.php');
|
|
|
|
/**
|
|
* Webservice class for handling associations of cohorts and users to a studyplan
|
|
*/
|
|
class associationservice extends \external_api {
|
|
/**
|
|
* Capability required to edit study plans
|
|
* @var string
|
|
*/
|
|
const CAP_EDIT = "local/treestudyplan:editstudyplan";
|
|
/**
|
|
* Capability required to view studyplans (for other users)
|
|
* @var string
|
|
*/
|
|
const CAP_VIEW = "local/treestudyplan:viewuserreports";
|
|
/**
|
|
* Capability required to be linked as coach to a studyplan
|
|
* @var string
|
|
*/
|
|
const CAP_COACH = "local/treestudyplan:coach";
|
|
/**
|
|
* Webservice structure to use in describing a user
|
|
*/
|
|
public static function user_structure(): \external_description {
|
|
return new \external_single_structure([
|
|
"id" => new \external_value(PARAM_INT, 'user id'),
|
|
"username" => new \external_value(PARAM_TEXT, 'username'),
|
|
"firstname" => new \external_value(PARAM_TEXT, 'first name'),
|
|
"lastname" => new \external_value(PARAM_TEXT, 'last name'),
|
|
"idnumber" => new \external_value(PARAM_TEXT, 'id number'),
|
|
"email" => new \external_value(PARAM_TEXT, 'email address'),
|
|
"lastaccess" => new \external_value(PARAM_INT,
|
|
'id of last access this user had to any course in the studyplan', VALUE_OPTIONAL),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Make a webservice user model for a given user
|
|
* @param stdClass $r User DB record
|
|
*/
|
|
public static function make_user_model($r) {
|
|
return [
|
|
"id" => $r->id,
|
|
"username" => $r->username,
|
|
"firstname" => $r->firstname,
|
|
"lastname" => $r->lastname,
|
|
"idnumber" => $r->idnumber,
|
|
"email" => $r->email,
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Webservice structure to use in describing a cohort
|
|
*/
|
|
public static function cohort_structure(): \external_description {
|
|
return new \external_single_structure([
|
|
"id" => new \external_value(PARAM_INT, 'cohort id'),
|
|
"name" => new \external_value(PARAM_TEXT, 'name'),
|
|
"idnumber" => new \external_value(PARAM_TEXT, 'id number'),
|
|
"description" => new \external_value(PARAM_TEXT, 'description'),
|
|
"visible" => new \external_value(PARAM_BOOL, 'is visible'),
|
|
"context" => new \external_single_structure([
|
|
"name" => new \external_value(PARAM_TEXT, 'context name'),
|
|
"shortname" => new \external_value(PARAM_TEXT, 'context short name'),
|
|
"path" => new \external_multiple_structure( new \external_value(PARAM_TEXT)),
|
|
"shortpath" => new \external_multiple_structure( new \external_value(PARAM_TEXT)),
|
|
], 'context information', VALUE_OPTIONAL),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Make a webservice cohort model for a given cohort
|
|
* @param stdClass $r Cohort DB record
|
|
*/
|
|
public static function make_cohort_model($r) {
|
|
global $DB;
|
|
|
|
$ctx = \context::instance_by_id($r->contextid);
|
|
if (is_object($ctx)) {
|
|
$ctxpath = array_reverse($ctx->get_parent_context_ids(true));
|
|
if (count($ctxpath) > 1 && $ctxpath[0] == 1) {
|
|
array_shift($ctxpath);
|
|
}
|
|
|
|
$result = [
|
|
"id" => $r->id,
|
|
"name" => $r->name,
|
|
"idnumber" => $r->idnumber,
|
|
"description" => $r->description,
|
|
"visible" => $r->visible,
|
|
"context" => [
|
|
"name" => $ctx->get_context_name(false, false),
|
|
"shortname" => $ctx->get_context_name(false, true),
|
|
"path" => array_map(function($c) {
|
|
$cx = (object)\context::instance_by_id($c);
|
|
return $cx->get_context_name(false, false);
|
|
}, $ctxpath),
|
|
"shortpath" => array_map(function($c) {
|
|
$cx = (object)\context::instance_by_id($c);
|
|
return $cx->get_context_name(false, true);
|
|
}, $ctxpath),
|
|
],
|
|
];
|
|
|
|
return $result;
|
|
} else {
|
|
throw new \moodle_exception("");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get last access time of user to any of the courses in the studyplan
|
|
* @param int $userid ID of user
|
|
* @param int $studyplanid ID of studyplan
|
|
*/
|
|
public static function user_lastaccess($userid, $studyplanid=null) {
|
|
global $DB;
|
|
if (!empty($studyplanid)) {
|
|
$lasql = "SELECT MAX(a.timeaccess) FROM {user_lastaccess} a
|
|
INNER JOIN {local_treestudyplan_item} i ON i.course_id = a.courseid
|
|
INNER JOIN {local_treestudyplan_line} l ON l.id = i.line_id
|
|
INNER JOIN {local_treestudyplan_page} p ON l.page_id = p.id
|
|
WHERE a.userid = :userid AND p.studyplan_id = :studyplanid";
|
|
$lastaccess = $DB->get_field_sql($lasql, ["userid" => $userid, "studyplanid" => $studyplanid]);
|
|
} else {
|
|
$lasql = "SELECT MAX(a.timeaccess) FROM {user_lastaccess} a
|
|
WHERE a.userid = :userid";
|
|
$lastaccess = $DB->get_field_sql($lasql, ["userid" => $userid]);
|
|
}
|
|
|
|
return $lastaccess;
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function list_cohort
|
|
*/
|
|
public static function list_cohort_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
'like' => new \external_value(PARAM_TEXT, 'search text'),
|
|
'studyplan_id' => new \external_value(PARAM_INT, 'id of studyplan to list for'),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function list_cohort
|
|
*/
|
|
public static function list_cohort_returns(): \external_description {
|
|
return new \external_multiple_structure(self::cohort_structure());
|
|
}
|
|
|
|
/**
|
|
* Search cohorts for matching string
|
|
* @param string $like String to match cohorts with
|
|
* @param int $studyplanid Do not include these cohorts
|
|
* @return array
|
|
*/
|
|
public static function list_cohort($like, $studyplanid) {
|
|
global $CFG, $DB;
|
|
|
|
// Only allow this if the user has the right to edit in this context.
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
$context = $studyplan->context();
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $context);
|
|
|
|
$pattern = "%{$like}%";
|
|
|
|
$params = ["pattern_nm" => $pattern, "pattern_id" => $pattern ];
|
|
|
|
$sql = "SELECT DISTINCT c.* from {cohort} c LEFT JOIN {local_treestudyplan_cohort} j ON c.id = j.cohort_id
|
|
WHERE c.visible = 1 AND(name LIKE :pattern_nm OR idnumber LIKE :pattern_id)
|
|
AND (j.studyplan_id IS NULL OR j.studyplan_id != :exclude_id)";
|
|
$params['exclude_id'] = $studyplanid;
|
|
|
|
$cohorts = [];
|
|
$rs = $DB->get_recordset_sql($sql, $params);
|
|
foreach ($rs as $r) {
|
|
$cohorts[] = static::make_cohort_model($r);
|
|
}
|
|
$rs->close();
|
|
return $cohorts;
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function find_user
|
|
*/
|
|
public static function find_user_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
'like' => new \external_value(PARAM_TEXT, 'search text'),
|
|
'studyplan_id' => new \external_value(PARAM_INT, 'id of studyplan to list for'),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function find_user
|
|
*/
|
|
public static function find_user_returns(): \external_description {
|
|
return new \external_multiple_structure(self::user_structure());
|
|
}
|
|
|
|
/**
|
|
* Search users for match
|
|
* @param string $like String to match user firstname/lastname with
|
|
* @param int $studyplanid Id of studyplan to search for
|
|
* @return array
|
|
*/
|
|
public static function find_user($like, $studyplanid) {
|
|
global $CFG, $DB;
|
|
|
|
// Only allow this if the user has the right to edit in this context.
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
$context = $studyplan->context();
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $context);
|
|
|
|
$pattern = "%{$like}%";
|
|
$params = ["pattern_fn" => $pattern,
|
|
"pattern_ln" => $pattern,
|
|
"pattern_un" => $pattern,
|
|
];
|
|
$sql = "SELECT DISTINCT u.* from {user} u LEFT JOIN {local_treestudyplan_user} j ON u.id = j.user_id
|
|
WHERE u.deleted != 1 AND (firstname LIKE :pattern_fn OR lastname LIKE :pattern_ln OR username LIKE :pattern_un)
|
|
AND (j.studyplan_id IS NULL OR j.studyplan_id != :exclude_id)";
|
|
$params['exclude_id'] = $studyplanid;
|
|
|
|
$users = [];
|
|
$rs = $DB->get_recordset_sql($sql, $params);
|
|
foreach ($rs as $r) {
|
|
$users[] = static::make_user_model($r);
|
|
}
|
|
$rs->close();
|
|
|
|
self::sortusermodels($users);
|
|
return $users;
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function connect_cohort
|
|
*/
|
|
public static function connect_cohort_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
"cohort_id" => new \external_value(PARAM_INT, 'id of cohort to link', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function connect_cohort
|
|
*/
|
|
public static function connect_cohort_returns(): \external_description {
|
|
return new \external_single_structure([
|
|
"success" => new \external_value(PARAM_BOOL, 'operation completed succesfully'),
|
|
"msg" => new \external_value(PARAM_TEXT, 'message'),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Connect a cohort to a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @param mixed $cohortid Id of cohort
|
|
* @return array Success/fail model
|
|
*/
|
|
public static function connect_cohort($studyplanid, $cohortid) {
|
|
global $CFG, $DB;
|
|
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $studyplan->context());
|
|
|
|
if (!$DB->record_exists('local_treestudyplan_cohort', ['studyplan_id' => $studyplanid, 'cohort_id' => $cohortid])) {
|
|
$id = $DB->insert_record('local_treestudyplan_cohort', [
|
|
'studyplan_id' => $studyplanid,
|
|
'cohort_id' => $cohortid,
|
|
]);
|
|
|
|
$studyplan->mark_csync_changed();
|
|
return ['success' => true, 'msg' => 'Cohort connected'];
|
|
|
|
} else {
|
|
return ['success' => true, 'msg' => 'Cohort already connected'];
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function disconnect_cohort
|
|
*/
|
|
public static function disconnect_cohort_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
"cohort_id" => new \external_value(PARAM_INT, 'id of cohort to link', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function disconnect_cohort
|
|
*/
|
|
public static function disconnect_cohort_returns(): \external_description {
|
|
return new \external_single_structure([
|
|
"success" => new \external_value(PARAM_BOOL, 'operation completed succesfully'),
|
|
"msg" => new \external_value(PARAM_TEXT, 'message'),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Disconnect a cohort from a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @param mixed $cohortid Id of cohort
|
|
* @return array Success/fail model
|
|
*/
|
|
public static function disconnect_cohort($studyplanid, $cohortid) {
|
|
global $CFG, $DB;
|
|
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $studyplan->context());
|
|
|
|
if ($DB->record_exists('local_treestudyplan_cohort', ['studyplan_id' => $studyplanid, 'cohort_id' => $cohortid])) {
|
|
$DB->delete_records('local_treestudyplan_cohort', [
|
|
'studyplan_id' => $studyplanid,
|
|
'cohort_id' => $cohortid,
|
|
]);
|
|
|
|
$studyplan->mark_csync_changed();
|
|
|
|
return ['success' => true, 'msg' => 'Cohort Disconnected'];
|
|
} else {
|
|
return ['success' => true, 'msg' => 'Connection does not exist'];
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function connect_user
|
|
*/
|
|
public static function connect_user_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
"user_id" => new \external_value(PARAM_INT, 'id of user to link', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function connect_user
|
|
*/
|
|
public static function connect_user_returns(): \external_description {
|
|
return new \external_single_structure([
|
|
"success" => new \external_value(PARAM_BOOL, 'operation completed succesfully'),
|
|
"msg" => new \external_value(PARAM_TEXT, 'message'),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Connect a user to a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @param mixed $userid Id of user
|
|
* @return array Success/fail model
|
|
*/
|
|
public static function connect_user($studyplanid, $userid) {
|
|
global $CFG, $DB;
|
|
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $studyplan->context());
|
|
|
|
if (!$DB->record_exists('local_treestudyplan_user', ['studyplan_id' => $studyplanid, 'user_id' => $userid])) {
|
|
$id = $DB->insert_record('local_treestudyplan_user', [
|
|
'studyplan_id' => $studyplanid,
|
|
'user_id' => $userid,
|
|
]);
|
|
$studyplan->mark_csync_changed();
|
|
|
|
return ['success' => true, 'msg' => 'Cohort connected'];
|
|
|
|
} else {
|
|
return ['success' => true, 'msg' => 'Cohort already connected'];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function disconnect_user
|
|
*/
|
|
public static function disconnect_user_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
"user_id" => new \external_value(PARAM_INT, 'id of user to link', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function disconnect_user
|
|
*/
|
|
public static function disconnect_user_returns(): \external_description {
|
|
return new \external_single_structure([
|
|
"success" => new \external_value(PARAM_BOOL, 'operation completed succesfully'),
|
|
"msg" => new \external_value(PARAM_TEXT, 'message'),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Disconnect a user from a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @param mixed $userid Id of user
|
|
* @return array Success/fail model
|
|
*/
|
|
public static function disconnect_user($studyplanid, $userid) {
|
|
global $CFG, $DB;
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $studyplan->context());
|
|
|
|
if ($DB->record_exists('local_treestudyplan_user', ['studyplan_id' => $studyplanid, 'user_id' => $userid])) {
|
|
$DB->delete_records('local_treestudyplan_user', [
|
|
'studyplan_id' => $studyplanid,
|
|
'user_id' => $userid,
|
|
]);
|
|
|
|
$studyplan->mark_csync_changed();
|
|
|
|
return ['success' => true, 'msg' => 'User Disconnected'];
|
|
} else {
|
|
return ['success' => true, 'msg' => 'Connection does not exist'];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function associated_users
|
|
*/
|
|
public static function associated_users_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function associated_users
|
|
*/
|
|
public static function associated_users_returns(): \external_description {
|
|
return new \external_multiple_structure(self::user_structure());
|
|
}
|
|
|
|
/**
|
|
* List all users associated to a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @return array
|
|
*/
|
|
public static function associated_users($studyplanid) {
|
|
global $CFG, $DB;
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
if ($studyplan->is_coach()) {
|
|
\external_api::validate_context($studyplan->context());
|
|
} else {
|
|
webservicehelper::require_capabilities(self::CAP_VIEW, $studyplan->context());
|
|
}
|
|
|
|
$sql = "SELECT DISTINCT u.* FROM {user} u INNER JOIN {local_treestudyplan_user} j ON j.user_id = u.id
|
|
WHERE j.studyplan_id = :studyplan_id AND u.deleted != 1
|
|
ORDER BY u.lastname, u.firstname";
|
|
$rs = $DB->get_recordset_sql($sql, ['studyplan_id' => $studyplanid]);
|
|
|
|
/*
|
|
ID: 30
|
|
page: 33
|
|
plan: 28
|
|
*/
|
|
$users = [];
|
|
foreach ($rs as $u) {
|
|
$user = self::make_user_model($u);
|
|
$user["lastaccess"] = self::user_lastaccess($u->id, $studyplanid);
|
|
$users[] = $user;
|
|
|
|
}
|
|
$rs->close();
|
|
self::sortusermodels($users);
|
|
return $users;
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function associated_cohorts
|
|
*/
|
|
public static function associated_cohorts_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function associated_cohorts
|
|
*/
|
|
public static function associated_cohorts_returns(): \external_description {
|
|
return new \external_multiple_structure(self::cohort_structure());
|
|
}
|
|
|
|
/**
|
|
* List all cohorts associated to a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @return array
|
|
*/
|
|
public static function associated_cohorts($studyplanid) {
|
|
global $CFG, $DB;
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_VIEW, $studyplan->context());
|
|
|
|
$sql = "SELECT DISTINCT c.* FROM {cohort} c INNER JOIN {local_treestudyplan_cohort} j ON j.cohort_id = c.id
|
|
WHERE j.studyplan_id = :studyplan_id";
|
|
$rs = $DB->get_recordset_sql($sql, ['studyplan_id' => $studyplanid]);
|
|
$cohorts = [];
|
|
foreach ($rs as $c) {
|
|
$cohorts[] = self::make_cohort_model($c);
|
|
}
|
|
$rs->close();
|
|
return $cohorts;
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function all_associated
|
|
*/
|
|
public static function all_associated_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function all_associated
|
|
*/
|
|
public static function all_associated_returns(): \external_description {
|
|
return new \external_multiple_structure(self::user_structure());
|
|
}
|
|
|
|
/**
|
|
* List all users associated to a studyplan through either a cohort or directly
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @return array
|
|
*/
|
|
public static function all_associated($studyplanid) {
|
|
global $CFG, $DB;
|
|
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_VIEW, $studyplan->context());
|
|
|
|
$users = [];
|
|
// SQL JOIN script selecting all users that have a cohort linked to this studyplan .
|
|
// Or are directly linked.
|
|
$sql = "SELECT DISTINCT u.id, u.username, u.firstname, u.lastname, u.idnumber, u.email
|
|
FROM {user} u
|
|
LEFT JOIN {cohort_members} cm ON u.id = cm.userid
|
|
LEFT JOIN {local_treestudyplan_cohort} tc ON cm.cohortid = tc.cohort_id
|
|
LEFT JOIN {local_treestudyplan_user} tu ON u.id = tu.user_id
|
|
WHERE ( tc.studyplan_id = :studyplanid
|
|
OR tu.studyplan_id = :studyplanidtoo )
|
|
AND u.deleted != 1
|
|
ORDER BY u.lastname, u.firstname";
|
|
$rs = $DB->get_recordset_sql($sql, ["studyplanid" => $studyplan->id(), "studyplanidtoo" => $studyplan->id()]);
|
|
|
|
foreach ($rs as $u) {
|
|
$users[] = self::make_user_model($u);
|
|
}
|
|
$rs->close();
|
|
|
|
self::sortusermodels($users);
|
|
return $users;
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function all_associated
|
|
*/
|
|
public static function all_associated_grouped_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan'),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function all_associated
|
|
*/
|
|
public static function all_associated_grouped_returns(): \external_description {
|
|
return new \external_multiple_structure(new \external_single_structure([
|
|
'id' => new \external_value(PARAM_INT, 'id of group'),
|
|
'label' => new \external_value(PARAM_TEXT, 'group label'),
|
|
'users' => new \external_multiple_structure(self::user_structure()),
|
|
]));
|
|
}
|
|
|
|
/**
|
|
* List all users associated to a studyplan through either a cohort or directly
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @return array
|
|
*/
|
|
public static function all_associated_grouped($studyplanid) {
|
|
global $CFG, $DB;
|
|
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
if ($studyplan->is_coach()) {
|
|
\external_api::validate_context($studyplan->context());
|
|
} else {
|
|
webservicehelper::require_capabilities(self::CAP_VIEW, $studyplan->context());
|
|
}
|
|
|
|
$userlist = [
|
|
[
|
|
'id' => 0,
|
|
'label' => get_string("individuals", 'local_treestudyplan'),
|
|
'users' => self::associated_users($studyplanid),
|
|
],
|
|
];
|
|
|
|
$sql = "SELECT DISTINCT c.* FROM {cohort} c INNER JOIN {local_treestudyplan_cohort} j ON j.cohort_id = c.id
|
|
WHERE j.studyplan_id = :studyplan_id";
|
|
$crs = $DB->get_recordset_sql($sql, ['studyplan_id' => $studyplanid]);
|
|
foreach ($crs as $c) {
|
|
$users = [];
|
|
$sql = "SELECT DISTINCT u.id, u.username, u.firstname, u.lastname, u.idnumber, u.email
|
|
FROM {user} u
|
|
LEFT JOIN {cohort_members} cm ON u.id = cm.userid
|
|
WHERE cm.cohortid = :cohortid AND u.deleted != 1
|
|
ORDER BY u.lastname, u.firstname";
|
|
$rs = $DB->get_recordset_sql($sql, ["cohortid" => $c->id]);
|
|
|
|
foreach ($rs as $u) {
|
|
$user = self::make_user_model($u);
|
|
$user["lastaccess"] = self::user_lastaccess($u->id, $studyplanid);
|
|
$users[] = $user;
|
|
}
|
|
$rs->close();
|
|
|
|
$userlist[] = [
|
|
'id' => $c->id,
|
|
'label' => $c->name,
|
|
'users' => $users,
|
|
];
|
|
}
|
|
$crs->close();
|
|
|
|
return $userlist;
|
|
}
|
|
|
|
/**
|
|
* Sort a list of user models by firstname->lastname
|
|
* @param array $list Reference to list of user models
|
|
*/
|
|
public static function sortusermodels(&$list) {
|
|
return usort($list, function($a, $b) {
|
|
$m = [];
|
|
if (preg_match("/.*?([A-Z].*)/", $a['lastname'], $m)) {
|
|
$sortlna = $m[1];
|
|
} else {
|
|
$sortlna = $a['lastname'];
|
|
}
|
|
if (preg_match("/.*?([A-Z].*)/", $b['lastname'], $m)) {
|
|
$sortlnb = $m[1];
|
|
} else {
|
|
$sortlnb = $b['lastname'];
|
|
}
|
|
$cmp = $sortlna <=> $sortlnb;
|
|
return ($cmp != 0) ? $cmp : $a['firstname'] <=> $b['firstname'];
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function cascade_cohortsync
|
|
*/
|
|
public static function cascade_cohortsync_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function cascade_cohortsync
|
|
*/
|
|
public static function cascade_cohortsync_returns(): \external_description {
|
|
return success::structure();
|
|
}
|
|
|
|
/**
|
|
* Perform cascading cohort sync for a given studyplan
|
|
* This adds a cohort sync enrolment to all linked cohorts for each course listed in the studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @return array Success/fail model
|
|
*/
|
|
public static function cascade_cohortsync($studyplanid) {
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $studyplan->context());
|
|
|
|
autocohortsync::syncplan($studyplan);
|
|
|
|
return success::success()->model();
|
|
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function connect_user
|
|
*/
|
|
public static function connect_coach_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
"user_id" => new \external_value(PARAM_INT, 'id of user to link', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function connect_user
|
|
*/
|
|
public static function connect_coach_returns(): \external_description {
|
|
return new \external_single_structure([
|
|
"success" => new \external_value(PARAM_BOOL, 'operation completed succesfully'),
|
|
"msg" => new \external_value(PARAM_TEXT, 'message'),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Connect a user to a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @param mixed $userid Id of user
|
|
* @return array Success/fail model
|
|
*/
|
|
public static function connect_coach($studyplanid, $userid) {
|
|
global $CFG, $DB;
|
|
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $studyplan->context());
|
|
|
|
$user = $DB->get_record("user", ["id" => $userid]);
|
|
if (has_capability(self::CAP_COACH, $studyplan->context(), $user)) {
|
|
if (!$DB->record_exists('local_treestudyplan_coach', ['studyplan_id' => $studyplanid, 'user_id' => $userid])) {
|
|
$id = $DB->insert_record('local_treestudyplan_coach', [
|
|
'studyplan_id' => $studyplanid,
|
|
'user_id' => $userid,
|
|
]);
|
|
|
|
return ['success' => true, 'msg' => 'User connected as coach'];
|
|
|
|
} else {
|
|
return ['success' => true, 'msg' => 'User already connected as coach'];
|
|
}
|
|
} else {
|
|
return ['success' => false, 'msg' => 'User does not have coach capability in this studyplan\'s context'];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function disconnect_user
|
|
*/
|
|
public static function disconnect_coach_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
"user_id" => new \external_value(PARAM_INT, 'id of user to link', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function disconnect_user
|
|
*/
|
|
public static function disconnect_coach_returns(): \external_description {
|
|
return new \external_single_structure([
|
|
"success" => new \external_value(PARAM_BOOL, 'operation completed succesfully'),
|
|
"msg" => new \external_value(PARAM_TEXT, 'message'),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Disconnect a user from a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @param mixed $userid Id of user
|
|
* @return array Success/fail model
|
|
*/
|
|
public static function disconnect_coach($studyplanid, $userid) {
|
|
global $CFG, $DB;
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $studyplan->context());
|
|
|
|
if ($DB->record_exists('local_treestudyplan_coach', ['studyplan_id' => $studyplanid, 'user_id' => $userid])) {
|
|
$DB->delete_records('local_treestudyplan_coach', [
|
|
'studyplan_id' => $studyplanid,
|
|
'user_id' => $userid,
|
|
]);
|
|
|
|
return ['success' => true, 'msg' => 'User Disconnected as coach'];
|
|
} else {
|
|
return ['success' => true, 'msg' => 'Connection does not exist'];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parameter description for webservice function find_user
|
|
*/
|
|
public static function find_coach_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
'like' => new \external_value(PARAM_TEXT, 'search text'),
|
|
'studyplan_id' => new \external_value(PARAM_INT, 'studyplan id to associate for'),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function find_user
|
|
*/
|
|
public static function find_coach_returns(): \external_description {
|
|
return new \external_multiple_structure(self::user_structure());
|
|
}
|
|
|
|
/**
|
|
* Search users for match
|
|
* @param string $like String to match user firstname/lastname with
|
|
* @param int $studyplanid Id of studyplan to search for
|
|
* @return array
|
|
*/
|
|
public static function find_coach($like, $studyplanid) {
|
|
global $CFG, $DB;
|
|
|
|
// Only allow this if the user has the right to edit in this context.
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
$context = $studyplan->context();
|
|
webservicehelper::require_capabilities(self::CAP_EDIT, $context);
|
|
|
|
$pattern = "%{$like}%";
|
|
$params = ["pattern_fn" => $pattern,
|
|
"pattern_ln" => $pattern,
|
|
"pattern_un" => $pattern,
|
|
];
|
|
$sql = "SELECT DISTINCT u.* FROM {user} u LEFT JOIN {local_treestudyplan_coach} j ON u.id = j.user_id
|
|
WHERE u.deleted != 1 AND (firstname LIKE :pattern_fn OR lastname LIKE :pattern_ln OR username LIKE :pattern_un)";
|
|
if (isset($excludeid) && is_numeric($excludeid)) {
|
|
$sql .= " AND (j.studyplan_id IS NULL OR j.studyplan_id != :exclude_id)";
|
|
$params['exclude_id'] = $studyplan;
|
|
}
|
|
|
|
$users = [];
|
|
$rs = $DB->get_recordset_sql($sql, $params);
|
|
foreach ($rs as $r) {
|
|
if (has_capability(self::CAP_COACH, $context, $r)) {
|
|
$users[] = static::make_user_model($r);
|
|
}
|
|
}
|
|
$rs->close();
|
|
|
|
self::sortusermodels($users);
|
|
return $users;
|
|
}
|
|
/**
|
|
* Parameter description for webservice function associated_users
|
|
*/
|
|
public static function associated_coaches_parameters(): \external_function_parameters {
|
|
return new \external_function_parameters( [
|
|
"studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan', VALUE_OPTIONAL),
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Return value description for webservice function associated_users
|
|
*/
|
|
public static function associated_coaches_returns(): \external_description {
|
|
return new \external_multiple_structure(self::user_structure());
|
|
}
|
|
|
|
/**
|
|
* List all users associated to a studyplan
|
|
* @param mixed $studyplanid Id of studyplan
|
|
* @return array
|
|
*/
|
|
public static function associated_coaches($studyplanid) {
|
|
global $CFG, $DB;
|
|
$studyplan = studyplan::find_by_id($studyplanid);
|
|
webservicehelper::require_capabilities(self::CAP_VIEW, $studyplan->context());
|
|
|
|
$sql = "SELECT DISTINCT u.* FROM {user} u INNER JOIN {local_treestudyplan_coach} j ON j.user_id = u.id
|
|
WHERE j.studyplan_id = :studyplan_id AND u.deleted != 1
|
|
ORDER BY u.lastname, u.firstname";
|
|
$rs = $DB->get_recordset_sql($sql, ['studyplan_id' => $studyplanid]);
|
|
|
|
$users = [];
|
|
foreach ($rs as $u) {
|
|
$user = self::make_user_model($u);
|
|
$users[] = $user;
|
|
}
|
|
$rs->close();
|
|
self::sortusermodels($users);
|
|
return $users;
|
|
}
|
|
|
|
}
|