mirror of https://github.com/Aidaho12/haproxy-wi
v2.5.3
Now ssh credentials keep in DB, it's more user-frendly and and saferpull/19/head
parent
ff069069b5
commit
02d5f9ab6b
|
@ -126,6 +126,12 @@ def create_table():
|
|||
`user_id` INTEGER NOT NULL,
|
||||
`uuid` varchar ( 64 )
|
||||
);
|
||||
CREATE TABLE IF NOT EXISTS `cred` (
|
||||
`enable` INTEGER NOT NULL DEFAULT 1,
|
||||
`username` VARCHAR ( 64 ) NOT NULL,
|
||||
`password` VARCHAR ( 64 ) NOT NULL
|
||||
);
|
||||
insert into cred('enable','username','password') values ('1', 'root','password');
|
||||
"""
|
||||
try:
|
||||
cur.executescript(sql)
|
||||
|
@ -155,6 +161,7 @@ def update_db_v_2_0_1(**kwargs):
|
|||
"""
|
||||
try:
|
||||
cur.execute(sql)
|
||||
con.commit()
|
||||
except sqltool.Error as e:
|
||||
if kwargs.get('silent') != 1:
|
||||
if e.args[0] == 'duplicate column name: type_ip':
|
||||
|
@ -175,6 +182,7 @@ def update_db_v_2_0_1_1(**kwargs):
|
|||
"""
|
||||
try:
|
||||
cur.execute(sql)
|
||||
con.commit()
|
||||
except sqltool.Error as e:
|
||||
if kwargs.get('silent') != 1:
|
||||
if e.args[0] == 'duplicate column name: enable' or e == "1060 (42S21): Duplicate column name 'enable' ":
|
||||
|
@ -195,6 +203,7 @@ def update_db_v_2_0_5(**kwargs):
|
|||
"""
|
||||
try:
|
||||
cur.execute(sql)
|
||||
con.commit()
|
||||
except sqltool.Error as e:
|
||||
if kwargs.get('silent') != 1:
|
||||
if e.args[0] == 'duplicate column name: master' or e == "1060 (42S21): Duplicate column name 'master' ":
|
||||
|
@ -215,11 +224,12 @@ def update_db_v_2_4(**kwargs):
|
|||
"""
|
||||
try:
|
||||
cur.execute(sql)
|
||||
con.commit()
|
||||
except sqltool.Error as e:
|
||||
print(kwargs.get('silent'))
|
||||
if kwargs.get('silent') != 1:
|
||||
if e.args[0] == 'duplicate column name: user_id':
|
||||
print('Already updated. No run more. Thx =^.^=')
|
||||
print('Updating... go to version 2.5.3')
|
||||
else:
|
||||
print("An error occurred:", e)
|
||||
return False
|
||||
|
@ -230,17 +240,47 @@ def update_db_v_2_4(**kwargs):
|
|||
cur.close()
|
||||
con.close()
|
||||
|
||||
def update_db_v_2_5_3(**kwargs):
|
||||
con, cur = get_cur()
|
||||
sql = """
|
||||
CREATE TABLE IF NOT EXISTS `cred` (`enable` INTEGER NOT NULL DEFAULT 1, `username` VARCHAR ( 64 ) NOT NULL, `password` VARCHAR ( 64 ) NOT NULL );
|
||||
"""
|
||||
try:
|
||||
cur.execute(sql)
|
||||
con.commit()
|
||||
except sqltool.Error as e:
|
||||
print(kwargs.get('silent'))
|
||||
if kwargs.get('silent') != 1:
|
||||
if e.args[0] == 'duplicate column name: enable':
|
||||
print('Already updated. No run more. Thx =^.^=')
|
||||
else:
|
||||
print("An error occurred:", e)
|
||||
return False
|
||||
else:
|
||||
if kwargs.get('silent') != 1:
|
||||
print("DB was update to 2.5.3<br />")
|
||||
sql2 = """
|
||||
insert into `cred` (`enable`,`username`,`password`) values ('1', 'root','password');
|
||||
"""
|
||||
cur.execute(sql2)
|
||||
con.commit()
|
||||
return True
|
||||
cur.close()
|
||||
con.close()
|
||||
|
||||
def update_all():
|
||||
update_db_v_2_0_1()
|
||||
update_db_v_2_0_1_1()
|
||||
update_db_v_2_0_5()
|
||||
update_db_v_2_4()
|
||||
update_db_v_2_5_3()
|
||||
|
||||
def update_all_silent():
|
||||
update_db_v_2_0_1(silent=1)
|
||||
update_db_v_2_0_1_1(silent=1)
|
||||
update_db_v_2_0_5(silent=1)
|
||||
update_db_v_2_4(silent=1)
|
||||
update_db_v_2_5_3(silent=1)
|
||||
|
||||
#if check_db():
|
||||
# create_table()
|
||||
|
|
35
app/funct.py
35
app/funct.py
|
@ -18,7 +18,6 @@ fullpath = config.get('main', 'fullpath')
|
|||
time_zone = config.get('main', 'time_zone')
|
||||
proxy = config.get('main', 'proxy')
|
||||
ssh_keys = config.get('ssh', 'ssh_keys')
|
||||
ssh_user_name = config.get('ssh', 'ssh_user_name')
|
||||
haproxy_configs_server = config.get('configs', 'haproxy_configs_server')
|
||||
hap_configs_dir = config.get('configs', 'haproxy_save_configs_dir')
|
||||
haproxy_config_path = config.get('haproxy', 'haproxy_config_path')
|
||||
|
@ -123,44 +122,63 @@ def get_button(button, **kwargs):
|
|||
print('<button type="submit" value="%s" name="%s" class="btn btn-default">%s</button>' % (value, value, button))
|
||||
|
||||
def ssh_connect(serv, **kwargs):
|
||||
import sql
|
||||
ssh_enable = sql.ssh_enable()
|
||||
ssh_user_name = sql.select_ssh_username()
|
||||
ssh = SSHClient()
|
||||
ssh.load_system_host_keys()
|
||||
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
||||
try:
|
||||
if config.get('ssh', 'ssh_keys_enable') == "1":
|
||||
if ssh_enable == 1:
|
||||
k = paramiko.RSAKey.from_private_key_file(ssh_keys)
|
||||
ssh.connect(hostname = serv, username = ssh_user_name, pkey = k )
|
||||
else:
|
||||
ssh.connect(hostname = serv, username = ssh_user_name, password = config.get('ssh', 'ssh_pass'))
|
||||
ssh.connect(hostname = serv, username = ssh_user_name, password = sql.select_ssh_password())
|
||||
if kwargs.get('check'):
|
||||
return True
|
||||
else:
|
||||
return ssh
|
||||
except paramiko.AuthenticationException:
|
||||
if kwargs.get('check'):
|
||||
print('<div class="alert alert-danger">Authentication failed, please verify your credentials</div>')
|
||||
error = 'Authentication failed, please verify your credentials'
|
||||
return False
|
||||
else:
|
||||
return 'Authentication failed, please verify your credentials'
|
||||
except paramiko.SSHException as sshException:
|
||||
if kwargs.get('check'):
|
||||
print('<div class="alert alert-danger">Unable to establish SSH connection: %s </div>' % sshException)
|
||||
error = 'Unable to establish SSH connection: %s ' % sshException
|
||||
return False
|
||||
else:
|
||||
return 'Unable to establish SSH connection: %s ' % sshException
|
||||
except paramiko.BadHostKeyException as badHostKeyException:
|
||||
if kwargs.get('check'):
|
||||
print('<div class="alert alert-danger">Unable to verify server\'s host key: %s </div>' % badHostKeyException)
|
||||
error = 'Unable to verify server\'s host key: %s ' % badHostKeyException
|
||||
return False
|
||||
else:
|
||||
return 'Unable to verify server\'s host key: %s ' % badHostKeyException
|
||||
except Exception as e:
|
||||
if e.args[1] == "No such file or directory":
|
||||
if kwargs.get('check'):
|
||||
print('<div class="alert alert-danger">{}. Check ssh key</div>'.format(e.args[1]))
|
||||
error = '{}. Check ssh key'.format(e.args[1])
|
||||
else:
|
||||
return '{}. Check ssh key'.format(e.args[1])
|
||||
elif e.args[1] == "Invalid argument":
|
||||
if kwargs.get('check'):
|
||||
print('<div class="alert alert-danger">Check the IP of the new server</div>')
|
||||
else:
|
||||
error = 'Check the IP of the new server'
|
||||
else:
|
||||
if kwargs.get('check'):
|
||||
print('<div class="alert alert-danger">{}</div>'.format(e.args[1]))
|
||||
else:
|
||||
error = e.args[1]
|
||||
if kwargs.get('check'):
|
||||
return False
|
||||
else:
|
||||
return error
|
||||
|
||||
def get_config(serv, cfg, **kwargs):
|
||||
error = ""
|
||||
if kwargs.get("keepalived"):
|
||||
config_path = "/etc/keepalived/keepalived.conf"
|
||||
else:
|
||||
|
@ -173,7 +191,8 @@ def get_config(serv, cfg, **kwargs):
|
|||
sftp.close()
|
||||
ssh.close()
|
||||
except Exception as e:
|
||||
return str(e)
|
||||
ssh += str(e)
|
||||
return ssh
|
||||
|
||||
def show_config(cfg):
|
||||
print('<div style="margin-left: 16%" class="configShow">')
|
||||
|
|
|
@ -25,13 +25,8 @@ mysql_db = haproxywi
|
|||
mysql_host = 127.0.0.1
|
||||
|
||||
[ssh]
|
||||
#If ssh connect disable entare password for ssh connect. Default enable
|
||||
ssh_keys_enable = 1
|
||||
#SSH keys to connect without password to HAproxy servers
|
||||
ssh_keys = ${main:fullpath}/app/id_rsa.pem
|
||||
#Username for connect ssh
|
||||
ssh_user_name = root
|
||||
ssh_pass =
|
||||
|
||||
[logs]
|
||||
#Logs save locally, disable by default
|
||||
|
|
|
@ -44,15 +44,14 @@ if form.getvalue('serv') is not None and form.getvalue('open') is not None :
|
|||
funct.logging(serv, "keepalivedconfig.py open config")
|
||||
except:
|
||||
pass
|
||||
funct.get_config(serv, cfg, keepalived=1)
|
||||
error = funct.get_config(serv, cfg, keepalived=1)
|
||||
|
||||
try:
|
||||
conf = open(cfg, "r",encoding='utf-8', errors='ignore')
|
||||
config_read = conf.read()
|
||||
except IOError:
|
||||
print('<div class="alert alert-danger">Can\'t read import config file</div>')
|
||||
|
||||
conf.close
|
||||
except IOError:
|
||||
error += "<br>Can't read import config file"
|
||||
|
||||
os.system("/bin/mv %s %s.old" % (cfg, cfg))
|
||||
|
||||
|
@ -70,7 +69,7 @@ if form.getvalue('serv') is not None and form.getvalue('config') is not None:
|
|||
with open(cfg, "a") as conf:
|
||||
conf.write(config)
|
||||
except IOError:
|
||||
print("Can't read import config file")
|
||||
error += "Can't read import config file"
|
||||
|
||||
stderr = funct.upload_and_restart(serv, cfg, just_save=save, keepalived=1)
|
||||
|
||||
|
|
11
app/login.py
11
app/login.py
|
@ -49,12 +49,6 @@ except:
|
|||
user = ""
|
||||
pass
|
||||
|
||||
if create_db.check_db():
|
||||
if create_db.create_table():
|
||||
create_db.update_all()
|
||||
db_create = '<div class="alert alert-success">DB was created<br /><br />Now you can login, default: admin/admin</div>'
|
||||
create_db.update_all_silent()
|
||||
|
||||
if form.getvalue('logout'):
|
||||
try:
|
||||
sql.delete_uuid(user_id.value)
|
||||
|
@ -91,6 +85,11 @@ if login is not None and password is not None:
|
|||
|
||||
if login is None:
|
||||
print("Content-type: text/html\n")
|
||||
if create_db.check_db():
|
||||
if create_db.create_table():
|
||||
create_db.update_all()
|
||||
db_create = '<div class="alert alert-success">DB was created<br /><br />Now you can login, default: admin/admin</div>'
|
||||
|
||||
output_from_parsed_template = template.render(h2 = 1, title = "Login page. Enter please",
|
||||
role = role,
|
||||
user = user,
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
#!/usr/bin/env python3
|
||||
import funct, sql
|
||||
import create_db
|
||||
import os, http
|
||||
from jinja2 import Environment, FileSystemLoader
|
||||
env = Environment(loader=FileSystemLoader('templates/'))
|
||||
|
@ -7,6 +8,7 @@ template = env.get_template('ovw.html')
|
|||
|
||||
print('Content-type: text/html\n')
|
||||
funct.check_login()
|
||||
create_db.update_all_silent()
|
||||
|
||||
try:
|
||||
cookie = http.cookies.SimpleCookie(os.environ.get("HTTP_COOKIE"))
|
||||
|
|
68
app/sql.py
68
app/sql.py
|
@ -214,7 +214,7 @@ def select_user_name_group(id):
|
|||
try:
|
||||
cur.execute(sql)
|
||||
except sqltool.Error as e:
|
||||
print('<span class="alert alert-danger" id="error">An error occurred: ' + e.args[0] + ' <a title="Close" id="errorMess"><b>X</b></a></span>')
|
||||
print('<span class="alert alert-danger" id="error">An error occurred: ' + e + ' <a title="Close" id="errorMess"><b>X</b></a></span>')
|
||||
else:
|
||||
for group in cur.fetchone():
|
||||
return group
|
||||
|
@ -390,6 +390,61 @@ def is_master(ip, **kwargs):
|
|||
cur.close()
|
||||
con.close()
|
||||
|
||||
def ssh_enable():
|
||||
con, cur = create_db.get_cur()
|
||||
sql = """select enable from cred """
|
||||
try:
|
||||
cur.execute(sql)
|
||||
except sqltool.Error as e:
|
||||
print('<span class="alert alert-danger" id="error">An error occurred: ' + e + ' <a title="Close" id="errorMess"><b>X</b></a></span>')
|
||||
else:
|
||||
for enable in cur.fetchone():
|
||||
return enable
|
||||
cur.close()
|
||||
con.close()
|
||||
|
||||
def select_ssh_username():
|
||||
con, cur = create_db.get_cur()
|
||||
sql = """select username from cred """
|
||||
try:
|
||||
cur.execute(sql)
|
||||
except sqltool.Error as e:
|
||||
print('<span class="alert alert-danger" id="error">An error occurred: ' + e + ' <a title="Close" id="errorMess"><b>X</b></a></span>')
|
||||
else:
|
||||
for username in cur.fetchone():
|
||||
return username
|
||||
cur.close()
|
||||
con.close()
|
||||
|
||||
def select_ssh_password():
|
||||
con, cur = create_db.get_cur()
|
||||
sql = """select password from cred """
|
||||
try:
|
||||
cur.execute(sql)
|
||||
except sqltool.Error as e:
|
||||
print('<span class="alert alert-danger" id="error">An error occurred: ' + e + ' <a title="Close" id="errorMess"><b>X</b></a></span>')
|
||||
else:
|
||||
for password in cur.fetchone():
|
||||
return password
|
||||
cur.close()
|
||||
con.close()
|
||||
|
||||
def update_ssh(enable, username, password):
|
||||
con, cur = create_db.get_cur()
|
||||
sql = """
|
||||
update cred set
|
||||
enable = '%s',
|
||||
username = '%s',
|
||||
password = '%s' """ % (enable, username, password)
|
||||
try:
|
||||
cur.execute(sql)
|
||||
con.commit()
|
||||
except sqltool.Error as e:
|
||||
print('<span class="alert alert-danger" id="error">An error occurred: ' + e + ' <a title="Close" id="errorMess"><b>X</b></a></span>')
|
||||
con.rollback()
|
||||
cur.close()
|
||||
con.close()
|
||||
|
||||
def show_update_servers():
|
||||
SERVERS = select_servers()
|
||||
print('<tr class="overviewHead">'
|
||||
|
@ -613,3 +668,14 @@ if form.getvalue('updateserver') is not None:
|
|||
update_server(name, ip, group, typeip, enable, master, id)
|
||||
else:
|
||||
print('<span class="alert alert-danger" id="error"><a title="Close" id="errorMess"><b>X</b></a></span>')
|
||||
|
||||
if form.getvalue('updatessh'):
|
||||
enable = form.getvalue('ssh_enable')
|
||||
username = form.getvalue('ssh_user')
|
||||
password = form.getvalue('ssh_pass')
|
||||
if username is None:
|
||||
print('Content-type: text/html\n')
|
||||
print(error_mess)
|
||||
else:
|
||||
print('Content-type: text/html\n')
|
||||
update_ssh(enable, username, password)
|
|
@ -7,7 +7,7 @@
|
|||
<li><a href="#groups">Groups</a></li>
|
||||
<li><a href="#servers">Servers</a></li>
|
||||
<li><a href="#roles">Roles</a></li>
|
||||
<li><a href="#cert">SSH key</a></li>
|
||||
<li><a href="#ssh">SSH credentials</a></li>
|
||||
</ul>
|
||||
<div id="users">
|
||||
<table class="overview" id="ajax-users">
|
||||
|
@ -277,8 +277,34 @@
|
|||
{% endfor %}
|
||||
</table>
|
||||
</div>
|
||||
<div id="cert">
|
||||
<table id="ssh">
|
||||
<div id="ssh">
|
||||
<table id="ssh_enable_table">
|
||||
<tr class="overviewHead" style="width: 50%;">
|
||||
<td class="padding10 first-collumn">SSH key</td>
|
||||
<td>
|
||||
<span title="Enter SSH user name. If SSH key disabled, enter password for ssh user">Credentials(?)</span>
|
||||
</td>
|
||||
<td></td>
|
||||
</tr>
|
||||
<tr style="width: 50%;">
|
||||
<td class="first-collumn" valign="top" style="padding-top: 15px;">
|
||||
{% if ssh_enable == 1 %}
|
||||
<label for="ssh_enable">Enable SSH<input type="checkbox" id="ssh_enable" checked>
|
||||
{% else %}
|
||||
<label for="ssh_enable">Enable SSH<input type="checkbox" id="ssh_enable">
|
||||
{% endif %}
|
||||
</td>
|
||||
<td style="padding-top: 15px;">
|
||||
<p>
|
||||
<input type="text" id="ssh_user" class="form-control" value="{{ssh_user}}">
|
||||
</p>
|
||||
<input type="password" id="ssh_pass" class="form-control" value="{{ssh_pass}}" style="display: none;">
|
||||
<br>
|
||||
</td>
|
||||
<td></td>
|
||||
</tr>
|
||||
</table>
|
||||
<table id="ssh_key">
|
||||
<tr class="overviewHead" style="width: 50%;">
|
||||
<td class="padding10 first-collumn">Upload SSH Key</td>
|
||||
<td>
|
||||
|
|
|
@ -87,6 +87,7 @@
|
|||
<li><a href=/app/users.py#groups title="Actions with groups" class="group head-submenu">Groups</a></li>
|
||||
<li><a href=/app/users.py#servers title="Actions with servers" class="runtime head-submenu">Servers</a></li>
|
||||
<li><a href=/app/users.py#roles title="Users roles" class="role head-submenu">Roles</a></li>
|
||||
<li><a href=/app/users.py#ssh title="Users roles" class="admin head-submenu">SSH credentials</a></li>
|
||||
<li><a href=/app/settings.py title="View settings" class="settings head-submenu">View settings</a></li>
|
||||
<li><a href=/app/viewlogs.py title="View users actions logs" class="logs head-submenu">View logs</a></li>
|
||||
</li>
|
||||
|
@ -94,7 +95,7 @@
|
|||
{% endif %}
|
||||
</ul>
|
||||
</nav>
|
||||
<div class="copyright-menu">HAproxy-WI v2.5.2</div>
|
||||
<div class="copyright-menu">HAproxy-WI v2.5.3</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="container">
|
||||
|
|
|
@ -28,5 +28,8 @@ output_from_parsed_template = template.render(title = "Admin area: users manage"
|
|||
groups = sql.select_groups(),
|
||||
servers = sql.select_servers(full=1),
|
||||
roles = sql.select_roles(),
|
||||
masters = sql.select_servers(get_master_servers=1))
|
||||
masters = sql.select_servers(get_master_servers=1),
|
||||
ssh_enable = sql.ssh_enable(),
|
||||
ssh_user = sql.select_ssh_username(),
|
||||
ssh_pass = sql.select_ssh_password())
|
||||
print(output_from_parsed_template)
|
||||
|
|
45
inc/users.js
45
inc/users.js
|
@ -252,6 +252,21 @@ $( function() {
|
|||
var id = $(this).attr('id').split('-');
|
||||
updateServer(id[1])
|
||||
});
|
||||
$( "#ssh_enable_table input" ).change(function() {
|
||||
updateSSH()
|
||||
});
|
||||
$('#ssh_enable').click(function() {
|
||||
if ($('#ssh_enable').is(':checked')) {
|
||||
$('#ssh_pass').css('display', 'none');
|
||||
} else {
|
||||
$('#ssh_pass').css('display', 'block');
|
||||
}
|
||||
});
|
||||
if ($('#ssh_enable').is(':checked')) {
|
||||
$('#ssh_pass').css('display', 'none');
|
||||
} else {
|
||||
$('#ssh_pass').css('display', 'block');
|
||||
}
|
||||
} );
|
||||
function removeUser(id) {
|
||||
$("#user-"+id).css("background-color", "#f2dede");
|
||||
|
@ -420,3 +435,33 @@ function uploadSsh() {
|
|||
}
|
||||
} );
|
||||
}
|
||||
function updateSSH() {
|
||||
$('#error').remove();
|
||||
var ssh_enable = 0;
|
||||
if ($('#ssh_enable').is(':checked')) {
|
||||
ssh_enable = '1';
|
||||
}
|
||||
$.ajax( {
|
||||
url: "sql.py",
|
||||
data: {
|
||||
updatessh: 1,
|
||||
ssh_enable: ssh_enable,
|
||||
ssh_user: $('#ssh_user').val(),
|
||||
ssh_pass: $('#ssh_pass').val(),
|
||||
},
|
||||
type: "GET",
|
||||
success: function( data ) {
|
||||
data = data.replace(/\s+/g,' ');
|
||||
if (data.indexOf('error') != '-1') {
|
||||
$("#ajax-ssh").append(data);
|
||||
$.getScript(users);
|
||||
} else {
|
||||
$('.alert-danger').remove();
|
||||
$("#ssh_enable_table").addClass( "update", 1000 );
|
||||
setTimeout(function() {
|
||||
$( "#ssh_enable_table" ).removeClass( "update" );
|
||||
}, 2500 );
|
||||
}
|
||||
}
|
||||
} );
|
||||
}
|
Loading…
Reference in New Issue