add js linting, fix js linting issues
This commit is contained in:
		| @@ -2,7 +2,7 @@ import { requestAPI, goToPage, getCookie, setTitleAndHeader } from "./utils.js"; | ||||
|  | ||||
| window.addEventListener("DOMContentLoaded", init); | ||||
|  | ||||
| let prefixes = { | ||||
| const prefixes = { | ||||
| 	1024: [ | ||||
| 		"", | ||||
| 		"Ki", | ||||
| @@ -17,17 +17,17 @@ let prefixes = { | ||||
| 		"G", | ||||
| 		"T" | ||||
| 	] | ||||
| } | ||||
| }; | ||||
|  | ||||
| async function init() { | ||||
| async function init () { | ||||
| 	setTitleAndHeader(); | ||||
| 	let cookie = document.cookie; | ||||
| 	const cookie = document.cookie; | ||||
| 	if (cookie === "") { | ||||
| 		goToPage("login.html"); | ||||
| 	} | ||||
| 	let resources = await requestAPI("/user/resources"); | ||||
| 	let instances = await requestAPI("/user/config/cluster"); | ||||
| 	let nodes = await requestAPI("/user/config/nodes"); | ||||
| 	const resources = await requestAPI("/user/resources"); | ||||
| 	const instances = await requestAPI("/user/config/cluster"); | ||||
| 	const nodes = await requestAPI("/user/config/nodes"); | ||||
| 	document.querySelector("#username").innerText = `Username: ${getCookie("username")}`; | ||||
| 	document.querySelector("#pool").innerText = `Pool: ${instances.pool}`; | ||||
| 	document.querySelector("#vmid").innerText = `VMID Range: ${instances.vmid.min} - ${instances.vmid.max}`; | ||||
| @@ -35,24 +35,25 @@ async function init() { | ||||
| 	buildResourceTable(resources, "#resource-table"); | ||||
| } | ||||
|  | ||||
| function buildResourceTable(resources, tableid) { | ||||
| function buildResourceTable (resources, tableid) { | ||||
| 	if (resources instanceof Object) { | ||||
| 		let table = document.querySelector(tableid); | ||||
| 		let tbody = table.querySelector("tbody"); | ||||
| 		const table = document.querySelector(tableid); | ||||
| 		const tbody = table.querySelector("tbody"); | ||||
| 		Object.keys(resources.resources).forEach((element) => { | ||||
| 			if (resources.resources[element].display) { | ||||
| 				if (resources.resources[element].type === "list") { | ||||
|  | ||||
| 					// TODO | ||||
| 					console.error("Unimplemented"); | ||||
| 				} | ||||
| 				else { | ||||
| 					let row = tbody.insertRow(); | ||||
| 					let key = row.insertCell(); | ||||
| 					const row = tbody.insertRow(); | ||||
| 					const key = row.insertCell(); | ||||
| 					key.innerText = `${element}`; | ||||
| 					let used = row.insertCell(); | ||||
| 					const used = row.insertCell(); | ||||
| 					used.innerText = `${parseNumber(resources.used[element], resources.resources[element])}`; | ||||
| 					let val = row.insertCell(); | ||||
| 					const val = row.insertCell(); | ||||
| 					val.innerText = `${parseNumber(resources.avail[element], resources.resources[element])}`; | ||||
| 					let total = row.insertCell(); | ||||
| 					const total = row.insertCell(); | ||||
| 					total.innerText = `${parseNumber(resources.max[element], resources.resources[element])}`; | ||||
| 				} | ||||
| 			} | ||||
| @@ -60,22 +61,22 @@ function buildResourceTable(resources, tableid) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| function parseNumber(value, unitData) { | ||||
| 	let compact = unitData.compact; | ||||
| 	let multiplier = unitData.multiplier; | ||||
| 	let base = unitData.base; | ||||
| 	let unit = unitData.unit; | ||||
| function parseNumber (value, unitData) { | ||||
| 	const compact = unitData.compact; | ||||
| 	const multiplier = unitData.multiplier; | ||||
| 	const base = unitData.base; | ||||
| 	const unit = unitData.unit; | ||||
| 	value = multiplier * value; | ||||
| 	if (value <= 0) { | ||||
| 		return `0 ${unit}`; | ||||
| 	} | ||||
| 	else if (compact) { | ||||
| 		let exponent = Math.floor(Math.log(value) / Math.log(base)); | ||||
| 		const exponent = Math.floor(Math.log(value) / Math.log(base)); | ||||
| 		value = value / base ** exponent; | ||||
| 		let unitPrefix = prefixes[base][exponent]; | ||||
| 		return `${value} ${unitPrefix}${unit}` | ||||
| 		const unitPrefix = prefixes[base][exponent]; | ||||
| 		return `${value} ${unitPrefix}${unit}`; | ||||
| 	} | ||||
| 	else { | ||||
| 		return `${value} ${unit}`; | ||||
| 	} | ||||
| } | ||||
| } | ||||
|   | ||||
							
								
								
									
										1654
									
								
								scripts/config.js
									
									
									
									
									
								
							
							
						
						
									
										1654
									
								
								scripts/config.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,5 +1,5 @@ | ||||
| export function dialog(header, body, callback = async (result, form) => { }) { | ||||
| 	let dialog = document.createElement("dialog"); | ||||
| export function dialog (header, body, callback = async (result, form) => { }) { | ||||
| 	const dialog = document.createElement("dialog"); | ||||
| 	dialog.innerHTML = ` | ||||
| 		<p class="w3-large" id="prompt" style="text-align: center;"></p> | ||||
| 		<form method="dialog" class="input-grid" style="grid-template-columns: auto 1fr;" id="form"></form> | ||||
| @@ -23,8 +23,8 @@ export function dialog(header, body, callback = async (result, form) => { }) { | ||||
| 	return dialog; | ||||
| } | ||||
|  | ||||
| export function alert(message) { | ||||
| 	let dialog = document.createElement("dialog"); | ||||
| export function alert (message) { | ||||
| 	const dialog = document.createElement("dialog"); | ||||
| 	dialog.innerHTML = ` | ||||
| 		<form method="dialog"> | ||||
| 			<p class="w3-center" style="margin-bottom: 0px;">${message}</p> | ||||
| @@ -40,7 +40,7 @@ export function alert(message) { | ||||
|  | ||||
| 	dialog.addEventListener("close", () => { | ||||
| 		dialog.parentElement.removeChild(dialog); | ||||
| 	}) | ||||
| 	}); | ||||
|  | ||||
| 	return dialog; | ||||
| } | ||||
| } | ||||
|   | ||||
							
								
								
									
										840
									
								
								scripts/index.js
									
									
									
									
									
								
							
							
						
						
									
										840
									
								
								scripts/index.js
									
									
									
									
									
								
							| @@ -1,416 +1,424 @@ | ||||
| import { requestPVE, requestAPI, goToPage, goToURL, instances_config, nodes_config, setTitleAndHeader } from "./utils.js"; | ||||
| import { alert, dialog } from "./dialog.js"; | ||||
| import { PVE } from "../vars.js" | ||||
|  | ||||
| window.addEventListener("DOMContentLoaded", init); | ||||
|  | ||||
| async function init() { | ||||
| 	setTitleAndHeader(); | ||||
| 	let cookie = document.cookie; | ||||
| 	if (cookie === "") { | ||||
| 		goToPage("login.html"); | ||||
| 	} | ||||
|  | ||||
| 	await populateInstances(); | ||||
|  | ||||
| 	let addInstanceBtn = document.querySelector("#instance-add"); | ||||
| 	addInstanceBtn.addEventListener("click", handleInstanceAdd); | ||||
| } | ||||
|  | ||||
| async function populateInstances() { | ||||
| 	let resources = await requestPVE("/cluster/resources", "GET"); | ||||
| 	let instanceContainer = document.getElementById("instance-container"); | ||||
| 	let instances = []; | ||||
|  | ||||
| 	resources.data.forEach((element) => { | ||||
| 		if (element.type === "lxc" || element.type === "qemu") { | ||||
| 			let nodeName = element.node; | ||||
| 			let nodeStatus = resources.data.find(item => item.node === nodeName && item.type === "node").status; | ||||
| 			element.node = { name: nodeName, status: nodeStatus }; | ||||
| 			instances.push(element); | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	instances.sort((a, b) => (a.vmid > b.vmid) ? 1 : -1); | ||||
|  | ||||
| 	instanceContainer.innerHTML = ` | ||||
| 		<div class="w3-row w3-hide-small" style="border-bottom: 1px solid;"> | ||||
| 			<div class="w3-col l1 m2"> | ||||
| 				<p>VM ID</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3"> | ||||
| 				<p>VM Name</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l1 m2"> | ||||
| 				<p>VM Type</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3"> | ||||
| 				<p>VM Status</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium"> | ||||
| 				<p>Host Name</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium"> | ||||
| 				<p>Host Status</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m2"> | ||||
| 				<p>Actions</p> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	`; | ||||
| 	for (let i = 0; i < instances.length; i++) { | ||||
| 		let newInstance = new Instance(); | ||||
| 		newInstance.data = instances[i]; | ||||
| 		instanceContainer.append(newInstance.shadowElement); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| async function handleInstanceAdd() { | ||||
| 	let header = "Create New Instance"; | ||||
|  | ||||
| 	let body = ` | ||||
| 		<label for="type">Instance Type</label> | ||||
| 		<select class="w3-select w3-border" name="type" id="type" required> | ||||
| 			<option value="lxc">Container</option> | ||||
| 			<option value="qemu">Virtual Machine</option> | ||||
| 		</select> | ||||
| 		<label for="node">Node</label> | ||||
| 		<select class="w3-select w3-border" name="node" id="node" required></select> | ||||
| 		<label for="name">Name</label> | ||||
| 		<input class="w3-input w3-border" name="name" id="name" required></input> | ||||
| 		<label for="vmid">ID</label> | ||||
| 		<input class="w3-input w3-border" name="vmid" id="vmid" type="number" required></input> | ||||
| 		<label for="cores">Cores (Threads)</label> | ||||
| 		<input class="w3-input w3-border" name="cores" id="cores" type="number" min="1" max="8192" required></input> | ||||
| 		<label for="memory">Memory (MiB)</label> | ||||
| 		<input class="w3-input w3-border" name="memory" id="memory" type="number" min="16", step="1" required></input> | ||||
| 		<p class="container-specific none" style="grid-column: 1 / span 2; text-align: center;">Container Options</p> | ||||
| 		<label class="container-specific none" for="swap">Swap (MiB)</label> | ||||
| 		<input class="w3-input w3-border container-specific none" name="swap" id="swap" type="number" min="0" step="1" required disabled></input> | ||||
| 		<label class="container-specific none" for="template-storage">Template Storage</label> | ||||
| 		<select class="w3-select w3-border container-specific none" name="template-storage" id="template-storage" required disabled></select> | ||||
| 		<label class="container-specific none" for="template-image">Template Image</label> | ||||
| 		<select class="w3-select w3-border container-specific none" name="template-image" id="template-image" required disabled></select> | ||||
| 		<label class="container-specific none" for="rootfs-storage">ROOTFS Storage</label> | ||||
| 		<select class="w3-select w3-border container-specific none" name="rootfs-storage" id="rootfs-storage" required disabled></select>				 | ||||
| 		<label class="container-specific none" for="rootfs-size">ROOTFS Size (GiB)</label> | ||||
| 		<input class="w3-input w3-border container-specific none" name="rootfs-size" id="rootfs-size" type="number" min="0" max="131072" required disabled></input> | ||||
| 		<label class="container-specific none" for="password">Password</label> | ||||
| 		<input class="w3-input w3-border container-specific none" name="password" id="password" type="password" required disabled></input> | ||||
| 	`; | ||||
|  | ||||
| 	let d = dialog(header, body, async (result, form) => { | ||||
| 		if (result === "confirm") { | ||||
| 			let body = { | ||||
| 				node: form.get("node"), | ||||
| 				type: form.get("type"), | ||||
| 				name: form.get("name"), | ||||
| 				vmid: form.get("vmid"), | ||||
| 				cores: form.get("cores"), | ||||
| 				memory: form.get("memory") | ||||
| 			}; | ||||
| 			if (form.get("type") === "lxc") { | ||||
| 				body.swap = form.get("swap"); | ||||
| 				body.password = form.get("password"); | ||||
| 				body.ostemplate = form.get("template-image"); | ||||
| 				body.rootfslocation = form.get("rootfs-storage"); | ||||
| 				body.rootfssize = form.get("rootfs-size"); | ||||
| 			} | ||||
| 			let result = await requestAPI("/instance", "POST", body); | ||||
| 			if (result.status === 200) { | ||||
| 				populateInstances(); | ||||
| 			} | ||||
| 			else { | ||||
| 				alert(result.error); | ||||
| 				populateInstances(); | ||||
| 			} | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	let typeSelect = d.querySelector("#type"); | ||||
| 	typeSelect.selectedIndex = -1; | ||||
| 	typeSelect.addEventListener("change", () => { | ||||
| 		if (typeSelect.value === "qemu") { | ||||
| 			d.querySelectorAll(".container-specific").forEach((element) => { | ||||
| 				element.classList.add("none"); | ||||
| 				element.disabled = true; | ||||
| 			}); | ||||
| 		} | ||||
| 		else { | ||||
| 			d.querySelectorAll(".container-specific").forEach((element) => { | ||||
| 				element.classList.remove("none"); | ||||
| 				element.disabled = false; | ||||
| 			}); | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	let templateContent = "iso"; | ||||
| 	let templateStorage = d.querySelector("#template-storage"); | ||||
| 	templateStorage.selectedIndex = -1; | ||||
|  | ||||
| 	let rootfsContent = "rootdir"; | ||||
| 	let rootfsStorage = d.querySelector("#rootfs-storage"); | ||||
| 	rootfsStorage.selectedIndex = -1; | ||||
|  | ||||
| 	let nodeSelect = d.querySelector("#node"); | ||||
| 	let clusterNodes = await requestPVE("/nodes", "GET"); | ||||
| 	let allowedNodes = await requestAPI("/user/config/nodes", "GET"); | ||||
| 	clusterNodes.data.forEach((element) => { | ||||
| 		if (element.status === "online" && allowedNodes.includes(element.node)) { | ||||
| 			nodeSelect.add(new Option(element.node)); | ||||
| 		} | ||||
| 	}); | ||||
| 	nodeSelect.selectedIndex = -1; | ||||
| 	nodeSelect.addEventListener("change", async () => { // change template and rootfs storage based on node | ||||
| 		let node = nodeSelect.value; | ||||
| 		let storage = await requestPVE(`/nodes/${node}/storage`, "GET"); | ||||
| 		storage.data.forEach((element) => { | ||||
| 			if (element.content.includes(templateContent)) { | ||||
| 				templateStorage.add(new Option(element.storage)); | ||||
| 			} | ||||
| 			if (element.content.includes(rootfsContent)) { | ||||
| 				rootfsStorage.add(new Option(element.storage)); | ||||
| 			} | ||||
| 		}); | ||||
| 		templateStorage.selectedIndex = -1; | ||||
| 		rootfsStorage.selectedIndex = -1; | ||||
| 	}); | ||||
|  | ||||
| 	let templateImage = d.querySelector("#template-image"); // populate templateImage depending on selected image storage | ||||
| 	templateStorage.addEventListener("change", async () => { | ||||
| 		templateImage.innerHTML = ``; | ||||
| 		let content = "vztmpl"; | ||||
| 		let images = await requestPVE(`/nodes/${nodeSelect.value}/storage/${templateStorage.value}/content`, "GET"); | ||||
| 		images.data.forEach((element) => { | ||||
| 			if (element.content.includes(content)) { | ||||
| 				templateImage.append(new Option(element.volid.replace(`${templateStorage.value}:${content}/`, ""), element.volid)); | ||||
| 			} | ||||
| 		}); | ||||
| 		templateImage.selectedIndex = -1; | ||||
| 	}); | ||||
|  | ||||
| 	let userResources = await requestAPI("/user/resources", "GET"); | ||||
| 	let userInstances = await requestAPI("/user/config/instances", "GET"); | ||||
| 	d.querySelector("#cores").max = userResources.avail.cores; | ||||
| 	d.querySelector("#memory").max = userResources.avail.memory; | ||||
| 	d.querySelector("#vmid").min = userInstances.vmid.min; | ||||
| 	d.querySelector("#vmid").max = userInstances.vmid.max; | ||||
| } | ||||
|  | ||||
| class Instance { | ||||
| 	constructor() { | ||||
| 		let shadowRoot = document.createElement("div"); | ||||
| 		shadowRoot.classList.add("w3-row"); | ||||
|  | ||||
| 		shadowRoot.innerHTML = ` | ||||
| 			<div class="w3-col l1 m2 s6"> | ||||
| 				<p id="instance-id"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3 s6"> | ||||
| 				<p id="instance-name"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l1 m2 w3-hide-small"> | ||||
| 				<p id="instance-type"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3 s6 flex row nowrap"> | ||||
| 				<img id="instance-status-icon"> | ||||
| 				<p id="instance-status"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium w3-hide-small"> | ||||
| 				<p id="node-name"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium w3-hide-small flex row nowrap"> | ||||
| 				<img id="node-status-icon"> | ||||
| 				<p id="node-status"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m2 s6 flex row nowrap" style="height: 1lh; margin-top: 15px; margin-bottom: 15px;"> | ||||
| 				<img id="power-btn"> | ||||
| 				<img id="console-btn"> | ||||
| 				<img id="configure-btn"> | ||||
| 				<img id="delete-btn"> | ||||
| 			</div> | ||||
| 		`; | ||||
|  | ||||
| 		this.shadowElement = shadowRoot; | ||||
| 		this.actionLock = false; | ||||
| 	} | ||||
|  | ||||
| 	set data(data) { | ||||
| 		if (data.status === "unknown") { | ||||
| 			data.status = "stopped"; | ||||
| 		} | ||||
| 		this.type = data.type; | ||||
| 		this.status = data.status; | ||||
| 		this.vmid = data.vmid; | ||||
| 		this.name = data.name; | ||||
| 		this.node = data.node; | ||||
| 		this.update(); | ||||
| 	} | ||||
|  | ||||
| 	update() { | ||||
| 		let vmidParagraph = this.shadowElement.querySelector("#instance-id"); | ||||
| 		vmidParagraph.innerText = this.vmid; | ||||
|  | ||||
| 		let nameParagraph = this.shadowElement.querySelector("#instance-name"); | ||||
| 		nameParagraph.innerText = this.name ? this.name : ""; | ||||
|  | ||||
| 		let typeParagraph = this.shadowElement.querySelector("#instance-type"); | ||||
| 		typeParagraph.innerText = this.type; | ||||
|  | ||||
| 		let statusParagraph = this.shadowElement.querySelector("#instance-status"); | ||||
| 		statusParagraph.innerText = this.status; | ||||
|  | ||||
| 		let statusIcon = this.shadowElement.querySelector("#instance-status-icon"); | ||||
| 		statusIcon.src = instances_config[this.status].status.src; | ||||
| 		statusIcon.alt = instances_config[this.status].status.alt; | ||||
|  | ||||
| 		let nodeNameParagraph = this.shadowElement.querySelector("#node-name"); | ||||
| 		nodeNameParagraph.innerText = this.node.name; | ||||
|  | ||||
| 		let nodeStatusParagraph = this.shadowElement.querySelector("#node-status"); | ||||
| 		nodeStatusParagraph.innerText = this.node.status; | ||||
|  | ||||
| 		let nodeStatusIcon = this.shadowElement.querySelector("#node-status-icon"); | ||||
| 		nodeStatusIcon.src = nodes_config[this.node.status].status.src; | ||||
| 		nodeStatusIcon.alt = nodes_config[this.node.status].status.src; | ||||
|  | ||||
| 		let powerButton = this.shadowElement.querySelector("#power-btn"); | ||||
| 		powerButton.src = instances_config[this.status].power.src; | ||||
| 		powerButton.alt = instances_config[this.status].power.alt; | ||||
| 		powerButton.title = instances_config[this.status].power.alt; | ||||
| 		if (instances_config[this.status].power.clickable) { | ||||
| 			powerButton.classList.add("clickable"); | ||||
| 			powerButton.onclick = this.handlePowerButton.bind(this) | ||||
| 		} | ||||
|  | ||||
| 		let configButton = this.shadowElement.querySelector("#configure-btn"); | ||||
| 		configButton.src = instances_config[this.status].config.src; | ||||
| 		configButton.alt = instances_config[this.status].config.alt; | ||||
| 		configButton.title = instances_config[this.status].config.alt; | ||||
| 		if (instances_config[this.status].config.clickable) { | ||||
| 			configButton.classList.add("clickable"); | ||||
| 			configButton.onclick = this.handleConfigButton.bind(this); | ||||
| 		} | ||||
|  | ||||
| 		let consoleButton = this.shadowElement.querySelector("#console-btn"); | ||||
| 		consoleButton.src = instances_config[this.status].console.src; | ||||
| 		consoleButton.alt = instances_config[this.status].console.alt; | ||||
| 		consoleButton.title = instances_config[this.status].console.alt; | ||||
| 		if (instances_config[this.status].console.clickable) { | ||||
| 			consoleButton.classList.add("clickable"); | ||||
| 			consoleButton.onclick = this.handleConsoleButton.bind(this); | ||||
| 		} | ||||
|  | ||||
| 		let deleteButton = this.shadowElement.querySelector("#delete-btn"); | ||||
| 		deleteButton.src = instances_config[this.status].delete.src; | ||||
| 		deleteButton.alt = instances_config[this.status].delete.alt; | ||||
| 		deleteButton.title = instances_config[this.status].delete.alt; | ||||
| 		if (instances_config[this.status].delete.clickable) { | ||||
| 			deleteButton.classList.add("clickable"); | ||||
| 			deleteButton.onclick = this.handleDeleteButton.bind(this); | ||||
| 		} | ||||
|  | ||||
| 		if (this.node.status !== "online") { | ||||
| 			powerButton.classList.add("hidden"); | ||||
| 			configButton.classList.add("hidden"); | ||||
| 			consoleButton.classList.add("hidden"); | ||||
| 			deleteButton.classList.add("hidden"); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	async handlePowerButton() { | ||||
| 		if (!this.actionLock) { | ||||
| 			let header = `${this.status === "running" ? "Stop" : "Start"} VM ${this.vmid}`; | ||||
| 			let body = `<p>Are you sure you want to ${this.status === "running" ? "stop" : "start"} VM</p><p>${this.vmid}</p>` | ||||
|  | ||||
| 			dialog(header, body, async (result, form) => { | ||||
| 				if (result === "confirm") { | ||||
| 					this.actionLock = true; | ||||
| 					let targetAction = this.status === "running" ? "stop" : "start"; | ||||
| 					let targetStatus = this.status === "running" ? "stopped" : "running"; | ||||
| 					let prevStatus = this.status; | ||||
| 					this.status = "loading"; | ||||
|  | ||||
| 					this.update(); | ||||
|  | ||||
| 					let result = await requestPVE(`/nodes/${this.node.name}/${this.type}/${this.vmid}/status/${targetAction}`, "POST", { node: this.node.name, vmid: this.vmid }); | ||||
|  | ||||
| 					const waitFor = delay => new Promise(resolve => setTimeout(resolve, delay)); | ||||
|  | ||||
| 					while (true) { | ||||
| 						let taskStatus = await requestPVE(`/nodes/${this.node.name}/tasks/${result.data}/status`, "GET"); | ||||
| 						if (taskStatus.data.status === "stopped" && taskStatus.data.exitstatus === "OK") { // task stopped and was successful | ||||
| 							this.status = targetStatus; | ||||
| 							this.update(); | ||||
| 							this.actionLock = false; | ||||
| 							break; | ||||
| 						} | ||||
| 						else if (taskStatus.data.status === "stopped") { // task stopped but was not successful | ||||
| 							this.status = prevStatus; | ||||
| 							alert(`attempted to ${targetAction} ${this.vmid} but process returned stopped:${result.data.exitstatus}`); | ||||
| 							this.update(); | ||||
| 							this.actionLock = false; | ||||
| 							break; | ||||
| 						} | ||||
| 						else { // task has not stopped | ||||
| 							await waitFor(1000); | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	handleConfigButton() { | ||||
| 		if (!this.actionLock && this.status === "stopped") { // if the action lock is false, and the node is stopped, then navigate to the conig page with the node infor in the search query | ||||
| 			goToPage("config.html", { node: this.node.name, type: this.type, vmid: this.vmid }); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	handleConsoleButton() { | ||||
| 		if (!this.actionLock && this.status === "running") { | ||||
| 			let data = { console: `${this.type === "qemu" ? "kvm" : "lxc"}`, vmid: this.vmid, vmname: this.name, node: this.node.name, resize: "off", cmd: "" }; | ||||
| 			data[`${this.type === "qemu" ? "novnc" : "xtermjs"}`] = 1; | ||||
| 			goToURL(PVE, data, true); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	handleDeleteButton() { | ||||
| 		if (!this.actionLock && this.status === "stopped") { | ||||
|  | ||||
| 			let header = `Delete VM ${this.vmid}`; | ||||
| 			let body = `<p>Are you sure you want to <strong>delete</strong> VM </p><p>${this.vmid}</p>` | ||||
|  | ||||
| 			dialog(header, body, async (result, form) => { | ||||
| 				if (result === "confirm") { | ||||
| 					this.actionLock = true; | ||||
| 					let prevStatus = this.status; | ||||
| 					this.status = "loading"; | ||||
| 					this.update(); | ||||
|  | ||||
| 					let action = {}; | ||||
| 					action.purge = 1; | ||||
| 					action["destroy-unreferenced-disks"] = 1; | ||||
|  | ||||
| 					let body = { | ||||
| 						node: this.node.name, | ||||
| 						type: this.type, | ||||
| 						vmid: this.vmid, | ||||
| 						action: JSON.stringify(action) | ||||
| 					}; | ||||
|  | ||||
| 					let result = await requestAPI("/instance", "DELETE", body); | ||||
| 					if (result.status === 200) { | ||||
| 						this.shadowElement.parentElement.removeChild(this.shadowElement); | ||||
| 					} | ||||
| 					else { | ||||
| 						alert(result.error); | ||||
| 						this.status = this.prevStatus; | ||||
| 						this.update(); | ||||
| 						this.actionLock = false; | ||||
| 					} | ||||
| 				} | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| import { requestPVE, requestAPI, goToPage, goToURL, instancesConfig, nodesConfig, setTitleAndHeader } from "./utils.js"; | ||||
| import { alert, dialog } from "./dialog.js"; | ||||
| import { PVE } from "../vars.js"; | ||||
|  | ||||
| window.addEventListener("DOMContentLoaded", init); | ||||
|  | ||||
| async function init () { | ||||
| 	setTitleAndHeader(); | ||||
| 	const cookie = document.cookie; | ||||
| 	if (cookie === "") { | ||||
| 		goToPage("login.html"); | ||||
| 	} | ||||
|  | ||||
| 	await populateInstances(); | ||||
|  | ||||
| 	const addInstanceBtn = document.querySelector("#instance-add"); | ||||
| 	addInstanceBtn.addEventListener("click", handleInstanceAdd); | ||||
| } | ||||
|  | ||||
| async function populateInstances () { | ||||
| 	const resources = await requestPVE("/cluster/resources", "GET"); | ||||
| 	const instanceContainer = document.getElementById("instance-container"); | ||||
| 	const instances = []; | ||||
|  | ||||
| 	resources.data.forEach((element) => { | ||||
| 		if (element.type === "lxc" || element.type === "qemu") { | ||||
| 			const nodeName = element.node; | ||||
| 			const nodeStatus = resources.data.find(item => item.node === nodeName && item.type === "node").status; | ||||
| 			element.node = { name: nodeName, status: nodeStatus }; | ||||
| 			instances.push(element); | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	instances.sort((a, b) => (a.vmid > b.vmid) ? 1 : -1); | ||||
|  | ||||
| 	instanceContainer.innerHTML = ` | ||||
| 		<div class="w3-row w3-hide-small" style="border-bottom: 1px solid;"> | ||||
| 			<div class="w3-col l1 m2"> | ||||
| 				<p>VM ID</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3"> | ||||
| 				<p>VM Name</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l1 m2"> | ||||
| 				<p>VM Type</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3"> | ||||
| 				<p>VM Status</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium"> | ||||
| 				<p>Host Name</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium"> | ||||
| 				<p>Host Status</p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m2"> | ||||
| 				<p>Actions</p> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	`; | ||||
| 	for (let i = 0; i < instances.length; i++) { | ||||
| 		const newInstance = new Instance(); | ||||
| 		newInstance.data = instances[i]; | ||||
| 		instanceContainer.append(newInstance.shadowElement); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| async function handleInstanceAdd () { | ||||
| 	const header = "Create New Instance"; | ||||
|  | ||||
| 	const body = ` | ||||
| 		<label for="type">Instance Type</label> | ||||
| 		<select class="w3-select w3-border" name="type" id="type" required> | ||||
| 			<option value="lxc">Container</option> | ||||
| 			<option value="qemu">Virtual Machine</option> | ||||
| 		</select> | ||||
| 		<label for="node">Node</label> | ||||
| 		<select class="w3-select w3-border" name="node" id="node" required></select> | ||||
| 		<label for="name">Name</label> | ||||
| 		<input class="w3-input w3-border" name="name" id="name" required></input> | ||||
| 		<label for="vmid">ID</label> | ||||
| 		<input class="w3-input w3-border" name="vmid" id="vmid" type="number" required></input> | ||||
| 		<label for="cores">Cores (Threads)</label> | ||||
| 		<input class="w3-input w3-border" name="cores" id="cores" type="number" min="1" max="8192" required></input> | ||||
| 		<label for="memory">Memory (MiB)</label> | ||||
| 		<input class="w3-input w3-border" name="memory" id="memory" type="number" min="16", step="1" required></input> | ||||
| 		<p class="container-specific none" style="grid-column: 1 / span 2; text-align: center;">Container Options</p> | ||||
| 		<label class="container-specific none" for="swap">Swap (MiB)</label> | ||||
| 		<input class="w3-input w3-border container-specific none" name="swap" id="swap" type="number" min="0" step="1" required disabled></input> | ||||
| 		<label class="container-specific none" for="template-storage">Template Storage</label> | ||||
| 		<select class="w3-select w3-border container-specific none" name="template-storage" id="template-storage" required disabled></select> | ||||
| 		<label class="container-specific none" for="template-image">Template Image</label> | ||||
| 		<select class="w3-select w3-border container-specific none" name="template-image" id="template-image" required disabled></select> | ||||
| 		<label class="container-specific none" for="rootfs-storage">ROOTFS Storage</label> | ||||
| 		<select class="w3-select w3-border container-specific none" name="rootfs-storage" id="rootfs-storage" required disabled></select> | ||||
| 		<label class="container-specific none" for="rootfs-size">ROOTFS Size (GiB)</label> | ||||
| 		<input class="w3-input w3-border container-specific none" name="rootfs-size" id="rootfs-size" type="number" min="0" max="131072" required disabled></input> | ||||
| 		<label class="container-specific none" for="password">Password</label> | ||||
| 		<input class="w3-input w3-border container-specific none" name="password" id="password" type="password" required disabled></input> | ||||
| 	`; | ||||
|  | ||||
| 	const d = dialog(header, body, async (result, form) => { | ||||
| 		if (result === "confirm") { | ||||
| 			const body = { | ||||
| 				node: form.get("node"), | ||||
| 				type: form.get("type"), | ||||
| 				name: form.get("name"), | ||||
| 				vmid: form.get("vmid"), | ||||
| 				cores: form.get("cores"), | ||||
| 				memory: form.get("memory") | ||||
| 			}; | ||||
| 			if (form.get("type") === "lxc") { | ||||
| 				body.swap = form.get("swap"); | ||||
| 				body.password = form.get("password"); | ||||
| 				body.ostemplate = form.get("template-image"); | ||||
| 				body.rootfslocation = form.get("rootfs-storage"); | ||||
| 				body.rootfssize = form.get("rootfs-size"); | ||||
| 			} | ||||
| 			const result = await requestAPI("/instance", "POST", body); | ||||
| 			if (result.status === 200) { | ||||
| 				populateInstances(); | ||||
| 			} | ||||
| 			else { | ||||
| 				alert(result.error); | ||||
| 				populateInstances(); | ||||
| 			} | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	const typeSelect = d.querySelector("#type"); | ||||
| 	typeSelect.selectedIndex = -1; | ||||
| 	typeSelect.addEventListener("change", () => { | ||||
| 		if (typeSelect.value === "qemu") { | ||||
| 			d.querySelectorAll(".container-specific").forEach((element) => { | ||||
| 				element.classList.add("none"); | ||||
| 				element.disabled = true; | ||||
| 			}); | ||||
| 		} | ||||
| 		else { | ||||
| 			d.querySelectorAll(".container-specific").forEach((element) => { | ||||
| 				element.classList.remove("none"); | ||||
| 				element.disabled = false; | ||||
| 			}); | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	const templateContent = "iso"; | ||||
| 	const templateStorage = d.querySelector("#template-storage"); | ||||
| 	templateStorage.selectedIndex = -1; | ||||
|  | ||||
| 	const rootfsContent = "rootdir"; | ||||
| 	const rootfsStorage = d.querySelector("#rootfs-storage"); | ||||
| 	rootfsStorage.selectedIndex = -1; | ||||
|  | ||||
| 	const nodeSelect = d.querySelector("#node"); | ||||
| 	const clusterNodes = await requestPVE("/nodes", "GET"); | ||||
| 	const allowedNodes = await requestAPI("/user/config/nodes", "GET"); | ||||
| 	clusterNodes.data.forEach((element) => { | ||||
| 		if (element.status === "online" && allowedNodes.includes(element.node)) { | ||||
| 			nodeSelect.add(new Option(element.node)); | ||||
| 		} | ||||
| 	}); | ||||
| 	nodeSelect.selectedIndex = -1; | ||||
| 	nodeSelect.addEventListener("change", async () => { // change template and rootfs storage based on node | ||||
| 		const node = nodeSelect.value; | ||||
| 		const storage = await requestPVE(`/nodes/${node}/storage`, "GET"); | ||||
| 		storage.data.forEach((element) => { | ||||
| 			if (element.content.includes(templateContent)) { | ||||
| 				templateStorage.add(new Option(element.storage)); | ||||
| 			} | ||||
| 			if (element.content.includes(rootfsContent)) { | ||||
| 				rootfsStorage.add(new Option(element.storage)); | ||||
| 			} | ||||
| 		}); | ||||
| 		templateStorage.selectedIndex = -1; | ||||
| 		rootfsStorage.selectedIndex = -1; | ||||
| 	}); | ||||
|  | ||||
| 	const templateImage = d.querySelector("#template-image"); // populate templateImage depending on selected image storage | ||||
| 	templateStorage.addEventListener("change", async () => { | ||||
| 		templateImage.innerHTML = ""; | ||||
| 		const content = "vztmpl"; | ||||
| 		const images = await requestPVE(`/nodes/${nodeSelect.value}/storage/${templateStorage.value}/content`, "GET"); | ||||
| 		images.data.forEach((element) => { | ||||
| 			if (element.content.includes(content)) { | ||||
| 				templateImage.append(new Option(element.volid.replace(`${templateStorage.value}:${content}/`, ""), element.volid)); | ||||
| 			} | ||||
| 		}); | ||||
| 		templateImage.selectedIndex = -1; | ||||
| 	}); | ||||
|  | ||||
| 	const userResources = await requestAPI("/user/resources", "GET"); | ||||
| 	const userInstances = await requestAPI("/user/config/instances", "GET"); | ||||
| 	d.querySelector("#cores").max = userResources.avail.cores; | ||||
| 	d.querySelector("#memory").max = userResources.avail.memory; | ||||
| 	d.querySelector("#vmid").min = userInstances.vmid.min; | ||||
| 	d.querySelector("#vmid").max = userInstances.vmid.max; | ||||
| } | ||||
|  | ||||
| class Instance { | ||||
| 	constructor () { | ||||
| 		const shadowRoot = document.createElement("div"); | ||||
| 		shadowRoot.classList.add("w3-row"); | ||||
|  | ||||
| 		shadowRoot.innerHTML = ` | ||||
| 			<div class="w3-col l1 m2 s6"> | ||||
| 				<p id="instance-id"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3 s6"> | ||||
| 				<p id="instance-name"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l1 m2 w3-hide-small"> | ||||
| 				<p id="instance-type"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m3 s6 flex row nowrap"> | ||||
| 				<img id="instance-status-icon"> | ||||
| 				<p id="instance-status"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium w3-hide-small"> | ||||
| 				<p id="node-name"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 w3-hide-medium w3-hide-small flex row nowrap"> | ||||
| 				<img id="node-status-icon"> | ||||
| 				<p id="node-status"></p> | ||||
| 			</div> | ||||
| 			<div class="w3-col l2 m2 s6 flex row nowrap" style="height: 1lh; margin-top: 15px; margin-bottom: 15px;"> | ||||
| 				<img id="power-btn"> | ||||
| 				<img id="console-btn"> | ||||
| 				<img id="configure-btn"> | ||||
| 				<img id="delete-btn"> | ||||
| 			</div> | ||||
| 		`; | ||||
|  | ||||
| 		this.shadowElement = shadowRoot; | ||||
| 		this.actionLock = false; | ||||
| 	} | ||||
|  | ||||
| 	get data () { | ||||
| 		return { | ||||
| 			type: this.type, | ||||
| 			status: this.status, | ||||
| 			vmid: this.status, | ||||
| 			name: this.name, | ||||
| 			node: this.node | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| 	set data (data) { | ||||
| 		if (data.status === "unknown") { | ||||
| 			data.status = "stopped"; | ||||
| 		} | ||||
| 		this.type = data.type; | ||||
| 		this.status = data.status; | ||||
| 		this.vmid = data.vmid; | ||||
| 		this.name = data.name; | ||||
| 		this.node = data.node; | ||||
| 		this.update(); | ||||
| 	} | ||||
|  | ||||
| 	update () { | ||||
| 		const vmidParagraph = this.shadowElement.querySelector("#instance-id"); | ||||
| 		vmidParagraph.innerText = this.vmid; | ||||
|  | ||||
| 		const nameParagraph = this.shadowElement.querySelector("#instance-name"); | ||||
| 		nameParagraph.innerText = this.name ? this.name : ""; | ||||
|  | ||||
| 		const typeParagraph = this.shadowElement.querySelector("#instance-type"); | ||||
| 		typeParagraph.innerText = this.type; | ||||
|  | ||||
| 		const statusParagraph = this.shadowElement.querySelector("#instance-status"); | ||||
| 		statusParagraph.innerText = this.status; | ||||
|  | ||||
| 		const statusIcon = this.shadowElement.querySelector("#instance-status-icon"); | ||||
| 		statusIcon.src = instancesConfig[this.status].status.src; | ||||
| 		statusIcon.alt = instancesConfig[this.status].status.alt; | ||||
|  | ||||
| 		const nodeNameParagraph = this.shadowElement.querySelector("#node-name"); | ||||
| 		nodeNameParagraph.innerText = this.node.name; | ||||
|  | ||||
| 		const nodeStatusParagraph = this.shadowElement.querySelector("#node-status"); | ||||
| 		nodeStatusParagraph.innerText = this.node.status; | ||||
|  | ||||
| 		const nodeStatusIcon = this.shadowElement.querySelector("#node-status-icon"); | ||||
| 		nodeStatusIcon.src = nodesConfig[this.node.status].status.src; | ||||
| 		nodeStatusIcon.alt = nodesConfig[this.node.status].status.src; | ||||
|  | ||||
| 		const powerButton = this.shadowElement.querySelector("#power-btn"); | ||||
| 		powerButton.src = instancesConfig[this.status].power.src; | ||||
| 		powerButton.alt = instancesConfig[this.status].power.alt; | ||||
| 		powerButton.title = instancesConfig[this.status].power.alt; | ||||
| 		if (instancesConfig[this.status].power.clickable) { | ||||
| 			powerButton.classList.add("clickable"); | ||||
| 			powerButton.onclick = this.handlePowerButton.bind(this); | ||||
| 		} | ||||
|  | ||||
| 		const configButton = this.shadowElement.querySelector("#configure-btn"); | ||||
| 		configButton.src = instancesConfig[this.status].config.src; | ||||
| 		configButton.alt = instancesConfig[this.status].config.alt; | ||||
| 		configButton.title = instancesConfig[this.status].config.alt; | ||||
| 		if (instancesConfig[this.status].config.clickable) { | ||||
| 			configButton.classList.add("clickable"); | ||||
| 			configButton.onclick = this.handleConfigButton.bind(this); | ||||
| 		} | ||||
|  | ||||
| 		const consoleButton = this.shadowElement.querySelector("#console-btn"); | ||||
| 		consoleButton.src = instancesConfig[this.status].console.src; | ||||
| 		consoleButton.alt = instancesConfig[this.status].console.alt; | ||||
| 		consoleButton.title = instancesConfig[this.status].console.alt; | ||||
| 		if (instancesConfig[this.status].console.clickable) { | ||||
| 			consoleButton.classList.add("clickable"); | ||||
| 			consoleButton.onclick = this.handleConsoleButton.bind(this); | ||||
| 		} | ||||
|  | ||||
| 		const deleteButton = this.shadowElement.querySelector("#delete-btn"); | ||||
| 		deleteButton.src = instancesConfig[this.status].delete.src; | ||||
| 		deleteButton.alt = instancesConfig[this.status].delete.alt; | ||||
| 		deleteButton.title = instancesConfig[this.status].delete.alt; | ||||
| 		if (instancesConfig[this.status].delete.clickable) { | ||||
| 			deleteButton.classList.add("clickable"); | ||||
| 			deleteButton.onclick = this.handleDeleteButton.bind(this); | ||||
| 		} | ||||
|  | ||||
| 		if (this.node.status !== "online") { | ||||
| 			powerButton.classList.add("hidden"); | ||||
| 			configButton.classList.add("hidden"); | ||||
| 			consoleButton.classList.add("hidden"); | ||||
| 			deleteButton.classList.add("hidden"); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	async handlePowerButton () { | ||||
| 		if (!this.actionLock) { | ||||
| 			const header = `${this.status === "running" ? "Stop" : "Start"} VM ${this.vmid}`; | ||||
| 			const body = `<p>Are you sure you want to ${this.status === "running" ? "stop" : "start"} VM</p><p>${this.vmid}</p>`; | ||||
|  | ||||
| 			dialog(header, body, async (result, form) => { | ||||
| 				if (result === "confirm") { | ||||
| 					this.actionLock = true; | ||||
| 					const targetAction = this.status === "running" ? "stop" : "start"; | ||||
| 					const targetStatus = this.status === "running" ? "stopped" : "running"; | ||||
| 					const prevStatus = this.status; | ||||
| 					this.status = "loading"; | ||||
|  | ||||
| 					this.update(); | ||||
|  | ||||
| 					const result = await requestPVE(`/nodes/${this.node.name}/${this.type}/${this.vmid}/status/${targetAction}`, "POST", { node: this.node.name, vmid: this.vmid }); | ||||
|  | ||||
| 					const waitFor = delay => new Promise(resolve => setTimeout(resolve, delay)); | ||||
|  | ||||
| 					while (true) { | ||||
| 						const taskStatus = await requestPVE(`/nodes/${this.node.name}/tasks/${result.data}/status`, "GET"); | ||||
| 						if (taskStatus.data.status === "stopped" && taskStatus.data.exitstatus === "OK") { // task stopped and was successful | ||||
| 							this.status = targetStatus; | ||||
| 							this.update(); | ||||
| 							this.actionLock = false; | ||||
| 							break; | ||||
| 						} | ||||
| 						else if (taskStatus.data.status === "stopped") { // task stopped but was not successful | ||||
| 							this.status = prevStatus; | ||||
| 							alert(`attempted to ${targetAction} ${this.vmid} but process returned stopped:${result.data.exitstatus}`); | ||||
| 							this.update(); | ||||
| 							this.actionLock = false; | ||||
| 							break; | ||||
| 						} | ||||
| 						else { // task has not stopped | ||||
| 							await waitFor(1000); | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	handleConfigButton () { | ||||
| 		if (!this.actionLock && this.status === "stopped") { // if the action lock is false, and the node is stopped, then navigate to the conig page with the node infor in the search query | ||||
| 			goToPage("config.html", { node: this.node.name, type: this.type, vmid: this.vmid }); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	handleConsoleButton () { | ||||
| 		if (!this.actionLock && this.status === "running") { | ||||
| 			const data = { console: `${this.type === "qemu" ? "kvm" : "lxc"}`, vmid: this.vmid, vmname: this.name, node: this.node.name, resize: "off", cmd: "" }; | ||||
| 			data[`${this.type === "qemu" ? "novnc" : "xtermjs"}`] = 1; | ||||
| 			goToURL(PVE, data, true); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	handleDeleteButton () { | ||||
| 		if (!this.actionLock && this.status === "stopped") { | ||||
| 			const header = `Delete VM ${this.vmid}`; | ||||
| 			const body = `<p>Are you sure you want to <strong>delete</strong> VM </p><p>${this.vmid}</p>`; | ||||
|  | ||||
| 			dialog(header, body, async (result, form) => { | ||||
| 				if (result === "confirm") { | ||||
| 					this.actionLock = true; | ||||
| 					this.status = "loading"; | ||||
| 					this.update(); | ||||
|  | ||||
| 					const action = {}; | ||||
| 					action.purge = 1; | ||||
| 					action["destroy-unreferenced-disks"] = 1; | ||||
|  | ||||
| 					const body = { | ||||
| 						node: this.node.name, | ||||
| 						type: this.type, | ||||
| 						vmid: this.vmid, | ||||
| 						action: JSON.stringify(action) | ||||
| 					}; | ||||
|  | ||||
| 					const result = await requestAPI("/instance", "DELETE", body); | ||||
| 					if (result.status === 200) { | ||||
| 						this.shadowElement.parentElement.removeChild(this.shadowElement); | ||||
| 					} | ||||
| 					else { | ||||
| 						alert(result.error); | ||||
| 						this.status = this.prevStatus; | ||||
| 						this.update(); | ||||
| 						this.actionLock = false; | ||||
| 					} | ||||
| 				} | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,43 +1,43 @@ | ||||
| import { requestTicket, goToPage, deleteAllCookies, requestPVE, setTitleAndHeader } from "./utils.js"; | ||||
| import { alert } from "./dialog.js"; | ||||
|  | ||||
| window.addEventListener("DOMContentLoaded", init); | ||||
|  | ||||
| async function init() { | ||||
| 	setTitleAndHeader(); | ||||
| 	await deleteAllCookies(); | ||||
| 	let formSubmitButton = document.querySelector("#submit"); | ||||
| 	let realms = await requestPVE("/access/domains", "GET"); | ||||
| 	let realmSelect = document.querySelector("#realm"); | ||||
| 	realms.data.forEach((element) => { | ||||
| 		realmSelect.add(new Option(element.comment, element.realm)); | ||||
| 		if ("default" in element && element.default === 1) { | ||||
| 			realmSelect.value = element.realm; | ||||
| 		} | ||||
| 	}); | ||||
| 	formSubmitButton.addEventListener("click", async (e) => { | ||||
| 		e.preventDefault(); | ||||
| 		let form = document.querySelector("form"); | ||||
| 		let formData = new FormData(form); | ||||
|  | ||||
| 		formSubmitButton.innerText = "Authenticating..."; | ||||
| 		let ticket = await requestTicket(formData.get("username"), formData.get("password"), formData.get("realm")); | ||||
| 		if (ticket.status === 200) { | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 			goToPage("index.html"); | ||||
| 		} | ||||
| 		else if (ticket.status === 401) { | ||||
| 			alert("Authenticaton failed."); | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 		} | ||||
| 		else if (ticket.status === 408) { | ||||
| 			alert("Network error."); | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 		} | ||||
| 		else { | ||||
| 			alert("An error occured."); | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 			console.error(ticket.error); | ||||
| 		} | ||||
| 	}); | ||||
| } | ||||
| import { requestTicket, goToPage, deleteAllCookies, requestPVE, setTitleAndHeader } from "./utils.js"; | ||||
| import { alert } from "./dialog.js"; | ||||
|  | ||||
| window.addEventListener("DOMContentLoaded", init); | ||||
|  | ||||
| async function init () { | ||||
| 	setTitleAndHeader(); | ||||
| 	await deleteAllCookies(); | ||||
| 	const formSubmitButton = document.querySelector("#submit"); | ||||
| 	const realms = await requestPVE("/access/domains", "GET"); | ||||
| 	const realmSelect = document.querySelector("#realm"); | ||||
| 	realms.data.forEach((element) => { | ||||
| 		realmSelect.add(new Option(element.comment, element.realm)); | ||||
| 		if ("default" in element && element.default === 1) { | ||||
| 			realmSelect.value = element.realm; | ||||
| 		} | ||||
| 	}); | ||||
| 	formSubmitButton.addEventListener("click", async (e) => { | ||||
| 		e.preventDefault(); | ||||
| 		const form = document.querySelector("form"); | ||||
| 		const formData = new FormData(form); | ||||
|  | ||||
| 		formSubmitButton.innerText = "Authenticating..."; | ||||
| 		const ticket = await requestTicket(formData.get("username"), formData.get("password"), formData.get("realm")); | ||||
| 		if (ticket.status === 200) { | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 			goToPage("index.html"); | ||||
| 		} | ||||
| 		else if (ticket.status === 401) { | ||||
| 			alert("Authenticaton failed."); | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 		} | ||||
| 		else if (ticket.status === 408) { | ||||
| 			alert("Network error."); | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 		} | ||||
| 		else { | ||||
| 			alert("An error occured."); | ||||
| 			formSubmitButton.innerText = "LOGIN"; | ||||
| 			console.error(ticket.error); | ||||
| 		} | ||||
| 	}); | ||||
| } | ||||
|   | ||||
							
								
								
									
										506
									
								
								scripts/utils.js
									
									
									
									
									
								
							
							
						
						
									
										506
									
								
								scripts/utils.js
									
									
									
									
									
								
							| @@ -1,253 +1,253 @@ | ||||
| import { API, organization } from "/vars.js"; | ||||
|  | ||||
| export const resources_config = { | ||||
| 	disk: { | ||||
| 		actionBarOrder: ["move", "resize", "detach_attach", "delete"], | ||||
| 		lxc: { | ||||
| 			prefixOrder: ["rootfs", "mp", "unused"], | ||||
| 			rootfs: { name: "ROOTFS", icon: "images/resources/drive.svg", actions: ["move", "resize"] }, | ||||
| 			mp: { name: "MP", icon: "images/resources/drive.svg", actions: ["detach", "move", "reassign", "resize"] }, | ||||
| 			unused: { name: "UNUSED", icon: "images/resources/drive.svg", actions: ["attach", "delete", "reassign"] } | ||||
| 		}, | ||||
| 		qemu: { | ||||
| 			prefixOrder: ["ide", "sata", "unused"], | ||||
| 			ide: { name: "IDE", icon: "images/resources/disk.svg", actions: ["delete"] }, | ||||
| 			sata: { name: "SATA", icon: "images/resources/drive.svg", actions: ["detach", "move", "reassign", "resize"] }, | ||||
| 			unused: { name: "UNUSED", icon: "images/resources/drive.svg", actions: ["attach", "delete", "reassign"] } | ||||
| 		} | ||||
| 	}, | ||||
| 	network: { | ||||
| 		prefix: "net" | ||||
| 	}, | ||||
| 	pcie: { | ||||
| 		prefix: "hostpci" | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export const instances_config = { | ||||
| 	running: { | ||||
| 		status: { | ||||
| 			src: "images/status/active.svg", | ||||
| 			alt: "Instance is running", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		power: { | ||||
| 			src: "images/actions/instance/stop.svg", | ||||
| 			alt: "Shutdown Instance", | ||||
| 			clickable: true, | ||||
| 		}, | ||||
| 		config: { | ||||
| 			src: "images/actions/instance/config-inactive.svg", | ||||
| 			alt: "Change Configuration (Inactive)", | ||||
| 			clickable: false, | ||||
| 		}, | ||||
| 		console: { | ||||
| 			src: "images/actions/instance/console-active.svg", | ||||
| 			alt: "Open Console", | ||||
| 			clickable: true, | ||||
| 		}, | ||||
| 		delete: { | ||||
| 			src: "images/actions/delete-inactive.svg", | ||||
| 			alt: "Delete Instance (Inactive)", | ||||
| 			clickable: false, | ||||
| 		} | ||||
| 	}, | ||||
| 	stopped: { | ||||
| 		status: { | ||||
| 			src: "images/status/inactive.svg", | ||||
| 			alt: "Instance is stopped", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		power: { | ||||
| 			src: "images/actions/instance/start.svg", | ||||
| 			alt: "Start Instance", | ||||
| 			clickable: true, | ||||
| 		}, | ||||
| 		config: { | ||||
| 			src: "images/actions/instance/config-active.svg", | ||||
| 			alt: "Change Configuration", | ||||
| 			clickable: true, | ||||
| 		}, | ||||
| 		console: { | ||||
| 			src: "images/actions/instance/console-inactive.svg", | ||||
| 			alt: "Open Console (Inactive)", | ||||
| 			clickable: false, | ||||
| 		}, | ||||
| 		delete: { | ||||
| 			src: "images/actions/delete-active.svg", | ||||
| 			alt: "Delete Instance", | ||||
| 			clickable: true, | ||||
| 		} | ||||
| 	}, | ||||
| 	loading: { | ||||
| 		status: { | ||||
| 			src: "images/status/loading.svg", | ||||
| 			alt: "Instance is loading", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		power: { | ||||
| 			src: "images/status/loading.svg", | ||||
| 			alt: "Loading Instance", | ||||
| 			clickable: false, | ||||
| 		}, | ||||
| 		config: { | ||||
| 			src: "images/actions/instance/config-inactive.svg", | ||||
| 			alt: "Change Configuration (Inactive)", | ||||
| 			clickable: false, | ||||
| 		}, | ||||
| 		console: { | ||||
| 			src: "images/actions/instance/console-inactive.svg", | ||||
| 			alt: "Open Console (Inactive)", | ||||
| 			clickable: false, | ||||
| 		}, | ||||
| 		delete: { | ||||
| 			src: "images/actions/delete-inactive.svg", | ||||
| 			alt: "Delete Instance (Inactive)", | ||||
| 			clickable: false, | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export const nodes_config = { | ||||
| 	online: { | ||||
| 		status: { | ||||
| 			src: "images/status/active.svg", | ||||
| 			alt: "Node is online" | ||||
| 		} | ||||
| 	}, | ||||
| 	offline: { | ||||
| 		status: { | ||||
| 			src: "images/status/inactive.svg", | ||||
| 			alt: "Node is offline" | ||||
| 		} | ||||
| 	}, | ||||
| 	uknown: { | ||||
| 		status: { | ||||
| 			src: "images/status/inactive.svg", | ||||
| 			alt: "Node is offline" | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function getCookie(cname) { | ||||
| 	let name = cname + "="; | ||||
| 	let decodedCookie = decodeURIComponent(document.cookie); | ||||
| 	let ca = decodedCookie.split(";"); | ||||
| 	for (let i = 0; i < ca.length; i++) { | ||||
| 		let c = ca[i]; | ||||
| 		while (c.charAt(0) === " ") { | ||||
| 			c = c.substring(1); | ||||
| 		} | ||||
| 		if (c.indexOf(name) === 0) { | ||||
| 			return c.substring(name.length, c.length); | ||||
| 		} | ||||
| 	} | ||||
| 	return ""; | ||||
| } | ||||
|  | ||||
| export async function requestTicket(username, password, realm) { | ||||
| 	let response = await requestAPI("/ticket", "POST", { username: `${username}@${realm}`, password: password }, false); | ||||
| 	return response; | ||||
| } | ||||
|  | ||||
| export async function requestPVE(path, method, body = null) { | ||||
| 	let prms = new URLSearchParams(body); | ||||
| 	let content = { | ||||
| 		method: method, | ||||
| 		mode: "cors", | ||||
| 		credentials: "include", | ||||
| 		headers: { | ||||
| 			"Content-Type": "application/x-www-form-urlencoded" | ||||
| 		} | ||||
| 	} | ||||
| 	if (method === "POST") { | ||||
| 		content.body = prms.toString(); | ||||
| 		content.headers.CSRFPreventionToken = getCookie("CSRFPreventionToken"); | ||||
| 	} | ||||
|  | ||||
| 	let response = await request(`${API}/proxmox${path}`, content); | ||||
| 	return response; | ||||
| } | ||||
|  | ||||
| export async function requestAPI(path, method, body = null) { | ||||
| 	let prms = new URLSearchParams(body); | ||||
| 	let content = { | ||||
| 		method: method, | ||||
| 		mode: "cors", | ||||
| 		credentials: "include", | ||||
| 		headers: { | ||||
| 			"Content-Type": "application/x-www-form-urlencoded" | ||||
| 		} | ||||
| 	} | ||||
| 	if (method === "POST" || method === "DELETE") { | ||||
| 		content.headers.CSRFPreventionToken = getCookie("CSRFPreventionToken"); | ||||
| 	} | ||||
| 	if (body) { | ||||
| 		content.body = prms.toString(); | ||||
| 	} | ||||
|  | ||||
| 	let response = await request(`${API}${path}`, content); | ||||
| 	return response; | ||||
| } | ||||
|  | ||||
| async function request(url, content) { | ||||
| 	let response = await fetch(url, content); | ||||
| 	let data = null; | ||||
| 	try { | ||||
| 		data = await response.json(); | ||||
| 		data.status = response.status; | ||||
| 	} | ||||
| 	catch { | ||||
| 		data = null; | ||||
| 	} | ||||
|  | ||||
| 	if (!response.ok) { | ||||
| 		return { status: response.status, error: data ? data.error : response.status }; | ||||
| 	} | ||||
| 	else { | ||||
| 		data.status = response.status; | ||||
| 		return data ? data : response; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function goToPage(page, data = {}, newwindow = false) { | ||||
| 	let url = new URL(`https://${window.location.host}/${page}`); | ||||
| 	for (let k in data) { | ||||
| 		url.searchParams.append(k, data[k]); | ||||
| 	} | ||||
|  | ||||
| 	if (newwindow) { | ||||
| 		window.open(url, `${organization} - client`, "height=480,width=848"); | ||||
| 	} | ||||
| 	else { | ||||
| 		window.location.assign(url.toString()); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function goToURL(href, data = {}, newwindow = false) { | ||||
| 	let url = new URL(href); | ||||
| 	for (let k in data) { | ||||
| 		url.searchParams.append(k, data[k]); | ||||
| 	} | ||||
|  | ||||
| 	if (newwindow) { | ||||
| 		window.open(url, `${organization} - client`, "height=480,width=848"); | ||||
| 	} | ||||
| 	else { | ||||
| 		window.location.assign(url.toString()); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function getURIData() { | ||||
| 	let url = new URL(window.location.href); | ||||
| 	return Object.fromEntries(url.searchParams); | ||||
| } | ||||
|  | ||||
| export async function deleteAllCookies() { | ||||
| 	await requestAPI("/ticket", "DELETE"); | ||||
| } | ||||
|  | ||||
| export function setTitleAndHeader() { | ||||
| 	document.title = `${organization} - client`; | ||||
| 	document.querySelector("h1").innerText = organization; | ||||
| } | ||||
| import { API, organization } from "../../../../../vars.js"; | ||||
|  | ||||
| export const resourcesConfig = { | ||||
| 	disk: { | ||||
| 		actionBarOrder: ["move", "resize", "detach_attach", "delete"], | ||||
| 		lxc: { | ||||
| 			prefixOrder: ["rootfs", "mp", "unused"], | ||||
| 			rootfs: { name: "ROOTFS", icon: "images/resources/drive.svg", actions: ["move", "resize"] }, | ||||
| 			mp: { name: "MP", icon: "images/resources/drive.svg", actions: ["detach", "move", "reassign", "resize"] }, | ||||
| 			unused: { name: "UNUSED", icon: "images/resources/drive.svg", actions: ["attach", "delete", "reassign"] } | ||||
| 		}, | ||||
| 		qemu: { | ||||
| 			prefixOrder: ["ide", "sata", "unused"], | ||||
| 			ide: { name: "IDE", icon: "images/resources/disk.svg", actions: ["delete"] }, | ||||
| 			sata: { name: "SATA", icon: "images/resources/drive.svg", actions: ["detach", "move", "reassign", "resize"] }, | ||||
| 			unused: { name: "UNUSED", icon: "images/resources/drive.svg", actions: ["attach", "delete", "reassign"] } | ||||
| 		} | ||||
| 	}, | ||||
| 	network: { | ||||
| 		prefix: "net" | ||||
| 	}, | ||||
| 	pcie: { | ||||
| 		prefix: "hostpci" | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| export const instancesConfig = { | ||||
| 	running: { | ||||
| 		status: { | ||||
| 			src: "images/status/active.svg", | ||||
| 			alt: "Instance is running", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		power: { | ||||
| 			src: "images/actions/instance/stop.svg", | ||||
| 			alt: "Shutdown Instance", | ||||
| 			clickable: true | ||||
| 		}, | ||||
| 		config: { | ||||
| 			src: "images/actions/instance/config-inactive.svg", | ||||
| 			alt: "Change Configuration (Inactive)", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		console: { | ||||
| 			src: "images/actions/instance/console-active.svg", | ||||
| 			alt: "Open Console", | ||||
| 			clickable: true | ||||
| 		}, | ||||
| 		delete: { | ||||
| 			src: "images/actions/delete-inactive.svg", | ||||
| 			alt: "Delete Instance (Inactive)", | ||||
| 			clickable: false | ||||
| 		} | ||||
| 	}, | ||||
| 	stopped: { | ||||
| 		status: { | ||||
| 			src: "images/status/inactive.svg", | ||||
| 			alt: "Instance is stopped", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		power: { | ||||
| 			src: "images/actions/instance/start.svg", | ||||
| 			alt: "Start Instance", | ||||
| 			clickable: true | ||||
| 		}, | ||||
| 		config: { | ||||
| 			src: "images/actions/instance/config-active.svg", | ||||
| 			alt: "Change Configuration", | ||||
| 			clickable: true | ||||
| 		}, | ||||
| 		console: { | ||||
| 			src: "images/actions/instance/console-inactive.svg", | ||||
| 			alt: "Open Console (Inactive)", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		delete: { | ||||
| 			src: "images/actions/delete-active.svg", | ||||
| 			alt: "Delete Instance", | ||||
| 			clickable: true | ||||
| 		} | ||||
| 	}, | ||||
| 	loading: { | ||||
| 		status: { | ||||
| 			src: "images/status/loading.svg", | ||||
| 			alt: "Instance is loading", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		power: { | ||||
| 			src: "images/status/loading.svg", | ||||
| 			alt: "Loading Instance", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		config: { | ||||
| 			src: "images/actions/instance/config-inactive.svg", | ||||
| 			alt: "Change Configuration (Inactive)", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		console: { | ||||
| 			src: "images/actions/instance/console-inactive.svg", | ||||
| 			alt: "Open Console (Inactive)", | ||||
| 			clickable: false | ||||
| 		}, | ||||
| 		delete: { | ||||
| 			src: "images/actions/delete-inactive.svg", | ||||
| 			alt: "Delete Instance (Inactive)", | ||||
| 			clickable: false | ||||
| 		} | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| export const nodesConfig = { | ||||
| 	online: { | ||||
| 		status: { | ||||
| 			src: "images/status/active.svg", | ||||
| 			alt: "Node is online" | ||||
| 		} | ||||
| 	}, | ||||
| 	offline: { | ||||
| 		status: { | ||||
| 			src: "images/status/inactive.svg", | ||||
| 			alt: "Node is offline" | ||||
| 		} | ||||
| 	}, | ||||
| 	uknown: { | ||||
| 		status: { | ||||
| 			src: "images/status/inactive.svg", | ||||
| 			alt: "Node is offline" | ||||
| 		} | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| export function getCookie (cname) { | ||||
| 	const name = cname + "="; | ||||
| 	const decodedCookie = decodeURIComponent(document.cookie); | ||||
| 	const ca = decodedCookie.split(";"); | ||||
| 	for (let i = 0; i < ca.length; i++) { | ||||
| 		let c = ca[i]; | ||||
| 		while (c.charAt(0) === " ") { | ||||
| 			c = c.substring(1); | ||||
| 		} | ||||
| 		if (c.indexOf(name) === 0) { | ||||
| 			return c.substring(name.length, c.length); | ||||
| 		} | ||||
| 	} | ||||
| 	return ""; | ||||
| } | ||||
|  | ||||
| export async function requestTicket (username, password, realm) { | ||||
| 	const response = await requestAPI("/ticket", "POST", { username: `${username}@${realm}`, password }, false); | ||||
| 	return response; | ||||
| } | ||||
|  | ||||
| export async function requestPVE (path, method, body = null) { | ||||
| 	const prms = new URLSearchParams(body); | ||||
| 	const content = { | ||||
| 		method, | ||||
| 		mode: "cors", | ||||
| 		credentials: "include", | ||||
| 		headers: { | ||||
| 			"Content-Type": "application/x-www-form-urlencoded" | ||||
| 		} | ||||
| 	}; | ||||
| 	if (method === "POST") { | ||||
| 		content.body = prms.toString(); | ||||
| 		content.headers.CSRFPreventionToken = getCookie("CSRFPreventionToken"); | ||||
| 	} | ||||
|  | ||||
| 	const response = await request(`${API}/proxmox${path}`, content); | ||||
| 	return response; | ||||
| } | ||||
|  | ||||
| export async function requestAPI (path, method, body = null) { | ||||
| 	const prms = new URLSearchParams(body); | ||||
| 	const content = { | ||||
| 		method, | ||||
| 		mode: "cors", | ||||
| 		credentials: "include", | ||||
| 		headers: { | ||||
| 			"Content-Type": "application/x-www-form-urlencoded" | ||||
| 		} | ||||
| 	}; | ||||
| 	if (method === "POST" || method === "DELETE") { | ||||
| 		content.headers.CSRFPreventionToken = getCookie("CSRFPreventionToken"); | ||||
| 	} | ||||
| 	if (body) { | ||||
| 		content.body = prms.toString(); | ||||
| 	} | ||||
|  | ||||
| 	const response = await request(`${API}${path}`, content); | ||||
| 	return response; | ||||
| } | ||||
|  | ||||
| async function request (url, content) { | ||||
| 	const response = await fetch(url, content); | ||||
| 	let data = null; | ||||
| 	try { | ||||
| 		data = await response.json(); | ||||
| 		data.status = response.status; | ||||
| 	} | ||||
| 	catch { | ||||
| 		data = null; | ||||
| 	} | ||||
|  | ||||
| 	if (!response.ok) { | ||||
| 		return { status: response.status, error: data ? data.error : response.status }; | ||||
| 	} | ||||
| 	else { | ||||
| 		data.status = response.status; | ||||
| 		return data || response; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function goToPage (page, data = {}, newwindow = false) { | ||||
| 	const url = new URL(`https://${window.location.host}/${page}`); | ||||
| 	for (const k in data) { | ||||
| 		url.searchParams.append(k, data[k]); | ||||
| 	} | ||||
|  | ||||
| 	if (newwindow) { | ||||
| 		window.open(url, `${organization} - client`, "height=480,width=848"); | ||||
| 	} | ||||
| 	else { | ||||
| 		window.location.assign(url.toString()); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function goToURL (href, data = {}, newwindow = false) { | ||||
| 	const url = new URL(href); | ||||
| 	for (const k in data) { | ||||
| 		url.searchParams.append(k, data[k]); | ||||
| 	} | ||||
|  | ||||
| 	if (newwindow) { | ||||
| 		window.open(url, `${organization} - client`, "height=480,width=848"); | ||||
| 	} | ||||
| 	else { | ||||
| 		window.location.assign(url.toString()); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| export function getURIData () { | ||||
| 	const url = new URL(window.location.href); | ||||
| 	return Object.fromEntries(url.searchParams); | ||||
| } | ||||
|  | ||||
| export async function deleteAllCookies () { | ||||
| 	await requestAPI("/ticket", "DELETE"); | ||||
| } | ||||
|  | ||||
| export function setTitleAndHeader () { | ||||
| 	document.title = `${organization} - client`; | ||||
| 	document.querySelector("h1").innerText = organization; | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user