/** * configuration.js - Copyright(C) 2021 Donald Zou [https://github.com/donaldzou] * Under Apache-2.0 License */ let peers = []; (function() { /** * Definitions */ let configuration_name; let configuration_interval; let configuration_timeout = window.localStorage.getItem("configurationTimeout"); if (configuration_timeout === null || !["5000", "10000", "30000", "60000"].includes(configuration_timeout)) { window.localStorage.setItem("configurationTimeout", "10000"); configuration_timeout = window.localStorage.getItem("configurationTimeout"); } document.querySelector(`button[data-refresh-interval="${configuration_timeout}"]`).classList.add("active"); let display_mode = window.localStorage.getItem("displayMode"); if (display_mode === null || !["grid", "list"].includes(display_mode)) { window.localStorage.setItem("displayMode", "grid"); display_mode = "grid"; } document.querySelectorAll(".display-btn-group button").forEach(ele => ele.classList.remove("active")); document.querySelector(`button[data-display-mode="${display_mode}"]`).classList.add("active"); let $progress_bar = $(".progress-bar"); let bootstrapModalConfig = { keyboard: false, backdrop: 'static' }; let addModal = new bootstrap.Modal(document.getElementById('add_modal'), bootstrapModalConfig); let deleteBulkModal = new bootstrap.Modal(document.getElementById('delete_bulk_modal'), bootstrapModalConfig); let ipModal = new bootstrap.Modal(document.getElementById('available_ip_modal'), bootstrapModalConfig); let qrcodeModal = new bootstrap.Modal(document.getElementById('qrcode_modal'), bootstrapModalConfig); let settingModal = new bootstrap.Modal(document.getElementById('setting_modal'), bootstrapModalConfig); let deleteModal = new bootstrap.Modal(document.getElementById('delete_modal'), bootstrapModalConfig); let configurationDeleteModal = new bootstrap.Modal(document.getElementById('configuration_delete_modal'), bootstrapModalConfig); $("[data-toggle='tooltip']").tooltip(); $("[data-toggle='popover']").popover(); /** * Chart!!!!!! * @type {any} */ let chartUnit = window.localStorage.chartUnit; let chartUnitAvailable = ["GB", "MB", "KB"]; if (chartUnit === null || !chartUnitAvailable.includes(chartUnit)) { window.localStorage.setItem("chartUnit", "GB"); $('.switchUnit[data-unit="GB"]').addClass("active"); } else { $(`.switchUnit[data-unit="${chartUnit}"]`).addClass("active"); } chartUnit = window.localStorage.getItem("chartUnit"); const totalDataUsageChart = document.getElementById('totalDataUsageChartObj').getContext('2d'); const totalDataUsageChartObj = new Chart(totalDataUsageChart, { type: 'line', data: { labels: [], datasets: [{ label: 'Data Sent', data: [], stroke: '#FFFFFF', borderColor: '#28a745', tension: 0.1, borderWidth: 2 }, { label: 'Data Received', data: [], stroke: '#FFFFFF', borderColor: '#007bff', tension: 0.1, borderWidth: 2 } ] }, options: { maintainAspectRatio: false, showScale: false, responsive: false, scales: { y: { min: 0, ticks: { min: 0, callback: function(value, index, ticks) { return `${value} ${chartUnit}`; } } } }, plugins: { tooltip: { callbacks: { label: function(context) { return `${context.dataset.label}: ${context.parsed.y} ${chartUnit}`; } } } } } }); let $totalDataUsageChartObj = $("#totalDataUsageChartObj"); $totalDataUsageChartObj.css("width", "100%"); totalDataUsageChartObj.width = $totalDataUsageChartObj.parent().width(); totalDataUsageChartObj.resize(); $(window).on("resize", function() { totalDataUsageChartObj.resize(); }); $(".fullScreen").on("click", function() { let $chartContainer = $(".chartContainer"); if ($chartContainer.hasClass("fullScreen")) { $(this).children().removeClass("bi-fullscreen-exit").addClass("bi-fullscreen"); $chartContainer.removeClass("fullScreen"); } else { $(this).children().removeClass("bi-fullscreen").addClass("bi-fullscreen-exit"); $chartContainer.addClass("fullScreen"); } totalDataUsageChartObj.resize(); }); let mul = 1; $(".switchUnit").on("click", function() { $(".switchUnit").removeClass("active"); $(this).addClass("active"); if ($(this).data('unit') !== chartUnit) { switch ($(this).data('unit')) { case "GB": if (chartUnit === "MB") { mul = 1 / 1024; } if (chartUnit === "KB") { mul = 1 / 1048576; } break; case "MB": if (chartUnit === "GB") { mul = 1024; } if (chartUnit === "KB") { mul = 1 / 1024; } break; case "KB": if (chartUnit === "GB") { mul = 1048576; } if (chartUnit === "MB") { mul = 1024; } break; default: break; } window.localStorage.setItem("chartUnit", $(this).data('unit')); chartUnit = $(this).data('unit'); totalDataUsageChartObj.data.datasets[0].data = totalDataUsageChartObj.data.datasets[0].data.map(x => x * mul); totalDataUsageChartObj.data.datasets[1].data = totalDataUsageChartObj.data.datasets[1].data.map(x => x * mul); totalDataUsageChartObj.update(); } }); /** * To show alert on the configuration page * @param response */ function configurationAlert(response) { if (response.listen_port === "" && response.status === "stopped") { let configAlert = document.createElement("div"); configAlert.classList.add("alert"); configAlert.classList.add("alert-warning"); configAlert.setAttribute("role", "alert"); configAlert.innerHTML = 'Peer QR Code and configuration file download required a specified Listen Port.'; document.querySelector("#config_info_alert").appendChild(configAlert); } if (response.conf_address === "N/A") { let configAlert = document.createElement("div"); configAlert.classList.add("alert"); configAlert.classList.add("alert-warning"); configAlert.setAttribute("role", "alert"); configAlert.innerHTML = 'Configuration Address need to be specified to have peers connect to it.'; document.querySelector("#config_info_alert").appendChild(configAlert); } } function setActiveConfigurationName() { $(".nav-conf-link").removeClass("active"); $(`.sb-${configuration_name}-url`).addClass("active"); } let firstLoading = true; $(".nav-conf-link").on("click", function(e) { e.preventDefault(); if (configuration_name !== $(this).data("conf-id")) { firstLoading = true; $("#config_body").addClass("firstLoading"); configuration_name = $(this).data("conf-id"); if (loadPeers($('#search_peer_textbox').val())) { setActiveConfigurationName(); window.history.pushState(null, null, `/configuration/${configuration_name}`); $("title").text(`${configuration_name} | WGDashboard`); $(".index-alert").addClass("d-none").text(``); totalDataUsageChartObj.data.labels = []; totalDataUsageChartObj.data.datasets[0].data = []; totalDataUsageChartObj.data.datasets[1].data = []; totalDataUsageChartObj.update(); } } }); /** * Parse all responded information onto the configuration header * @param response */ function configurationHeader(response) { let $conf_status_btn = $(".toggle--switch"); if (response.checked === "checked") { $conf_status_btn.prop("checked", true) }else{ $conf_status_btn.prop("checked", false) } $conf_status_btn.data("conf-id", configuration_name) if (response.running_peer > 0) { let d = new Date(); let time = d.toLocaleString("en-us", { hour: '2-digit', minute: '2-digit', second: "2-digit", hourCycle: 'h23' }); totalDataUsageChartObj.data.labels.push(`${time}`); if (totalDataUsageChartObj.data.datasets[0].data.length === 0) { totalDataUsageChartObj.data.datasets[1].lastData = response.total_data_usage[2]; totalDataUsageChartObj.data.datasets[0].lastData = response.total_data_usage[1]; totalDataUsageChartObj.data.datasets[0].data.push(0); totalDataUsageChartObj.data.datasets[1].data.push(0); } else { if (totalDataUsageChartObj.data.datasets[0].data.length === 50 && totalDataUsageChartObj.data.datasets[1].data.length === 50) { totalDataUsageChartObj.data.labels.shift(); totalDataUsageChartObj.data.datasets[0].data.shift(); totalDataUsageChartObj.data.datasets[1].data.shift(); } let newTotalReceive = response.total_data_usage[2] - totalDataUsageChartObj.data.datasets[1].lastData; let newTotalSent = response.total_data_usage[1] - totalDataUsageChartObj.data.datasets[0].lastData; let k = 0; if (chartUnit === "MB") { k = 1024; } else if (chartUnit === "KB") { k = 1048576; } else { k = 1; } totalDataUsageChartObj.data.datasets[1].data.push(newTotalReceive * k); totalDataUsageChartObj.data.datasets[0].data.push(newTotalSent * k); totalDataUsageChartObj.data.datasets[0].lastData = response.total_data_usage[1]; totalDataUsageChartObj.data.datasets[1].lastData = response.total_data_usage[2]; } totalDataUsageChartObj.update(); } document.querySelector("#conf_name").textContent = configuration_name; $("#switch").removeClass("info_loading"); document.querySelectorAll("#sort_by_dropdown option").forEach(ele => ele.removeAttribute("selected")); document.querySelector(`#sort_by_dropdown option[value="${response.sort_tag}"]`).setAttribute("selected", "selected"); document.querySelector("#conf_status").innerHTML = `${response.status}`; document.querySelector("#conf_connected_peers").innerHTML = response.running_peer; document.querySelector("#conf_total_data_usage").innerHTML = `${response.total_data_usage[0]} GB`; document.querySelector("#conf_total_data_received").innerHTML = `${response.total_data_usage[2]} GB`; document.querySelector("#conf_total_data_sent").innerHTML = `${response.total_data_usage[1]} GB`; document.querySelector("#conf_public_key").innerHTML = response.public_key; document.querySelector("#conf_listen_port").innerHTML = response.listen_port === "" ? "N/A" : response.listen_port; document.querySelector("#conf_address").innerHTML = response.conf_address; let delay = 0; let h6 = $(".info h6"); for (let i = 0; i < h6.length; i++){ setTimeout(function(){ $(h6[i]).removeClass("info_loading"); }, delay) delay += 40 } } /** * Parse all responded information onto the peers list * @param response */ function configurationPeers(response) { let result = ""; if (response.peer_data.length === 0) { document.querySelector(".peer_list").innerHTML = `

Oops! No peers found ‘︿’

`; } else { let mode = display_mode === "list" ? "col-12" : "col-sm-6 col-lg-4"; response.peer_data.forEach(function(peer) { let total_r = 0; let total_s = 0; total_r += peer.cumu_receive; total_s += peer.cumu_sent; let spliter = '
'; let peer_name = `
${peer.name === "" ? "Untitled" : peer.name}
`; let peer_transfer = `

${roundN(peer.total_receive + total_r, 4)} GB

${roundN(peer.total_sent + total_s, 4)} GB

`; let peer_key = `
PEER CLICK TO COPY
${peer.id}
`; let peer_allowed_ip = `
ALLOWED IP
${peer.allowed_ip}
`; let peer_latest_handshake = `
LATEST HANDSHAKE
${peer.latest_handshake}
`; let peer_endpoint = `
END POINT
${peer.endpoint}
`; let peer_control = `

`; if (peer.private_key !== "") { peer_control += ``; } peer_control += '
'; let html = `
` + peer_name + spliter + peer_transfer + peer_key + peer_allowed_ip + peer_latest_handshake + spliter + peer_endpoint + spliter + peer_control + `
`; result += html; }); response.lock_access_peers.forEach(function(peer) { let total_r = 0; let total_s = 0; total_r += peer.cumu_receive; total_s += peer.cumu_sent; let spliter = '
'; let peer_name = `
${peer.name === "" ? "Untitled" : peer.name}
`; let peer_transfer = `

${roundN(peer.total_receive + total_r, 4)} GB

${roundN(peer.total_sent + total_s, 4)} GB

`; let peer_key = '
PEERCLICK TO COPY
' + peer.id + '
'; let peer_allowed_ip = '
ALLOWED IP
' + peer.allowed_ip + '
'; let peer_latest_handshake = '
LATEST HANDSHAKE
' + peer.latest_handshake + '
'; let peer_endpoint = '
END POINT
' + peer.endpoint + '
'; let peer_control = `

Peer Disabled
`; let html = '
' + '
' + '
' + '
' + peer_name + spliter + peer_transfer + peer_key + peer_allowed_ip + peer_latest_handshake + spliter + peer_endpoint + spliter + peer_control + '
' + '
' + '
' + '
'; result += html; }); document.querySelector(".peer_list").innerHTML = result; if (configuration_interval === undefined) { setConfigurationInterval(); } } } /** * Handle when adding peers by bulk */ function addPeersByBulk() { let $new_add_amount = $("#new_add_amount"); $add_peer.setAttribute("disabled", "disabled"); $add_peer.innerHTML = `Adding ${$new_add_amount.val()} peers...`; let $new_add_DNS = $("#new_add_DNS"); $new_add_DNS.val(window.configurations.cleanIp($new_add_DNS.val())); let $new_add_endpoint_allowed_ip = $("#new_add_endpoint_allowed_ip"); $new_add_endpoint_allowed_ip.val(window.configurations.cleanIp($new_add_endpoint_allowed_ip.val())); let $new_add_MTU = $("#new_add_MTU"); let $new_add_keep_alive = $("#new_add_keep_alive"); let $enable_preshare_key = $("#enable_preshare_key"); let data_list = [$new_add_DNS, $new_add_endpoint_allowed_ip, $new_add_MTU, $new_add_keep_alive]; if ($new_add_amount.val() > 0 && !$new_add_amount.hasClass("is-invalid")) { if ($new_add_DNS.val() !== "" && $new_add_endpoint_allowed_ip.val() !== "") { let conf = configuration_name; let keys = []; for (let i = 0; i < $new_add_amount.val(); i++) { keys.push(window.wireguard.generateKeypair()); } $.ajax({ method: "POST", url: "/add_peer_bulk/" + conf, headers: { "Content-Type": "application/json" }, data: JSON.stringify({ "DNS": $new_add_DNS.val(), "endpoint_allowed_ip": $new_add_endpoint_allowed_ip.val(), "MTU": $new_add_MTU.val(), "keep_alive": $new_add_keep_alive.val(), "enable_preshared_key": $enable_preshare_key.prop("checked"), "keys": keys, "amount": $new_add_amount.val() }), success: function(response) { if (response !== "true") { $("#add_peer_alert").html(response).removeClass("d-none"); data_list.forEach((ele) => ele.removeAttr("disabled")); $add_peer.removeAttribute("disabled"); $add_peer.innerHTML = "Save"; } else { window.configurations.loadPeers(""); data_list.forEach((ele) => ele.removeAttr("disabled")); $("#add_peer_form").trigger("reset"); $add_peer.removeAttribute("disabled"); $add_peer.innerHTML = "Save"; window.configurations.showToast($new_add_amount.val() + " peers added successful!"); window.configurations.addModal().toggle(); } } }); } else { $("#add_peer_alert").html("Please fill in all required box.").removeClass("d-none"); $add_peer.removeAttribute("disabled"); $add_peer.innerHTML = "Add"; } } else { $add_peer.removeAttribute("disabled"); $add_peer.innerHTML = "Add"; } } /** * Delete one peer or by bulk * @param config * @param peer_ids */ function deletePeers(config, peer_ids) { $.ajax({ method: "POST", url: "/remove_peer/" + config, headers: { "Content-Type": "application/json" }, data: JSON.stringify({ "action": "delete", "peer_ids": peer_ids }), success: function(response) { if (response !== "true") { if (window.configurations.deleteModal()._isShown) { $("#remove_peer_alert").html(response + $("#add_peer_alert").html()) .removeClass("d-none"); $("#delete_peer").removeAttr("disabled").html("Delete"); } if (window.configurations.deleteBulkModal()._isShown) { let $bulk_remove_peer_alert = $("#bulk_remove_peer_alert"); $bulk_remove_peer_alert.html(response + $bulk_remove_peer_alert.html()) .removeClass("d-none"); $("#confirm_delete_bulk_peers").removeAttr("disabled").html("Delete"); } } else { if (window.configurations.deleteModal()._isShown) { window.configurations.deleteModal().toggle(); } if (window.configurations.deleteBulkModal()._isShown) { $("#confirm_delete_bulk_peers").removeAttr("disabled").html("Delete"); $("#selected_peer_list").html(''); $(".delete-bulk-peer-item.active").removeClass('active'); window.configurations.deleteBulkModal().toggle(); } window.configurations.loadPeers($('#search_peer_textbox').val()); window.configurations.showToast(`Deleted ${peer_ids.length} peers`) $("#delete_peer").removeAttr("disabled").html("Delete"); } } }); } /** * Handle when the server is not responding */ function noResponding(message = "Opps!
I can't connect to the server.") { document.querySelectorAll(".no-response").forEach(ele => ele.classList.add("active")); setTimeout(function() { document.querySelectorAll(".no-response").forEach(ele => ele.classList.add("show")); document.querySelector("#right_body").classList.add("no-responding"); document.querySelector(".navbar").classList.add("no-responding"); document.querySelector(".no-response .container h4").innerHTML = message; }, 10); } /** * Remove no responding */ function removeNoResponding() { document.querySelectorAll(".no-response").forEach(ele => ele.classList.remove("show")); document.querySelector("#right_body").classList.remove("no-responding"); document.querySelector(".navbar").classList.remove("no-responding"); setTimeout(function() { document.querySelectorAll(".no-response").forEach(ele => ele.classList.remove("active")); }, 1010); } /** * Set configuration refresh Interval */ function setConfigurationInterval() { configuration_interval = setInterval(function() { loadPeers($('#search_peer_textbox').val()); }, configuration_timeout); } /** * Remove configuration refresh interval */ function removeConfigurationInterval() { clearInterval(configuration_interval); } /** * Start Progress Bar */ function startProgressBar() { $progress_bar.css("width", "0%") .css("opacity", "100") .css("background", "rgb(255,69,69)") .css("background", "linear-gradient(145deg, rgba(255,69,69,1) 0%, rgba(0,115,186,1) 100%)") .css("width", "25%"); setTimeout(function() { stillLoadingProgressBar(); }, 300); } /** * Still Loading Progress Bar */ function stillLoadingProgressBar() { $progress_bar.css("transition", "3s ease-in-out").css("width", "75%"); } /** * End Progress Bar */ function endProgressBar() { $progress_bar.css("transition", "0.3s ease-in-out").css("width", "100%"); setTimeout(function() { $progress_bar.css("opacity", "0"); }, 250); } /** * Round Transfer number into 4 digits * @param value * @param digits * @returns {number} */ function roundN(value, digits) { let tenToN = 10 ** digits; return (Math.round(value * tenToN)) / tenToN; } /** * Load Peers from server to configuration page * @param searchString */ let time = 0; let count = 0; let d1 = new Date(); function loadPeers(searchString) { d1 = new Date(); let good = true; $.ajax({ method: "GET", url: `/get_config/${configuration_name}?search=${encodeURIComponent(searchString)}`, headers: { "Content-Type": "application/json" } }).done(function(response) { console.log(response); parsePeers(response); }).fail(function() { noResponding(); good = false; }); return good; } function parsePeers(response) { if (response.status) { removeAllTooltips(); let d2 = new Date(); let seconds = (d2 - d1); time += seconds; count += 1; window.console.log(`Average time: ${time/count}ms`); $("#peer_loading_time").html(`Peer Loading Time: ${seconds}ms`); removeNoResponding(); peers = response.data.peer_data; configurationAlert(response.data); configurationHeader(response.data); configurationPeers(response.data); $(".dot.dot-running").attr("title", "Peer Connected").tooltip(); $(".dot.dot-stopped").attr("title", "Peer Disconnected").tooltip(); $("i[data-toggle='tooltip']").tooltip(); $("#configuration_name").text(configuration_name); if (firstLoading) { firstLoading = false; $("#config_body").removeClass("firstLoading"); } } else { noResponding(response.message); removeConfigurationInterval(); } } function removeAllTooltips(){ $(".tooltip").remove(); } function toggleAccess(peerID){ $.ajax({ url: "/api/togglePeerAccess", method: "POST", headers: {"Content-Type": "application/json"}, data: JSON.stringify({"peerID": peerID, "config": configuration_name}) }).done(function(res){ if(res.status){ loadPeers($('#search_peer_textbox').val()); }else{ showToast(res.reason); } }); } /** * Generate Private and Public key for a new peer */ function generate_key() { let keys = window.wireguard.generateKeypair(); document.querySelector("#private_key").value = keys.privateKey; document.querySelector("#public_key").value = keys.publicKey; document.querySelector("#add_peer_alert").classList.add("d-none"); document.querySelector("#re_generate_key i").classList.remove("rotating"); document.querySelector("#enable_preshare_key").value = keys.presharedKey; } /** * Show toast * @param msg */ let numberToast = 0; function showToast(msg) { $(".toastContainer").append( `` ) $(`#${numberToast}-toast`).toast('show'); $(`#${numberToast}-toast .toast-body`).html(msg); $(`#${numberToast}-toast .toast-progressbar`).css("transition", `width ${$(`#${numberToast}-toast .toast-progressbar`).parent().data('delay')}ms cubic-bezier(0, 0, 0, 0)`); $(`#${numberToast}-toast .toast-progressbar`).css("width", "0px"); numberToast++; } /** * Update peer's refresh interval * @param res * @param interval */ function updateRefreshInterval(interval) { configuration_timeout = interval; window.localStorage.setItem("configurationTimeout", configuration_timeout.toString()); removeConfigurationInterval(); setConfigurationInterval(); showToast("Refresh Interval set to " + Math.round(interval / 1000) + " seconds"); } /** * Clean IP * @param val * @returns {string} */ function cleanIp(val) { let clean_ip = val.split(','); for (let i = 0; i < clean_ip.length; i++) { clean_ip[i] = clean_ip[i].trim(' '); } return clean_ip.filter(Boolean).join(","); } /** * Trigger IP badge and item * @param ip */ function trigger_ip(ip) { let $ip_ele = document.querySelector(`.available-ip-item[data-ip='${ip}']`); if ($ip_ele) { if ($ip_ele.classList.contains("active")) { $ip_ele.classList.remove("active"); document.querySelector(`#selected_ip_list .badge[data-ip='${ip}']`).remove(); } else { $ip_ele.classList.add("active"); document.querySelector("#selected_ip_list").innerHTML += `${ip}`; } } } /** * Download single configuration file * @param conf */ function download_one_config(conf) { let link = document.createElement('a'); link.download = conf.filename; let blob = new Blob([conf.content], { type: 'text/conf' }); link.href = window.URL.createObjectURL(blob); link.click(); } /** * Toggle delete by bulk IP * @param element */ function toggleBulkIP(element) { let $selected_peer_list = $("#selected_peer_list"); let id = element.data("id"); let name = element.data("name") === "" ? "Untitled Peer" : element.data("name"); if (element.hasClass("active")) { element.removeClass("active"); $("#selected_peer_list .badge[data-id='" + id + "']").remove(); } else { element.addClass("active"); $selected_peer_list.append('' + name + ' - ' + id + ''); } } /** * Copy public keys to clipboard * @param element */ function copyToClipboard(element) { let $temp = $(""); $body.append($temp); $temp.val($(element).text()).trigger("select"); document.execCommand("copy"); $temp.remove(); } /** * Get all available IP for this configuration */ function getAvailableIps() { $.ajax({ "url": `/available_ips/${configuration_name}`, "method": "GET", }).done(function(res) { if (res.status === true) { available_ips = res.data; let $list_group = document.querySelector("#available_ip_modal .modal-body .list-group"); $list_group.innerHTML = ""; document.querySelector("#allowed_ips").value = available_ips[0]; available_ips.forEach((ip) => $list_group.innerHTML += `${ip}`); } else { document.querySelector("#allowed_ips").value = res.message; document.querySelector("#search_available_ip").setAttribute("disabled", "disabled"); } }); } function deleteConfiguration(){ } window.configurations = { addModal: () => { return addModal; }, deleteBulkModal: () => { return deleteBulkModal; }, deleteModal: () => { return deleteModal; }, configurationDeleteModal: () => { return configurationDeleteModal; }, ipModal: () => { return ipModal; }, qrcodeModal: () => { return qrcodeModal; }, settingModal: () => { return settingModal; }, configurationTimeout: () => { return configuration_timeout; }, updateDisplayMode: () => { display_mode = window.localStorage.getItem("displayMode"); }, removeConfigurationInterval: () => { removeConfigurationInterval(); }, loadPeers: (searchString) => { loadPeers(searchString); }, addPeersByBulk: () => { addPeersByBulk(); }, deletePeers: (config, peers_ids) => { deletePeers(config, peers_ids); }, deleteConfiguration: () => { deleteConfiguration() }, parsePeers: (response) => { parsePeers(response); }, toggleAccess: (peerID) => { toggleAccess(peerID); }, setConfigurationName: (confName) => { configuration_name = confName; }, getConfigurationName: () => { return configuration_name; }, setActiveConfigurationName: () => { setActiveConfigurationName(); }, getAvailableIps: () => { getAvailableIps(); }, generateKeyPair: () => { generate_key(); }, showToast: (message) => { showToast(message); }, updateRefreshInterval: (interval) => { updateRefreshInterval(interval); }, copyToClipboard: (element) => { copyToClipboard(element); }, toggleDeleteByBulkIP: (element) => { toggleBulkIP(element); }, downloadOneConfig: (conf) => { download_one_config(conf); }, triggerIp: (ip) => { trigger_ip(ip); }, cleanIp: (val) => { return cleanIp(val); }, startProgressBar: () => { startProgressBar(); }, stillLoadingProgressBar: () => { stillLoadingProgressBar(); }, endProgressBar: () => { endProgressBar(); } }; })(); let $body = $("body"); let available_ips = []; let $add_peer = document.getElementById("save_peer"); $("#configuration_delete").on("click", function(){ window.configurations.configurationDeleteModal().toggle(); }); function ajaxPostJSON(url, data, doneFunc){ $.ajax({ url: url, method: "POST", data: JSON.stringify(data), headers: {"Content-Type": "application/json"} }).done(function (res) { doneFunc(res); }); } $("#sure_delete_configuration").on("click", function () { window.configurations.removeConfigurationInterval(); let ele = $(this) ele.attr("disabled", "disabled"); function done(res){ if (res.status){ $('#configuration_delete_modal button[data-dismiss="modal"]').remove(); ele.text("Delete Successful! Redirecting in 5 seconds."); setTimeout(function(){ window.location.replace('/'); }, 5000) }else{ $("#remove_configuration_alert").removeClass("d-none").text(res.reason); } } ajaxPostJSON("/api/deleteConfiguration", {"name": window.configurations.getConfigurationName()}, done); }); /** * ========== * Add peers * ========== */ /** * Toggle add peers modal when add button clicked */ document.querySelector(".add_btn").addEventListener("click", () => { window.configurations.addModal().toggle(); }); /** * When configuration switch got click */ $(".toggle--switch").on("change", function(){ console.log('lol') $(this).addClass("waiting").attr("disabled", "disabled"); let id = window.configurations.getConfigurationName(); let status = $(this).prop("checked"); let ele = $(this); $.ajax({ url: `/switch/${id}` }).done(function(res){ if (res.status){ if (status){ window.configurations.showToast(`${id} is running.`) }else{ window.configurations.showToast(`${id} is stopped.`) } }else{ if (status){ ele.prop("checked", false) }else{ ele.prop("checked", true) } window.configurations.showToast(res.reason); $(".index-alert").removeClass("d-none").text(`Configuration toggle failed. Please check the following error message:\n${res.message}`); } ele.removeClass("waiting"); ele.removeAttr("disabled"); window.configurations.loadPeers($('#search_peer_textbox').val()) }); }); /** * Generate Public key when private got change */ document.querySelector("#private_key").addEventListener("change", (event) => { let publicKey = document.querySelector("#public_key"); if (event.target.value.length === 44) { publicKey.value = window.wireguard.generatePublicKey(event.target.value); publicKey.setAttribute("disabled", "disabled"); } else { publicKey.attributes.removeNamedItem("disabled"); publicKey.value = ""; } }); /** * Handle when add modal is show and hide */ $('#add_modal').on('show.bs.modal', function() { window.configurations.generateKeyPair(); window.configurations.getAvailableIps(); }).on('hide.bs.modal', function() { $("#allowed_ips_indicator").html(''); }); /** * Handle when user clicked the regenerate button */ $("#re_generate_key").on("click", function() { $("#public_key").attr("disabled", "disabled"); $("#re_generate_key i").addClass("rotating"); window.configurations.generateKeyPair(); }); /** * Handle when user is editing in allowed ips textbox */ $("#allowed_ips").on("keyup", function() { let s = window.configurations.cleanIp($(this).val()); s = s.split(","); if (available_ips.includes(s[s.length - 1])) { $("#allowed_ips_indicator").removeClass().addClass("text-success") .html(''); } else { $("#allowed_ips_indicator").removeClass().addClass("text-warning") .html(''); } }); /** * Change peer name when user typing in peer name textbox */ $("#peer_name_textbox").on("keyup", function() { $(".peer_name").html($(this).val()); }); /** * When Add Peer button got clicked */ $add_peer.addEventListener("click", function() { let $bulk_add = $("#bulk_add"); if ($bulk_add.prop("checked")) { if (!$("#new_add_amount").hasClass("is-invalid")) { window.configurations.addPeersByBulk(); } } else { let $public_key = $("#public_key"); let $private_key = $("#private_key"); let $allowed_ips = $("#allowed_ips"); $allowed_ips.val(window.configurations.cleanIp($allowed_ips.val())); let $new_add_DNS = $("#new_add_DNS"); $new_add_DNS.val(window.configurations.cleanIp($new_add_DNS.val())); let $new_add_endpoint_allowed_ip = $("#new_add_endpoint_allowed_ip"); $new_add_endpoint_allowed_ip.val(window.configurations.cleanIp($new_add_endpoint_allowed_ip.val())); let $new_add_name = $("#new_add_name"); let $new_add_MTU = $("#new_add_MTU"); let $new_add_keep_alive = $("#new_add_keep_alive"); let $enable_preshare_key = $("#enable_preshare_key"); $add_peer.setAttribute("disabled", "disabled"); $add_peer.innerHTML = "Adding..."; if ($allowed_ips.val() !== "" && $public_key.val() !== "" && $new_add_DNS.val() !== "" && $new_add_endpoint_allowed_ip.val() !== "") { let conf = window.configurations.getConfigurationName(); let data_list = [$private_key, $allowed_ips, $new_add_name, $new_add_DNS, $new_add_endpoint_allowed_ip, $new_add_MTU, $new_add_keep_alive]; data_list.forEach((ele) => ele.attr("disabled", "disabled")); $.ajax({ method: "POST", url: "/add_peer/" + conf, headers: { "Content-Type": "application/json" }, data: JSON.stringify({ "private_key": $private_key.val(), "public_key": $public_key.val(), "allowed_ips": $allowed_ips.val(), "name": $new_add_name.val(), "DNS": $new_add_DNS.val(), "endpoint_allowed_ip": $new_add_endpoint_allowed_ip.val(), "MTU": $new_add_MTU.val(), "keep_alive": $new_add_keep_alive.val(), "enable_preshared_key": $enable_preshare_key.prop("checked"), "preshared_key": $enable_preshare_key.val() }), success: function(response) { if (response !== "true") { $("#add_peer_alert").html(response).removeClass("d-none"); data_list.forEach((ele) => ele.removeAttr("disabled")); $add_peer.removeAttribute("disabled"); $add_peer.innerHTML = "Save"; } else { window.configurations.loadPeers(""); data_list.forEach((ele) => ele.removeAttr("disabled")); $("#add_peer_form").trigger("reset"); $add_peer.removeAttribute("disabled"); $add_peer.innerHTML = "Save"; window.configurations.showToast("Add peer successful!"); window.configurations.addModal().toggle(); } } }); } else { $("#add_peer_alert").html("Please fill in all required box.").removeClass("d-none"); $add_peer.removeAttribute("disabled"); $add_peer.innerHTML = "Add"; } } }); /** * Handle when user is typing the amount of peers they want to add, and will check if the amount is less than 1 or * is larger than the amount of available ips */ $("#new_add_amount").on("keyup", function() { let $bulk_amount_validation = $("#bulk_amount_validation"); // $(this).removeClass("is-valid").addClass("is-invalid"); if ($(this).val().length > 0) { if (isNaN($(this).val())) { $(this).removeClass("is-valid").addClass("is-invalid"); $bulk_amount_validation.html("Please enter a valid integer"); } else if ($(this).val() > available_ips.length) { $(this).removeClass("is-valid").addClass("is-invalid"); $bulk_amount_validation.html(`Cannot create more than ${available_ips.length} peers.`); } else if ($(this).val() < 1) { $(this).removeClass("is-valid").addClass("is-invalid"); $bulk_amount_validation.html("Please enter at least 1 or more."); } else { $(this).removeClass("is-invalid").addClass("is-valid"); } } else { $(this).removeClass("is-invalid").removeClass("is-valid"); } }); /** * Handle when user toggled add peers by bulk */ $("#bulk_add").on("change", function() { let hide = $(".non-bulk"); let amount = $("#new_add_amount"); if ($(this).prop("checked") === true) { for (let i = 0; i < hide.length; i++) { $(hide[i]).attr("disabled", "disabled"); } amount.removeAttr("disabled"); } else { for (let i = 0; i < hide.length; i++) { if ($(hide[i]).attr('id') !== "public_key") { $(hide[i]).removeAttr("disabled"); } } amount.attr("disabled", "disabled"); } }); /** * ======================= * Available IP Related * ======================= */ /** * Handle when available ip modal show and hide */ $("#available_ip_modal").on("show.bs.modal", () => { document.querySelector('#add_modal').classList.add("ip_modal_open"); }).on("hidden.bs.modal", () => { document.querySelector('#add_modal').classList.remove("ip_modal_open"); let ips = []; let $selected_ip_list = document.querySelector("#selected_ip_list"); for (let i = 0; i < $selected_ip_list.childElementCount; i++) { ips.push($selected_ip_list.children[i].dataset.ip); } ips.forEach((ele) => window.configurations.triggerIp(ele)); }); /** * When IP Badge got click */ $body.on("click", ".available-ip-badge", function() { $(".available-ip-item[data-ip='" + $(this).data("ip") + "']").removeClass("active"); $(this).remove(); }); /** * When available ip item got click */ $body.on("click", ".available-ip-item", function() { window.configurations.triggerIp($(this).data("ip")); }); /** * When search IP button got clicked */ $("#search_available_ip").on("click", function() { window.configurations.ipModal().toggle(); let $allowed_ips = document.querySelector("#allowed_ips"); if ($allowed_ips.value.length > 0) { let s = $allowed_ips.value.split(","); for (let i = 0; i < s.length; i++) { s[i] = s[i].trim(); window.configurations.triggerIp(s[i]); } } }).tooltip(); /** * When confirm IP is clicked */ $("#confirm_ip").on("click", () => { window.configurations.ipModal().toggle(); let ips = []; let $selected_ip_list = $("#selected_ip_list"); $selected_ip_list.children().each(function() { ips.push($(this).data("ip")); }); $("#allowed_ips").val(ips.join(", ")); ips.forEach((ele) => window.configurations.triggerIp(ele)); }); /** * ======= * QR Code * ======= */ /** * When the QR-code button got clicked on each peer */ $body.on("click", ".btn-qrcode-peer", function() { let src = $(this).data('imgsrc'); $.ajax({ "url": src, "method": "GET" }).done(function(res) { $("#qrcode_img").attr('src', res); window.configurations.qrcodeModal().toggle(); }); }); /** * =========== * Delete Peer * =========== */ /** * When the delete button got clicked on each peer */ $body.on("click", ".btn-delete-peer", function() { let peer_id = $(this).data('peer-id') $("#delete_peer").data("peer-id", peer_id); window.configurations.deleteModal().toggle(); }); $body.on("click", ".btn-lock-peer", function() { window.configurations.toggleAccess($(this).data('peer-id'), window.configurations.getConfigurationName()); if ($(this).hasClass("lock")) { console.log($(this).data("peer-name")) window.configurations.showToast(`Enabled ${$(this).children().data("peer-name")}`) $(this).removeClass("lock") $(this).children().tooltip('hide').attr('data-original-title', 'Peer enabled. Click to disable peer.').tooltip('show'); } else { // Currently unlocked window.configurations.showToast(`Disabled ${$(this).children().data("peer-name")}`) $(this).addClass("lock"); $(this).children().tooltip('hide').attr('data-original-title', 'Peer disabled. Click to enable peer.').tooltip('show'); } }); /** * When the confirm delete button clicked */ $("#delete_peer").on("click", function() { $(this).attr("disabled", "disabled"); $(this).html("Deleting..."); let config = window.configurations.getConfigurationName(); let peer_ids = [$(this).data("peer-id")]; window.configurations.deletePeers(config, peer_ids); }); /** * ============= * Peer Settings * ============= */ /** * Handle when setting button got clicked for each peer */ $body.on("click", ".btn-setting-peer", function() { // window.configurations.startProgressBar(); let peer_id = $(this).data("peer-id"); $("#save_peer_setting").attr("peer_id", peer_id); $.ajax({ method: "POST", url: "/get_peer_data/" + window.configurations.getConfigurationName(), headers: { "Content-Type": "application/json" }, data: JSON.stringify({ "id": peer_id }), success: function(response) { let peer_name = ((response.name === "") ? "Untitled" : response.name); $("#setting_modal .peer_name").html(peer_name); $("#setting_modal #peer_name_textbox").val(response.name); $("#setting_modal #peer_private_key_textbox").val(response.private_key); $("#setting_modal #peer_DNS_textbox").val(response.DNS); $("#setting_modal #peer_allowed_ip_textbox").val(response.allowed_ip); $("#setting_modal #peer_endpoint_allowed_ips").val(response.endpoint_allowed_ip); $("#setting_modal #peer_mtu").val(response.mtu); $("#setting_modal #peer_keep_alive").val(response.keep_alive); $("#setting_modal #peer_preshared_key_textbox").val(response.preshared_key); window.configurations.settingModal().toggle(); window.configurations.endProgressBar(); } }); }); /** * Handle when setting modal is closing */ $('#setting_modal').on('hidden.bs.modal', function() { $("#setting_peer_alert").addClass("d-none"); }); /** * Handle when private key text box in setting modal got changed */ $("#peer_private_key_textbox").on("change", function() { let $save_peer_setting = $("#save_peer_setting"); if ($(this).val().length > 0) { $.ajax({ "url": "/check_key_match/" + window.configurations.getConfigurationName(), "method": "POST", "headers": { "Content-Type": "application/json" }, "data": JSON.stringify({ "private_key": $("#peer_private_key_textbox").val(), "public_key": $save_peer_setting.attr("peer_id") }) }).done(function(res) { if (res.status === "failed") { $("#setting_peer_alert").html(res.status).removeClass("d-none"); } else { $("#setting_peer_alert").addClass("d-none"); } }); } }); /** * When save peer setting button got clicked */ $("#save_peer_setting").on("click", function() { $(this).attr("disabled", "disabled"); $(this).html("Saving..."); let $peer_DNS_textbox = $("#peer_DNS_textbox"); let $peer_allowed_ip_textbox = $("#peer_allowed_ip_textbox"); let $peer_endpoint_allowed_ips = $("#peer_endpoint_allowed_ips"); let $peer_name_textbox = $("#peer_name_textbox"); let $peer_private_key_textbox = $("#peer_private_key_textbox"); let $peer_preshared_key_textbox = $("#peer_preshared_key_textbox"); let $peer_mtu = $("#peer_mtu"); let $peer_keep_alive = $("#peer_keep_alive"); if ($peer_DNS_textbox.val() !== "" && $peer_allowed_ip_textbox.val() !== "" && $peer_endpoint_allowed_ips.val() !== "") { let peer_id = $(this).attr("peer_id"); let conf_id = $(this).attr("conf_id"); let data_list = [$peer_name_textbox, $peer_DNS_textbox, $peer_private_key_textbox, $peer_preshared_key_textbox, $peer_allowed_ip_textbox, $peer_endpoint_allowed_ips, $peer_mtu, $peer_keep_alive]; data_list.forEach((ele) => ele.attr("disabled", "disabled")); $.ajax({ method: "POST", url: "/save_peer_setting/" + conf_id, headers: { "Content-Type": "application/json" }, data: JSON.stringify({ id: peer_id, name: $peer_name_textbox.val(), DNS: $peer_DNS_textbox.val(), private_key: $peer_private_key_textbox.val(), allowed_ip: $peer_allowed_ip_textbox.val(), endpoint_allowed_ip: $peer_endpoint_allowed_ips.val(), MTU: $peer_mtu.val(), keep_alive: $peer_keep_alive.val(), preshared_key: $peer_preshared_key_textbox.val() }), success: function(response) { if (response.status === "failed") { $("#setting_peer_alert").html(response.msg).removeClass("d-none"); } else { window.configurations.settingModal().toggle(); window.configurations.loadPeers($('#search_peer_textbox').val()); $('#alertToast').toast('show'); $('#alertToast .toast-body').html("Peer Saved!"); } $("#save_peer_setting").removeAttr("disabled").html("Save"); data_list.forEach((ele) => ele.removeAttr("disabled")); } }); } else { $("#setting_peer_alert").html("Please fill in all required box.").removeClass("d-none"); $("#save_peer_setting").removeAttr("disabled").html("Save"); } }); /** * Toggle show or hide for the private key textbox in the setting modal */ $(".peer_private_key_textbox_switch").on("click", function() { let $peer_private_key_textbox = $("#peer_private_key_textbox"); let mode = (($peer_private_key_textbox.attr('type') === 'password') ? "text" : "password"); let icon = (($peer_private_key_textbox.attr('type') === 'password') ? "bi bi-eye-slash-fill" : "bi bi-eye-fill"); $peer_private_key_textbox.attr('type', mode); $(".peer_private_key_textbox_switch i").removeClass().addClass(icon); }); /** * =========== * Search Peer * =========== */ let typingTimer; // Timeout object let doneTypingInterval = 200; // Timeout interval /** * Handle when the user keyup and keydown on the search textbox */ $('#search_peer_textbox').on('keyup', function() { clearTimeout(typingTimer); typingTimer = setTimeout(() => { window.configurations.loadPeers($(this).val()); }, doneTypingInterval); }).on('keydown', function() { clearTimeout(typingTimer); }); /** * Manage Peers */ /** * Handle when sort peers changed */ $body.on("change", "#sort_by_dropdown", function() { $.ajax({ method: "POST", data: JSON.stringify({ 'sort': $("#sort_by_dropdown option:selected").val() }), headers: { "Content-Type": "application/json" }, url: "/update_dashboard_sort", success: function() { window.configurations.loadPeers($('#search_peer_textbox').val()); } }); }); /** * Handle copy public key */ $body.on("mouseenter", ".key", function() { let label = $(this).parent().siblings().children()[1]; label.style.opacity = "100"; }).on("mouseout", ".key", function() { let label = $(this).parent().siblings().children()[1]; label.style.opacity = "0"; setTimeout(function() { label.innerHTML = "CLICK TO COPY"; }, 200); }).on("click", ".key", function() { let label = $(this).parent().siblings().children()[1]; window.configurations.copyToClipboard($(this)); label.innerHTML = "COPIED!"; }); /** * Handle when interval button got clicked */ $body.on("click", ".update_interval", function() { $(".interval-btn-group button").removeClass("active"); let _new = $(this); _new.addClass("active"); let interval = $(this).data("refresh-interval"); if ([5000, 10000, 30000, 60000].includes(interval)) { window.configurations.updateRefreshInterval(interval); } // $.ajax({ // method:"POST", // data: "interval="+$(this).data("refresh-interval"), // url: "/update_dashboard_refresh_interval", // success: function (res){ // window.configurations.updateRefreshInterval(res, interval); // } // }); }); /** * Handle when refresh button got clicked */ $body.on("click", ".refresh", function() { window.configurations.loadPeers($('#search_peer_textbox').val()); }); /** * Handle when display mode button got clicked */ $body.on("click", ".display_mode", function() { $(".display-btn-group button").removeClass("active"); $(this).addClass("active"); window.localStorage.setItem("displayMode", $(this).data("display-mode")); window.configurations.updateDisplayMode(); if ($(this).data("display-mode") === "list") { Array($(".peer_list").children()).forEach(function(child) { $(child).removeClass().addClass("col-12"); }); window.configurations.showToast("Displaying as List"); } else { Array($(".peer_list").children()).forEach(function(child) { $(child).removeClass().addClass("col-sm-6 col-lg-4"); }); window.configurations.showToast("Displaying as Grids"); } }); /** * ================= * Configuration Menu * ================= */ let $setting_btn_menu = $(".setting_btn_menu"); $setting_btn_menu.css("top", ($setting_btn_menu.height() + 54) * (-1)); let $setting_btn = $(".setting_btn"); /** * When the menu button got clicked */ $setting_btn.on("click", function() { if ($setting_btn_menu.hasClass("show")) { $setting_btn_menu.removeClass("showing"); setTimeout(function() { $setting_btn_menu.removeClass("show"); }, 201); } else { $setting_btn_menu.addClass("show"); setTimeout(function() { $setting_btn_menu.addClass("showing"); }, 10); } }); /** * Whenever the user clicked, if it is outside the menu and the menu is opened, hide the menu */ $("html").on("click", function(r) { if (document.querySelector(".setting_btn") !== r.target) { if (!document.querySelector(".setting_btn").contains(r.target)) { if (!document.querySelector(".setting_btn_menu").contains(r.target)) { $setting_btn_menu.removeClass("showing"); setTimeout(function() { $setting_btn_menu.removeClass("show"); }, 310); } } } }); /** * ==================== * Delete Peers by Bulk * ==================== */ /** * When delete peers by bulk clicked */ $("#delete_peers_by_bulk_btn").on("click", () => { let $delete_bulk_modal_list = $("#delete_bulk_modal .list-group"); $delete_bulk_modal_list.html(''); peers.forEach((peer) => { let name; if (peer.name === "") { name = "Untitled Peer"; } else { name = peer.name; } $delete_bulk_modal_list.append('' + name + '
' + peer.id + '
'); }); window.configurations.deleteBulkModal().toggle(); }); /** * When the item or tag of delete peers by bulk got clicked */ $body.on("click", ".delete-bulk-peer-item", function() { window.configurations.toggleDeleteByBulkIP($(this)); }).on("click", ".delete-peer-bulk-badge", function() { window.configurations.toggleDeleteByBulkIP($(".delete-bulk-peer-item[data-id='" + $(this).data("id") + "']")); }); let $selected_peer_list = document.getElementById("selected_peer_list"); /** * The change observer to observe when user choose 1 or more peers to delete * @type {MutationObserver} */ let changeObserver = new MutationObserver(function() { if ($selected_peer_list.hasChildNodes()) { $("#confirm_delete_bulk_peers").removeAttr("disabled"); } else { $("#confirm_delete_bulk_peers").attr("disabled", "disabled"); } }); changeObserver.observe($selected_peer_list, { attributes: true, childList: true, characterData: true }); let confirm_delete_bulk_peers_interval; /** * When the user clicked the delete button in the delete peers by bulk */ $("#confirm_delete_bulk_peers").on("click", function() { let btn = $(this); if (confirm_delete_bulk_peers_interval !== undefined) { clearInterval(confirm_delete_bulk_peers_interval); confirm_delete_bulk_peers_interval = undefined; btn.html("Delete"); } else { let timer = 5; btn.html(`Deleting in ${timer} secs... Click to cancel`); confirm_delete_bulk_peers_interval = setInterval(function() { timer -= 1; btn.html(`Deleting in ${timer} secs... Click to cancel`); if (timer === 0) { btn.html(`Deleting...`); btn.attr("disabled", "disabled"); let ips = []; $selected_peer_list.childNodes.forEach((ele) => ips.push(ele.dataset.id)); window.configurations.deletePeers(window.configurations.getConfigurationName(), ips); clearInterval(confirm_delete_bulk_peers_interval); confirm_delete_bulk_peers_interval = undefined; } }, 1000); } }); /** * Select all peers to delete */ $("#select_all_delete_bulk_peers").on("click", function() { $(".delete-bulk-peer-item").each(function() { if (!$(this).hasClass("active")) { window.configurations.toggleDeleteByBulkIP($(this)); } }); }); /** * When delete peers by bulk window is hidden */ $(window.configurations.deleteBulkModal()._element).on("hidden.bs.modal", function() { $(".delete-bulk-peer-item").each(function() { if ($(this).hasClass("active")) { window.configurations.toggleDeleteByBulkIP($(this)); } }); }); /** * ============== * Download Peers * ============== */ /** * When the download peers button got clicked */ $body.on("click", ".btn-download-peer", function(e) { e.preventDefault(); let link = $(this).attr("href"); $.ajax({ "url": link, "method": "GET", success: function(res) { window.configurations.downloadOneConfig(res); } }); }); /** * When the download all peers got clicked */ $("#download_all_peers").on("click", function() { $.ajax({ "url": `/download_all/${window.configurations.getConfigurationName()}`, "method": "GET", success: function(res) { if (res.peers.length > 0) { window.wireguard.generateZipFiles(res); window.configurations.showToast("Peers' zip file download successful!"); } else { window.configurations.showToast("Oops! There are no peer can be download."); } } }); });