You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
526 lines
21 KiB
526 lines
21 KiB
<?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/>.
|
|
/**
|
|
* Externallib.php file for attendance plugin.
|
|
*
|
|
* @package mod_attendance
|
|
* @copyright 2015 Caio Bressan Doneda
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
|
|
defined('MOODLE_INTERNAL') || die;
|
|
|
|
require_once($CFG->libdir . '/externallib.php');
|
|
require_once($CFG->libdir . '/filelib.php');
|
|
require_once(dirname(__FILE__).'/classes/attendance_webservices_handler.php');
|
|
|
|
/**
|
|
* Class mod_attendance_external
|
|
* @copyright 2015 Caio Bressan Doneda
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
class mod_attendance_external extends external_api {
|
|
|
|
/**
|
|
* Describes the parameters for add_attendance.
|
|
*
|
|
* @return external_function_parameters
|
|
*/
|
|
public static function add_attendance_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'courseid' => new external_value(PARAM_INT, 'course id'),
|
|
'name' => new external_value(PARAM_TEXT, 'attendance name'),
|
|
'intro' => new external_value(PARAM_RAW, 'attendance description', VALUE_DEFAULT, ''),
|
|
'groupmode' => new external_value(PARAM_INT, 'group mode (0 - no groups, 1 - separate groups, 2 - visible groups)', VALUE_DEFAULT, 0),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Adds attendance instance to course.
|
|
*
|
|
* @param int $courseid
|
|
* @param string $name
|
|
* @param string $intro
|
|
* @param int $groupmode
|
|
* @return array
|
|
*/
|
|
public static function add_attendance(int $courseid, $name, $intro, int $groupmode) {
|
|
global $CFG, $DB;
|
|
require_once($CFG->dirroot.'/course/modlib.php');
|
|
|
|
$params = self::validate_parameters(self::add_attendance_parameters(), array(
|
|
'courseid' => $courseid,
|
|
'name' => $name,
|
|
'intro' => $intro,
|
|
'groupmode' => $groupmode,
|
|
));
|
|
|
|
// Get course.
|
|
$course = $DB->get_record('course', array('id' => $params['courseid']), '*', MUST_EXIST);
|
|
|
|
// Verify permissions.
|
|
list($module, $context) = can_add_moduleinfo($course, 'attendance', 0);
|
|
self::validate_context($context);
|
|
require_capability('mod/attendance:addinstance', $context);
|
|
|
|
// Verify group mode.
|
|
if (!in_array($params['groupmode'], array(NOGROUPS, SEPARATEGROUPS, VISIBLEGROUPS))) {
|
|
throw new invalid_parameter_exception('Group mode is invalid.');
|
|
}
|
|
|
|
// Populate modinfo object.
|
|
$moduleinfo = new stdClass();
|
|
$moduleinfo->modulename = 'attendance';
|
|
$moduleinfo->module = $module->id;
|
|
|
|
$moduleinfo->name = $params['name'];
|
|
$moduleinfo->intro = $params['intro'];
|
|
$moduleinfo->introformat = FORMAT_HTML;
|
|
|
|
$moduleinfo->section = 0;
|
|
$moduleinfo->visible = 1;
|
|
$moduleinfo->visibleoncoursepage = 1;
|
|
$moduleinfo->cmidnumber = '';
|
|
$moduleinfo->groupmode = $params['groupmode'];
|
|
$moduleinfo->groupingid = 0;
|
|
|
|
// Add the module to the course.
|
|
$moduleinfo = add_moduleinfo($moduleinfo, $course);
|
|
|
|
return array('attendanceid' => $moduleinfo->instance);
|
|
}
|
|
|
|
/**
|
|
* Describes add_attendance return values.
|
|
*
|
|
* @return external_multiple_structure
|
|
*/
|
|
public static function add_attendance_returns() {
|
|
return new external_single_structure(array(
|
|
'attendanceid' => new external_value(PARAM_INT, 'instance id of the created attendance'),
|
|
));
|
|
}
|
|
|
|
/**
|
|
* Describes the parameters for remove_attendance.
|
|
*
|
|
* @return external_function_parameters
|
|
*/
|
|
public static function remove_attendance_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'attendanceid' => new external_value(PARAM_INT, 'attendance instance id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Remove attendance instance.
|
|
*
|
|
* @param int $attendanceid
|
|
*/
|
|
public static function remove_attendance(int $attendanceid) {
|
|
$params = self::validate_parameters(self::remove_attendance_parameters(), array(
|
|
'attendanceid' => $attendanceid,
|
|
));
|
|
|
|
$cm = get_coursemodule_from_instance('attendance', $params['attendanceid'], 0, false, MUST_EXIST);
|
|
|
|
// Check permissions.
|
|
$context = context_module::instance($cm->id);
|
|
self::validate_context($context);
|
|
require_capability('mod/attendance:manageattendances', $context);
|
|
|
|
// Delete attendance instance.
|
|
attendance_delete_instance($params['attendanceid']);
|
|
rebuild_course_cache($cm->course, true);
|
|
}
|
|
|
|
/**
|
|
* Describes remove_attendance return values.
|
|
*
|
|
* @return void
|
|
*/
|
|
public static function remove_attendance_returns() {
|
|
}
|
|
|
|
/**
|
|
* Describes the parameters for add_session.
|
|
*
|
|
* @return external_function_parameters
|
|
*/
|
|
public static function add_session_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'attendanceid' => new external_value(PARAM_INT, 'attendance instance id'),
|
|
'description' => new external_value(PARAM_RAW, 'description', VALUE_DEFAULT, ''),
|
|
'sessiontime' => new external_value(PARAM_INT, 'session start timestamp'),
|
|
'duration' => new external_value(PARAM_INT, 'session duration (seconds)', VALUE_DEFAULT, 0),
|
|
'groupid' => new external_value(PARAM_INT, 'group id', VALUE_DEFAULT, 0),
|
|
'addcalendarevent' => new external_value(PARAM_BOOL, 'add calendar event', VALUE_DEFAULT, true),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Adds session to attendance instance.
|
|
*
|
|
* @param int $attendanceid
|
|
* @param string $description
|
|
* @param int $sessiontime
|
|
* @param int $duration
|
|
* @param int $groupid
|
|
* @param bool $addcalendarevent
|
|
* @return array
|
|
*/
|
|
public static function add_session(int $attendanceid, $description, int $sessiontime, int $duration, int $groupid, bool $addcalendarevent) {
|
|
global $USER, $DB;
|
|
|
|
$params = self::validate_parameters(self::add_session_parameters(), array(
|
|
'attendanceid' => $attendanceid,
|
|
'description' => $description,
|
|
'sessiontime' => $sessiontime,
|
|
'duration' => $duration,
|
|
'groupid' => $groupid,
|
|
'addcalendarevent' => $addcalendarevent,
|
|
));
|
|
|
|
$cm = get_coursemodule_from_instance('attendance', $params['attendanceid'], 0, false, MUST_EXIST);
|
|
$course = $DB->get_record('course', array('id' => $cm->course), '*', MUST_EXIST);
|
|
$attendance = $DB->get_record('attendance', array('id' => $cm->instance), '*', MUST_EXIST);
|
|
|
|
// Check permissions.
|
|
$context = context_module::instance($cm->id);
|
|
self::validate_context($context);
|
|
require_capability('mod/attendance:manageattendances', $context);
|
|
|
|
// Validate group.
|
|
$groupid = $params['groupid'];
|
|
$groupmode = (int)groups_get_activity_groupmode($cm);
|
|
if ($groupmode === NOGROUPS && $groupid > 0) {
|
|
throw new invalid_parameter_exception('Group id is specified, but group mode is disabled for activity');
|
|
} else if ($groupmode === SEPARATEGROUPS && $groupid === 0) {
|
|
throw new invalid_parameter_exception('Group id is not specified (or 0) in separate groups mode.');
|
|
}
|
|
if ($groupmode === SEPARATEGROUPS || ($groupmode === VISIBLEGROUPS && $groupid > 0)) {
|
|
// Determine valid groups
|
|
$userid = has_capability('moodle/site:accessallgroups', $context) ? 0 : $USER->id;
|
|
$validgroupids = array_map(function($group) {
|
|
return $group->id;
|
|
}, groups_get_all_groups($course->id, $userid, $cm->groupingid));
|
|
if (!in_array($groupid, $validgroupids)) {
|
|
throw new invalid_parameter_exception('Invalid group id');
|
|
}
|
|
}
|
|
|
|
// Get attendance.
|
|
$attendance = new mod_attendance_structure($attendance, $cm, $course, $context);
|
|
|
|
// Create session.
|
|
$sess = new stdClass();
|
|
$sess->sessdate = $params['sessiontime'];
|
|
$sess->duration = $params['duration'];
|
|
$sess->descriptionitemid = 0;
|
|
$sess->description = $params['description'];
|
|
$sess->descriptionformat = FORMAT_HTML;
|
|
$sess->calendarevent = (int) $params['addcalendarevent'];
|
|
$sess->timemodified = time();
|
|
$sess->studentscanmark = 0;
|
|
$sess->autoassignstatus = 0;
|
|
$sess->subnet = '';
|
|
$sess->studentpassword = '';
|
|
$sess->automark = 0;
|
|
$sess->automarkcompleted = 0;
|
|
$sess->absenteereport = get_config('attendance', 'absenteereport_default');
|
|
$sess->includeqrcode = 0;
|
|
$sess->subnet = $attendance->subnet;
|
|
$sess->statusset = 0;
|
|
$sess->groupid = $groupid;
|
|
|
|
$sessionid = $attendance->add_session($sess);
|
|
return array('sessionid' => $sessionid);
|
|
}
|
|
|
|
/**
|
|
* Describes add_session return values.
|
|
*
|
|
* @return external_multiple_structure
|
|
*/
|
|
public static function add_session_returns() {
|
|
return new external_single_structure(array(
|
|
'sessionid' => new external_value(PARAM_INT, 'id of the created session'),
|
|
));
|
|
}
|
|
|
|
/**
|
|
* Describes the parameters for remove_session.
|
|
*
|
|
* @return external_function_parameters
|
|
*/
|
|
public static function remove_session_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'sessionid' => new external_value(PARAM_INT, 'session id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Delete session from attendance instance.
|
|
*
|
|
* @param int $sessionid
|
|
* @return int $sessionid
|
|
*/
|
|
public static function remove_session(int $sessionid) {
|
|
global $DB;
|
|
|
|
$params = self::validate_parameters(self::remove_session_parameters(),
|
|
array('sessionid' => $sessionid));
|
|
|
|
$session = $DB->get_record('attendance_sessions', array('id' => $params['sessionid']), '*', MUST_EXIST);
|
|
$attendance = $DB->get_record('attendance', array('id' => $session->attendanceid), '*', MUST_EXIST);
|
|
$cm = get_coursemodule_from_instance('attendance', $attendance->id, 0, false, MUST_EXIST);
|
|
$course = $DB->get_record('course', array('id' => $cm->course), '*', MUST_EXIST);
|
|
|
|
// Check permissions.
|
|
$context = context_module::instance($cm->id);
|
|
self::validate_context($context);
|
|
require_capability('mod/attendance:manageattendances', $context);
|
|
|
|
// Get attendance.
|
|
$attendance = new mod_attendance_structure($attendance, $cm, $course, $context);
|
|
|
|
// Delete session.
|
|
$attendance->delete_sessions(array($sessionid));
|
|
attendance_update_users_grade($attendance);
|
|
}
|
|
|
|
/**
|
|
* Describes remove_session return values.
|
|
*
|
|
* @return void
|
|
*/
|
|
public static function remove_session_returns() {
|
|
}
|
|
|
|
/**
|
|
* Get parameter list.
|
|
* @return external_function_parameters
|
|
*/
|
|
public static function get_courses_with_today_sessions_parameters() {
|
|
return new external_function_parameters (
|
|
array('userid' => new external_value(PARAM_INT, 'User id.', VALUE_DEFAULT, 0)));
|
|
}
|
|
|
|
/**
|
|
* Get list of courses with active sessions for today.
|
|
* @param int $userid
|
|
* @return array
|
|
*/
|
|
public static function get_courses_with_today_sessions($userid) {
|
|
global $DB;
|
|
|
|
$params = self::validate_parameters(self::get_courses_with_today_sessions_parameters(), array(
|
|
'userid' => $userid,
|
|
));
|
|
|
|
// Check user id is valid.
|
|
$user = $DB->get_record('user', array('id' => $params['userid']), '*', MUST_EXIST);
|
|
|
|
// Capability check is done in get_courses_with_today_sessions
|
|
// as it switches contexts in loop for each course.
|
|
return attendance_handler::get_courses_with_today_sessions($params['userid']);
|
|
}
|
|
|
|
/**
|
|
* Get structure of an attendance session.
|
|
*
|
|
* @return array
|
|
*/
|
|
private static function get_session_structure() {
|
|
$session = array('id' => new external_value(PARAM_INT, 'Session id.'),
|
|
'attendanceid' => new external_value(PARAM_INT, 'Attendance id.'),
|
|
'groupid' => new external_value(PARAM_INT, 'Group id.'),
|
|
'sessdate' => new external_value(PARAM_INT, 'Session date.'),
|
|
'duration' => new external_value(PARAM_INT, 'Session duration.'),
|
|
'lasttaken' => new external_value(PARAM_INT, 'Session last taken time.'),
|
|
'lasttakenby' => new external_value(PARAM_INT, 'ID of the last user that took this session.'),
|
|
'timemodified' => new external_value(PARAM_INT, 'Time modified.'),
|
|
'description' => new external_value(PARAM_TEXT, 'Session description.'),
|
|
'descriptionformat' => new external_value(PARAM_INT, 'Session description format.'),
|
|
'studentscanmark' => new external_value(PARAM_INT, 'Students can mark their own presence.'),
|
|
'absenteereport' => new external_value(PARAM_INT, 'Session included in absetee reports.'),
|
|
'autoassignstatus' => new external_value(PARAM_INT, 'Automatically assign a status to students.'),
|
|
'preventsharedip' => new external_value(PARAM_INT, 'Prevent students from sharing IP addresses.'),
|
|
'preventsharediptime' => new external_value(PARAM_INT, 'Time delay before IP address is allowed again.'),
|
|
'statusset' => new external_value(PARAM_INT, 'Session statusset.'),
|
|
'includeqrcode' => new external_value(PARAM_INT, 'Include QR code when displaying password'));
|
|
|
|
return $session;
|
|
}
|
|
|
|
/**
|
|
* Show structure of return.
|
|
* @return external_multiple_structure
|
|
*/
|
|
public static function get_courses_with_today_sessions_returns() {
|
|
$todaysessions = self::get_session_structure();
|
|
|
|
$attendanceinstances = array('name' => new external_value(PARAM_TEXT, 'Attendance name.'),
|
|
'today_sessions' => new external_multiple_structure(
|
|
new external_single_structure($todaysessions)));
|
|
|
|
$courses = array('shortname' => new external_value(PARAM_TEXT, 'short name of a moodle course.'),
|
|
'fullname' => new external_value(PARAM_TEXT, 'full name of a moodle course.'),
|
|
'attendance_instances' => new external_multiple_structure(
|
|
new external_single_structure($attendanceinstances)));
|
|
|
|
return new external_multiple_structure(new external_single_structure(($courses)));
|
|
}
|
|
|
|
/**
|
|
* Get session params.
|
|
*
|
|
* @return external_function_parameters
|
|
*/
|
|
public static function get_session_parameters() {
|
|
return new external_function_parameters (
|
|
array('sessionid' => new external_value(PARAM_INT, 'session id')));
|
|
}
|
|
|
|
/**
|
|
* Get session.
|
|
*
|
|
* @param int $sessionid
|
|
* @return mixed
|
|
*/
|
|
public static function get_session($sessionid) {
|
|
global $DB;
|
|
|
|
$params = self::validate_parameters(self::get_session_parameters(), array(
|
|
'sessionid' => $sessionid,
|
|
));
|
|
|
|
$session = $DB->get_record('attendance_sessions', array('id' => $params['sessionid']), '*', MUST_EXIST);
|
|
$attendance = $DB->get_record('attendance', array('id' => $session->attendanceid), '*', MUST_EXIST);
|
|
$cm = get_coursemodule_from_instance('attendance', $attendance->id, 0, false, MUST_EXIST);
|
|
|
|
// Check permissions.
|
|
$context = context_module::instance($cm->id);
|
|
self::validate_context($context);
|
|
$capabilities = array(
|
|
'mod/attendance:manageattendances',
|
|
'mod/attendance:takeattendances',
|
|
'mod/attendance:changeattendances'
|
|
);
|
|
if (!has_any_capability($capabilities, $context)) {
|
|
throw new invalid_parameter_exception('Invalid session id or no permissions.');
|
|
}
|
|
|
|
return attendance_handler::get_session($sessionid);
|
|
}
|
|
|
|
/**
|
|
* Show return values of get_session.
|
|
*
|
|
* @return external_single_structure
|
|
*/
|
|
public static function get_session_returns() {
|
|
$statuses = array('id' => new external_value(PARAM_INT, 'Status id.'),
|
|
'attendanceid' => new external_value(PARAM_INT, 'Attendance id.'),
|
|
'acronym' => new external_value(PARAM_TEXT, 'Status acronym.'),
|
|
'description' => new external_value(PARAM_TEXT, 'Status description.'),
|
|
'grade' => new external_value(PARAM_FLOAT, 'Status grade.'),
|
|
'visible' => new external_value(PARAM_INT, 'Status visibility.'),
|
|
'deleted' => new external_value(PARAM_INT, 'informs if this session was deleted.'),
|
|
'setnumber' => new external_value(PARAM_INT, 'Set number.'));
|
|
|
|
$users = array('id' => new external_value(PARAM_INT, 'User id.'),
|
|
'firstname' => new external_value(PARAM_TEXT, 'User first name.'),
|
|
'lastname' => new external_value(PARAM_TEXT, 'User last name.'));
|
|
|
|
$attendancelog = array('studentid' => new external_value(PARAM_INT, 'Student id.'),
|
|
'statusid' => new external_value(PARAM_TEXT, 'Status id (last time).'),
|
|
'remarks' => new external_value(PARAM_TEXT, 'Last remark.'),
|
|
'id' => new external_value(PARAM_TEXT, 'log id.'));
|
|
|
|
$session = self::get_session_structure();
|
|
$session['courseid'] = new external_value(PARAM_INT, 'Course moodle id.');
|
|
$session['statuses'] = new external_multiple_structure(new external_single_structure($statuses));
|
|
$session['attendance_log'] = new external_multiple_structure(new external_single_structure($attendancelog));
|
|
$session['users'] = new external_multiple_structure(new external_single_structure($users));
|
|
|
|
return new external_single_structure($session);
|
|
}
|
|
|
|
/**
|
|
* Update user status params.
|
|
*
|
|
* @return external_function_parameters
|
|
*/
|
|
public static function update_user_status_parameters() {
|
|
return new external_function_parameters(
|
|
array('sessionid' => new external_value(PARAM_INT, 'Session id'),
|
|
'studentid' => new external_value(PARAM_INT, 'Student id'),
|
|
'takenbyid' => new external_value(PARAM_INT, 'Id of the user who took this session'),
|
|
'statusid' => new external_value(PARAM_INT, 'Status id'),
|
|
'statusset' => new external_value(PARAM_TEXT, 'Status set of session')));
|
|
}
|
|
|
|
/**
|
|
* Update user status.
|
|
*
|
|
* @param int $sessionid
|
|
* @param int $studentid
|
|
* @param int $takenbyid
|
|
* @param int $statusid
|
|
* @param int $statusset
|
|
*/
|
|
public static function update_user_status($sessionid, $studentid, $takenbyid, $statusid, $statusset) {
|
|
$params = self::validate_parameters(self::update_user_status_parameters(), array(
|
|
'sessionid' => $sessionid,
|
|
'studentid' => $studentid,
|
|
'takenbyid' => $takenbyid,
|
|
'statusid' => $statusid,
|
|
'statusset' => $statusset,
|
|
));
|
|
|
|
// Make sure session is open for marking.
|
|
$session = $DB->get_record('attendance_sessions', array('id' => $params['sessionid']), '*', MUST_EXIST);
|
|
list($canmark, $reason) = attendance_can_student_mark($attforsession);
|
|
if (!$canmark) {
|
|
throw new invalid_parameter_exception($reason);
|
|
}
|
|
|
|
// Check user id is valid.
|
|
$student = $DB->get_record('user', array('id' => $params['studentid']), '*', MUST_EXIST);
|
|
$takenby = $DB->get_record('user', array('id' => $params['takenbyid']), '*', MUST_EXIST);
|
|
|
|
// TODO: Verify statusset and statusid.
|
|
|
|
return attendance_handler::update_user_status($params['sessionid'], $params['studentid'], $params['takenbyid'], $params['statusid'], $params['statusset']);
|
|
}
|
|
|
|
/**
|
|
* Show return values.
|
|
* @return external_value
|
|
*/
|
|
public static function update_user_status_returns() {
|
|
return new external_value(PARAM_TEXT, 'Http code');
|
|
}
|
|
}
|
|
|