mirror of
https://gitlab.com/davical-project/davical.git
synced 2026-05-28 03:04:15 +00:00
Add support for Digest authentication.
Signed-off-by: Andrew McMillan <andrew@morphoss.com>
This commit is contained in:
parent
1816d313bd
commit
ba1e80d185
@ -45,12 +45,18 @@ class HTTPAuthSession {
|
|||||||
/**#@-*/
|
/**#@-*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The constructor, which just calls the actual type configured
|
* The constructor, which just calls the type supplied or configured
|
||||||
*/
|
*/
|
||||||
function HTTPAuthSession() {
|
function HTTPAuthSession() {
|
||||||
global $c;
|
global $c;
|
||||||
|
|
||||||
if ( isset($c->http_auth_mode) && $c->http_auth_mode == "Digest" ) {
|
if ( ! empty($_SERVER['PHP_AUTH_DIGEST'])) {
|
||||||
|
$this->DigestAuthSession();
|
||||||
|
}
|
||||||
|
// else if ( isset($_SERVER["AUTHORIZATION"]) || isset($_SERVER['PHP_AUTH_USER']) ) {
|
||||||
|
// $this->BasicAuthSession();
|
||||||
|
// }
|
||||||
|
else if ( isset($c->http_auth_mode) && $c->http_auth_mode == "Digest" ) {
|
||||||
$this->DigestAuthSession();
|
$this->DigestAuthSession();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -73,7 +79,7 @@ class HTTPAuthSession {
|
|||||||
$auth_realm .= ' - ' . $principal_name;
|
$auth_realm .= ' - ' . $principal_name;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
dbg_error_log( "HTTPAuth", ":AuthFailedResponse Requesting authentictaion in the '%s' realm", $auth_realm );
|
dbg_error_log( "HTTPAuth", ":AuthFailedResponse Requesting authentication in the '%s' realm", $auth_realm );
|
||||||
$auth_header = sprintf( 'WWW-Authenticate: Basic realm="%s"', $auth_realm );
|
$auth_header = sprintf( 'WWW-Authenticate: Basic realm="%s"', $auth_realm );
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -81,7 +87,7 @@ class HTTPAuthSession {
|
|||||||
header('Content-type: text/plain; ; charset="utf-8"' );
|
header('Content-type: text/plain; ; charset="utf-8"' );
|
||||||
header( $auth_header );
|
header( $auth_header );
|
||||||
echo 'Please log in for access to this system.';
|
echo 'Please log in for access to this system.';
|
||||||
dbg_error_log( "HTTPAuth", ":Session: User is not authorised" );
|
dbg_error_log( "HTTPAuth", ":Session: User is not authorised: %s ", $_SERVER['REMOTE_ADDR'] );
|
||||||
exit;
|
exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -168,30 +174,55 @@ class HTTPAuthSession {
|
|||||||
function DigestAuthSession() {
|
function DigestAuthSession() {
|
||||||
global $c;
|
global $c;
|
||||||
|
|
||||||
|
$realm = $c->system_name;
|
||||||
|
$opaque = $realm;
|
||||||
|
if ( isset($_SERVER['HTTP_USER_AGENT']) ) $opaque .= $_SERVER['HTTP_USER_AGENT'];
|
||||||
|
if ( isset($_SERVER['REMOTE_ADDR']) ) $opaque .= $_SERVER['REMOTE_ADDR'];
|
||||||
|
$opaque = sha1($opaque);
|
||||||
|
|
||||||
if ( ! empty($_SERVER['PHP_AUTH_DIGEST'])) {
|
if ( ! empty($_SERVER['PHP_AUTH_DIGEST'])) {
|
||||||
// analyze the PHP_AUTH_DIGEST variable
|
// analyze the PHP_AUTH_DIGEST variable
|
||||||
if ( $data = $this->ParseDigestHeader($_SERVER['PHP_AUTH_DIGEST']) ) {
|
if ( $data = $this->ParseDigestHeader($_SERVER['PHP_AUTH_DIGEST']) ) {
|
||||||
// generate the valid response
|
|
||||||
$user_password = "Don't be silly! Why would a user have a password like this!!?";
|
|
||||||
/**
|
|
||||||
* @todo At this point we need to query the database for something fitting
|
|
||||||
* either strategy (A) or (B) above, in order to set $user_password to
|
|
||||||
* something useful!
|
|
||||||
*/
|
|
||||||
$A1 = md5($data['username'] . ':' . $c->system_name . ':' . $user_password);
|
|
||||||
$A2 = md5($_SERVER['REQUEST_METHOD'].':'.$data['uri']);
|
|
||||||
$valid_response = md5($A1.':'.$data['nonce'].':'.$data['nc'].':'.$data['cnonce'].':'.$data['qop'].':'.$A2);
|
|
||||||
|
|
||||||
if ( $data['response'] == $valid_response ) {
|
if ( $data['uri'] != $_SERVER['REQUEST_URI'] ) {
|
||||||
$this->AssignSessionDetails($u);
|
dbg_error_log( "ERROR", " DigestAuth: WTF! URI is '%s' and request URI is '%s'!?!" );
|
||||||
return;
|
$this->AuthFailedResponse();
|
||||||
|
// Does not return
|
||||||
|
}
|
||||||
|
|
||||||
|
// generate the valid response
|
||||||
|
$test_user = new Principal('username', $data['username']);
|
||||||
|
|
||||||
|
if ( preg_match( '{\*(Digest)?\*(.*)}', $test_user->password, $matches ) ) {
|
||||||
|
if ( $matches[1] == 'Digest' )
|
||||||
|
$A1 = $matches[2];
|
||||||
|
else {
|
||||||
|
dbg_error_log( "HTTPAuth", "Constructing A1 from md5(%s:%s:%s)", $data['username'], $realm, $matches[2] );
|
||||||
|
$A1 = md5($data['username'] . ':' . $realm . ':' . $matches[2]);
|
||||||
|
}
|
||||||
|
$A2 = md5($_SERVER['REQUEST_METHOD'].':'.$data['uri']);
|
||||||
|
$valid_response = md5($A1.':'.$data['nonce'].':'.$data['nc'].':'.$data['cnonce'].':'.$data['qop'].':'.$A2);
|
||||||
|
|
||||||
|
if ( $data['response'] == $valid_response ) {
|
||||||
|
$this->AssignSessionDetails($test_user);
|
||||||
|
dbg_error_log( "HTTPAuth", "Success!!!" );
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// Their account is not configured for Digest auth so we need to use Basic.
|
||||||
|
$this->AuthFailedResponse();
|
||||||
|
// Does not return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$nonce = uniqid();
|
$nonce = sha1(uniqid('',true));
|
||||||
$opaque = md5($c->system_name);
|
$authheader = sprintf('WWW-Authenticate: Digest realm="%s", qop="auth", nonce="%s", opaque="%s", algorithm="MD5"',
|
||||||
$this->AuthFailedResponse( sprintf('WWW-Authenticate: Digest realm="%s", qop="auth,auth-int", nonce="%s", opaque="%s"', $c->system_name, $nonce, $opaque ) );
|
$realm, $nonce, $opaque );
|
||||||
|
dbg_error_log( "HTTPAuth", $authheader );
|
||||||
|
$this->AuthFailedResponse( $authheader );
|
||||||
|
// Does not return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -204,16 +235,30 @@ class HTTPAuthSession {
|
|||||||
$needed_parts = array('nonce'=>1, 'nc'=>1, 'cnonce'=>1, 'qop'=>1, 'username'=>1, 'uri'=>1, 'response'=>1);
|
$needed_parts = array('nonce'=>1, 'nc'=>1, 'cnonce'=>1, 'qop'=>1, 'username'=>1, 'uri'=>1, 'response'=>1);
|
||||||
$data = array();
|
$data = array();
|
||||||
|
|
||||||
preg_match_all('@(\w+)=(?:([\'"])([^\2]+)\2|([^\s,]+))@', $auth_header, $matches, PREG_SET_ORDER);
|
preg_match_all('{(\w+)="([^"]+)"}', $auth_header, $matches, PREG_SET_ORDER);
|
||||||
|
|
||||||
foreach ($matches as $m) {
|
foreach ($matches as $m) {
|
||||||
$data[$m[1]] = $m[3] ? $m[3] : $m[4];
|
// dbg_error_log( "HTTPAuth", 'Match: "%s"', $m[0] );
|
||||||
|
$data[$m[1]] = $m[2];
|
||||||
unset($needed_parts[$m[1]]);
|
unset($needed_parts[$m[1]]);
|
||||||
|
dbg_error_log( "HTTPAuth", 'Received: %s: %s', $m[1], $m[2] );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
preg_match_all('{(\w+)=([^" ,]+)}', $auth_header, $matches, PREG_SET_ORDER);
|
||||||
|
foreach ($matches as $m) {
|
||||||
|
// dbg_error_log( "HTTPAuth", 'Match: "%s"', $m[0] );
|
||||||
|
$data[$m[1]] = $m[2];
|
||||||
|
unset($needed_parts[$m[1]]);
|
||||||
|
dbg_error_log( "HTTPAuth", 'Received: %s: %s', $m[1], $m[2] );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@dbg_error_log( "HTTPAuth", 'Received: nonce: %s, nc: %s, cnonce: %s, qop: %s, username: %s, uri: %s, response: %s',
|
||||||
|
$data['nonce'], $data['nc'], $data['cnonce'], $data['qop'], $data['username'], $data['uri'], $data['response']
|
||||||
|
);
|
||||||
return $needed_parts ? false : $data;
|
return $needed_parts ? false : $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* CheckPassword does all of the password checking and
|
* CheckPassword does all of the password checking and
|
||||||
* returns a user record object, or false if it all ends in tears.
|
* returns a user record object, or false if it all ends in tears.
|
||||||
@ -254,6 +299,7 @@ class HTTPAuthSession {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks whether a user is allowed to do something.
|
* Checks whether a user is allowed to do something.
|
||||||
*
|
*
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user