MDL-29020 add CLI verbose mode and stop requiring course idnumber when external column specified

This commit is contained in:
Petr Skoda 2011-08-20 11:22:32 +02:00
parent 4f3632441a
commit 6650f3dedf
2 changed files with 141 additions and 32 deletions

View File

@ -1,5 +1,4 @@
<?php <?php
// This file is part of Moodle - http://moodle.org/ // This file is part of Moodle - http://moodle.org/
// //
// Moodle is free software: you can redistribute it and/or modify // Moodle is free software: you can redistribute it and/or modify
@ -36,12 +35,47 @@
define('CLI_SCRIPT', true); define('CLI_SCRIPT', true);
require(dirname(dirname(dirname(dirname(__FILE__)))).'/config.php'); require(dirname(dirname(dirname(dirname(__FILE__)))).'/config.php');
require_once($CFG->libdir.'/clilib.php');
if (!enrol_is_enabled('database')) { // now get cli options
die('enrol_database plugin is disabled, sync is disabled'); list($options, $unrecognized) = cli_get_params(array('verbose'=>false, 'help'=>false), array('v'=>'verbose', 'h'=>'help'));
if ($unrecognized) {
$unrecognized = implode("\n ", $unrecognized);
cli_error(get_string('cliunknowoption', 'admin', $unrecognized));
} }
if ($options['help']) {
$help =
"Execute enrol sync with external database.
The enrol_database plugin must be enabled and properly configured.
Options:
-v, --verbose Print verbose progess information
-h, --help Print out this help
Example:
\$sudo -u www-data /usr/bin/php enrol/database/cli/sync.php
Sample cron entry:
# 5 minutes past 4am
5 4 * * * \$sudo -u www-data /usr/bin/php /var/www/moodle/enrol/database/cli/sync.php
";
echo $help;
die;
}
if (!enrol_is_enabled('database')) {
echo('enrol_database plugin is disabled, sync is disabled'."\n");
exit(1);
}
$verbose = !empty($options['verbose']);
$enrol = enrol_get_plugin('database'); $enrol = enrol_get_plugin('database');
$enrol->sync_courses(); $result = 0;
$enrol->sync_enrolments();
$result = $result | $enrol->sync_courses($verbose);
$result = $result | $enrol->sync_enrolments($verbose);
exit($result);

View File

