libdir.'/authlib.php'); require_once($CFG->libdir.'/adodb/adodb.inc.php'); /** * External database authentication plugin. */ class auth_plugin_db extends auth_plugin_base { /** * Constructor. */ function auth_plugin_db() { $this->authtype = 'db'; $this->config = get_config('auth/db'); if (empty($this->config->extencoding)) { $this->config->extencoding = 'utf-8'; } } /** * Returns true if the username and password work and false if they are * wrong or don't exist. * * @param string $username The username * @param string $password The password * * @return bool Authentication success or failure. */ function user_login($username, $password) { global $CFG, $DB; $textlib = textlib_get_instance(); $extusername = $textlib->convert($username, 'utf-8', $this->config->extencoding); $extpassword = $textlib->convert($password, 'utf-8', $this->config->extencoding); $authdb = $this->db_init(); if ($this->config->passtype === 'internal') { // lookup username externally, but resolve // password locally -- to support backend that // don't track passwords $rs = $authdb->Execute("SELECT * FROM {$this->config->table} WHERE {$this->config->fielduser} = '".$this->ext_addslashes($extusername)."' "); if (!$rs) { $authdb->Close(); debugging(get_string('auth_dbcantconnect','auth_db')); return false; } if ( !$rs->EOF ) { $rs->Close(); $authdb->Close(); // user exists exterally // check username/password internally if ($user = $DB->get_record('user', array('username'=>$username, 'mnethostid'=>$CFG->mnet_localhost_id))) { return validate_internal_user_password($user, $password); } } else { $rs->Close(); $authdb->Close(); // user does not exist externally return false; } } else { // normal case: use external db for passwords if ($this->config->passtype === 'md5') { // Re-format password accordingly $extpassword = md5($extpassword); } else if ($this->config->passtype === 'sha1') { $extpassword = sha1($extpassword); } $rs = $authdb->Execute("SELECT * FROM {$this->config->table} WHERE {$this->config->fielduser} = '".$this->ext_addslashes($extusername)."' AND {$this->config->fieldpass} = '".$this->ext_addslashes($extpassword)."' "); if (!$rs) { $authdb->Close(); debugging(get_string('auth_dbcantconnect','auth_db')); return false; } if (!$rs->EOF) { $rs->Close(); $authdb->Close(); return true; } else { $rs->Close(); $authdb->Close(); return false; } } } function db_init() { // Connect to the external database (forcing new connection) $authdb = &ADONewConnection($this->config->type); if (!empty($this->config->debugauthdb)) { $authdb->debug = true; ob_start();//start output buffer to allow later use of the page headers } $authdb->Connect($this->config->host, $this->config->user, $this->config->pass, $this->config->name, true); $authdb->SetFetchMode(ADODB_FETCH_ASSOC); if (!empty($this->config->setupsql)) { $authdb->Execute($this->config->setupsql); } return $authdb; } /** * retuns user attribute mappings between moodle and ldap * * @return array */ function db_attributes() { $moodleattributes = array(); foreach ($this->userfields as $field) { if (!empty($this->config->{"field_map_$field"})) { $moodleattributes[$field] = $this->config->{"field_map_$field"}; } } $moodleattributes['username'] = $this->config->fielduser; return $moodleattributes; } /** * Reads any other information for a user from external database, * then returns it in an array * * @param string $username * * @return array without magic quotes */ function get_userinfo($username) { global $CFG; $textlib = textlib_get_instance(); $extusername = $textlib->convert($username, 'utf-8', $this->config->extencoding); $authdb = $this->db_init(); //Array to map local fieldnames we want, to external fieldnames $selectfields = $this->db_attributes(); $result = array(); //If at least one field is mapped from external db, get that mapped data: if ($selectfields) { $select = ''; foreach ($selectfields as $localname=>$externalname) { $select .= ", $externalname AS $localname"; } $select = 'SELECT ' . substr($select,1); $sql = $select . " FROM {$this->config->table}" . " WHERE {$this->config->fielduser} = '".$this->ext_addslashes($extusername)."'"; if ($rs = $authdb->Execute($sql)) { if ( !$rs->EOF ) { $fields_obj = $rs->FetchObj(); $fields_obj = (object)array_change_key_case((array)$fields_obj , CASE_LOWER); foreach ($selectfields as $localname=>$externalname) { $result[$localname] = $textlib->convert($fields_obj->{$localname}, $this->config->extencoding, 'utf-8'); } } $rs->Close(); } } $authdb->Close(); return $result; } /** * Change a user's password * * @param object $user User table object * @param string $newpassword Plaintext password * * @return bool True on success */ function user_update_password($user, $newpassword) { global $CFG; if ($this->config->passtype === 'internal') { return update_internal_user_password($user, $newpassword); } else { // we should have never been called! return false; } } /** * syncronizes user fron external db to moodle user table * * Sync shouid be done by using idnumber attribute, not username. * You need to pass firstsync parameter to function to fill in * idnumbers if they dont exists in moodle user table. * * Syncing users removes (disables) users that dont exists anymore in external db. * Creates new users and updates coursecreator status of users. * * @param bool $do_updates Optional: set to true to force an update of existing accounts * * This implementation is simpler but less scalable than the one found in the LDAP module. * */ function sync_users($do_updates=false) { global $CFG, $DB; $pcfg = get_config('auth/db'); /// list external users $userlist = $this->get_userlist(); /// delete obsolete internal users if (!empty($this->config->removeuser)) { // find obsolete users if (count($userlist)) { list($notin_sql, $params) = $DB->get_in_or_equal($userlist, SQL_PARAMS_NAMED, 'u0000', false); $params['authtype'] = $this->authtype; $sql = "SELECT u.id, u.username, u.email, u.auth FROM {user} u WHERE u.auth=:authtype AND u.deleted=0 AND u.username $notin_sql"; } else { $sql = "SELECT u.id, u.username, u.email, u.auth FROM {user} u WHERE u.auth=:authtype AND u.deleted=0"; $params = array(); $params['authtype'] = $this->authtype; } $remove_users = $DB->get_records_sql($sql, $params); if (!empty($remove_users)) { print_string('auth_dbuserstoremove','auth_db', count($remove_users)); echo "\n"; foreach ($remove_users as $user) { if ($this->config->removeuser == AUTH_REMOVEUSER_FULLDELETE) { if (delete_user($user)) { echo "\t"; print_string('auth_dbdeleteuser', 'auth_db', array('name'=>$user->username, 'id'=>$user->id)); echo "\n"; } else { echo "\t"; print_string('auth_dbdeleteusererror', 'auth_db', $user->username); echo "\n"; } } else if ($this->config->removeuser == AUTH_REMOVEUSER_SUSPEND) { $updateuser = new stdClass(); $updateuser->id = $user->id; $updateuser->auth = 'nologin'; $DB->update_record('user', $updateuser); echo "\t"; print_string('auth_dbsuspenduser', 'auth_db', array('name'=>$user->username, 'id'=>$user->id)); echo "\n"; } } } unset($remove_users); // free mem! } if (!count($userlist)) { // exit right here // nothing else to do return true; } /// /// update existing accounts /// if ($do_updates) { // narrow down what fields we need to update $all_keys = array_keys(get_object_vars($this->config)); $updatekeys = array(); foreach ($all_keys as $key) { if (preg_match('/^field_updatelocal_(.+)$/',$key, $match)) { if ($this->config->{$key} === 'onlogin') { array_push($updatekeys, $match[1]); // the actual key name } } } // print_r($all_keys); print_r($updatekeys); unset($all_keys); unset($key); // only go ahead if we actually // have fields to update locally if (!empty($updatekeys)) { list($in_sql, $params) = $DB->get_in_or_equal($userlist, SQL_PARAMS_NAMED, 'u0000', true); $params['authtype'] = $this->authtype; $sql = "SELECT u.id, u.username FROM {user} u WHERE u.auth=:authtype AND u.deleted=0 AND u.username {$in_sql}"; if ($update_users = $DB->get_records_sql($sql, $params)) { print "User entries to update: ". count($update_users). "\n"; foreach ($update_users as $user) { echo "\t"; print_string('auth_dbupdatinguser', 'auth_db', array('name'=>$user->username, 'id'=>$user->id)); if (!$this->update_user_record($user->username, $updatekeys)) { echo " - ".get_string('skipped'); } echo "\n"; } unset($update_users); // free memory } } } /// /// create missing accounts /// // NOTE: this is very memory intensive // and generally inefficient $sql = 'SELECT u.id, u.username FROM {user} u WHERE u.auth=\'' . $this->authtype . '\' AND u.deleted=\'0\''; $users = $DB->get_records_sql($sql); // simplify down to usernames $usernames = array(); if (!empty($users)) { foreach ($users as $user) { array_push($usernames, $user->username); } unset($users); } $add_users = array_diff($userlist, $usernames); unset($usernames); if (!empty($add_users)) { print_string('auth_dbuserstoadd','auth_db',count($add_users)); echo "\n"; $transaction = $DB->start_delegated_transaction(); foreach($add_users as $user) { $username = $user; $user = $this->get_userinfo_asobj($user); // prep a few params $user->username = $username; $user->modified = time(); $user->confirmed = 1; $user->auth = $this->authtype; $user->mnethostid = $CFG->mnet_localhost_id; if (empty($user->lang)) { $user->lang = $CFG->lang; } // maybe the user has been deleted before if ($old_user = $DB->get_record('user', array('username'=>$user->username, 'deleted'=>1, 'mnethostid'=>$user->mnethostid))) { $user->id = $old_user->id; $DB->set_field('user', 'deleted', 0, array('username'=>$user->username)); echo "\t"; print_string('auth_dbreviveduser', 'auth_db', array('name'=>$user->username, 'id'=>$user->id)); echo "\n"; } else { $id = $DB->insert_record ('user',$user); // it is truly a new user echo "\t"; print_string('auth_dbinsertuser','auth_db',array('name'=>$user->username, 'id'=>$id)); echo "\n"; // if relevant, tag for password generation if ($this->config->passtype === 'internal') { set_user_preference('auth_forcepasswordchange', 1, $id); set_user_preference('create_password', 1, $id); } } } $transaction->allow_commit(); unset($add_users); // free mem } return true; } function user_exists($username) { /// Init result value $result = false; $textlib = textlib_get_instance(); $extusername = $textlib->convert($username, 'utf-8', $this->config->extencoding); $authdb = $this->db_init(); $rs = $authdb->Execute("SELECT * FROM {$this->config->table} WHERE {$this->config->fielduser} = '".$this->ext_addslashes($extusername)."' "); if (!$rs) { print_error('auth_dbcantconnect','auth_db'); } else if ( !$rs->EOF ) { // user exists exterally $result = true; } $authdb->Close(); return $result; } function get_userlist() { /// Init result value $result = array(); $authdb = $this->db_init(); // fetch userlist $rs = $authdb->Execute("SELECT {$this->config->fielduser} AS username FROM {$this->config->table} "); if (!$rs) { print_error('auth_dbcantconnect','auth_db'); } else if ( !$rs->EOF ) { while ($rec = $rs->FetchRow()) { $rec = (object)array_change_key_case((array)$rec , CASE_LOWER); array_push($result, $rec->username); } } $authdb->Close(); return $result; } /** * reads userinformation from DB and return it in an object * * @param string $username username (with system magic quotes) * @return array */ function get_userinfo_asobj($username) { $user_array = truncate_userinfo($this->get_userinfo($username)); $user = new stdClass(); foreach($user_array as $key=>$value) { $user->{$key} = $value; } return $user; } /** * will update a local user record from an external source. * is a lighter version of the one in moodlelib -- won't do * expensive ops such as enrolment * * If you don't pass $updatekeys, there is a performance hit and * values removed from DB won't be removed from moodle. * * @param string $username username */ function update_user_record($username, $updatekeys=false) { global $CFG, $DB; //just in case check text case $username = trim(moodle_strtolower($username)); // get the current user record $user = $DB->get_record('user', array('username'=>$username, 'mnethostid'=>$CFG->mnet_localhost_id)); if (empty($user)) { // trouble error_log("Cannot update non-existent user: $username"); print_error('auth_dbusernotexist','auth_db',$username); die; } // Ensure userid is not overwritten $userid = $user->id; if ($newinfo = $this->get_userinfo($username)) { $newinfo = truncate_userinfo($newinfo); if (empty($updatekeys)) { // all keys? this does not support removing values $updatekeys = array_keys($newinfo); } foreach ($updatekeys as $key) { if (isset($newinfo[$key])) { $value = $newinfo[$key]; } else { $value = ''; } if (!empty($this->config->{'field_updatelocal_' . $key})) { if (isset($user->{$key}) and $user->{$key} != $value) { // only update if it's changed $DB->set_field('user', $key, $value, array('id'=>$userid)); } } } } return $DB->get_record('user', array('id'=>$userid, 'deleted'=>0)); } /** * Called when the user record is updated. * Modifies user in external database. It takes olduser (before changes) and newuser (after changes) * conpares information saved modified information to external db. * * @param mixed $olduser Userobject before modifications * @param mixed $newuser Userobject new modified userobject * @return boolean result * */ function user_update($olduser, $newuser) { if (isset($olduser->username) and isset($newuser->username) and $olduser->username != $newuser->username) { error_log("ERROR:User renaming not allowed in ext db"); return false; } if (isset($olduser->auth) and $olduser->auth != $this->authtype) { return true; // just change auth and skip update } $curruser = $this->get_userinfo($olduser->username); if (empty($curruser)) { error_log("ERROR:User $olduser->username found in ext db"); return false; } $textlib = textlib_get_instance(); $extusername = $textlib->convert($olduser->username, 'utf-8', $this->config->extencoding); $authdb = $this->db_init(); $update = array(); foreach($curruser as $key=>$value) { if ($key == 'username') { continue; // skip this } if (empty($this->config->{"field_updateremote_$key"})) { continue; // remote update not requested } if (!isset($newuser->$key)) { continue; } $nuvalue = $newuser->$key; if ($nuvalue != $value) { $update[] = $this->config->{"field_map_$key"}."='".$this->ext_addslashes($textlib->convert($nuvalue, 'utf-8', $this->config->extencoding))."'"; } } if (!empty($update)) { $authdb->Execute("UPDATE {$this->config->table} SET ".implode(',', $update)." WHERE {$this->config->fielduser}='".$this->ext_addslashes($extusername)."'"); } $authdb->Close(); return true; } /** * A chance to validate form data, and last chance to * do stuff before it is inserted in config_plugin */ function validate_form(&$form, &$err) { if ($form->passtype === 'internal') { $this->config->changepasswordurl = ''; set_config('changepasswordurl', '', 'auth/db'); } } function prevent_local_passwords() { if (!isset($this->config->passtype)) { return false; } return ($this->config->passtype != 'internal'); } /** * Returns true if this authentication plugin is 'internal'. * * @return bool */ function is_internal() { if (!isset($this->config->passtype)) { return true; } return ($this->config->passtype == 'internal'); } /** * Returns true if this authentication plugin can change the user's * password. * * @return bool */ function can_change_password() { return ($this->config->passtype == 'internal' or !empty($this->config->changepasswordurl)); } /** * Returns the URL for changing the user's pw, or empty if the default can * be used. * * @return moodle_url */ function change_password_url() { if ($this->config->passtype == 'internal') { // standard form return null; } else { // use custom url return new moodle_url($this->config->changepasswordurl); } } /** * Returns true if plugin allows resetting of internal password. * * @return bool */ function can_reset_password() { return ($this->config->passtype == 'internal'); } /** * Prints a form for configuring this authentication plugin. * * This function is called from admin/auth.php, and outputs a full page with * a form for configuring this plugin. * * @param array $page An object containing all the data for this page. */ function config_form($config, $err, $user_fields) { include 'config.html'; } /** * Processes and stores configuration data for this authentication plugin. */ function process_config($config) { // set to defaults if undefined if (!isset($config->host)) { $config->host = 'localhost'; } if (!isset($config->type)) { $config->type = 'mysql'; } if (!isset($config->sybasequoting)) { $config->sybasequoting = 0; } if (!isset($config->name)) { $config->name = ''; } if (!isset($config->user)) { $config->user = ''; } if (!isset($config->pass)) { $config->pass = ''; } if (!isset($config->table)) { $config->table = ''; } if (!isset($config->fielduser)) { $config->fielduser = ''; } if (!isset($config->fieldpass)) { $config->fieldpass = ''; } if (!isset($config->passtype)) { $config->passtype = 'plaintext'; } if (!isset($config->extencoding)) { $config->extencoding = 'utf-8'; } if (!isset($config->setupsql)) { $config->setupsql = ''; } if (!isset($config->debugauthdb)) { $config->debugauthdb = 0; } if (!isset($config->removeuser)) { $config->removeuser = AUTH_REMOVEUSER_KEEP; } if (!isset($config->changepasswordurl)) { $config->changepasswordurl = ''; } // save settings set_config('host', $config->host, 'auth/db'); set_config('type', $config->type, 'auth/db'); set_config('sybasequoting', $config->sybasequoting, 'auth/db'); set_config('name', $config->name, 'auth/db'); set_config('user', $config->user, 'auth/db'); set_config('pass', $config->pass, 'auth/db'); set_config('table', $config->table, 'auth/db'); set_config('fielduser', $config->fielduser, 'auth/db'); set_config('fieldpass', $config->fieldpass, 'auth/db'); set_config('passtype', $config->passtype, 'auth/db'); set_config('extencoding', trim($config->extencoding), 'auth/db'); set_config('setupsql', trim($config->setupsql),'auth/db'); set_config('debugauthdb', $config->debugauthdb, 'auth/db'); set_config('removeuser', $config->removeuser, 'auth/db'); set_config('changepasswordurl', trim($config->changepasswordurl), 'auth/db'); return true; } function ext_addslashes($text) { // using custom made function for now if (empty($this->config->sybasequoting)) { $text = str_replace('\\', '\\\\', $text); $text = str_replace(array('\'', '"', "\0"), array('\\\'', '\\"', '\\0'), $text); } else { $text = str_replace("'", "''", $text); } return $text; } }