AutorÃa | Ultima modificación | Ver Log |
<?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/>.
namespace enrol_lti\local\ltiadvantage\repository;
use enrol_lti\local\ltiadvantage\entity\user;
/**
* Class user_repository.
*
* This class encapsulates persistence logic for \enrol_lti\local\entity\user type objects.
*
* @package enrol_lti
* @copyright 2021 Jake Dallimore <jrhdallimore@gmail.com>
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
class user_repository {
/** @var string $ltiuserstable the name of the table to which the entity will be persisted.*/
private $ltiuserstable = 'enrol_lti_users';
/** @var string $userresourcelinkidtable the name of the join table mapping users to resource links.*/
private $userresourcelinkidtable = 'enrol_lti_user_resource_link';
/**
* Convert a record into a user object and return it.
*
* @param \stdClass $userrecord the raw data from relevant tables required to instantiate a user.
* @return user a user object.
*/
private function user_from_record(\stdClass $userrecord): user {
return user::create(
$userrecord->toolid,
$userrecord->localid,
$userrecord->ltideploymentid,
$userrecord->sourceid,
$userrecord->lang,
$userrecord->timezone,
$userrecord->city,
$userrecord->country,
$userrecord->institution,
$userrecord->maildisplay,
$userrecord->lastgrade,
$userrecord->lastaccess,
$userrecord->resourcelinkid ?? null,
(int) $userrecord->id
);
}
/**
* Create a list of user instances from a list of records.
*
* @param array $records the array of records.
* @return array of user instances.
*/
private function users_from_records(array $records): array {
$users = [];
foreach ($records as $record) {
$users[] = $this->user_from_record($record);
}
return $users;
}
/**
* Get a stdClass object ready for persisting, based on the supplied user object.
*
* @param user $user the user instance.
* @return \stdClass the record.
*/
private function user_record_from_user(user $user): \stdClass {
return (object) [
'id' => $user->get_localid(),
'city' => $user->get_city(),
'country' => $user->get_country(),
'institution' => $user->get_institution(),
'timezone' => $user->get_timezone(),
'maildisplay' => $user->get_maildisplay(),
'lang' => $user->get_lang()
];
}
/**
* Create the corresponding enrol_lti_user record from a user instance.
*
* @param user $user the user instance.
* @return \stdClass the record.
*/
private function lti_user_record_from_user(user $user): \stdClass {
$record = [
'toolid' => $user->get_resourceid(),
'ltideploymentid' => $user->get_deploymentid(),
'sourceid' => $user->get_sourceid(),
'lastgrade' => $user->get_lastgrade(),
'lastaccess' => $user->get_lastaccess(),
];
if ($user->get_id()) {
$record['id'] = $user->get_id();
}
return (object) $record;
}
/**
* Helper to validate user:tool uniqueness across a deployment.
*
* The DB cannot be relied on to do this uniqueness check, since the table is shared by LTI 1.1/2.0 data.
*
* @param user $user the user instance.
* @return bool true if found, false otherwise.
*/
private function user_exists_for_tool(user $user): bool {
// Lack of an id doesn't preclude the object from existence in the store. It may be stale, without an id.
// The user can still be found by checking their lti advantage user creds and correlating that to the relevant
// lti_user entry (where tool matches the user object's resource).
global $DB;
$uniquesql = "SELECT lu.id
FROM {{$this->ltiuserstable}} lu
WHERE lu.toolid = :toolid
AND lu.userid = :userid";
$params = ['toolid' => $user->get_resourceid(), 'userid' => $user->get_localid()];
return $DB->record_exists_sql($uniquesql, $params);
}
/**
* Save a user instance in the store.
*
* @param user $user the object to save.
* @return user the saved object.
*/
public function save(user $user): user {
global $DB;
$id = $user->get_id();
$exists = !is_null($id) && $this->exists($id);
if ($id && !$exists) {
throw new \coding_exception("Cannot save lti user with id '{$id}'. The record does not exist.");
}
$userrecord = $this->user_record_from_user($user);
$ltiuserrecord = $this->lti_user_record_from_user($user);
$timenow = time();
global $CFG;
require_once($CFG->dirroot . '/user/lib.php');
if ($exists) {
$ltiuser = $DB->get_record($this->ltiuserstable, ['id' => $ltiuserrecord->id]);
$userid = $ltiuser->userid;
// Warn about localid vs ltiuser->userid mismatches here. Callers shouldn't be able to force updates using
// localid. Only new user associations can be created that way.
if (!empty($userrecord->id) && $userid != $userrecord->id) {
throw new \coding_exception("Cannot update user mapping. LTI user '{$ltiuser->id}' is already mapped " .
"to user '{$ltiuser->userid}' and can't be associated with another user '{$userrecord->id}'.");
}
// Only update the Moodle user record if something has changed.
$rawuser = \core_user::get_user($userrecord->id);
$userfieldstocompare = array_intersect_key(
(array) $rawuser,
(array) $userrecord
);
if (!empty(array_diff((array) $userrecord, $userfieldstocompare))) {
\user_update_user($userrecord);
}
unset($userrecord->id);
$ltiuserrecord->timemodified = $timenow;
$DB->update_record($this->ltiuserstable, $ltiuserrecord);
} else {
// Validate uniqueness of the lti user, in the case of a stale object coming in to be saved.
if ($this->user_exists_for_tool($user)) {
throw new \coding_exception("Cannot create duplicate LTI user '{$user->get_localid()}' for resource " .
"'{$user->get_resourceid()}'.");
}
// Only update the Moodle user record if something has changed.
$userid = $userrecord->id;
$rawuser = \core_user::get_user($userid);
$userfieldstocompare = array_intersect_key(
(array) $rawuser,
(array) $userrecord
);
if (!empty(array_diff((array) $userrecord, $userfieldstocompare))) {
\user_update_user($userrecord);
}
unset($userrecord->id);
// Create the lti_user record, holding details that have a lifespan equal to that of the enrolment instance.
$ltiuserrecord->timecreated = $ltiuserrecord->timemodified = $timenow;
$ltiuserrecord->userid = $userid;
$ltiuserrecord->id = $DB->insert_record($this->ltiuserstable, $ltiuserrecord);
}
// If the user was created via a resource_link, create that association.
if ($reslinkid = $user->get_resourcelinkid()) {
$resourcelinkmap = ['ltiuserid' => $ltiuserrecord->id, 'resourcelinkid' => $reslinkid];
if (!$DB->record_exists($this->userresourcelinkidtable, $resourcelinkmap)) {
$DB->insert_record($this->userresourcelinkidtable, $resourcelinkmap);
}
}
$resourcelinkmap = $resourcelinkmap ?? [];
// Transform the data into something that looks like a read and can be processed by user_from_record.
$record = (object) array_merge(
(array) $userrecord,
(array) $ltiuserrecord,
$resourcelinkmap,
['localid' => $userid]
);
return $this->user_from_record($record);
}
/**
* Find and return a user by id.
*
* @param int $id the id of the user object.
* @return user|null the user object, or null if the object cannot be found.
*/
public function find(int $id): ?user {
global $DB;
try {
$sql = "SELECT lu.id, u.id as localid, u.username, u.firstname, u.lastname, u.email, u.city, u.country,
u.institution, u.timezone, u.maildisplay, u.lang, lu.sourceid, lu.toolid, lu.lastgrade,
lu.lastaccess, lu.ltideploymentid
FROM {{$this->ltiuserstable}} lu
JOIN {user} u
ON (u.id = lu.userid)
WHERE lu.id = :id
AND lu.ltideploymentid IS NOT NULL";
$record = $DB->get_record_sql($sql, ['id' => $id], MUST_EXIST);
return $this->user_from_record($record);
} catch (\dml_missing_record_exception $ex) {
return null;
}
}
/**
* Find an lti user instance by resource.
*
* @param int $userid the id of the moodle user to look for.
* @param int $resourceid the id of the published resource.
* @return user|null the lti user instance, or null if not found.
*/
public function find_single_user_by_resource(int $userid, int $resourceid): ?user {
global $DB;
try {
// Find the lti advantage user record.
$sql = "SELECT lu.id, u.id as localid, u.username, u.firstname, u.lastname, u.email, u.city, u.country,
u.institution, u.timezone, u.maildisplay, u.lang, lu.sourceid, lu.toolid, lu.lastgrade,
lu.lastaccess, lu.ltideploymentid
FROM {{$this->ltiuserstable}} lu
JOIN {user} u
ON (u.id = lu.userid)
WHERE lu.userid = :userid
AND lu.toolid = :resourceid
AND lu.ltideploymentid IS NOT NULL";
$params = ['userid' => $userid, 'resourceid' => $resourceid];
$record = $DB->get_record_sql($sql, $params, MUST_EXIST);
return $this->user_from_record($record);
} catch (\dml_missing_record_exception $ex) {
return null;
}
}
/**
* Find all users for a particular shared resource.
*
* @param int $resourceid the id of the shared resource.
* @return array the array of users, empty if none were found.
*/
public function find_by_resource(int $resourceid): array {
global $DB;
$sql = "SELECT lu.id, u.id as localid, u.username, u.firstname, u.lastname, u.email, u.city, u.country,
u.institution, u.timezone, u.maildisplay, u.lang, lu.sourceid, lu.toolid, lu.lastgrade,
lu.lastaccess, lu.ltideploymentid
FROM {{$this->ltiuserstable}} lu
JOIN {user} u
ON (u.id = lu.userid)
WHERE lu.toolid = :resourceid
AND lu.ltideploymentid IS NOT NULL
ORDER BY lu.lastaccess DESC";
$records = $DB->get_records_sql($sql, ['resourceid' => $resourceid]);
return $this->users_from_records($records);
}
/**
* Get a list of users associated with the given resource link.
*
* @param int $resourcelinkid the id of the resource_link instance with which the users are associated.
* @return array the array of users, empty if none were found.
*/
public function find_by_resource_link(int $resourcelinkid) {
global $DB;
$sql = "SELECT lu.id, u.id as localid, u.username, u.firstname, u.lastname, u.email, u.city, u.country,
u.institution, u.timezone, u.maildisplay, u.lang, lu.sourceid, lu.toolid, lu.lastgrade,
lu.lastaccess, lu.ltideploymentid
FROM {{$this->ltiuserstable}} lu
JOIN {user} u
ON (u.id = lu.userid)
JOIN {{$this->userresourcelinkidtable}} url
ON (url.ltiuserid = lu.id)
WHERE url.resourcelinkid = :resourcelinkid
ORDER BY lu.lastaccess DESC";
$records = $DB->get_records_sql($sql, ['resourcelinkid' => $resourcelinkid]);
return $this->users_from_records($records);
}
/**
* Check whether or not the given user object exists.
*
* @param int $id the unique id the user.
* @return bool true if found, false otherwise.
*/
public function exists(int $id): bool {
global $DB;
return $DB->record_exists($this->ltiuserstable, ['id' => $id]);
}
/**
* Delete a user based on id.
*
* @param int $id the id of the user to remove.
*/
public function delete(int $id) {
global $DB;
$DB->delete_records($this->ltiuserstable, ['id' => $id]);
$DB->delete_records($this->userresourcelinkidtable, ['ltiuserid' => $id]);
}
/**
* Delete all lti user instances based on a given local deployment instance id.
*
* @param int $deploymentid the local id of the deployment instance to which the users belong.
*/
public function delete_by_deployment(int $deploymentid): void {
global $DB;
$DB->delete_records($this->ltiuserstable, ['ltideploymentid' => $deploymentid]);
}
}