@ -1,5 +1,4 @@
<?php <?php
// This file is part of Moodle - http://moodle.org/ // This file is part of Moodle - http://moodle.org/
// //
// Moodle is free software: you can redistribute it and/or modify // Moodle is free software: you can redistribute it and/or modify
@ -103,7 +102,10 @@ class enrol_database_plugin extends enrol_plugin {
$enrols = array(); $enrols = array();
$instances = array(); $instances = array();
$extdb = $this->db_init(); if (!$extdb = $this->db_init()) {
// can not connect to database, sorry
return;
}
// read remote enrols and create instances // read remote enrols and create instances
$sql = $this->db_get_sql($table, array($userfield=>$user->$localuserfield), array(), false); $sql = $this->db_get_sql($table, array($userfield=>$user->$localuserfield), array(), false);
@ -242,22 +244,33 @@ class enrol_database_plugin extends enrol_plugin {
/** /**
* Forces synchronisation of all enrolments with external database. * Forces synchronisation of all enrolments with external database.
* *
* @return void * @param bool $verbose
* @return int 0 means success, 1 db connect failure, 2 db read failure
*/ */
public function sync_enrolments() { public function sync_enrolments($verbose = false) {
global $CFG, $DB; global $CFG, $DB;
// we do not create courses here intentionally because it requires full sync and is slow // we do not create courses here intentionally because it requires full sync and is slow
if (!$this->get_config('dbtype') or !$this->get_config('dbhost') or !$this->get_config('remoteenroltable') or !$this->get_config('remotecoursefield') or !$this->get_config('remoteuserfield')) { if (!$this->get_config('dbtype') or !$this->get_config('dbhost') or !$this->get_config('remoteenroltable') or !$this->get_config('remotecoursefield') or !$this->get_config('remoteuserfield')) {
return; if ($verbose) {
mtrace('User enrolment synchronisation skipped.');
}
return 0;
}
if ($verbose) {
mtrace('Starting user enrolment synchronisation...');
}
if (!$extdb = $this->db_init()) {
mtrace('Error while communicating with external enrolment database');
return 1;
} }
// we may need a lot of memory here // we may need a lot of memory here
@set_time_limit(0); @set_time_limit(0);
raise_memory_limit(MEMORY_HUGE); raise_memory_limit(MEMORY_HUGE);
$extdb = $this->db_init();
// second step is to sync instances and users // second step is to sync instances and users
$table = $this->get_config('remoteenroltable'); $table = $this->get_config('remoteenroltable');
$coursefield = strtolower($this->get_config('remotecoursefield')); $coursefield = strtolower($this->get_config('remotecoursefield'));
@ -298,18 +311,20 @@ class enrol_database_plugin extends enrol_plugin {
} }
$rs->Close(); $rs->Close();
} else { } else {
debugging('Error while communicating with external enrolment database'); mtrace('Error reading data from the external enrolment table');
$extdb->Close(); $extdb->Close();
return; return 2;
} }
$preventfullunenrol = empty($externalcourses); $preventfullunenrol = empty($externalcourses);
if ($preventfullunenrol and $unenrolaction == ENROL_EXT_REMOVED_UNENROL) { if ($preventfullunenrol and $unenrolaction == ENROL_EXT_REMOVED_UNENROL) {
debugging('Preventing unenrolment of all current users, because it might result in major data loss, there has to be at least one record in external enrol table, sorry.'); if ($verbose) {
mtrace(' Preventing unenrolment of all current users, because it might result in major data loss, there has to be at least one record in external enrol table, sorry.');
}
} }
// first find all existing courses with enrol instance // first find all existing courses with enrol instance
$existing = array(); $existing = array();
$sql = "SELECT c.id, c.visible, c.$localcoursefield AS mapping, e.id AS enrolid $sql = "SELECT c.id, c.visible, c.$localcoursefield AS mapping, e.id AS enrolid, c.shortname
FROM {course} c FROM {course} c
JOIN {enrol} e ON (e.courseid = c.id AND e.enrol = 'database')"; JOIN {enrol} e ON (e.courseid = c.id AND e.enrol = 'database')";
$rs = $DB->get_recordset_sql($sql); // watch out for idnumber duplicates $rs = $DB->get_recordset_sql($sql); // watch out for idnumber duplicates
@ -328,7 +343,7 @@ class enrol_database_plugin extends enrol_plugin {
$localnotempty = "AND c.$localcoursefield <> :lcfe"; $localnotempty = "AND c.$localcoursefield <> :lcfe";
$params['lcfe'] = $DB->sql_empty(); $params['lcfe'] = $DB->sql_empty();
} }
$sql = "SELECT c.id, c.visible, c.$localcoursefield AS mapping $sql = "SELECT c.id, c.visible, c.$localcoursefield AS mapping, c.shortname
FROM {course} c FROM {course} c
LEFT JOIN {enrol} e ON (e.courseid = c.id AND e.enrol = 'database') LEFT JOIN {enrol} e ON (e.courseid = c.id AND e.enrol = 'database')
WHERE e.id IS NULL $localnotempty"; WHERE e.id IS NULL $localnotempty";
@ -430,7 +445,7 @@ class enrol_database_plugin extends enrol_plugin {
} }
$rs->Close(); $rs->Close();
} else { } else {
debugging('Error while communicating with external enrolment database'); mtrace('Error while communicating with external enrolment database');
$extdb->Close(); $extdb->Close();
return; return;
} }
@ -443,6 +458,9 @@ class enrol_database_plugin extends enrol_plugin {
$this->enrol_user($instance, $userid, $roleid); $this->enrol_user($instance, $userid, $roleid);
$current_roles[$userid][$roleid] = $roleid; $current_roles[$userid][$roleid] = $roleid;
$current_status[$userid] = ENROL_USER_ACTIVE; $current_status[$userid] = ENROL_USER_ACTIVE;
if ($verbose) {
mtrace(" enrolling: $userid ==> $course->shortname as ".$allroles[$roleid]->shortname);
}
} }
} }
@ -451,12 +469,18 @@ class enrol_database_plugin extends enrol_plugin {
if (empty($userroles[$cr])) { if (empty($userroles[$cr])) {
role_unassign($cr, $userid, $context->id, 'enrol_database', $instance->id); role_unassign($cr, $userid, $context->id, 'enrol_database', $instance->id);
unset($current_roles[$userid][$cr]); unset($current_roles[$userid][$cr]);
if ($verbose) {
mtrace(" unsassigning roles: $userid ==> $course->shortname");
}
} }
} }
// reenable enrolment when previously disable enrolment refreshed // reenable enrolment when previously disable enrolment refreshed
if ($current_status[$userid] == ENROL_USER_SUSPENDED) { if ($current_status[$userid] == ENROL_USER_SUSPENDED) {
$DB->set_field('user_enrolments', 'status', ENROL_USER_ACTIVE, array('enrolid'=>$instance->id, 'userid'=>$userid)); $DB->set_field('user_enrolments', 'status', ENROL_USER_ACTIVE, array('enrolid'=>$instance->id, 'userid'=>$userid));
if ($verbose) {
mtrace(" unsuspending: $userid ==> $course->shortname");
}
} }
} }
@ -469,6 +493,9 @@ class enrol_database_plugin extends enrol_plugin {
continue; continue;
} }
$this->unenrol_user($instance, $userid); $this->unenrol_user($instance, $userid);
if ($verbose) {
mtrace(" unenrolling: $userid ==> $course->shortname");
}
} }
} }
@ -483,9 +510,15 @@ class enrol_database_plugin extends enrol_plugin {
} }
if ($status != ENROL_USER_SUSPENDED) { if ($status != ENROL_USER_SUSPENDED) {
$DB->set_field('user_enrolments', 'status', ENROL_USER_SUSPENDED, array('enrolid'=>$instance->id, 'userid'=>$userid)); $DB->set_field('user_enrolments', 'status', ENROL_USER_SUSPENDED, array('enrolid'=>$instance->id, 'userid'=>$userid));
if ($verbose) {
mtrace(" suspending: $userid ==> $course->shortname");
}
} }
if ($unenrolaction == ENROL_EXT_REMOVED_SUSPENDNOROLES) { if ($unenrolaction == ENROL_EXT_REMOVED_SUSPENDNOROLES) {
role_unassign_all(array('contextid'=>$context->id, 'userid'=>$userid, 'component'=>'enrol_database', 'itemid'=>$instance->id)); role_unassign_all(array('contextid'=>$context->id, 'userid'=>$userid, 'component'=>'enrol_database', 'itemid'=>$instance->id));
if ($verbose) {
mtrace(" unsassigning all roles: $userid ==> $course->shortname");
}
} }
} }
} }
@ -493,6 +526,12 @@ class enrol_database_plugin extends enrol_plugin {
// close db connection // close db connection
$extdb->Close(); $extdb->Close();
if ($verbose) {
mtrace('...user enrolment synchronisation finished.');
}
return 0;
} }
/** /**
@ -500,21 +539,33 @@ class enrol_database_plugin extends enrol_plugin {
* *
* First it creates new courses if necessary, then * First it creates new courses if necessary, then
* enrols and unenrols users. * enrols and unenrols users.
* @return void *
* @param bool $verbose
* @return int 0 means success, 1 db connect failure, 4 db read failure
*/ */
public function sync_courses() { public function sync_courses($verbose = false) {
global $CFG, $DB; global $CFG, $DB;
// make sure we sync either enrolments or courses // make sure we sync either enrolments or courses
if (!$this->get_config('dbtype') or !$this->get_config('dbhost') or !$this->get_config('newcoursetable') or !$this->get_config('newcoursefullname') or !$this->get_config('newcourseshortname')) { if (!$this->get_config('dbtype') or !$this->get_config('dbhost') or !$this->get_config('newcoursetable') or !$this->get_config('newcoursefullname') or !$this->get_config('newcourseshortname')) {
return; if ($verbose) {
mtrace('Course synchronisation skipped.');
}
return 0;
}
if ($verbose) {
mtrace('Starting course synchronisation...');
} }
// we may need a lot of memory here // we may need a lot of memory here
@set_time_limit(0); @set_time_limit(0);
raise_memory_limit(MEMORY_HUGE); raise_memory_limit(MEMORY_HUGE);
$extdb = $this->db_init(); if (!$extdb = $this->db_init()) {
mtrace('Error while communicating with external enrolment database');
return 1;
}
// first create new courses // first create new courses
$table = $this->get_config('newcoursetable'); $table = $this->get_config('newcoursetable');
@ -534,24 +585,30 @@ class enrol_database_plugin extends enrol_plugin {
$createcourses = array(); $createcourses = array();
if ($rs = $extdb->Execute($sql)) { if ($rs = $extdb->Execute($sql)) {
if (!$rs->EOF) { if (!$rs->EOF) {
$courselist = array();
while ($fields = $rs->FetchRow()) { while ($fields = $rs->FetchRow()) {
$fields = array_change_key_case($fields, CASE_LOWER); $fields = array_change_key_case($fields, CASE_LOWER);
$fields = $this->db_decode($fields);
if (empty($fields[$shortname]) or empty($fields[$fullname])) { if (empty($fields[$shortname]) or empty($fields[$fullname])) {
//invalid record - these two are mandatory if ($verbose) {
mtrace(' error: invalid external course record, shortname and fullname are mandatory: ' . json_encode($fields)); // hopefully every geek can read JS, right?
}
continue; continue;
} }
$fields = $this->db_decode($fields);
if ($DB->record_exists('course', array('shortname'=>$fields[$shortname]))) { if ($DB->record_exists('course', array('shortname'=>$fields[$shortname]))) {
// already exists // already exists
continue; continue;
} }
if ($idnumber and $DB->record_exists('course', array('idnumber'=>$fields[$idnumber]))) { // allow empty idnumber but not duplicates
// idnumber duplicates are not allowed if ($idnumber and $fields[$idnumber] !== '' and $fields[$idnumber] !== null and $DB->record_exists('course', array('idnumber'=>$fields[$idnumber]))) {
if ($verbose) {
mtrace(' error: duplicate idnumber, can not create course: '.$fields[$shortname].' ['.$fields[$idnumber].']');
}
continue; continue;
} }
if ($category and !$DB->record_exists('course_categories', array('id'=>$fields[$category]))) { if ($category and !$DB->record_exists('course_categories', array('id'=>$fields[$category]))) {
// invalid category id, better to skip if ($verbose) {
mtrace(' error: invalid category id, can not create course: '.$fields[$shortname]);
}
continue; continue;
} }
$course = new stdClass(); $course = new stdClass();
@ -564,9 +621,9 @@ class enrol_database_plugin extends enrol_plugin {
} }
$rs->Close(); $rs->Close();
} else { } else {
debugging('Error while communicating with external enrolment database'); mtrace('Error reading data from the external course table');
$extdb->Close(); $extdb->Close();
return; return 4;
} }
if ($createcourses) { if ($createcourses) {
require_once("$CFG->dirroot/course/lib.php"); require_once("$CFG->dirroot/course/lib.php");
@ -599,7 +656,10 @@ class enrol_database_plugin extends enrol_plugin {
$newcourse->idnumber = $fields->idnumber; $newcourse->idnumber = $fields->idnumber;
$newcourse->category = $fields->category ? $fields->category : $defaultcategory; $newcourse->category = $fields->category ? $fields->category : $defaultcategory;
create_course($newcourse); $c = create_course($newcourse);
if ($verbose) {
mtrace(" creating course: $c->id, $c->fullname, $c->shortname, $c->idnumber, $c->category");
}
} }
unset($createcourses); unset($createcourses);
@ -608,6 +668,12 @@ class enrol_database_plugin extends enrol_plugin {
// close db connection // close db connection
$extdb->Close(); $extdb->Close();
if ($verbose) {
mtrace('...course synchronisation finished.');
}
return 0;
} }
protected function db_get_sql($table, array $conditions, array $fields, $distinct = false, $sort = "") { protected function db_get_sql($table, array $conditions, array $fields, $distinct = false, $sort = "") {
@ -631,6 +697,11 @@ class enrol_database_plugin extends enrol_plugin {
return $sql; return $sql;
} }
/**
* Tries to make connection to the external database.
*
* @return null|ADONewConnection
*/
protected function db_init() { protected function db_init() {
global $CFG; global $CFG;
@ -643,7 +714,11 @@ class enrol_database_plugin extends enrol_plugin {
ob_start(); //start output buffer to allow later use of the page headers ob_start(); //start output buffer to allow later use of the page headers
} }
$extdb->Connect($this->get_config('dbhost'), $this->get_config('dbuser'), $this->get_config('dbpass'), $this->get_config('dbname'), true); $result = $extdb->Connect($this->get_config('dbhost'), $this->get_config('dbuser'), $this->get_config('dbpass'), $this->get_config('dbname'), true);
if (!$result) {
return null;
}
$extdb->SetFetchMode(ADODB_FETCH_ASSOC); $extdb->SetFetchMode(ADODB_FETCH_ASSOC);
if ($this->get_config('dbsetupsql')) { if ($this->get_config('dbsetupsql')) {
$extdb->Execute($this->get_config('dbsetupsql')); $extdb->Execute($this->get_config('dbsetupsql'));