fixy z listy todo
This commit is contained in:
parent
2904209332
commit
a5c59f8a64
1
.gitignore
vendored
1
.gitignore
vendored
@ -2,3 +2,4 @@ __pycache__
|
|||||||
data/
|
data/
|
||||||
instance/
|
instance/
|
||||||
venv/
|
venv/
|
||||||
|
todo.txt
|
@ -26,4 +26,6 @@ ADD COLUMN preferred_hostfile_id INTEGER
|
|||||||
REFERENCES host_file(id);
|
REFERENCES host_file(id);
|
||||||
|
|
||||||
ALTER TABLE user_settings ADD COLUMN global_ssh_key TEXT;
|
ALTER TABLE user_settings ADD COLUMN global_ssh_key TEXT;
|
||||||
ALTER TABLE user_settings ADD COLUMN global_key_passphrase VARCHAR(200);
|
ALTER TABLE user_settings ADD COLUMN global_key_passphrase VARCHAR(200);
|
||||||
|
|
||||||
|
ALTER TABLE host ADD COLUMN disable_regex_deploy BOOLEAN NOT NULL DEFAULT 0;
|
||||||
|
53
app.py
53
app.py
@ -46,6 +46,7 @@ class Host(db.Model):
|
|||||||
use_daemon = db.Column(db.Boolean, default=False)
|
use_daemon = db.Column(db.Boolean, default=False)
|
||||||
daemon_url = db.Column(db.String(255), nullable=True)
|
daemon_url = db.Column(db.String(255), nullable=True)
|
||||||
daemon_token = db.Column(db.String(255), nullable=True)
|
daemon_token = db.Column(db.String(255), nullable=True)
|
||||||
|
disable_regex_deploy = db.Column(db.Boolean, default=False)
|
||||||
@property
|
@property
|
||||||
def resolved_hostname(self):
|
def resolved_hostname(self):
|
||||||
try:
|
try:
|
||||||
@ -53,6 +54,18 @@ class Host(db.Model):
|
|||||||
except Exception:
|
except Exception:
|
||||||
return self.hostname
|
return self.hostname
|
||||||
|
|
||||||
|
@property
|
||||||
|
def resolved_daemon(self):
|
||||||
|
if self.use_daemon and self.daemon_url:
|
||||||
|
try:
|
||||||
|
daemon_str = self.daemon_url.split("://")[-1]
|
||||||
|
daemon_ip = daemon_str.split(":")[0]
|
||||||
|
return socket.gethostbyaddr(daemon_ip)[0]
|
||||||
|
except Exception:
|
||||||
|
return daemon_ip
|
||||||
|
return self.resolved_hostname
|
||||||
|
|
||||||
|
|
||||||
class DeployLog(db.Model):
|
class DeployLog(db.Model):
|
||||||
id = db.Column(db.Integer, primary_key=True)
|
id = db.Column(db.Integer, primary_key=True)
|
||||||
timestamp = db.Column(db.DateTime, default=db.func.current_timestamp())
|
timestamp = db.Column(db.DateTime, default=db.func.current_timestamp())
|
||||||
@ -86,6 +99,7 @@ class Backup(db.Model):
|
|||||||
content = db.Column(db.Text, nullable=False)
|
content = db.Column(db.Text, nullable=False)
|
||||||
description = db.Column(db.String(255), nullable=True)
|
description = db.Column(db.String(255), nullable=True)
|
||||||
host = db.relationship('Host', backref='backups', lazy=True)
|
host = db.relationship('Host', backref='backups', lazy=True)
|
||||||
|
|
||||||
class RegexHostEntry(db.Model):
|
class RegexHostEntry(db.Model):
|
||||||
__tablename__ = 'regex_host_entry'
|
__tablename__ = 'regex_host_entry'
|
||||||
id = db.Column(db.Integer, primary_key=True)
|
id = db.Column(db.Integer, primary_key=True)
|
||||||
@ -736,6 +750,8 @@ def delete_hosts_file(file_id):
|
|||||||
# -------------------
|
# -------------------
|
||||||
# WDROŻENIE WYBRANEGO PLIKU HOSTS NA WYBRANE SERWERY
|
# WDROŻENIE WYBRANEGO PLIKU HOSTS NA WYBRANE SERWERY
|
||||||
# -------------------
|
# -------------------
|
||||||
|
import socket
|
||||||
|
|
||||||
@app.route('/deploy-hosts-file/<int:file_id>', methods=['GET', 'POST'])
|
@app.route('/deploy-hosts-file/<int:file_id>', methods=['GET', 'POST'])
|
||||||
def deploy_hosts_file(file_id):
|
def deploy_hosts_file(file_id):
|
||||||
if 'user_id' not in session:
|
if 'user_id' not in session:
|
||||||
@ -746,6 +762,17 @@ def deploy_hosts_file(file_id):
|
|||||||
return redirect(url_for('list_hosts_files'))
|
return redirect(url_for('list_hosts_files'))
|
||||||
hosts = Host.query.filter_by(user_id=session['user_id']).all()
|
hosts = Host.query.filter_by(user_id=session['user_id']).all()
|
||||||
|
|
||||||
|
# Dla hostów z demonem obliczamy IP i rozwiązaną nazwę z IP
|
||||||
|
for host in hosts:
|
||||||
|
if host.use_daemon and host.type == 'linux' and host.daemon_url:
|
||||||
|
daemon_str = host.daemon_url.split("://")[-1]
|
||||||
|
daemon_ip = daemon_str.split(":")[0]
|
||||||
|
host.daemon_ip = daemon_ip
|
||||||
|
try:
|
||||||
|
host.resolved_daemon = socket.gethostbyaddr(daemon_ip)[0]
|
||||||
|
except Exception:
|
||||||
|
host.resolved_daemon = daemon_ip # jeśli nie uda się rozwiązać, pozostaw IP
|
||||||
|
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
selected_host_ids = request.form.getlist('hosts')
|
selected_host_ids = request.form.getlist('hosts')
|
||||||
for host in hosts:
|
for host in hosts:
|
||||||
@ -764,11 +791,10 @@ def deploy_hosts_file(file_id):
|
|||||||
if resp.status_code != 200:
|
if resp.status_code != 200:
|
||||||
raise Exception(f"Daemon POST error: {resp.status_code} - {resp.text}")
|
raise Exception(f"Daemon POST error: {resp.status_code} - {resp.text}")
|
||||||
db.session.add(DeployLog(
|
db.session.add(DeployLog(
|
||||||
details=f'[LINUX/DAEMON] Deployed file "{file.title}" to {host.hostname} for user {session["user_id"]}',
|
details=f'[LINUX/DAEMON] Deployed file "{file.title}" to {host.daemon_ip} - {host.resolved_daemon} for user {session["user_id"]}',
|
||||||
user_id=session['user_id']
|
user_id=session['user_id']
|
||||||
))
|
))
|
||||||
elif host.type == 'mikrotik':
|
elif host.type == 'mikrotik':
|
||||||
# Mikrotik
|
|
||||||
wrapped_mikro = wrap_mikrotik_content(file.content)
|
wrapped_mikro = wrap_mikrotik_content(file.content)
|
||||||
deploy_mikrotik(host, wrapped_mikro)
|
deploy_mikrotik(host, wrapped_mikro)
|
||||||
db.session.add(DeployLog(
|
db.session.add(DeployLog(
|
||||||
@ -776,7 +802,6 @@ def deploy_hosts_file(file_id):
|
|||||||
user_id=session['user_id']
|
user_id=session['user_id']
|
||||||
))
|
))
|
||||||
else:
|
else:
|
||||||
# Standard Linux (SSH)
|
|
||||||
ssh = open_ssh_connection(host)
|
ssh = open_ssh_connection(host)
|
||||||
with tempfile.NamedTemporaryFile(mode='w', delete=False) as tmpf:
|
with tempfile.NamedTemporaryFile(mode='w', delete=False) as tmpf:
|
||||||
tmpf.write(wrapped_content)
|
tmpf.write(wrapped_content)
|
||||||
@ -790,7 +815,6 @@ def deploy_hosts_file(file_id):
|
|||||||
details=f'[LINUX] Deployed file "{file.title}" to {host.hostname} for user {session["user_id"]}',
|
details=f'[LINUX] Deployed file "{file.title}" to {host.hostname} for user {session["user_id"]}',
|
||||||
user_id=session['user_id']
|
user_id=session['user_id']
|
||||||
))
|
))
|
||||||
|
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
flash(f'Deployed file "{file.title}" to {host.hostname}', 'success')
|
flash(f'Deployed file "{file.title}" to {host.hostname}', 'success')
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
@ -1176,13 +1200,11 @@ def delete_backup(backup_id):
|
|||||||
|
|
||||||
def deploy_user(user_id):
|
def deploy_user(user_id):
|
||||||
user_settings = UserSettings.query.filter_by(user_id=user_id).first()
|
user_settings = UserSettings.query.filter_by(user_id=user_id).first()
|
||||||
# domyślny plik "Default Hosts" (np. szukasz title=="Default Hosts")
|
|
||||||
default_file = HostFile.query.filter_by(user_id=user_id, title="Default Hosts").first()
|
default_file = HostFile.query.filter_by(user_id=user_id, title="Default Hosts").first()
|
||||||
if not default_file:
|
if not default_file:
|
||||||
# jeśli nie ma w ogóle, nic nie robimy
|
|
||||||
return
|
return
|
||||||
|
|
||||||
# ewentualnie regex
|
# Pobieramy regex tylko raz – globalnie
|
||||||
regex_lines = ""
|
regex_lines = ""
|
||||||
if user_settings and user_settings.regex_deploy_enabled:
|
if user_settings and user_settings.regex_deploy_enabled:
|
||||||
regex_lines = generate_regex_hosts(user_id)
|
regex_lines = generate_regex_hosts(user_id)
|
||||||
@ -1192,17 +1214,15 @@ def deploy_user(user_id):
|
|||||||
if not h.auto_deploy_enabled:
|
if not h.auto_deploy_enabled:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# Który plik wgrywać?
|
|
||||||
if h.preferred_hostfile_id:
|
if h.preferred_hostfile_id:
|
||||||
chosen_file = HostFile.query.filter_by(id=h.preferred_hostfile_id, user_id=user_id).first()
|
chosen_file = HostFile.query.filter_by(id=h.preferred_hostfile_id, user_id=user_id).first()
|
||||||
if not chosen_file:
|
if not chosen_file:
|
||||||
# fallback do default
|
|
||||||
chosen_file = default_file
|
chosen_file = default_file
|
||||||
else:
|
else:
|
||||||
chosen_file = default_file
|
chosen_file = default_file
|
||||||
|
|
||||||
# final_content
|
# Dołączamy regex_lines tylko, jeśli dla hosta nie wyłączono tej opcji
|
||||||
final_content = regex_lines + chosen_file.content
|
final_content = ("" if h.disable_regex_deploy else regex_lines) + chosen_file.content
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if h.type == 'mikrotik':
|
if h.type == 'mikrotik':
|
||||||
@ -1213,7 +1233,6 @@ def deploy_user(user_id):
|
|||||||
user_id=user_id
|
user_id=user_id
|
||||||
))
|
))
|
||||||
elif h.use_daemon and h.type == 'linux':
|
elif h.use_daemon and h.type == 'linux':
|
||||||
# Demon
|
|
||||||
import requests
|
import requests
|
||||||
adjusted_content = ensure_local_defaults(final_content)
|
adjusted_content = ensure_local_defaults(final_content)
|
||||||
wrapped_content = wrap_content_with_comments(adjusted_content)
|
wrapped_content = wrap_content_with_comments(adjusted_content)
|
||||||
@ -1223,12 +1242,13 @@ def deploy_user(user_id):
|
|||||||
if resp.status_code != 200:
|
if resp.status_code != 200:
|
||||||
raise Exception(f"Daemon POST error: {resp.status_code} - {resp.text}")
|
raise Exception(f"Daemon POST error: {resp.status_code} - {resp.text}")
|
||||||
|
|
||||||
|
daemon_str = h.daemon_url.split("://")[-1]
|
||||||
|
daemon_ip = daemon_str.split(":")[0]
|
||||||
db.session.add(DeployLog(
|
db.session.add(DeployLog(
|
||||||
details=f'[LINUX/DAEMON] Updated {h.hostname} for user {user_id}',
|
details=f'[LINUX/DAEMON] Updated {h.hostname} (Daemon IP: {daemon_ip}) for user {user_id}',
|
||||||
user_id=user_id
|
user_id=user_id
|
||||||
))
|
))
|
||||||
else:
|
else:
|
||||||
# standard linux - SSH
|
|
||||||
ssh = open_ssh_connection(h)
|
ssh = open_ssh_connection(h)
|
||||||
adjusted_content = ensure_local_defaults(final_content)
|
adjusted_content = ensure_local_defaults(final_content)
|
||||||
wrapped_content = wrap_content_with_comments(adjusted_content)
|
wrapped_content = wrap_content_with_comments(adjusted_content)
|
||||||
@ -1250,10 +1270,10 @@ def deploy_user(user_id):
|
|||||||
db.session.add(DeployLog(
|
db.session.add(DeployLog(
|
||||||
details=f'Failed to update {h.hostname}: {str(e)} for user {user_id}',
|
details=f'Failed to update {h.hostname}: {str(e)} for user {user_id}',
|
||||||
user_id=user_id
|
user_id=user_id
|
||||||
|
|
||||||
))
|
))
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
|
|
||||||
|
|
||||||
def deploy_mikrotik(host, hosts_content):
|
def deploy_mikrotik(host, hosts_content):
|
||||||
ssh = open_ssh_connection(host)
|
ssh = open_ssh_connection(host)
|
||||||
stdin, stdout, stderr = ssh.exec_command("/ip dns static export")
|
stdin, stdout, stderr = ssh.exec_command("/ip dns static export")
|
||||||
@ -1468,11 +1488,12 @@ def update_host_automation(id):
|
|||||||
host.auto_deploy_enabled = enabled
|
host.auto_deploy_enabled = enabled
|
||||||
elif setting == 'auto_backup':
|
elif setting == 'auto_backup':
|
||||||
host.auto_backup_enabled = enabled
|
host.auto_backup_enabled = enabled
|
||||||
|
elif setting == 'disable_regex':
|
||||||
|
host.disable_regex_deploy = enabled
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
flash('Ustawienia automatyzacji zostały zaktualizowane.', 'success')
|
flash('Ustawienia automatyzacji zostały zaktualizowane.', 'success')
|
||||||
return redirect(url_for('server_list'))
|
return redirect(url_for('server_list'))
|
||||||
|
|
||||||
|
|
||||||
# -------------------
|
# -------------------
|
||||||
# EDYCJA LOKALNEGO PLIKU HOSTS
|
# EDYCJA LOKALNEGO PLIKU HOSTS
|
||||||
# -------------------
|
# -------------------
|
||||||
|
@ -21,7 +21,14 @@
|
|||||||
{% for host in hosts %}
|
{% for host in hosts %}
|
||||||
<div class="form-check">
|
<div class="form-check">
|
||||||
<input class="form-check-input" type="checkbox" name="hosts" value="{{ host.id }}" id="host{{ host.id }}">
|
<input class="form-check-input" type="checkbox" name="hosts" value="{{ host.id }}" id="host{{ host.id }}">
|
||||||
<label class="form-check-label" for="host{{ host.id }}">{{ host.hostname }} ({{ host.type }})</label>
|
<label class="form-check-label" for="host{{ host.id }}">
|
||||||
|
{% if host.use_daemon and host.type == 'linux' and host.daemon_url %}
|
||||||
|
{{ host.daemon_ip }} - {{ host.resolved_daemon }}
|
||||||
|
{% else %}
|
||||||
|
{{ host.hostname }}
|
||||||
|
{% endif %}
|
||||||
|
({{ host.type }})
|
||||||
|
</label>
|
||||||
</div>
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
<th>Wybrany plik /etc/hosts</th>
|
<th>Wybrany plik /etc/hosts</th>
|
||||||
<th>Auto Deploy</th>
|
<th>Auto Deploy</th>
|
||||||
<th>Auto Backup</th>
|
<th>Auto Backup</th>
|
||||||
|
<th>Wyłącz regex deploy</th> <!-- Nowa kolumna -->
|
||||||
<th>Akcje</th>
|
<th>Akcje</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
@ -36,15 +37,16 @@
|
|||||||
{% for h in hosts %}
|
{% for h in hosts %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ h.id }}</td>
|
<td>{{ h.id }}</td>
|
||||||
<td data-bs-toggle="tooltip" data-bs-placement="top" title="{{ h.resolved_hostname }}">
|
<td data-bs-toggle="tooltip" data-bs-placement="top"
|
||||||
|
title="{% if h.use_daemon and h.type == 'linux' and h.daemon_url %}{{ h.resolved_daemon }}{% else %}{{ h.resolved_hostname }}{% endif %}">
|
||||||
{% if h.use_daemon and h.type == 'linux' and h.daemon_url %}
|
{% if h.use_daemon and h.type == 'linux' and h.daemon_url %}
|
||||||
{% set daemon_str = h.daemon_url.split('://') | last %}
|
{% set daemon_str = h.daemon_url.split('://') | last %}
|
||||||
{% set daemon_split = daemon_str.split(':') %}
|
{% set daemon_ip = daemon_str.split(':')[0] %}
|
||||||
<small>(Daemon IP: {{ daemon_split[0] }})</small>
|
{{ daemon_ip }}
|
||||||
{% else %}
|
{% else %}
|
||||||
{{ h.hostname }}
|
{{ h.hostname }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
{% if h.use_daemon and h.type == 'linux' %}
|
{% if h.use_daemon and h.type == 'linux' %}
|
||||||
<em>—</em>
|
<em>—</em>
|
||||||
@ -81,7 +83,6 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</td>
|
</td>
|
||||||
<!-- Nowa kolumna: preferowany plik hosts -->
|
|
||||||
<td>
|
<td>
|
||||||
{% if h.preferred_hostfile %}
|
{% if h.preferred_hostfile %}
|
||||||
{{ h.preferred_hostfile.title }}
|
{{ h.preferred_hostfile.title }}
|
||||||
@ -89,7 +90,6 @@
|
|||||||
(Default)
|
(Default)
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</td>
|
</td>
|
||||||
<!-- Formularz aktualizujący auto_deploy -->
|
|
||||||
<td>
|
<td>
|
||||||
<form method="POST" action="{{ url_for('update_host_automation', id=h.id) }}" style="display:inline;">
|
<form method="POST" action="{{ url_for('update_host_automation', id=h.id) }}" style="display:inline;">
|
||||||
<input type="hidden" name="setting" value="auto_deploy">
|
<input type="hidden" name="setting" value="auto_deploy">
|
||||||
@ -97,7 +97,6 @@
|
|||||||
onchange="this.form.submit()" {% if h.auto_deploy_enabled %}checked{% endif %}>
|
onchange="this.form.submit()" {% if h.auto_deploy_enabled %}checked{% endif %}>
|
||||||
</form>
|
</form>
|
||||||
</td>
|
</td>
|
||||||
<!-- Formularz aktualizujący auto_backup -->
|
|
||||||
<td>
|
<td>
|
||||||
<form method="POST" action="{{ url_for('update_host_automation', id=h.id) }}" style="display:inline;">
|
<form method="POST" action="{{ url_for('update_host_automation', id=h.id) }}" style="display:inline;">
|
||||||
<input type="hidden" name="setting" value="auto_backup">
|
<input type="hidden" name="setting" value="auto_backup">
|
||||||
@ -106,16 +105,26 @@
|
|||||||
</form>
|
</form>
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<a href="{{ url_for('edit_server', id=h.id) }}" class="btn btn-primary btn-sm">Edytuj</a>
|
<form method="POST" action="{{ url_for('update_host_automation', id=h.id) }}" style="display:inline;">
|
||||||
<a href="{{ url_for('test_server_connection', id=h.id) }}" class="btn btn-info btn-sm">Testuj</a>
|
<input type="hidden" name="setting" value="disable_regex">
|
||||||
<a href="{{ url_for('server_backup', host_id=h.id) }}" class="btn btn-success btn-sm">Backup</a>
|
<input type="checkbox" name="enabled" value="1"
|
||||||
<form method="GET" action="{{ url_for('delete_server', id=h.id) }}" style="display:inline;">
|
onchange="this.form.submit()" {% if h.disable_regex_deploy %}checked{% endif %}>
|
||||||
<button type="submit" class="btn btn-danger btn-sm">Usuń</button>
|
|
||||||
</form>
|
</form>
|
||||||
</td>
|
</td>
|
||||||
|
<td>
|
||||||
|
<div class="d-flex flex-wrap gap-1">
|
||||||
|
<a href="{{ url_for('edit_server', id=h.id) }}" class="btn btn-primary btn-sm">Edytuj</a>
|
||||||
|
<a href="{{ url_for('test_server_connection', id=h.id) }}" class="btn btn-info btn-sm">Testuj</a>
|
||||||
|
<a href="{{ url_for('server_backup', host_id=h.id) }}" class="btn btn-success btn-sm">Backup</a>
|
||||||
|
<form method="GET" action="{{ url_for('delete_server', id=h.id) }}" style="display:inline;">
|
||||||
|
<button type="submit" class="btn btn-danger btn-sm">Usuń</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user