crafty-4/app/frontend/templates/panel/server_backup.html

631 lines
23 KiB
HTML
Raw Normal View History

{% extends ../base.html %}
{% block meta %}
{% end %}
{% block title %}Crafty Controller - {{ translate('serverDetails', 'serverDetails', data['lang']) }}{% end %}
{% block content %}
<div class="content-wrapper">
<!-- Page Title Header Starts-->
<div class="row page-title-header">
<div class="col-12">
<div class="page-header">
<h4 class="page-title">
{{ translate('serverDetails', 'serverDetails', data['lang']) }} - {{
data['server_stats']['server_id']['server_name'] }}
<br />
2023-11-01 17:44:56 +00:00
<small>UUID: {{ data['server_stats']['server_id']['server_id'] }}</small>
</h4>
</div>
</div>
</div>
<!-- Page Title Header Ends-->
{% include "parts/details_stats.html %}
<div class="row">
<div class="col-sm-12 grid-margin">
<div class="card">
<div class="card-body pt-0">
<span class="d-none d-sm-block">
{% include "parts/server_controls_list.html %}
</span>
<span class="d-block d-sm-none">
{% include "parts/m_server_controls_list.html %}
</span>
2024-03-09 04:23:36 +00:00
<div class="row">
<div class="col-md-12 col-sm-12" style="overflow-x:auto;">
<div class="card">
<div class="card-header header-sm d-flex justify-content-between align-items-center">
2024-04-20 22:15:06 +00:00
<h4 class="card-title"><i class="fa-regular fa-bell"></i> {{ translate('serverBackups', 'backups',
data['lang']) }} </h4>
2024-03-09 04:23:36 +00:00
{% if data['user_data']['hints'] %}
2024-04-20 22:15:06 +00:00
<span class="too_small" title="{{ translate('serverSchedules', 'cannotSee', data['lang']) }}" ,
data-content="{{ translate('serverSchedules', 'cannotSeeOnMobile', data['lang']) }}" ,
data-placement="bottom"></span>
2024-03-09 04:23:36 +00:00
{% end %}
2024-04-20 22:15:06 +00:00
<div><a class="btn btn-info"
href="/panel/add_backup?id={{ data['server_stats']['server_id']['server_id'] }}"><i
class="fas fa-plus-circle"></i> {{ translate('serverBackups', 'newBackup', data['lang']) }}</a>
</div>
2024-03-09 04:23:36 +00:00
</div>
<div class="card-body">
{% if len(data['backups']) == 0 %}
<div style="text-align: center; color: grey;">
2024-04-20 22:15:06 +00:00
<h7>{{ translate('serverBackups', 'no-backup', data['lang']) }} <strong>{{
translate('serverBackups', 'newBackup',data['lang']) }}</strong>.</h7>
2024-03-09 04:23:36 +00:00
</div>
{% end %}
{% if len(data['backups']) > 0 %}
<div class="d-none d-lg-block">
2024-04-20 22:15:06 +00:00
<table class="table table-hover responsive-table" aria-label="backups list" id="backup_table"
width="100%" style="table-layout:fixed;">
2024-03-09 04:23:36 +00:00
<thead>
<tr class="rounded">
2024-04-20 22:15:06 +00:00
<th scope="col" style="width: 15%; min-width: 10px;">{{ translate('serverBackups', 'name',
data['lang']) }} </th>
<th scope="col" style="width: 60%; min-width: 50px;">{{ translate('serverBackups',
'storageLocation', data['lang']) }}</th>
<th scope="col" style="width: 10%; min-width: 50px;">{{ translate('serverBackups',
'maxBackups', data['lang']) }}</th>
<th scope="col" style="width: 10%; min-width: 50px;">{{ translate('serverBackups', 'actions',
data['lang']) }}</th>
2024-03-09 04:23:36 +00:00
</tr>
</thead>
<tbody>
{% for backup in data['backups'] %}
<tr>
<td id="{{backup.backup_name}}" class="id">
<p>{{backup.backup_name}}</p>
</td>
<td id="{{backup.backup_location}}" class="type">
<p>{{backup.backup_location}}</p>
</td>
<td id="{{backup.max_backups}}" class="trigger" style="overflow: scroll; max-width: 30px;">
<p>{{backup.max_backups}}</p>
</td>
<td id="backup_edit" class="action">
2024-04-20 22:15:06 +00:00
<button
2024-04-21 15:26:16 +00:00
onclick="window.location.href=`/panel/edit_backup?id={{ data['server_stats']['server_id']['server_id'] }}&backup_id={{backup.backup_id}}`"
2024-04-20 22:15:06 +00:00
class="btn btn-info">
2024-03-09 04:23:36 +00:00
<i class="fas fa-pencil-alt"></i>
</button>
2024-04-21 15:26:16 +00:00
<button data-backup={{ backup.backup_id }} class="btn btn-danger del_button">
2024-03-09 04:23:36 +00:00
<i class="fas fa-trash" aria-hidden="true"></i>
</button>
2024-04-21 15:26:16 +00:00
<button data-backup={{ backup.backup_id }} data-toggle="tooltip"
2024-04-20 22:15:06 +00:00
title="{{ translate('serverBackups', 'run', data['lang']) }}"
class="btn btn-outline-warning test-socket">
2024-03-09 04:23:36 +00:00
<i class="fa-solid fa-vial"></i>
</button>
</td>
</tr>
{% end %}
</tbody>
</table>
</div>
<div class="d-block d-lg-none">
2024-04-20 22:15:06 +00:00
<table aria-label="backups list" class="table table-hover responsive-table" id="backup_table_mini"
width="100%" style="table-layout:fixed;">
2024-03-09 04:23:36 +00:00
<thead>
<tr class="rounded">
<th style="width: 40%; min-width: 10px;">Name
</th>
2024-03-09 17:49:51 +00:00
<th style="width: 40%; min-width: 50px;">{{ translate('serverBackups', 'edit', data['lang'])
2024-03-09 04:23:36 +00:00
}}</th>
</tr>
</thead>
<tbody>
{% for backup in data['backups'] %}
<tr>
<td id="{{backup.backup_name}}" class="id">
<p>{{backup.backup_name}}</p>
</td>
<td id="backup_edit" class="action">
2024-04-20 22:15:06 +00:00
<button
2024-04-21 15:26:16 +00:00
onclick="window.location.href=`/panel/edit_backup?id={{ data['server_stats']['server_id']['server_id'] }}&backup_id={{backup.backup_id}}`"
2024-04-20 22:15:06 +00:00
class="btn btn-info">
2024-03-09 04:23:36 +00:00
<i class="fas fa-pencil-alt"></i>
</button>
<button data-backup={{ backup.backup_id }} class="btn btn-danger del_button">
<i class="fas fa-trash" aria-hidden="true"></i>
</button>
2024-04-20 22:15:06 +00:00
<button data-backup={{ backup.backup_id }} data-toggle="tooltip"
title="{{ translate('serverBackups', 'run', data['lang']) }}"
class="btn btn-outline-warning test-socket">
2024-03-09 04:23:36 +00:00
<i class="fa-solid fa-vial"></i>
</button>
</td>
</tr>
{% end %}
</tbody>
</table>
</div>
{% end %}
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<style>
/* Remove default bullets */
.tree-view,
.tree-nested {
list-style-type: none;
margin: 0;
padding: 0;
margin-left: 10px;
}
/* Style the items */
.tree-item,
.files-tree-title {
cursor: pointer;
user-select: none;
/* Prevent text selection */
}
/* Create the caret/arrow with a unicode, and style it */
.tree-caret .fa-folder {
display: inline-block;
}
.tree-caret .fa-folder-open {
display: none;
}
/* Rotate the caret/arrow icon when clicked on (using JavaScript) */
.tree-caret-down .fa-folder {
display: none;
}
.tree-caret-down .fa-folder-open {
display: inline-block;
}
/* Hide the nested list */
.tree-nested {
display: none;
}
</style>
<!-- content-wrapper ends -->
{% end %}
{% block js %}
<script>
const server_id = new URLSearchParams(document.location.search).get('id')
//used to get cookies from browser - this is part of tornados xsrf protection - it's for extra security
function getCookie(name) {
var r = document.cookie.match("\\b" + name + "=([^;]*)\\b");
return r ? r[1] : undefined;
}
async function backup_started() {
2023-09-04 23:32:56 +00:00
const token = getCookie("_xsrf")
let res = await fetch(`/api/v2/servers/${server_id}/action/backup_server`, {
2024-04-20 22:15:06 +00:00
method: 'POST',
headers: {
'X-XSRFToken': token
}
});
let responseData = await res.json();
if (responseData.status === "ok") {
console.log(responseData);
$("#backup_button").html(`<div class="progress" style="height: 15px;">
2023-09-27 01:07:56 +00:00
<div class="progress-bar progress-bar-striped progress-bar-animated" id="backup_progress_bar"
role="progressbar" style="width:{{data['backup_stats']['percent']}}%;"
aria-valuenow="{{data['backup_stats']['percent']}}" aria-valuemin="0" aria-valuemax="100">{{
data['backup_stats']['percent'] }}%</div>
</div>
<p>Backing up <i class="fas fa-spin fa-spinner"></i> <span
id="total_files">{{data['server_stats']['world_size']}}</span></p>`);
2024-04-20 22:15:06 +00:00
} else {
2024-04-20 22:15:06 +00:00
bootbox.alert({
title: responseData.status,
message: responseData.error
});
}
return;
}
async function del_backup(filename, id) {
2023-09-04 23:32:56 +00:00
const token = getCookie("_xsrf")
2024-04-20 22:15:06 +00:00
let contents = JSON.stringify({ "filename": filename })
let res = await fetch(`/api/v2/servers/${id}/backups/backup/`, {
method: 'DELETE',
headers: {
'token': token,
},
body: contents
});
let responseData = await res.json();
if (responseData.status === "ok") {
window.location.reload();
2024-04-20 22:15:06 +00:00
} else {
bootbox.alert({
"title": responseData.status,
"message": responseData.error
})
}
}
async function restore_backup(filename, id) {
2023-09-04 23:32:56 +00:00
const token = getCookie("_xsrf")
2024-04-20 22:15:06 +00:00
let contents = JSON.stringify({ "filename": filename })
var dialog = bootbox.dialog({
message: "<i class='fa fa-spin fa-spinner'></i> {{ translate('serverBackups', 'restoring', data['lang']) }}",
closeButton: false
});
let res = await fetch(`/api/v2/servers/${id}/backups/backup/`, {
method: 'POST',
headers: {
'token': token,
},
body: contents
});
let responseData = await res.json();
if (responseData.status === "ok") {
window.location.href = "/panel/dashboard";
2024-04-20 22:15:06 +00:00
} else {
bootbox.alert({
"title": responseData.status,
"message": responseData.error
})
}
}
2023-01-21 18:13:02 +00:00
$("#before-check").on("click", function () {
if ($("#before-check:checked").val()) {
$("#backup_before").css("display", "inline-block");
} else {
$("#backup_before").css("display", "none");
$("#backup_before").val("");
}
});
$("#after-check").on("click", function () {
if ($("#after-check:checked").val()) {
$("#backup_after").css("display", "inline-block");
} else {
2023-01-21 18:13:02 +00:00
$("#backup_after").css("display", "none");
$("#backup_after").val("");
}
});
2023-07-12 22:01:14 +00:00
function replacer(key, value) {
if (key != "backup_before" && key != "backup_after") {
if (typeof value == "boolean" || key === "executable_update_url") {
return value
} else {
return (isNaN(value) ? value : +value);
}
} else {
return value;
}
}
$(document).ready(function () {
2023-07-12 22:01:14 +00:00
$("#backup-form").on("submit", async function (e) {
2023-07-12 16:23:49 +00:00
e.preventDefault();
2023-09-04 23:32:56 +00:00
const token = getCookie("_xsrf")
2023-07-12 22:01:14 +00:00
let backupForm = document.getElementById("backup-form");
2023-07-12 16:23:49 +00:00
let formData = new FormData(backupForm);
2023-07-12 22:01:14 +00:00
//Remove checks that we don't need in form data.
formData.delete("after-check");
formData.delete("before-check");
2023-07-12 16:23:49 +00:00
//Create an object from the form data entries
let formDataObject = Object.fromEntries(formData.entries());
//We need to make sure these are sent regardless of whether or not they're checked
formDataObject.compress = $("#compress").prop('checked');
formDataObject.shutdown = $("#shutdown").prop('checked');
2023-07-12 22:01:14 +00:00
let excluded = [];
$('input.excluded:checkbox:checked').each(function () {
excluded.push($(this).val());
});
2024-04-20 22:15:06 +00:00
if ($("#root_files_button").hasClass("clicked")) {
2023-07-12 22:01:14 +00:00
formDataObject.exclusions = excluded;
}
2023-12-05 04:44:51 +00:00
delete formDataObject.root_path
2023-07-12 22:01:14 +00:00
console.log(excluded);
2023-07-12 16:23:49 +00:00
console.log(formDataObject);
// Format the plain form data as JSON
let formDataJsonString = JSON.stringify(formDataObject, replacer);
console.log(formDataJsonString);
2023-07-12 22:01:14 +00:00
let res = await fetch(`/api/v2/servers/${server_id}/backups/`, {
2023-07-12 16:23:49 +00:00
method: 'PATCH',
headers: {
'X-XSRFToken': token
},
body: formDataJsonString,
});
let responseData = await res.json();
if (responseData.status === "ok") {
window.location.reload();
} else {
bootbox.alert({
title: responseData.error,
message: responseData.error_data
});
}
});
2023-11-01 17:44:56 +00:00
2022-04-12 21:37:20 +00:00
try {
if ($('#backup_path').val() == '') {
console.log('true')
try {
document.getElementById('backup_now_button').disabled = true;
} catch {
}
} else {
document.getElementById('backup_now_button').disabled = false;
}
} catch {
try {
document.getElementById('backup_now_button').disabled = false;
} catch {
}
}
console.log("ready!");
$("#backup_config_box").hide();
$("#backup_save_note").hide();
$("#show_config").click(function () {
$("#backup_config_box").toggle();
$('#backup_button').hide();
$('#backup_save_note').show();
$('#backup_data').hide();
});
$('#backup_table').DataTable({
"order": [[1, "desc"]],
2022-06-22 03:18:17 +00:00
"paging": false,
"lengthChange": false,
"searching": true,
"ordering": true,
"info": true,
"autoWidth": false,
"responsive": true,
});
$(".del_button").click(function () {
var file_to_del = $(this).data("file");
var backup_path = $(this).data('backup_path');
console.log("file to delete is" + file_to_del);
bootbox.confirm({
title: "{% raw translate('serverBackups', 'destroyBackup', data['lang']) %}",
message: "{{ translate('serverBackups', 'confirmDelete', data['lang']) }}",
buttons: {
cancel: {
label: '<i class="fas fa-times"></i> {{ translate("serverBackups", "cancel", data['lang']) }}'
},
confirm: {
label: '<i class="fas fa-check"></i> {{ translate("serverBackups", "confirm", data['lang']) }}'
}
2021-11-29 21:22:46 +00:00
},
callback: function (result) {
console.log(result);
if (result == true) {
var full_path = backup_path + '/' + file_to_del;
del_backup(file_to_del, server_id);
}
}
2021-11-29 21:22:46 +00:00
});
});
2021-11-29 21:22:46 +00:00
$(".restore_button").click(function () {
var file_to_restore = $(this).data("file");
2021-11-29 21:22:46 +00:00
bootbox.confirm({
title: "{{ translate('serverBackups', 'restore', data['lang']) }} " + file_to_restore,
message: "{{ translate('serverBackups', 'confirmRestore', data['lang']) }}",
buttons: {
cancel: {
label: '<i class="fas fa-times"></i> {{ translate("serverBackups", "cancel", data['lang']) }}'
},
confirm: {
label: '<i class="fas fa-check"></i> {{ translate("serverBackups", "restore", data['lang']) }}',
className: 'btn-outline-danger'
}
},
callback: function (result) {
console.log(result);
if (result == true) {
restore_backup(file_to_restore, server_id);
}
}
});
});
$("#backup_now_button").click(function () {
backup_started();
});
2021-11-29 21:22:46 +00:00
});
document.getElementById("modal-cancel").addEventListener("click", function () {
document.getElementById("root_files_button").classList.remove('clicked');
document.getElementById("main-tree-div").innerHTML = '<input type="checkbox" id="main-tree-input" name="root_path" value="" disabled><span id="main-tree" class="files-tree-title tree-caret-down root-dir" data-path=""><i class="far fa-folder"></i><i class="far fa-folder-open"></i>{{ translate("serverFiles", "files", data["lang"]) }}</span></input>'
})
document.getElementById("root_files_button").addEventListener("click", function () {
if ($("#root_files_button").data('server_path') != "") {
if (document.getElementById('root_files_button').classList.contains('clicked')) {
show_file_tree();
return;
} else {
document.getElementById('root_files_button').classList.add('clicked');
}
path = $("#root_files_button").data('server_path')
console.log($("#root_files_button").data('server_path'))
2023-09-04 23:32:56 +00:00
const token = getCookie("_xsrf");
var dialog = bootbox.dialog({
message: '<p class="text-center mb-0"><i class="fa fa-spin fa-cog"></i> Please wait while we gather your files...</p>',
closeButton: false
});
setTimeout(function () {
var x = document.querySelector('.bootbox');
if (x) {
x.remove()
}
var x = document.querySelector('.modal-backdrop');
if (x) {
x.remove()
}
2023-07-06 00:11:27 +00:00
document.getElementById('main-tree-input').setAttribute('value', path)
getTreeView(path);
show_file_tree();
}, 5000);
2023-07-06 00:11:27 +00:00
} else {
bootbox.alert("You must input a path before selecting this button");
}
});
if (webSocket) {
webSocket.on('backup_status', function (backup) {
if (backup.percent >= 100) {
document.getElementById('backup_progress_bar').innerHTML = '100%';
document.getElementById('backup_progress_bar').style.width = '100%';
setTimeout(function () {
window.location.reload(1);
}, 5000);
} else {
document.getElementById('backup_progress_bar').innerHTML = backup.percent + '%';
document.getElementById('backup_progress_bar').style.width = backup.percent + '%';
document.getElementById('total_files').innerHTML = backup.total_files;
}
});
}
2024-04-20 22:15:06 +00:00
function getDirView(event) {
2023-06-04 18:43:45 +00:00
let path = event.target.parentElement.getAttribute("data-path");
if (document.getElementById(path).classList.contains('clicked')) {
return;
2024-04-20 22:15:06 +00:00
} else {
2023-06-04 18:43:45 +00:00
getTreeView(path);
}
2023-06-04 18:43:45 +00:00
}
2024-04-20 22:15:06 +00:00
async function getTreeView(path) {
2023-09-04 23:32:56 +00:00
console.log(path)
const token = getCookie("_xsrf");
2023-06-05 00:55:48 +00:00
let res = await fetch(`/api/v2/servers/${server_id}/files`, {
2024-04-20 22:15:06 +00:00
method: 'POST',
headers: {
'X-XSRFToken': token
},
body: JSON.stringify({ "page": "backups", "path": path }),
});
let responseData = await res.json();
if (responseData.status === "ok") {
console.log(responseData);
process_tree_response(responseData);
2024-04-20 22:15:06 +00:00
} else {
2024-04-20 22:15:06 +00:00
bootbox.alert({
title: responseData.status,
message: responseData.error
});
}
2023-06-04 18:43:45 +00:00
}
function process_tree_response(response) {
let path = response.data.root_path.path;
let text = `<ul class="tree-nested d-block" id="${path}ul">`;
2024-04-20 22:15:06 +00:00
Object.entries(response.data).forEach(([key, value]) => {
if (key === "root_path" || key === "db_stats") {
//continue is not valid in for each. Return acts as a continue.
return;
}
2023-06-04 18:43:45 +00:00
let checked = ""
let dpath = value.path;
let filename = key;
2024-04-20 22:15:06 +00:00
if (value.excluded) {
2023-12-05 04:44:51 +00:00
checked = "checked"
}
2024-04-20 22:15:06 +00:00
if (value.dir) {
2023-06-04 18:43:45 +00:00
text += `<li class="tree-item" data-path="${dpath}">
\n<div id="${dpath}" data-path="${dpath}" data-name="${filename}" class="tree-caret tree-ctx-item tree-folder">
2023-07-12 22:01:14 +00:00
<input type="checkbox" class="checkBoxClass excluded" value="${dpath}" ${checked}>
2023-06-04 18:43:45 +00:00
<span id="${dpath}span" class="files-tree-title" data-path="${dpath}" data-name="${filename}" onclick="getDirView(event)">
<i style="color: var(--info);" class="far fa-folder"></i>
<i style="color: var(--info);" class="far fa-folder-open"></i>
<strong>${filename}</strong>
</span>
</input></div><li>`
2024-04-20 22:15:06 +00:00
} else {
2023-06-04 18:43:45 +00:00
text += `<li
class="d-block tree-ctx-item tree-file"
data-path="${dpath}"
data-name="${filename}"
2023-07-12 22:01:14 +00:00
onclick=""><input type='checkbox' class="checkBoxClass excluded" name='root_path' value="${dpath}" ${checked}><span style="margin-right: 6px;">
2023-06-04 18:43:45 +00:00
<i class="far fa-file"></i></span></input>${filename}</li>`
}
});
text += `</ul>`;
2024-04-20 22:15:06 +00:00
if (response.data.root_path.top) {
2023-06-04 18:43:45 +00:00
try {
2024-04-20 22:15:06 +00:00
document.getElementById('main-tree-div').innerHTML += text;
document.getElementById('main-tree').parentElement.classList.add("clicked");
} catch {
document.getElementById('files-tree').innerHTML = text;
}
} else {
2023-06-04 18:43:45 +00:00
try {
2024-04-20 22:15:06 +00:00
document.getElementById(path + "span").classList.add('tree-caret-down');
document.getElementById(path).innerHTML += text;
document.getElementById(path).classList.add("clicked");
} catch {
console.log("Bad")
}
2024-04-20 22:15:06 +00:00
var toggler = document.getElementById(path + "span");
2024-04-20 22:15:06 +00:00
if (toggler.classList.contains('files-tree-title')) {
document.getElementById(path + "span").addEventListener("click", function caretListener() {
document.getElementById(path + "ul").classList.toggle("d-block");
document.getElementById(path + "span").classList.toggle("tree-caret-down");
});
}
}
}
2023-06-04 18:43:45 +00:00
function getToggleMain(event) {
path = event.target.parentElement.getAttribute('data-path');
document.getElementById("files-tree").classList.toggle("d-block");
document.getElementById(path + "span").classList.toggle("tree-caret-down");
document.getElementById(path + "span").classList.toggle("tree-caret");
}
function show_file_tree() {
$("#dir_select").modal();
}
2022-02-27 21:15:40 +00:00
</script>
{% end %}