This commit is contained in:
William Mantly 2020-04-10 17:04:50 -04:00
parent 0c337716d3
commit 5266aec2b1
14 changed files with 300 additions and 244 deletions

View File

@ -110,13 +110,17 @@ User.addByInvite = async function(data){
}; };
User.remove = async function(data){
try{
return await linuxUser.removeUser(this.username);
}catch(error){
throw error;
}
};
User.setPassword = async function(data){ User.setPassword = async function(data){
try{ try{
if(!data.password1 || data.password1 !== data.password2){ await linuxUser.setPassword(this.username, data.password);
throw new Error('PasswordMismatch');
}
await linuxUser.setPassword(this.username, data.password1);
return this; return this;
}catch(error){ }catch(error){

View File

@ -23,6 +23,45 @@ app.api = (function(app){
}); });
} }
function put(url, data, callack){
$.ajax({
type: 'PUT',
url: baseURL+url,
headers:{
'auth-token': app.auth.getToken()
},
data: JSON.stringify(data),
contentType: "application/json; charset=utf-8",
dataType: "json",
complete: function(res, text){
callack(
text !== 'success' ? res.statusText : null,
JSON.parse(res.responseText),
res.status
)
}
});
}
function remove(url, callack){
$.ajax({
type: 'delete',
url: baseURL+url,
headers:{
'auth-token': app.auth.getToken()
},
contentType: "application/json; charset=utf-8",
dataType: "json",
complete: function(res, text){
callack(
text !== 'success' ? res.statusText : null,
JSON.parse(res.responseText),
res.status
)
}
});
}
function get(url, callack){ function get(url, callack){
$.ajax({ $.ajax({
type: 'GET', type: 'GET',
@ -42,7 +81,7 @@ app.api = (function(app){
}); });
} }
return {post: post, get: get} return {post: post, get: get, put: put, delete: remove}
})(app) })(app)
app.auth = (function(app) { app.auth = (function(app) {
@ -56,11 +95,11 @@ app.auth = (function(app) {
function isLoggedIn(callack){ function isLoggedIn(callack){
if(getToken()){ if(getToken()){
return app.api.get('users/me', function(error, data){ return app.api.get('user/me', function(error, data){
return callack(error, data.username); return callack(error, data.username);
}) })
}else{ }else{
callack(false, false); callack(null, false);
} }
} }
@ -73,20 +112,50 @@ app.auth = (function(app) {
}); });
} }
function logOut(){
localStorage.removeItem('APIToken');
}
return { return {
getToken: getToken, getToken: getToken,
isLoggedIn: isLoggedIn, isLoggedIn: isLoggedIn,
logIn:logIn logIn: logIn,
logOut: logOut,
} }
})(app); })(app);
app.users = (function(app){ app.user = (function(app){
function list(callack){
app.api.get('user/?detail=true', function(error, data){
callack(error, data);
})
}
function add(args, callack){
app.api.post('user/', args, function(error, data){
callack(error, data);
});
}
function remove(args, callack){
app.api.delete('user/'+ args.username, function(error, data){
callack(error, data);
});
}
function changePassword(args, callack){
app.api.put('users/'+ arg.username || '', args, function(error, data){
callack(error, data);
});
}
function createInvite(callack){ function createInvite(callack){
app.api.post('users/invite', function(error, data, status){ app.api.post('user/invite', function(error, data, status){
callack(error, data.token); callack(error, data.token);
}); });
} }
function consumeInvite(args){ function consumeInvite(args){
app.api.post('/auth/invite/'+args.token, args, function(error, data){ app.api.post('/auth/invite/'+args.token, args, function(error, data){
if(data.token){ if(data.token){
@ -96,31 +165,48 @@ app.users = (function(app){
callack(error) callack(error)
}); });
} }
return {list, remove};
})(app); })(app);
app.hosts = (function(app){ app.host = (function(app){
function list(callack){ function list(callack){
app.api.get('hosts/?detail=true', function(error, data){ app.api.get('host/?detail=true', function(error, data){
callack(error, data.hosts) callack(error, data.hosts)
}); });
} }
function get(host, callack){ function get(host, callack){
app.api.get('hosts/' + host, function(error, data){ app.api.get('host/' + host, function(error, data){
callack(error, data) callack(error, data)
}); });
} }
function add(args, callack){ function add(args, callack){
app.api.post('hosts/', args, function(error, data){ app.api.post('host/', args, function(error, data){
callack(error, data); callack(error, data);
}); });
} }
function edit(args, callack){
app.api.put('host/' + args.edit_host, args, function(error, data){
callack(error, data);
});
}
function remove(args, callack){
app.api.delete('host/'+ args.host, function(error, data){
callack(error, data);
})
}
return { return {
list: list, list: list,
get: get, get: get,
add: add add: add,
edit: edit,
remove: remove,
} }
})(app); })(app);
@ -133,18 +219,21 @@ app.util = (function(app){
return results === null ? '' : decodeURIComponent(results[1].replace(/\+/g, ' ')); return results === null ? '' : decodeURIComponent(results[1].replace(/\+/g, ' '));
}; };
function actionMessage(message, $target){ function actionMessage(message, options){
$target = $target || $('div.actionMessage'); options = options || {};
$target = options.$target || $('div.actionMessage');
message = message || '';
if($target.html() === message) return; if($target.html() === message) return;
if($target.html()){ if($target.html()){
$target.slideUp('fast', function(){ $target.slideUp('fast', function(){
$target.html('') $target.html('')
if(message) actionMessage(message); if(message) actionMessage(message, options);
}) })
return; return;
}else{ }else{
if(options.type) $target.addClass('alert-' + options.type);
$target.html(message).slideDown('fast'); $target.html(message).slideDown('fast');
} }
} }
@ -173,7 +262,18 @@ app.util = (function(app){
} }
})(app); })(app);
$.holdReady( true );
if(!location.pathname.includes('/login')){
app.auth.isLoggedIn(function(error, isLoggedIn){
if(error || !isLoggedIn){
location.replace('/login/?redirect='+location.pathname);
}else{
$.holdReady( false );
}
})
}else{
$.holdReady( false );
}
$( document ).ready( function () { $( document ).ready( function () {
@ -195,9 +295,14 @@ $( document ).ready( function () {
//ajax form submit //ajax form submit
function formAJAX( btn, del ) { function formAJAX( btn, del ) {
var $form = $(btn).closest( '[action]' ), // gets the 'form' parent event.preventDefault(); // avoid to execute the actual submit of the form.
str = $form.find( '[name]' ).serialize(); // builds query string var $form = $(btn).closest( '[action]' ); // gets the 'form' parent
/*if( !$form.validate( var formData = $form.find( '[name]' ).serializeObject(); // builds query formDataing
var method = $form.attr('method') || 'post';
console.log('formAJAX method', method)
if( !$form.validate(
{ {
form: { form: {
alertCount: true, alertCount: true,
@ -205,21 +310,11 @@ function formAJAX( btn, del ) {
} }
}) ) { }) ) {
return false; return false;
}*/
if( del ){ // delete request
if( confirm( 'Are you sure?' ) ) {
str += "&deleteThis=true"; // adds delete to query string
}else{
event.preventDefault(); // avoid to execute the actual submit of the form
return false; //cancels function
}
} }
$.post( $form.attr( 'action' ), str, function ( data ) { // sends post data to server validate app.api[method]($form.attr( 'action' ), formData, function(error, data){
tableAJAX( data.action_message ); //re-populate table tableAJAX( data.message ); //re-populate table
eval( $form.attr( 'evalAJAX' ) ); //gets JS to run after completion
}); });
eval( $form.attr( 'evalAJAX' ) ); //gets JS to run after completion
event.preventDefault(); // avoid to execute the actual submit of the form.
return false; // avoid to execute the actual submit of the form.
} }

View File

@ -140,7 +140,7 @@ $.validateSettings({
password: function( value ) { password: function( value ) {
var reg = /^(?=[^\d_].*?\d)\w(\w|[!@#$%]){1,48}/; var reg = /^(?=[^\d_].*?\d)\w(\w|[!@#$%]){1,48}/;
if ( reg.test( value ) === false ) { if ( reg.test( value ) === false ) {
return "Try again"; return "Weak password, Try again";
} }
} }
} }

View File

@ -17,6 +17,18 @@ router.post('/login', async function(req, res, next){
} }
}); });
router.all('/logout', async function(req, res, next){
try{
if(req.user){
await req.user.logout();
}
res.json({message: 'Bye'})
}catch(error){
next(error);
}
});
router.post('/invite/:token', async function(req, res, next) { router.post('/invite/:token', async function(req, res, next) {
try{ try{
req.body.token = req.params.token; req.body.token = req.params.token;

View File

@ -30,7 +30,8 @@ router.get('/', async function(req, res, next){
router.put('/:host', async function(req, res, next){ router.put('/:host', async function(req, res, next){
try{ try{
req.body.updated_by = req.user.username; req.body.updated_by = req.user.username;
await Host.update(req.body, req.params.host); let host = await Host.get(req.params.host);
await host.update(req.body);
return res.json({ return res.json({
message: `Host "${req.params.host}" updated.` message: `Host "${req.params.host}" updated.`

View File

@ -3,8 +3,33 @@
const router = require('express').Router(); const router = require('express').Router();
const {User} = require('../models/user'); const {User} = require('../models/user');
router.get('/', async function(req, res, next){
try{
return res.json({results: await User.list()});
}catch(error){
next(error);
}
});
router.get('/me', async function(req, res){ router.post('/', async function(req, res, next){
try{
return res.json({results: await User.add(req.body)});
}catch(error){
next(error);
}
});
router.delete('/:username', async function(req, res, next){
try{
let user = await User.get(req.params.username);
return res.json({username: req.params.username, results: await user.remove()})
}catch(error){
next(error);
}
});
router.get('/me', async function(req, res, next){
try{ try{
return res.json({username: req.user.username}); return res.json({username: req.user.username});
}catch(error){ }catch(error){
@ -12,6 +37,23 @@ router.get('/me', async function(req, res){
} }
}); });
router.put('/password', async function(req, res, next){
try{
return res.json({results: await req.user.setPassword(req.body)})
}catch(error){
next(error);
}
});
router.put('/password/:username', async function(req, res, next){
try{
let user = await User.get(req.params.username);
return res.json({results: await user.setPassword(req.body)});
}catch(error){
next(error);
}
});
router.post('/invite', async function(req, res, next){ router.post('/invite', async function(req, res, next){
try{ try{
let token = await req.user.invite(); let token = await req.user.invite();

View File

@ -27,7 +27,7 @@ function processKeys(map, data, partial){
continue; continue;
} }
if(data.hasOwnProperty(key) && typeof(data[key]) !== map[key].type){ if(data.hasOwnProperty(key) && map[key].type && typeof(data[key]) !== map[key].type){
errors.push({key, message:`${key} is not ${map[key].type} type.`}); errors.push({key, message:`${key} is not ${map[key].type} type.`});
continue; continue;
} }

View File

@ -4,10 +4,10 @@
<input type="hidden" name="host" value="<< host >>" /> <input type="hidden" name="host" value="<< host >>" />
<td><a target="_blank" href="http://<<host>>"><<host>></a></td> <td><a target="_blank" href="http://<<host>>"><<host>></a></td>
<td><<ip>></td> <td><<ip>></td>
<td class="hidden-xs"><<date>></td> <td class="hidden-xs"><<updated_on>></td>
<td> <td>
<button type="button" data-host="<< host >>" onclick="editHost(this);" class="btn btn-sm btn-default">Edit</button> <button type="button" data-host="<< host >>" onclick="editHost(this);" class="btn btn-sm btn-default">Edit</button>
<button type="button" onclick="formAJAX(this, true)" class="btn btn-sm btn-default">Delete</button> <button type="button" onclick="app.host.remove({host:'<<host>>'}, function(){tableAJAX('Host <<host>> delete.')})" class="btn btn-sm btn-default">Delete</button>
</td> </td>
</tr> </tr>
</script> </script>
@ -18,22 +18,24 @@
var btn = $(btn); var btn = $(btn);
currentEditHost = btn.data('host'); currentEditHost = btn.data('host');
$('.editWindow').slideDown('fast'); $('.editWindow').slideDown('fast');
$('.editWindow .panel-body').slideDown('fast');
$('.editWindow .panel-title .pull-left').html("Edit "+ btn.data('host')) $('.editWindow .panel-title .pull-left').html("Edit "+ btn.data('host'))
$('div.editWindow .panel-body span').html($('#addHost').html()) $('div.editWindow .panel-body span').html($('#addHost').html())
$('div.editWindow .panel-body span button').remove() $('div.editWindow .panel-body span button').remove()
app.hosts.get(currentEditHost, function(error, data){ $(".editWindow input[name='edit_host']").val(btn.data('host'));
console.log(data)
$.each( data.results, function( key, val ) {
$(".editWindow input[name='" + key + "']").val(val); app.host.get(currentEditHost, function(error, data){
console.log(data)
$.each( data.results, function( key, value ) {
if(typeof value == "boolean"){
$(".editWindow #"+ key +"-"+ value).prop('checked', true)
}else{
$(".editWindow input[name='" + key + "']").val(value);
}
$('.editWindow .panel-body').slideDown('fast');
}); });
}); });
// $.getJSON('api/hosts/' + btn.data('host'), function( data ) {
// $("input[name='old_host']").attr('value', btn.data('host'));
// });
} }
function tableAJAX(actionMessage){ function tableAJAX(actionMessage){
@ -41,16 +43,17 @@
$('#tableAJAX').html('').hide(); $('#tableAJAX').html('').hide();
app.util.actionMessage('') app.util.actionMessage('')
app.hosts.list(function(err, data){ app.host.list(function(err, data){
if(err) return app.util.actionMessage(err); if(err) return app.util.actionMessage(err, {type: 'danger'});
if(data){ if(data){
$.each(data, function( key, value ) { $.each(data, function( key, value ) {
host_row = ich.rowTemplate(value); host_row = ich.rowTemplate(value);
$('#tableAJAX').append(host_row); $('#tableAJAX').append(host_row);
}); });
app.util.actionMessage(actionMessage || '', {type: 'info'});
$('#tableAJAX').fadeIn('slow'); $('#tableAJAX').fadeIn('slow');
}else{ }else{
app.util.actionMessage('No hosts...') app.util.actionMessage('No hosts...', {type: 'info'});
} }
}); });
} }
@ -59,16 +62,33 @@
tableAJAX(); //populate the table tableAJAX(); //populate the table
setInterval(tableAJAX, 30000); setInterval(tableAJAX, 30000);
$('#addHost').on('submit', function(){ $('form.addHost').on('submit', function(){
event.preventDefault(); event.preventDefault();
$form = $(this); $form = $(this);
var action = $($form.find('button[type="submit"]')[0]).data('type')
app.util.actionMessage('') app.util.actionMessage('')
if($form.attr('isValid') === 'true'){ if($form.attr('isValid') === 'true'){
app.hosts.add($form.serializeObject(), function(error, data){ var formdata = $form.serializeObject();
app.util.actionMessage(data.message || 'Error!'); if(formdata.targetport) formdata.targetport = Number(formdata.targetport);
if(!error) $form.trigger('reset'); if(formdata.targetssl) formdata.targetssl = formdata.targetssl == 'true' ? true : false;
if(formdata.forcessl) formdata.forcessl = formdata.forcessl == 'true' ? true : false;
app.host[action](formdata, function(error, data){
if(error){
console.log('error data',data)
app.util.actionMessage(error + data.message, {type: 'danger'});
return;
}
if(action == 'edit') $('.editWindow').slideUp('fast');
app.util.actionMessage(data.message || 'Error!', {type: 'info'});
tableAJAX("Added "+ formdata.host);
$form.trigger('reset');
}) })
} }
}); });
@ -88,9 +108,10 @@
</div> </div>
</div> </div>
<div class="panel-body"> <div class="panel-body">
<form action="api" evalAJAX="$('div.editWindow').slideUp();currentEditHost = null;"> <form class="addHost" onsubmit="$(this).validate()">
<span></span> <span></span>
<button type="button" class="btn btn-danger">Update</button> <input type="hidden" name="edit_host" />
<button type="submit" data-type="edit" class="btn btn-danger host-submit">Update</button>
<button class="btn btn-link" type="reset">Cancel</button> <button class="btn btn-link" type="reset">Cancel</button>
</form> </form>
</div> </div>
@ -106,19 +127,19 @@
</div> </div>
</div> </div>
<div class="panel-body"> <div class="panel-body">
<form id="addHost" onsubmit="$(this).validate()"> <form class="addHost" id="addHost" onsubmit="$(this).validate()">
<div class="form-group"> <div class="form-group">
<label class="control-label">Incoming SSL</label> <label class="control-label">Incoming SSL</label>
<div class="radio"> <div class="radio">
<label> <label>
<input type="radio" name="forceSSL" id="forceSSL1" value="true" checked> <input type="radio" name="forcessl" id="forcessl-true" value="true" checked>
Force incoming connections over HTTPS <b>Recommended</b> Force incoming connections over HTTPS <b>Recommended</b>
</label> </label>
</div> </div>
<div class="radio"> <div class="radio">
<label> <label>
<input type="radio" name="forceSSL" id="forceSSL2" value="false"> <input type="radio" name="forcessl" id="forcessl-false" value="false">
Allow use of both HTTP and HTTPS Allow use of both HTTP and HTTPS
</label> </label>
</div> </div>
@ -126,38 +147,38 @@
<div class="form-group"> <div class="form-group">
<label class="control-label">Hostname</label> <label class="control-label">Hostname</label>
<input type="text" name="host" class="form-control" placeholder="ex: proxy.cloud-ops.net" validate="host:3" > <input type="text" name="host" class="form-control" placeholder="ex: proxy.cloud-ops.net" validate=":3" >
</div> </div>
<hr /> <hr />
<div class="form-group"> <div class="form-group">
<label class="control-label">Target IP or Host Name</label> <label class="control-label">Target IP or Host Name</label>
<input type="text" name="ip" class="form-control" placeholder="ex: 10.10.10.10" /> <input type="text" name="ip" class="form-control" placeholder="ex: 10.10.10.10" validate=":3" />
</div> </div>
<div class="form-group"> <div class="form-group">
<label class="control-label">Target TCP Port</label> <label class="control-label">Target TCP Port</label>
<input type="number" name="targetPort" class="form-control" value="80" min="0" max="65535" /> <input type="number" name="targetport" class="form-control" value="80" min="0" max="65535" />
</div> </div>
<div class="form-group"> <div class="form-group">
<label class="control-label">Target SSL</label> <label class="control-label">Target SSL</label>
<div class="radio"> <div class="radio">
<label> <label>
<input type="radio" name="targetSSL" id="targetSSL1" value="true"> <input type="radio" name="targetssl" id="targetssl-true" value="true">
Proxt to HTTPS Proxt to HTTPS
</label> </label>
</div> </div>
<div class="radio"> <div class="radio">
<label> <label>
<input type="radio" name="targetSSL" id="targetSSL2" value="false" checked> <input type="radio" name="targetssl" id="targetssl-false" value="false" checked>
Proxy to HTTP <b>Recommended</b> Proxy to HTTP <b>Recommended</b>
</label> </label>
</div> </div>
</div> </div>
<button type="submit" class="btn btn-default">Add</button> <button type="submit" data-type="add" class="btn btn-default host-submit">Add</button>
</form> </form>
</div> </div>
</div> </div>

View File

@ -1,27 +1,32 @@
<%- include('top') %> <%- include('top') %>
<script type="text/javascript"> <script type="text/javascript">
app.auth.isLoggedIn(function(error, isLoggedIn){
if(isLoggedIn){
window.location.href = app.util.getUrlParameter('redirect') || '/';
}
})
$(document).ready(function(){ $(document).ready(function(){
$( "form[action='login']" ).submit(function( event ) { $( "form[action='login']" ).submit(function( event ) {
event.preventDefault();
$form = $(this); $form = $(this);
app.util.actionMessage('') app.util.actionMessage('')
if($form.attr('isValid') === 'true'){
if($form.attr('isValid') !== 'true'){
console.log('')
return app.util.actionMessage('Please fix the errors bellow!')
}
app.auth.logIn($form.serializeObject(), function(error, data){ app.auth.logIn($form.serializeObject(), function(error, data){
if(data){ if(data){
app.util.actionMessage('Login successful!'); app.util.actionMessage('Login successful!');
window.location.href = app.util.getUrlParameter('redirect') || '/hosts/'; window.location.href = app.util.getUrlParameter('redirect') || '/';
}else{ }else{
app.util.actionMessage('Login Failed, please try again'); app.util.actionMessage('Login Failed, please try again');
} }
}); });
}else{
app.util.actionMessage('Please fix the errors bellow!')
}
event.preventDefault();
}); });
}); });
</script> </script>
@ -35,7 +40,7 @@
<div class="panel-body"> <div class="panel-body">
<div class="alert alert-warning actionMessage" style="display:none"> <div class="alert alert-warning actionMessage" style="display:none">
</div> </div>
<form method="post" action="login" onsubmit="$(this).validate()"> <form action="login" onsubmit="$(this).validate()">
<input type="hidden" name="redirect" value="<%= redirect %>"> <input type="hidden" name="redirect" value="<%= redirect %>">
<div class="form-group"> <div class="form-group">
<label class="control-label">User name</label> <label class="control-label">User name</label>

View File

@ -1,27 +0,0 @@
<% include top %>
<div class="row">
<div class="col-md-4">
<div class="panel panel-default">
<div class="panel-heading">
<div class="panel-title">Log in</div>
</div>
<div class="panel-body">
<div class="alert alert-warning actionMessage" style="display:none">
</div>
<form method="post" action="login" onsubmit="$(this).validate()">
<input type="hidden" name="redirect" value="<%= redirect %>">
<div class="form-group">
<label class="control-label">Username</label>
<input type="text" name="username" class="form-control" placeholder="User" validate="user:3" />
</div>
<div class="form-group">
<label class="control-label">Password</label>
<input type="password" name="password" class="form-control" placeholder="Password" validate="password:5" />
</div>
<button type="submit" class="btn btn-default" >Log in</button>
</form>
</div>
</div>
</div>
</div>
<% include bottom %>

View File

@ -13,7 +13,7 @@
<script type="text/javascript" src='/static/js/jquery.min.js'></script> <script type="text/javascript" src='/static/js/jquery.min.js'></script>
<script type="text/javascript" src='/static/js/bootstrap.min.js'></script> <script type="text/javascript" src='/static/js/bootstrap.min.js'></script>
<script type="text/javascript" src='/static/js/ICanHaz.js'></script> <script type="text/javascript" src='/static/js/ICanHaz.js'></script>
<script type="text/javascript" src="/static/js/layout.js"></script> <script type="text/javascript" src="/static/js/app.js"></script>
<script type="text/javascript" src="/static/js/val.js"></script> <script type="text/javascript" src="/static/js/val.js"></script>
@ -21,7 +21,6 @@
<!--[if lt IE 9]> <!--[if lt IE 9]>
<script src="http://html5shim.googlecode.com/svn/trunk/html5.js"></script> <script src="http://html5shim.googlecode.com/svn/trunk/html5.js"></script>
<![endif]--> <![endif]-->
</head> </head>
<body> <body>
<nav class="navbar navbar-default" role="navigation"> <nav class="navbar navbar-default" role="navigation">
@ -37,9 +36,9 @@
</div> </div>
<div id="navbar-collapse-1" class="collapse navbar-collapse navbar-ex4-collapse"> <div id="navbar-collapse-1" class="collapse navbar-collapse navbar-ex4-collapse">
<ul class="nav navbar-nav navbar-right"> <ul class="nav navbar-nav navbar-right">
<li><a href="hosts">Hosts</a> <li><a href="/hosts">Hosts</a>
<li><a href="users">User Panel</a> <li><a href="/users">User Panel</a>
<li><a href="logout">Log out</a> <li><a href="/" onclick="app.auth.logOut()">Log out</a>
</ul> </ul>
</div> </div>
</div> </div>

View File

@ -1,38 +1,45 @@
<% include top %> <%- include('top') %>
<script id="rowTemplate" type="text/html"> <script id="rowTemplate" type="text/html">
<tr action="users" evalAJAX="$form.trigger('reset')"> <tr action="user/password/<< username >>" method="put" evalAJAX="$form.trigger('reset')">
<td><< user >></td>
<input type="hidden" name="username" value="<< user >>" />
<td> <td>
<label></label> << uid >>:<< username >>
<div class="col-xs-10 form-group"> </td>
<input type="password" size="30" class="form-control" name="password" placeholder="New password" validate="password:5"/> <td>
</div> <input type="hidden" name="username" value="<< username >>" />
<button type="button" onclick="formAJAX(this)" class="btn btn-sm btn-default">Change</button> <div class="col-xs-10 form-group">
<label class="control-label"></label>
<input type="password" size="15" class="form-control" name="password" placeholder="New password" validate="password:5"/>
</div>
<button type="button btn-warn" onclick="formAJAX(this)" class="btn btn-sm btn-warn">
Change
</button>
</td>
<td>
<button type="button" onclick="app.user.remove({username: '<<username>>'}, function(){tableAJAX('username <<username>> delete.')})" class="btn btn-sm btn-danger">
Delete
</button>
</td> </td>
<td><button type="button" onclick="formAJAX(this, true)" class="btn btn-sm btn-default">Delete</button><td>
</tr> </tr>
</script> </script>
<script type="text/javascript"> <script type="text/javascript">
function tableAJAX(actionMessage){ function tableAJAX(actionMessage){
$('#tableAJAX').html('').hide(); $('#tableAJAX').html('').hide();
$('div.actionMessage').html('Refreshing user list...').show(); app.util.actionMessage('Refreshing user list...')
$.get('users',function(data){
$.each( data, function(key, value) { app.user.list(function(error, data){
user_row = ich.rowTemplate({user : value}); $.each( data.results, function(key, value) {
if(value.uid+'' && value.uid < 1000) return;
user_row = ich.rowTemplate(value);
$('#tableAJAX').append(user_row); $('#tableAJAX').append(user_row);
}); });
$('#tableAJAX').fadeIn('slow'); $('#tableAJAX').fadeIn('slow');
if(!actionMessage){ app.util.actionMessage(actionMessage || '', {type: 'info'});
$('div.actionMessage').slideUp('fast');
}else{
$('div.actionMessage').html(actionMessage).slideDown('fast');
}
});
});
} }
$(document).ready(function(){ $(document).ready(function(){
@ -51,7 +58,7 @@
</div> </div>
</div> </div>
<div class="panel-body"> <div class="panel-body">
<form action="users" evalAJAX="$form.trigger('reset')"> <form action="user/" evalAJAX="$form.trigger('reset')">
<input type="hidden" class="form-control" name="delete" value="false" /> <input type="hidden" class="form-control" name="delete" value="false" />
<div class="form-group"> <div class="form-group">
<label class="control-label">User-name</label> <label class="control-label">User-name</label>
@ -100,4 +107,4 @@
</div> </div>
</div> </div>
</div> </div>
<% include bottom %> <%- include('bottom') %>

View File

@ -1,103 +0,0 @@
<% include top %>
<script id="rowTemplate" type="text/html">
<tr action="users" evalAJAX="$form.trigger('reset')">
<td><< user >></td>
<input type="hidden" name="username" value="<< user >>" />
<td>
<label></label>
<div class="col-xs-10 form-group">
<input type="password" size="30" class="form-control" name="password" placeholder="New password" validate="password:5"/>
</div>
<button type="button" onclick="formAJAX(this)" class="btn btn-sm btn-default">Change</button>
</td>
<td><button type="button" onclick="formAJAX(this, true)" class="btn btn-sm btn-default">Delete</button><td>
</tr>
</script>
<script type="text/javascript">
function tableAJAX(actionMessage){
$('#tableAJAX').html('').hide();
$('div.actionMessage').html('Refreshing user list...').show();
$.get('users',function(data){
$.each( data, function(key, value) {
user_row = ich.rowTemplate({user : value});
$('#tableAJAX').append(user_row);
});
$('#tableAJAX').fadeIn('slow');
if(!actionMessage){
$('div.actionMessage').slideUp('fast');
}else{
$('div.actionMessage').html(actionMessage).slideDown('fast');
}
});
}
$(document).ready(function(){
tableAJAX(); //populate the table
});
</script>
<div class="row" style="display:none">
<div class="col-md-4">
<div class="panel panel-default">
<div class="panel-heading">
<div class="panel-title">
Add new user
<div class="pull-right">
<label class="glyphicon glyphicon-circle-arrow-down panel-toggle"></label>
</div>
</div>
</div>
<div class="panel-body">
<form action="users" evalAJAX="$form.trigger('reset')">
<input type="hidden" class="form-control" name="delete" value="false" />
<div class="form-group">
<label class="control-label">User-name</label>
<input type="text" class="form-control" name="username" placeholder="Letter, numbers, -, _, . and @ only" validate="user:3" />
</div>
<div class="form-group">
<label class="control-label">Password</label>
<input type="password" class="form-control" name="password" placeholder="Atleast 5 char. long" validate="password:5"/>
</div>
<div class="form-group">
<label class="control-label">Again</label>
<input type="password" class="form-control" name="passwordMatch" placeholder="Retype password" validate="eq:password"/>
</div>
<button type="button" onclick="formAJAX(this)" class="btn btn-default">Add</button>
</form>
</div>
</div>
</div>
<div class="col-md-8">
<div class="panel panel-default">
<div class="panel-heading">
<div class="panel-title">
User List
<div class="pull-right">
<label class="glyphicon glyphicon-circle-arrow-down panel-toggle"></label>
</div>
</div>
</div>
<div class="panel-body" style="padding-bottom:0">
<div class="alert alert-warning actionMessage" style="display:none">
<!-- Message after AJAX action is preformed -->
</div>
<div class="table-responsive">
<table class="table">
<thead>
<th>Name</th>
<th>Password</th>
<th>Delete</th>
</thead>
<tbody id="tableAJAX">
<!-- ajax loaded table -->
</tbody>
</table>
</div>
</div>
</div>
</div>
</div>
<% include bottom %>