2011-08-18 00:31:15 +00:00
|
|
|
<?php
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
/* user_model.php
|
|
|
|
*
|
|
|
|
* This model implements user authentication and authorization
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2011-08-18 00:31:15 +00:00
|
|
|
// Uses 'phpass' from http://www.openwall.com/phpass/ to implement password hashing
|
|
|
|
require_once('application/third_party/PasswordHash.php');
|
|
|
|
|
|
|
|
class User_Model extends CI_Model {
|
|
|
|
|
|
|
|
function __construct()
|
|
|
|
{
|
|
|
|
// Call the Model constructor
|
|
|
|
parent::__construct();
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: object get($username)
|
2011-08-18 00:31:15 +00:00
|
|
|
// Retrieve a user
|
|
|
|
function get($username) {
|
|
|
|
$this->db->where('user_name', $username);
|
|
|
|
$r = $this->db->get($this->config->item('auth_table'));
|
|
|
|
return $r;
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: object get_by_id($id)
|
|
|
|
// Retrieve a user by user ID
|
2011-08-18 00:31:15 +00:00
|
|
|
function get_by_id($id) {
|
|
|
|
$this->db->where('user_id', $id);
|
|
|
|
$r = $this->db->get($this->config->item('auth_table'));
|
|
|
|
return $r;
|
|
|
|
}
|
|
|
|
|
2011-08-19 16:13:26 +00:00
|
|
|
// FUNCTION: object get_by_email($email)
|
|
|
|
// Retrieve a user by email address
|
|
|
|
function get_by_email($email) {
|
|
|
|
$this->db->where('user_email', $email);
|
|
|
|
$r = $this->db->get($this->config->item('auth_table'));
|
|
|
|
return $r;
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool exists($username)
|
|
|
|
// Check if a user exists (by username)
|
2011-08-18 00:31:15 +00:00
|
|
|
function exists($username) {
|
2011-08-18 01:27:53 +00:00
|
|
|
if($this->get($username)->num_rows == 0) {
|
2011-08-18 00:31:15 +00:00
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-19 16:13:26 +00:00
|
|
|
// FUNCTION: bool exists_by_id($id)
|
|
|
|
// Check if a user exists (by user ID)
|
|
|
|
function exists_by_id($id) {
|
|
|
|
if($this->get_by_id($id)->num_rows == 0) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FUNCTION: bool exists_by_email($email)
|
|
|
|
// Check if a user exists (by email address)
|
|
|
|
function exists_by_email($email) {
|
|
|
|
if($this->get_by_email($email)->num_rows == 0) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool add($username, $password, $email, $type)
|
|
|
|
// Add a user
|
2011-09-27 22:47:25 +00:00
|
|
|
function add($username, $password, $email, $type, $firstname, $lastname, $callsign, $locator, $timezone) {
|
2011-08-19 16:13:26 +00:00
|
|
|
// Check that the user isn't already used
|
2011-08-18 00:31:15 +00:00
|
|
|
if(!$this->exists($username)) {
|
|
|
|
$data = array(
|
|
|
|
'user_name' => $username,
|
|
|
|
'user_password' => $this->_hash($password),
|
|
|
|
'user_email' => $email,
|
2011-09-26 19:45:07 +00:00
|
|
|
'user_type' => $type,
|
|
|
|
'user_firstname' => $firstname,
|
|
|
|
'user_lastname' => $lastname,
|
|
|
|
'user_callsign' => $callsign,
|
2011-09-27 22:47:25 +00:00
|
|
|
'user_locator' => $locator,
|
|
|
|
'user_timezone' => $timezone
|
2011-08-18 00:31:15 +00:00
|
|
|
);
|
|
|
|
|
2011-08-19 16:13:26 +00:00
|
|
|
// Check the password is valid
|
|
|
|
if($data['user_password'] == EPASSWORDINVALID) {
|
|
|
|
return EPASSWORDINVALID;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the email address isn't in use
|
|
|
|
if($this->exists_by_email($email)) {
|
|
|
|
return EEMAILEXISTS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add user
|
2011-08-18 01:27:53 +00:00
|
|
|
$this->db->insert($this->config->item('auth_table'), $data);
|
2011-08-19 16:13:26 +00:00
|
|
|
return OK;
|
2011-08-18 00:31:15 +00:00
|
|
|
} else {
|
2011-08-19 16:13:26 +00:00
|
|
|
return EUSERNAMEEXISTS;
|
2011-08-18 00:31:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-19 16:13:26 +00:00
|
|
|
// FUNCTION: bool edit()
|
2011-08-18 19:57:27 +00:00
|
|
|
// Edit a user
|
2011-08-19 19:33:37 +00:00
|
|
|
function edit($fields) {
|
|
|
|
|
|
|
|
// Check user privileges
|
|
|
|
if(($this->session->userdata('user_type') == 99) || ($this->session->userdata('user_id') == $fields['id'])) {
|
|
|
|
if($this->exists_by_id($fields['id'])) {
|
|
|
|
$data = array(
|
|
|
|
'user_name' => $fields['user_name'],
|
|
|
|
'user_email' => $fields['user_email'],
|
|
|
|
'user_callsign' => $fields['user_callsign'],
|
|
|
|
'user_locator' => $fields['user_locator'],
|
|
|
|
'user_firstname' => $fields['user_firstname'],
|
2011-09-27 22:47:25 +00:00
|
|
|
'user_lastname' => $fields['user_lastname'],
|
2013-02-22 00:31:31 +00:00
|
|
|
'user_timezone' => $fields['user_timezone'],
|
|
|
|
'user_lotw_name' => $fields['user_lotw_name'],
|
|
|
|
'user_lotw_password' => $fields['user_lotw_password']
|
2011-08-19 19:33:37 +00:00
|
|
|
);
|
2011-08-19 16:13:26 +00:00
|
|
|
|
2011-08-19 19:33:37 +00:00
|
|
|
// Check to see if the user is allowed to change user levels
|
|
|
|
if($this->session->userdata('user_type') == 99) {
|
|
|
|
$data['user_type'] = $fields['user_type'];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check to see if username is used already
|
|
|
|
if($this->exists($fields['user_name']) && $this->get($fields['user_name'])->row()->user_id != $fields['id']) {
|
|
|
|
return EUSERNAMEEXISTS;
|
|
|
|
}
|
|
|
|
// Check to see if email address is used already
|
|
|
|
if($this->exists_by_email($fields['user_email']) && $this->get_by_email($fields['user_email'])->row()->user_id != $fields['id']) {
|
|
|
|
return EEMAILEXISTS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hash password
|
|
|
|
if($fields['user_password'] != NULL)
|
|
|
|
{
|
|
|
|
$data['user_password'] = $this->_hash($fields['user_password']);
|
|
|
|
if($data['user_password'] == EPASSWORDINVALID) {
|
|
|
|
return EPASSWORDINVALID;
|
|
|
|
}
|
2011-08-19 16:13:26 +00:00
|
|
|
}
|
|
|
|
|
2011-08-19 19:33:37 +00:00
|
|
|
// Update the user
|
|
|
|
$this->db->where('user_id', $fields['id']);
|
|
|
|
$this->db->update($this->config->item('auth_table'), $data);
|
|
|
|
return OK;
|
|
|
|
} else {
|
|
|
|
return ENOSUCHUSER;
|
|
|
|
}
|
2011-08-19 16:13:26 +00:00
|
|
|
} else {
|
2011-08-19 19:33:37 +00:00
|
|
|
return EFORBIDDEN;
|
|
|
|
}
|
2011-08-19 16:13:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// FUNCTION: bool delete()
|
|
|
|
// Deletes a user
|
|
|
|
function delete($user_id) {
|
2011-08-18 00:31:15 +00:00
|
|
|
|
2011-08-19 16:13:26 +00:00
|
|
|
if($this->exists_by_id($user_id)) {
|
|
|
|
$this->db->query("DELETE FROM ".$this->config->item('auth_table')." WHERE user_id = '".$user_id."'");
|
2011-08-18 00:31:15 +00:00
|
|
|
|
2011-08-19 16:13:26 +00:00
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2011-08-18 00:31:15 +00:00
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool login()
|
|
|
|
// Validates a username/password combination
|
|
|
|
// This is really just a wrapper around User_Model::authenticate
|
2011-08-18 00:31:15 +00:00
|
|
|
function login() {
|
|
|
|
|
|
|
|
$username = $this->input->post('user_name');
|
|
|
|
$password = $this->input->post('user_password');
|
|
|
|
|
|
|
|
return $this->authenticate($username, $password);
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: void clear_session()
|
|
|
|
// Clears a user's login session
|
|
|
|
// Nothing is returned - it can be assumed that if this is called, the user's
|
|
|
|
// login session *will* be cleared, no matter what state it is in
|
2011-08-18 00:31:15 +00:00
|
|
|
function clear_session() {
|
|
|
|
|
|
|
|
$this->session->unset_userdata(array('user_id' => '', 'user_type' => '', 'user_email' => '', 'user_hash' => ''));
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: void update_session()
|
|
|
|
// Updates a user's login session after they've logged in
|
|
|
|
// TODO: This should return bool TRUE/FALSE or 0/1
|
2011-08-18 00:31:15 +00:00
|
|
|
function update_session($id) {
|
|
|
|
|
|
|
|
$u = $this->get_by_id($id);
|
|
|
|
|
|
|
|
$userdata = array(
|
|
|
|
'user_id' => $u->row()->user_id,
|
|
|
|
'user_name' => $u->row()->user_name,
|
|
|
|
'user_type' => $u->row()->user_type,
|
2011-08-20 21:24:10 +00:00
|
|
|
'user_callsign' => $u->row()->user_callsign,
|
2011-09-27 19:25:21 +00:00
|
|
|
'user_locator' => $u->row()->user_locator,
|
2011-08-18 00:31:15 +00:00
|
|
|
'user_hash' => $this->_hash($u->row()->user_id."-".$u->row()->user_type)
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->session->set_userdata($userdata);
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool validate_session()
|
|
|
|
// Validate a user's login session
|
|
|
|
// If the user's session is corrupted in any way, it will clear the session
|
2011-08-18 00:31:15 +00:00
|
|
|
function validate_session() {
|
|
|
|
|
|
|
|
if($this->session->userdata('user_id'))
|
|
|
|
{
|
|
|
|
$user_id = $this->session->userdata('user_id');
|
|
|
|
$user_type = $this->session->userdata('user_type');
|
|
|
|
$user_hash = $this->session->userdata('user_hash');
|
|
|
|
|
|
|
|
if($this->_auth($user_id."-".$user_type, $user_hash)) {
|
2011-08-19 17:24:56 +00:00
|
|
|
// Freshen the session
|
|
|
|
$this->update_session($user_id);
|
2011-08-18 00:31:15 +00:00
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
$this->clear_session();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool authenticate($username, $password)
|
|
|
|
// Authenticate a user against the users table
|
2011-08-18 00:31:15 +00:00
|
|
|
function authenticate($username, $password) {
|
|
|
|
$u = $this->get($username);
|
|
|
|
if($u->num_rows != 0)
|
|
|
|
{
|
|
|
|
if($this->_auth($password, $u->row()->user_password)) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool authorize($level)
|
|
|
|
// Checks a user's level of access against the given $level
|
2011-08-18 00:31:15 +00:00
|
|
|
function authorize($level) {
|
|
|
|
$u = $this->get_by_id($this->session->userdata('user_id'));
|
2011-08-19 17:24:56 +00:00
|
|
|
$l = $this->config->item('auth_mode');
|
|
|
|
// Check to see if the minimum level of access is higher than
|
|
|
|
// the user's own level. If it is, use that.
|
|
|
|
if($this->config->item('auth_mode') > $level) {
|
|
|
|
$level = $this->config->item('auth_mode');
|
|
|
|
}
|
|
|
|
if(($this->validate_session()) && ($u->row()->user_type >= $level) || $this->config->item('use_auth') == FALSE || $level == 0) {
|
2011-08-18 00:31:15 +00:00
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool set($username, $data)
|
|
|
|
// Updates a user's record in the database
|
|
|
|
// TODO: This returns TRUE/1 no matter what at the moment - should
|
|
|
|
// TODO: return TRUE/FALSE or 0/1 depending on success/failure
|
2011-08-18 00:31:15 +00:00
|
|
|
function set($username, $data) {
|
|
|
|
$this->db->where('user_name', $username);
|
|
|
|
$this->db->update($this->config->item('auth_table', $data));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: object users()
|
|
|
|
// Returns a list of users
|
2011-08-18 00:31:15 +00:00
|
|
|
function users() {
|
|
|
|
$r = $this->db->get($this->config->item('auth_table'));
|
|
|
|
return $r;
|
|
|
|
}
|
|
|
|
|
2011-09-27 22:47:25 +00:00
|
|
|
// FUNCTION: array timezones()
|
|
|
|
// Returns a list of timezones
|
|
|
|
function timezones() {
|
|
|
|
$r = $this->db->query('SELECT id, name FROM timezones ORDER BY offset');
|
|
|
|
$ts = array();
|
|
|
|
foreach ($r->result_array() as $t) {
|
|
|
|
$ts[$t['id']] = $t['name'];
|
|
|
|
}
|
|
|
|
return $ts;
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: bool _auth($password, $hash)
|
|
|
|
// Checks a password against the stored hash
|
2011-08-18 00:31:15 +00:00
|
|
|
private function _auth($password, $hash) {
|
|
|
|
$h = new PasswordHash(8, FALSE);
|
|
|
|
if($h->CheckPassword($password, $hash)) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-18 19:57:27 +00:00
|
|
|
// FUNCTION: string _hash($password)
|
|
|
|
// Returns a hashed version of the supplied $password
|
|
|
|
// Will return '0' in the event of problems with the
|
|
|
|
// hashing function
|
2011-08-18 00:31:15 +00:00
|
|
|
private function _hash($password) {
|
|
|
|
$h = new PasswordHash(8, FALSE);
|
|
|
|
$hash = $h->HashPassword($password);
|
|
|
|
unset($h);
|
|
|
|
|
|
|
|
if(strlen($hash) < 20) {
|
2011-08-19 16:13:26 +00:00
|
|
|
return EPASSWORDINVALID;
|
2011-08-18 00:31:15 +00:00
|
|
|
} else {
|
|
|
|
return $hash;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
?>
|