mirror of
https://github.com/swissmakers/wireguard-manager.git
synced 2026-03-30 07:47:07 +02:00
745 lines
33 KiB
HTML
745 lines
33 KiB
HTML
{{define "base.html"}}
|
|
<!DOCTYPE html>
|
|
<html lang="en">
|
|
|
|
<head>
|
|
<meta charset="utf-8">
|
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
|
<title>{{template "title" .}}</title>
|
|
<!-- Tell the browser to be responsive to screen width -->
|
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
<!-- Favicon -->
|
|
<link rel="icon" href="{{.basePath}}/favicon">
|
|
|
|
<!-- Font Awesome -->
|
|
<link rel="stylesheet" href="{{.basePath}}/static/plugins/fontawesome-free/css/all.min.css">
|
|
<!-- iCheck for checkboxes and radio inputs -->
|
|
<link rel="stylesheet" href="{{.basePath}}/static/plugins/icheck-bootstrap/icheck-bootstrap.min.css">
|
|
<!-- Select2 -->
|
|
<link rel="stylesheet" href="{{.basePath}}/static/plugins/select2/css/select2.min.css">
|
|
<!-- Toastr -->
|
|
<link rel="stylesheet" href="{{.basePath}}/static/plugins/toastr/toastr.min.css">
|
|
<!-- Jquery Tags Input -->
|
|
<link rel="stylesheet" href="{{.basePath}}/static/plugins/jquery-tags-input/dist/jquery.tagsinput.min.css">
|
|
<!-- overlayScrollbars -->
|
|
<link rel="stylesheet" href="{{.basePath}}/static/dist/css/adminlte.min.css">
|
|
|
|
<!-- START: On page css -->
|
|
{{template "top_css" .}}
|
|
<!-- END: On page css -->
|
|
<style>
|
|
/* Base Dark Mode Styles */
|
|
body, .content-wrapper {
|
|
background-color: #121212;
|
|
color: #e0e0e0;
|
|
}
|
|
.navbar, .main-sidebar, .sidebar {
|
|
background-color: #1e1e1e;
|
|
}
|
|
.nav-link, .navbar-nav .nav-item .nav-link {
|
|
color: #b0b0b0;
|
|
}
|
|
.navbar-light .navbar-nav .nav-link {
|
|
color: #b0b0b0;
|
|
}
|
|
.navbar-light .navbar-nav .nav-link:hover {
|
|
color: #ffffff;
|
|
}
|
|
.main-footer {
|
|
background-color: #1c1c1c;
|
|
color: #e0e0e0;
|
|
}
|
|
.sidebar-dark-primary .nav-link {
|
|
color: #e0e0e0;
|
|
}
|
|
.sidebar-dark-primary .nav-link:hover {
|
|
background-color: #444444;
|
|
}
|
|
/* Dark mode for buttons */
|
|
.btn-outline-primary {
|
|
border-color: #4e73df;
|
|
color: #4e73df;
|
|
}
|
|
.btn-outline-primary:hover {
|
|
background-color: #4e73df;
|
|
color: #ffffff;
|
|
}
|
|
.btn-outline-danger {
|
|
border-color: #e74a3b;
|
|
color: #e74a3b;
|
|
}
|
|
.btn-outline-danger:hover {
|
|
background-color: #e74a3b;
|
|
color: #ffffff;
|
|
}
|
|
/* Modify inputs and form elements */
|
|
input, select, textarea, .form-control, .form-control:disabled, div.tagsinput {
|
|
background-color: #333333 !important;
|
|
color: #e0e0e0 !important;
|
|
border: 1px solid #555 !important;
|
|
}
|
|
input::placeholder, select::placeholder, textarea::placeholder, #search-input::placeholder {
|
|
color: #b0b0b0;
|
|
}
|
|
.custom-select {
|
|
background: #fff url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23dee2e6' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") right .75rem center/8px 10px no-repeat;
|
|
}
|
|
input[type="checkbox"], input[type="radio"] {
|
|
background-color: #444;
|
|
}
|
|
.form-control:focus {
|
|
background-color: #555555;
|
|
border-color: #80bdff;
|
|
}
|
|
.table-success td {
|
|
border-color: #8fd19e;
|
|
}
|
|
.table-success > td, .table-success>th {
|
|
background-color: #00430f;
|
|
}
|
|
div.tagsinput span.tag {
|
|
border: 1px solid #22923c;
|
|
background: #181b16;
|
|
color: #35d85b;
|
|
}
|
|
div.tagsinput span.tag a {
|
|
color: #35d85b;
|
|
}
|
|
.modal-content, .card, table, .select2-dropdown {
|
|
background-color: #2a2a2a;
|
|
color: #e0e0e0;
|
|
}
|
|
.modal-header {
|
|
border-bottom: 1px solid #555;
|
|
}
|
|
.modal-footer {
|
|
border-top: 1px solid #555;
|
|
}
|
|
/* Dark mode for the sidebar active state */
|
|
.nav-sidebar .nav-link.active {
|
|
background-color: #444;
|
|
}
|
|
table th, table td {
|
|
color: #e0e0e0;
|
|
border: 1px solid #444;
|
|
}
|
|
</style>
|
|
</head>
|
|
|
|
<body class="hold-transition sidebar-mini">
|
|
<!-- Site wrapper -->
|
|
<div class="wrapper">
|
|
<!-- Navbar -->
|
|
<nav class="main-header navbar navbar-expand navbar-white navbar-light">
|
|
<!-- Left navbar links -->
|
|
<ul class="navbar-nav">
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-widget="pushmenu" href="#" role="button"><i class="fas fa-bars"></i></a>
|
|
</li>
|
|
</ul>
|
|
|
|
<!-- SEARCH FORM -->
|
|
<form class="form-inline ml-3" style="display: none" id="search-form">
|
|
<div class="input-group input-group-sm">
|
|
<input class="form-control form-control-navbar" placeholder="Search"
|
|
aria-label="Search" id="search-input">
|
|
<div class="input-group-append">
|
|
<button class="btn-navbar" type="submit" disabled>
|
|
<i class="fas fa-search"></i>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<div class="form-group form-group-sm">
|
|
<select name="status-selector" id="status-selector" class="custom-select form-control-navbar" style="margin-left: 0.5em; height: 90%; font-size: 14px;">
|
|
<!-- THIS SECTION IS OVERRIDDEN BY JS. SEE updateSearchList() function in clients.html BEFORE EDITING -->
|
|
<option value="All">All</option>
|
|
<option value="Enabled">Enabled</option>
|
|
<option value="Disabled">Disabled</option>
|
|
<option value="Connected">Connected</option>
|
|
<option value="Disconnected">Disconnected</option>
|
|
<!-- THIS SECTION IS OVERRIDDEN BY JS. SEE updateSearchList() function in clients.html BEFORE EDITING -->
|
|
</select>
|
|
</div>
|
|
</form>
|
|
|
|
<!-- Right navbar links -->
|
|
<div class="navbar-nav ml-auto">
|
|
<button style="margin-left: 0.5em;" type="button" class="btn btn-outline-primary btn-sm" data-toggle="modal"
|
|
data-target="#modal_new_client"><i class="nav-icon fas fa-plus"></i> New Client</button>
|
|
<button id="apply-config-button" style="margin-left: 0.5em; display: none;" type="button" class="btn btn-outline-danger btn-sm" data-toggle="modal"
|
|
data-target="#modal_apply_config"><i class="nav-icon fas fa-check"></i> Apply Config</button>
|
|
{{if .baseData.CurrentUser}}
|
|
<button onclick="location.href='{{.basePath}}/logout';" style="margin-left: 0.5em;" type="button"
|
|
class="btn btn-outline-danger btn-sm"><i class="nav-icon fas fa-sign-out-alt"></i> Logout</button>
|
|
{{end}}
|
|
</div>
|
|
</nav>
|
|
<!-- /.navbar -->
|
|
|
|
<!-- Main Sidebar Container -->
|
|
<aside class="main-sidebar sidebar-dark-primary elevation-4">
|
|
<!-- Brand Logo -->
|
|
<a href="{{.basePath}}" class="brand-link">
|
|
<span class="brand-text"> WIREGUARD MANAGER</span>
|
|
</a>
|
|
|
|
<!-- Sidebar -->
|
|
<div class="sidebar">
|
|
<!-- Sidebar user (optional) -->
|
|
<div class="user-panel mt-3 pb-3 mb-3 d-flex">
|
|
<div class="image">
|
|
<i class="nav-icon fas fa-2x fa-user"></i>
|
|
</div>
|
|
<div class="info">
|
|
{{if .baseData.CurrentUser}}
|
|
<a href="{{.basePath}}/profile" class="d-block">My Account: {{.baseData.CurrentUser}}</a>
|
|
{{else}}
|
|
<a href="#" class="d-block">My Account</a>
|
|
{{end}}
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Sidebar Menu -->
|
|
<nav class="mt-2">
|
|
<ul class="nav nav-pills nav-sidebar flex-column" data-widget="treeview" role="menu" data-accordion="false">
|
|
<li class="nav-header">MAIN</li>
|
|
<li class="nav-item">
|
|
<a href="{{.basePath}}/" class="nav-link {{if eq .baseData.Active ""}}active{{end}}">
|
|
<i class="nav-icon fas fa-user-secret"></i>
|
|
<p>VPN Clients</p>
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a href="{{.basePath}}/status" class="nav-link {{if eq .baseData.Active "status" }}active{{end}}">
|
|
<i class="nav-icon fas fa-signal"></i>
|
|
<p>VPN Connected</p>
|
|
</a>
|
|
</li>
|
|
{{if .baseData.Admin}}
|
|
<li class="nav-header">SETTINGS</li>
|
|
<li class="nav-item">
|
|
<a href="{{.basePath}}/wg-server" class="nav-link {{if eq .baseData.Active "wg-server" }}active{{end}}">
|
|
<i class="nav-icon fas fa-server"></i>
|
|
<p>WireGuard Server</p>
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a href="{{.basePath}}/global-settings" class="nav-link {{if eq .baseData.Active "global-settings" }}active{{end}}">
|
|
<i class="nav-icon fas fa-cog"></i>
|
|
<p>Client Config Settings</p>
|
|
</a>
|
|
</li>
|
|
{{if not .loginDisabled}}
|
|
<li class="nav-item">
|
|
<a href="{{.basePath}}/users-settings" class="nav-link {{if eq .baseData.Active "users-settings" }}active{{end}}">
|
|
<i class="nav-icon fas fa-cog"></i>
|
|
<p>WGM User Accounts</p>
|
|
</a>
|
|
</li>
|
|
{{end}}
|
|
{{end}}
|
|
</ul>
|
|
</nav>
|
|
<!-- /.sidebar-menu -->
|
|
</div>
|
|
<!-- /.sidebar -->
|
|
</aside>
|
|
|
|
<div class="modal fade" id="modal_new_client">
|
|
<div class="modal-dialog">
|
|
<div class="modal-content">
|
|
<div class="modal-header">
|
|
<h4 class="modal-title">New WireGuard Client</h4>
|
|
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
|
|
<span aria-hidden="true">×</span>
|
|
</button>
|
|
</div>
|
|
<form name="frm_new_client" id="frm_new_client">
|
|
<div class="modal-body">
|
|
<!-- Form fields for new client go here -->
|
|
<div class="form-group">
|
|
<label for="client_name" class="control-label">Name</label>
|
|
<input type="text" class="form-control" id="client_name" name="client_name">
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="client_email" class="control-label">Email</label>
|
|
<input type="text" class="form-control" id="client_email" name="client_email">
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="subnet_ranges" class="control-label">Subnet range</label>
|
|
<select id="subnet_ranges" class="select2" data-placeholder="Select a subnet range" style="width: 100%;">
|
|
</select>
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="client_allocated_ips" class="control-label">IP Allocation</label>
|
|
<input type="text" data-role="tagsinput" class="form-control" id="client_allocated_ips">
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="client_allowed_ips" class="control-label">Allowed IPs - What subnet traffic should go through WireGuard.
|
|
<i class="fas fa-info-circle" data-toggle="tooltip" data-original-title="Specify a list of addresses that will get routed to the server. These addresses will be included in 'AllowedIPs' of client config">
|
|
</i>
|
|
</label>
|
|
<input type="text" data-role="tagsinput" class="form-control" id="client_allowed_ips" value="{{ StringsJoin .client_defaults.AllowedIPs "," }}">
|
|
</div>
|
|
<div class="form-group" style="display:none;">
|
|
<label for="client_extra_allowed_ips" class="control-label">Extra Allowed IPs
|
|
<i class="fas fa-info-circle" data-toggle="tooltip" data-original-title="Specify a list of addresses that will get routed to the client. These addresses will be included in 'AllowedIPs' of WG server config">
|
|
</i>
|
|
</label>
|
|
<input type="text" data-role="tagsinput" class="form-control" id="client_extra_allowed_ips" value="{{ StringsJoin .client_defaults.ExtraAllowedIPs "," }}">
|
|
</div>
|
|
<div class="form-group" style="display:none;">
|
|
<label for="client_endpoint" class="control-label">Endpoint</label>
|
|
<input type="text" class="form-control" id="client_endpoint" name="client_endpoint">
|
|
</div>
|
|
<div class="form-group">
|
|
<div class="icheck-primary d-inline">
|
|
<input type="checkbox" id="use_server_dns" {{ if .client_defaults.UseServerDNS }}checked{{ end }}>
|
|
<label for="use_server_dns">Use server DNS</label>
|
|
</div>
|
|
</div>
|
|
<div class="form-group">
|
|
<div class="icheck-primary d-inline">
|
|
<input type="checkbox" id="enabled" {{ if .client_defaults.EnableAfterCreation }}checked{{ end }}>
|
|
<label for="enabled">Enable after creation</label>
|
|
</div>
|
|
</div>
|
|
<details>
|
|
<summary>
|
|
<strong>Public and Preshared Keys</strong>
|
|
<i class="fas fa-info-circle" data-toggle="tooltip" data-original-title="If you don't want the server to generate and store the client's private key, you can manually specify its public and preshared key here. Note: QR code will not be generated">
|
|
</i>
|
|
</summary>
|
|
<div class="form-group" style="margin-top: 1rem">
|
|
<label for="client_public_key" class="control-label">Public Key</label>
|
|
<input type="text" class="form-control" id="client_public_key" name="client_public_key" placeholder="Autogenerated" aria-invalid="false">
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="client_preshared_key" class="control-label">Preshared Key</label>
|
|
<input type="text" class="form-control" id="client_preshared_key" name="client_preshared_key" placeholder="Autogenerated - enter "-" to skip generation">
|
|
</div>
|
|
</details>
|
|
<details style="margin-top: 0.5rem;">
|
|
<summary><strong>Additional configuration</strong></summary>
|
|
<div class="form-group">
|
|
<label for="additional_notes" class="control-label">Notes</label>
|
|
<textarea class="form-control" style="min-height: 6rem;" id="additional_notes" name="additional_notes" placeholder="Additional notes about this client"></textarea>
|
|
</div>
|
|
</details>
|
|
</div>
|
|
<div class="modal-footer justify-content-between">
|
|
<button type="button" class="btn btn-default" data-dismiss="modal">Cancel</button>
|
|
<button type="submit" class="btn btn-primary">Submit</button>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
<!-- /.modal-content -->
|
|
</div>
|
|
<!-- /.modal-dialog -->
|
|
</div>
|
|
<!-- /.modal -->
|
|
|
|
<div class="modal fade" id="modal_apply_config">
|
|
<div class="modal-dialog">
|
|
<div class="modal-content">
|
|
<div class="modal-header">
|
|
<h4 class="modal-title">Apply Config</h4>
|
|
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
|
|
<span aria-hidden="true">×</span>
|
|
</button>
|
|
</div>
|
|
<div class="modal-body">
|
|
<p>Do you want to write config file and restart WireGuard server?</p>
|
|
</div>
|
|
<div class="modal-footer justify-content-between">
|
|
<button type="button" class="btn btn-default" data-dismiss="modal">Cancel</button>
|
|
<button type="button" class="btn btn-danger" id="apply_config_confirm">Apply</button>
|
|
</div>
|
|
</div>
|
|
<!-- /.modal-content -->
|
|
</div>
|
|
<!-- /.modal-dialog -->
|
|
</div>
|
|
<!-- /.modal -->
|
|
|
|
<!-- Content Wrapper. Contains page content -->
|
|
<div class="content-wrapper">
|
|
<!-- Content Header (Page header) -->
|
|
<section class="content-header">
|
|
<div class="container-fluid">
|
|
<div class="row mb-2">
|
|
<div class="col-sm-6">
|
|
<h1>{{template "page_title" .}}</h1>
|
|
</div>
|
|
</div>
|
|
</div><!-- /.container-fluid -->
|
|
</section>
|
|
|
|
<!-- Main content -->
|
|
{{template "page_content" .}}
|
|
<!-- /.content -->
|
|
</div>
|
|
<!-- /.content-wrapper -->
|
|
|
|
<footer class="main-footer">
|
|
<div class="float-right d-none d-sm-block">
|
|
<b>Version</b> {{ .appVersion }}
|
|
</div>
|
|
<strong>Copyright © <script>document.write(new Date().getFullYear())</script> <a href="https://github.com/swissmakers/wireguard-manager">WireGuard Manager</a>.</strong> All rights reserved.
|
|
</footer>
|
|
|
|
<!-- Control Sidebar -->
|
|
<aside class="control-sidebar control-sidebar-dark">
|
|
<!-- Control sidebar content goes here -->
|
|
</aside>
|
|
<!-- /.control-sidebar -->
|
|
</div>
|
|
<!-- ./wrapper -->
|
|
|
|
<!-- jQuery -->
|
|
<script src="{{.basePath}}/static/plugins/jquery/jquery.min.js"></script>
|
|
<!-- Bootstrap 4 -->
|
|
<script src="{{.basePath}}/static/plugins/bootstrap/js/bootstrap.bundle.min.js"></script>
|
|
<!-- Select2 -->
|
|
<script src="{{.basePath}}/static/plugins/select2/js/select2.full.min.js"></script>
|
|
<!-- jquery-validation -->
|
|
<script src="{{.basePath}}/static/plugins/jquery-validation/jquery.validate.min.js"></script>
|
|
<!-- Toastr -->
|
|
<script src="{{.basePath}}/static/plugins/toastr/toastr.min.js"></script>
|
|
<!-- Jquery Tags Input -->
|
|
<script src="{{.basePath}}/static/plugins/jquery-tags-input/dist/jquery.tagsinput.min.js"></script>
|
|
<!-- AdminLTE App -->
|
|
<script src="{{.basePath}}/static/dist/js/adminlte.min.js"></script>
|
|
<!-- Custom js -->
|
|
<script src="{{.basePath}}/static/custom/js/helper.js"></script>
|
|
<script>
|
|
// initialize all tooltips
|
|
$(function () {
|
|
$('[data-toggle="tooltip"]').tooltip()
|
|
})
|
|
|
|
$(document).ready(function () {
|
|
|
|
addGlobalStyle(`
|
|
.toast-top-right-fix {
|
|
top: 67px;
|
|
right: 12px;
|
|
}
|
|
`, 'toastrToastStyleFix')
|
|
|
|
toastr.options.closeDuration = 100;
|
|
toastr.options.positionClass = 'toast-top-right-fix';
|
|
|
|
// Initial call, and then poll every 5 seconds for config changes
|
|
updateApplyConfigVisibility();
|
|
setInterval(updateApplyConfigVisibility, 5000);
|
|
|
|
// Only poll for updates if the current page is the status page.
|
|
if (window.location.pathname === "{{.basePath}}/status") {
|
|
updateStatusTable();
|
|
setInterval(updateStatusTable, 5000);
|
|
}
|
|
});
|
|
|
|
function addGlobalStyle(css, id) {
|
|
if (!document.querySelector('#' + id)) {
|
|
let head = document.head;
|
|
if (!head) { return; }
|
|
let style = document.createElement('style');
|
|
style.type = 'text/css';
|
|
style.id = id;
|
|
style.innerHTML = css;
|
|
head.appendChild(style);
|
|
}
|
|
}
|
|
|
|
function updateApplyConfigVisibility() {
|
|
$.ajax({
|
|
cache: false,
|
|
method: 'GET',
|
|
url: '{{.basePath}}/test-hash',
|
|
dataType: 'json',
|
|
contentType: "application/json",
|
|
success: function(data) {
|
|
console.log("Config check response:", data);
|
|
// Check the 'success' property returned by the endpoint.
|
|
if (data.success) {
|
|
$("#apply-config-button").show();
|
|
} else {
|
|
$("#apply-config-button").hide();
|
|
}
|
|
},
|
|
error: function(jqXHR, exception) {
|
|
try {
|
|
const responseJson = JSON.parse(jqXHR.responseText);
|
|
toastr.error(responseJson['message']);
|
|
} catch (e) {
|
|
toastr.error("Error checking config changes.");
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
|
|
// populateClient function for render new client info on the client page.
|
|
function populateClient(client_id) {
|
|
$.ajax({
|
|
cache: false,
|
|
method: 'GET',
|
|
url: '{{.basePath}}/api/client/' + client_id,
|
|
dataType: 'json',
|
|
contentType: "application/json",
|
|
success: function (resp) {
|
|
renderClientList([resp]);
|
|
},
|
|
error: function (jqXHR, exception) {
|
|
try {
|
|
const responseJson = JSON.parse(jqXHR.responseText);
|
|
toastr.error(responseJson['message']);
|
|
} catch (e) {
|
|
toastr.error("Error loading client data.");
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
// submitNewClient function for new client form submission.
|
|
function submitNewClient() {
|
|
const name = $("#client_name").val();
|
|
const email = $("#client_email").val();
|
|
const allocated_ips = $("#client_allocated_ips").val().split(",");
|
|
const allowed_ips = $("#client_allowed_ips").val().split(",");
|
|
const endpoint = $("#client_endpoint").val();
|
|
const use_server_dns = $("#use_server_dns").is(':checked');
|
|
const enabled = $("#enabled").is(':checked');
|
|
const public_key = $("#client_public_key").val();
|
|
const preshared_key = $("#client_preshared_key").val();
|
|
const additional_notes = $("#additional_notes").val();
|
|
|
|
const data = {
|
|
"name": name,
|
|
"email": email,
|
|
"allocated_ips": allocated_ips,
|
|
"allowed_ips": allowed_ips,
|
|
"extra_allowed_ips": $("#client_extra_allowed_ips").val().split(","),
|
|
"endpoint": endpoint,
|
|
"use_server_dns": use_server_dns,
|
|
"enabled": enabled,
|
|
"public_key": public_key,
|
|
"preshared_key": preshared_key,
|
|
"additional_notes": additional_notes
|
|
};
|
|
|
|
$.ajax({
|
|
cache: false,
|
|
method: 'POST',
|
|
url: '{{.basePath}}/new-client',
|
|
dataType: 'json',
|
|
contentType: "application/json",
|
|
data: JSON.stringify(data),
|
|
success: function(resp) {
|
|
$("#modal_new_client").modal('hide');
|
|
toastr.success('Created new client successfully');
|
|
// Update the home page (clients page) after adding successfully.
|
|
if (window.location.pathname === "{{.basePath}}/") {
|
|
populateClient(resp.id);
|
|
}
|
|
updateApplyConfigVisibility();
|
|
},
|
|
error: function(jqXHR, exception) {
|
|
try {
|
|
const responseJson = JSON.parse(jqXHR.responseText);
|
|
toastr.error(responseJson['message']);
|
|
} catch (e) {
|
|
toastr.error("Error creating client.");
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
// updateIPAllocationSuggestion function for automatically filling
|
|
// the IP Allocation input with suggested IP addresses.
|
|
function updateIPAllocationSuggestion(forceDefault = false) {
|
|
let subnetRange = $("#subnet_ranges").select2('val');
|
|
|
|
if (forceDefault || !subnetRange || subnetRange.length === 0) {
|
|
subnetRange = '__default_any__';
|
|
}
|
|
$.ajax({
|
|
cache: false,
|
|
method: 'GET',
|
|
url: `{{.basePath}}/api/suggest-client-ips?sr=${subnetRange}`,
|
|
dataType: 'json',
|
|
contentType: "application/json",
|
|
success: function(data) {
|
|
const allocated_ips = $("#client_allocated_ips").val().split(",");
|
|
allocated_ips.forEach(function (item) {
|
|
$('#client_allocated_ips').removeTag(item);
|
|
});
|
|
data.forEach(function (item) {
|
|
$('#client_allocated_ips').addTag(item);
|
|
});
|
|
},
|
|
error: function(jqXHR, exception) {
|
|
const allocated_ips = $("#client_allocated_ips").val().split(",");
|
|
allocated_ips.forEach(function (item) {
|
|
$('#client_allocated_ips').removeTag(item);
|
|
});
|
|
try {
|
|
const responseJson = JSON.parse(jqXHR.responseText);
|
|
toastr.error(responseJson['message']);
|
|
} catch (e) {
|
|
toastr.error("Error suggesting IP allocation.");
|
|
}
|
|
}
|
|
});
|
|
}
|
|
</script>
|
|
<script>
|
|
// Initialize Select2 Elements.
|
|
$(".select2").select2();
|
|
|
|
// IP Allocation tag input.
|
|
$("#client_allocated_ips").tagsInput({
|
|
'width': '100%',
|
|
'height': '75%',
|
|
'interactive': true,
|
|
'defaultText': 'Add More',
|
|
'removeWithBackspace': true,
|
|
'minChars': 0,
|
|
'minInputWidth': '100%',
|
|
'placeholderColor': '#666666'
|
|
});
|
|
|
|
// AllowedIPs tag input.
|
|
$("#client_allowed_ips").tagsInput({
|
|
'width': '100%',
|
|
'height': '75%',
|
|
'interactive': true,
|
|
'defaultText': 'Add More',
|
|
'removeWithBackspace': true,
|
|
'minChars': 0,
|
|
'minInputWidth': '100%',
|
|
'placeholderColor': '#666666'
|
|
});
|
|
|
|
$("#client_extra_allowed_ips").tagsInput({
|
|
'width': '100%',
|
|
'height': '75%',
|
|
'interactive': true,
|
|
'defaultText': 'Add More',
|
|
'removeWithBackspace': true,
|
|
'minChars': 0,
|
|
'minInputWidth': '100%',
|
|
'placeholderColor': '#666666'
|
|
});
|
|
|
|
// New client form validation.
|
|
$(document).ready(function () {
|
|
$.validator.setDefaults({
|
|
submitHandler: function () {
|
|
submitNewClient();
|
|
}
|
|
});
|
|
$("#frm_new_client").validate({
|
|
rules: {
|
|
client_name: {
|
|
required: true,
|
|
},
|
|
},
|
|
messages: {
|
|
client_name: {
|
|
required: "Please enter a name"
|
|
},
|
|
},
|
|
errorElement: 'span',
|
|
errorPlacement: function (error, element) {
|
|
error.addClass('invalid-feedback');
|
|
element.closest('.form-group').append(error);
|
|
},
|
|
highlight: function (element) {
|
|
$(element).addClass('is-invalid');
|
|
},
|
|
unhighlight: function (element) {
|
|
$(element).removeClass('is-invalid');
|
|
}
|
|
});
|
|
});
|
|
|
|
function updateSubnetRangesList(elementID, preselectedVal) {
|
|
$.getJSON("{{.basePath}}/api/subnet-ranges", null, function(data) {
|
|
$(`${elementID} option`).remove();
|
|
$(elementID).append(
|
|
$("<option></option>")
|
|
.text("Any")
|
|
.val("__default_any__")
|
|
);
|
|
$.each(data, function(index, item) {
|
|
$(elementID).append(
|
|
$("<option></option>")
|
|
.text(item)
|
|
.val(item)
|
|
);
|
|
if (item === preselectedVal) {
|
|
console.log(preselectedVal);
|
|
$(elementID).val(preselectedVal).trigger('change')
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
// New Client modal event.
|
|
$(document).ready(function () {
|
|
$("#modal_new_client").on('shown.bs.modal', function () {
|
|
$("#client_name").val("");
|
|
$("#client_email").val("");
|
|
$("#client_public_key").val("");
|
|
$("#client_preshared_key").val("");
|
|
$("#client_allocated_ips").importTags('');
|
|
$("#client_extra_allowed_ips").importTags('');
|
|
$("#client_endpoint").val('');
|
|
$("#additional_notes").val('');
|
|
updateSubnetRangesList("#subnet_ranges");
|
|
updateIPAllocationSuggestion(true);
|
|
});
|
|
});
|
|
|
|
// Handle subnet range select.
|
|
$('#subnet_ranges').on('select2:select', function () {
|
|
updateIPAllocationSuggestion();
|
|
});
|
|
|
|
// apply_config_confirm button event.
|
|
$(document).ready(function () {
|
|
$("#apply_config_confirm").click(function () {
|
|
$.ajax({
|
|
cache: false,
|
|
method: 'POST',
|
|
url: '{{.basePath}}/api/apply-wg-config',
|
|
dataType: 'json',
|
|
contentType: "application/json",
|
|
success: function(data) {
|
|
updateApplyConfigVisibility();
|
|
$("#modal_apply_config").modal('hide');
|
|
toastr.success('Applied config successfully');
|
|
},
|
|
error: function(jqXHR) {
|
|
try {
|
|
const responseJson = JSON.parse(jqXHR.responseText);
|
|
toastr.error(responseJson['message']);
|
|
} catch (e) {
|
|
toastr.error("Error applying config.");
|
|
}
|
|
}
|
|
});
|
|
});
|
|
});
|
|
</script>
|
|
<!-- START: On page script -->
|
|
{{template "bottom_js" .}}
|
|
<!-- END: On page script -->
|
|
</body>
|
|
|
|
</html>
|
|
{{end}}
|