Compare commits
3 Commits
master
...
developmen
Author | SHA1 | Date | |
---|---|---|---|
c0ffd0a7ba | |||
3b43538f90 | |||
2d73384e17 |
39
README.md
39
README.md
@ -1,7 +1,7 @@
|
|||||||
# lucidAuth [![](https://img.shields.io/badge/status-in%20production-%23003399.svg)](#) [![](https://img.shields.io/badge/contributors-1-green.svg) ](#)
|
# lucidAuth [![](https://img.shields.io/badge/status-in%20production-%23003399.svg)](#) [![](https://img.shields.io/badge/contributors-1-green.svg) ](#)
|
||||||
> *Respect* the unexpected, mitigate your risks
|
> *Respect* the unexpected, mitigate your risks
|
||||||
|
|
||||||
Forward Authentication for use with proxies (caddy, nginx, traefik, etc)
|
Forward Authentication for use with loadbalancers/proxies/webservers (Apache, ~~Caddy~~, Lighttpd, NGINX, Traefik, etc)
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
- Create a new folder, navigate to it in a commandprompt and run the following command:
|
- Create a new folder, navigate to it in a commandprompt and run the following command:
|
||||||
@ -9,12 +9,39 @@ Forward Authentication for use with proxies (caddy, nginx, traefik, etc)
|
|||||||
- Edit `include/lucidAuth.config.php.example` to reflect your configuration and save as `include/lucidAuth.config.php`
|
- Edit `include/lucidAuth.config.php.example` to reflect your configuration and save as `include/lucidAuth.config.php`
|
||||||
- Create a new website (within any php-capable webserver) and make sure that the documentroot points to the `public` folder
|
- Create a new website (within any php-capable webserver) and make sure that the documentroot points to the `public` folder
|
||||||
- Check if you are able to browse to `https://<fqdn>/lucidAuth.login.php` (where `<fqdn>` is the actual domain -or IP address- your webserver is listening on)
|
- Check if you are able to browse to `https://<fqdn>/lucidAuth.login.php` (where `<fqdn>` is the actual domain -or IP address- your webserver is listening on)
|
||||||
- Edit your proxy's configuration to use the new website as forward proxy:
|
- Edit your webserver's/proxy's configuration to use the new website for forward authentication:
|
||||||
#### ~~in Caddy/nginx~~ <small>(planned for a later stage)</small>
|
- #### ~~in Apache~~ <small>(Soon™)</small>
|
||||||
|
|
||||||
#### in Traefik
|
- #### ~~in Caddy~~ <small>(Never, due to lacking functionality)</small>
|
||||||
|
|
||||||
|
- #### ~~in Lighttpd~~ <small>(Soon™)</small>
|
||||||
|
|
||||||
|
- #### in NGINX
|
||||||
|
Add the following lines (adjust to reflect your existing configuration - more [details](https://docs.nginx.com/nginx/admin-guide/security-controls/configuring-subrequest-authentication/)):
|
||||||
|
```
|
||||||
|
http {
|
||||||
|
#...
|
||||||
|
server {
|
||||||
|
#...
|
||||||
|
location /private/ {
|
||||||
|
auth_request /auth;
|
||||||
|
auth_request_set $auth_status $upstream_status;
|
||||||
|
}
|
||||||
|
|
||||||
|
location = /auth {
|
||||||
|
internal;
|
||||||
|
proxy_pass https://<fqdn>/lucidAuth.validateRequest.php;
|
||||||
|
proxy_pass_request_body off;
|
||||||
|
proxy_set_header Content-Length "";
|
||||||
|
proxy_set_header X-Original-URI $request_uri;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
- #### in Traefik
|
||||||
Add the following lines (change to reflect your existing configuration):
|
Add the following lines (change to reflect your existing configuration):
|
||||||
##### 1.7
|
##### 1.7.x (more [details](https://docs.traefik.io/v1.7/configuration/entrypoints/#forward-authentication))
|
||||||
```
|
```
|
||||||
[frontends.server1]
|
[frontends.server1]
|
||||||
entrypoints = ["https"]
|
entrypoints = ["https"]
|
||||||
@ -25,7 +52,7 @@ Forward Authentication for use with proxies (caddy, nginx, traefik, etc)
|
|||||||
[frontends.server1.routes.ext]
|
[frontends.server1.routes.ext]
|
||||||
rule = "Host:<fqdn>"
|
rule = "Host:<fqdn>"
|
||||||
```
|
```
|
||||||
##### 2.0
|
##### 2.x (more [details](https://docs.traefik.io/middlewares/forwardauth/))
|
||||||
Either whitelist IP's which should be trusted to send `HTTP_X-Forwarded-*` headers, ór enable insecure-mode in your static configuration:
|
Either whitelist IP's which should be trusted to send `HTTP_X-Forwarded-*` headers, ór enable insecure-mode in your static configuration:
|
||||||
```
|
```
|
||||||
entryPoints:
|
entryPoints:
|
||||||
|
@ -106,7 +106,7 @@ function storeToken (string $secureToken, string $qualifiedUsername, string $htt
|
|||||||
// This might seem backwards, but relying on $_SERVER directly allows spoofed values with potential security risks
|
// This might seem backwards, but relying on $_SERVER directly allows spoofed values with potential security risks
|
||||||
return (strlen($value) > strlen($httpHost)) ? false : (0 === substr_compare($httpHost, $value, -strlen($value)));
|
return (strlen($value) > strlen($httpHost)) ? false : (0 === substr_compare($httpHost, $value, -strlen($value)));
|
||||||
}))[0];
|
}))[0];
|
||||||
if ($cookieDomain && setcookie('JWT', $secureToken, (time() + $settings->Session['Duration']), '/', '.' . $cookieDomain)) {
|
if ($cookieDomain && setcookie('JWT', $secureToken, (time() + $settings->Session['Duration']), '/', '.' . $cookieDomain, TRUE)) {
|
||||||
return ['status' => 'Success'];
|
return ['status' => 'Success'];
|
||||||
} else {
|
} else {
|
||||||
return ['status' => 'Fail', 'reason' => 'Unable to store cookie(s)'];
|
return ['status' => 'Fail', 'reason' => 'Unable to store cookie(s)'];
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
include_once('../include/lucidAuth.functions.php');
|
include_once('../include/lucidAuth.functions.php');
|
||||||
|
|
||||||
if ($_POST['do'] == 'login') {
|
if ($_POST['do'] === 'login') {
|
||||||
$result = authenticateLDAP($_POST['username'], $_POST['password']);
|
$result = authenticateLDAP($_POST['username'], $_POST['password']);
|
||||||
if ($result['status'] === 'Success') {
|
if ($result['status'] === 'Success') {
|
||||||
// Store authentication token; in database serverside & in cookie clientside
|
// Store authentication token; in database serverside & in cookie clientside
|
||||||
|
@ -8,7 +8,20 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ($validateTokenResult['status'] === "Success") {
|
if ($validateTokenResult['status'] === "Success") {
|
||||||
if ($_REQUEST['do'] === 'retrievesessions') {
|
switch ($_REQUEST['do']) {
|
||||||
|
case 'mutateusers':
|
||||||
|
if (isset($_REQUEST['new']) && isset($_REQUEST['removed'])) {
|
||||||
|
// Do magic!
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo json_encode([
|
||||||
|
"Result" => "Failure",
|
||||||
|
"Reason" => "Incomplete request data"
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'retrievesessions':
|
||||||
$storedTokens = [];
|
$storedTokens = [];
|
||||||
|
|
||||||
$pdoQuery = $pdoDB->prepare('
|
$pdoQuery = $pdoDB->prepare('
|
||||||
@ -41,7 +54,44 @@
|
|||||||
"SessionCount" => sizeof($storedTokens),
|
"SessionCount" => sizeof($storedTokens),
|
||||||
"UserSessions" => json_encode($storedTokens)
|
"UserSessions" => json_encode($storedTokens)
|
||||||
]);
|
]);
|
||||||
|
break;
|
||||||
|
case 'deletesession':
|
||||||
|
if (isset($_REQUEST['userid']) && isset($_REQUEST['tokenid'])) {
|
||||||
|
try {
|
||||||
|
$pdoQuery = $pdoDB->prepare('
|
||||||
|
DELETE FROM SecureToken
|
||||||
|
WHERE SecureToken.UserId = :userid AND SecureToken.Id = :tokenid
|
||||||
|
');
|
||||||
|
$pdoQuery->execute([
|
||||||
|
':userid' => (int) $_REQUEST['userid'],
|
||||||
|
':tokenid' => (int) $_REQUEST['tokenid']
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Return JSON object
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo json_encode([
|
||||||
|
"Result" => "Success",
|
||||||
|
"RowCount" => $pdoQuery->RowCount()
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
catch (Exception $e) {
|
||||||
|
// Return JSON object
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo json_encode([
|
||||||
|
"Result" => "Failure",
|
||||||
|
"Reason" => "Failed deleting tokens from database"
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
|
header('Content-Type: application/json');
|
||||||
|
echo json_encode([
|
||||||
|
"Result" => "Failure",
|
||||||
|
"Reason" => "Incomplete request data"
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
// No action requested, default action
|
// No action requested, default action
|
||||||
include_once('../include/lucidAuth.template.php');
|
include_once('../include/lucidAuth.template.php');
|
||||||
|
|
||||||
@ -74,6 +124,7 @@
|
|||||||
implode($tableRows)
|
implode($tableRows)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// No cookie containing valid authentication token found;
|
// No cookie containing valid authentication token found;
|
||||||
|
@ -1,22 +1,74 @@
|
|||||||
jQuery.fn.inlineConfirm = function() {
|
jQuery.fn.ConfirmDelete = function() {
|
||||||
return this.on('click', function(event) {
|
return this.on('click', function() {
|
||||||
sessionID = $(this).data('sessionid');
|
var sessionID = $(this).data('sessionid');
|
||||||
// event.preventDefault();
|
|
||||||
$(this).off('click').parent().empty().append(
|
$(this).off('click').parent().empty()
|
||||||
$('<button>', {
|
.append($('<button>', {
|
||||||
text: locales[(localStorage.getItem('language') !== null ? localStorage.getItem('language') : 'en')]['button_yes'],
|
text: locales[(localStorage.getItem('language') !== null ? localStorage.getItem('language') : 'en')]['button_yes'],
|
||||||
class: 'bttn-simple bttn-xs bttn-primary sessiondeleteconfirm',
|
class: 'bttn-simple bttn-xs bttn-primary sessiondeleteconfirm',
|
||||||
style: 'margin-right: 3px;',
|
style: 'margin-right: 3px;',
|
||||||
'data-translation': 'button_yes',
|
'data-translation': 'button_yes',
|
||||||
'data-sessionid': sessionID
|
'data-sessionid': sessionID
|
||||||
})).append(
|
}).on('click', function() {
|
||||||
|
// Move this out of 'ConfirmDelete()'
|
||||||
|
var pressedButton = $(this);
|
||||||
|
$(pressedButton).prop('disabled', true).css({
|
||||||
|
'background': '#999 url() no-repeat center',
|
||||||
|
'color': 'transparent',
|
||||||
|
'transform': 'rotateX(180deg)',
|
||||||
|
'cursor': 'default'
|
||||||
|
}).next().prop('disabled', true).addClass('disabled');
|
||||||
|
|
||||||
|
$.post("lucidAuth.manage.php", {
|
||||||
|
do: "deletesession",
|
||||||
|
tokenid: sessionID,
|
||||||
|
userid: $(pressedButton).closest('tr').data('userid')
|
||||||
|
})
|
||||||
|
.done(function(data,_status) {
|
||||||
|
if (data.Result === 'Success') {
|
||||||
|
$(pressedButton).css({
|
||||||
|
'background': 'green url() no-repeat center',
|
||||||
|
'transform': 'rotateX(0deg)'
|
||||||
|
});
|
||||||
|
setTimeout(function() {
|
||||||
|
$(pressedButton).closest('tr').fadeOut(500, function() {
|
||||||
|
$(this).remove()});
|
||||||
|
}, 2250);
|
||||||
|
} else {
|
||||||
|
$(pressedButton).css({
|
||||||
|
'background': 'red url() no-repeat center',
|
||||||
|
'transform': 'rotateX(0deg)'
|
||||||
|
});
|
||||||
|
setTimeout(function() {
|
||||||
|
$(pressedButton).closest('td').empty()
|
||||||
|
.append(
|
||||||
$('<button>', {
|
$('<button>', {
|
||||||
|
text: locales[(localStorage.getItem('language') !== null ? localStorage.getItem('language') : 'en')]['button_delete'],
|
||||||
|
class: 'bttn-simple bttn-xs bttn-primary sessiondelete',
|
||||||
|
'data-translation': 'button_delete',
|
||||||
|
'data-sessionid': sessionID
|
||||||
|
}).ConfirmDelete())
|
||||||
|
}, 2250);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}))
|
||||||
|
.append($('<button>', {
|
||||||
text: locales[(localStorage.getItem('language') !== null ? localStorage.getItem('language') : 'en')]['button_no'],
|
text: locales[(localStorage.getItem('language') !== null ? localStorage.getItem('language') : 'en')]['button_no'],
|
||||||
class: 'bttn-simple bttn-xs bttn-primary sessiondeletecancel',
|
class: 'bttn-simple bttn-xs bttn-primary sessiondeletecancel',
|
||||||
'data-translation': 'button_no',
|
'data-translation': 'button_no',
|
||||||
'data-sessionid': sessionID
|
'data-sessionid': sessionID
|
||||||
})
|
}).on('click', function() {
|
||||||
);
|
// Move this out of 'ConfirmDelete()'
|
||||||
|
var pressedButton = $(this);
|
||||||
|
$(pressedButton).closest('td').empty()
|
||||||
|
.append(
|
||||||
|
$('<button>', {
|
||||||
|
text: locales[(localStorage.getItem('language') !== null ? localStorage.getItem('language') : 'en')]['button_delete'],
|
||||||
|
class: 'bttn-simple bttn-xs bttn-primary sessiondelete',
|
||||||
|
'data-translation': 'button_delete',
|
||||||
|
'data-sessionid': sessionID
|
||||||
|
}).ConfirmDelete())
|
||||||
|
}))
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -35,9 +87,11 @@ $(document).ready(function(){
|
|||||||
$('#sessions tbody').empty();
|
$('#sessions tbody').empty();
|
||||||
$('#sessions').fadeToggle();
|
$('#sessions').fadeToggle();
|
||||||
|
|
||||||
|
var userID = $(this).closest('tr').find('td:nth-child(1)').data('userid');
|
||||||
|
|
||||||
$.post("lucidAuth.manage.php", {
|
$.post("lucidAuth.manage.php", {
|
||||||
do: "retrievesessions",
|
do: "retrievesessions",
|
||||||
userid: $(this).closest('tr').find('td:nth-child(1)').data('userid')
|
userid: userID
|
||||||
})
|
})
|
||||||
.done(function(data,_status) {
|
.done(function(data,_status) {
|
||||||
if (data.Result === 'Success') {
|
if (data.Result === 'Success') {
|
||||||
@ -51,7 +105,9 @@ $(document).ready(function(){
|
|||||||
} catch(e) {
|
} catch(e) {
|
||||||
// Do nothing
|
// Do nothing
|
||||||
}
|
}
|
||||||
$('#sessiontable tbody').append($('<tr>')
|
$('#sessiontable tbody').append($('<tr>', {
|
||||||
|
'data-userid': userID
|
||||||
|
})
|
||||||
.append($('<td>', {
|
.append($('<td>', {
|
||||||
text: new Date(Sessions[i]['iat'] * 1000).toLocaleString('en-GB')
|
text: new Date(Sessions[i]['iat'] * 1000).toLocaleString('en-GB')
|
||||||
}))
|
}))
|
||||||
@ -70,7 +126,7 @@ $(document).ready(function(){
|
|||||||
}))
|
}))
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
$('#sessiontable .sessiondelete').inlineConfirm();
|
$('#sessiontable .sessiondelete').ConfirmDelete();
|
||||||
} else {
|
} else {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -131,14 +187,15 @@ $(document).ready(function(){
|
|||||||
});
|
});
|
||||||
|
|
||||||
console.log({'new': newEntries, 'removed': removedEntries});
|
console.log({'new': newEntries, 'removed': removedEntries});
|
||||||
|
/*
|
||||||
/* $.get("psworker.php", {
|
$.post("lucidAuth.manage.php", {
|
||||||
do: "mutate",
|
do: "mutateusers",
|
||||||
mutations: {
|
|
||||||
new: newEntries,
|
new: newEntries,
|
||||||
removed: removedEntries
|
removed: removedEntries
|
||||||
|
})
|
||||||
|
.done(function(data,_status) {
|
||||||
}
|
}
|
||||||
})*/
|
*/
|
||||||
});
|
});
|
||||||
|
|
||||||
$('#btncancel').click(function() {
|
$('#btncancel').click(function() {
|
||||||
|
Loading…
Reference in New Issue
Block a user