add user form,
implement some admin handling, improve addResourceLine
This commit is contained in:
		| @@ -1,4 +1,4 @@ | |||||||
| import { setTitleAndHeader, setAppearance, requestAPI, goToPage } from "./utils.js"; | import { setTitleAndHeader, setAppearance, requestAPI, goToPage, isEmpty } from "./utils.js"; | ||||||
|  |  | ||||||
| window.addEventListener("DOMContentLoaded", init); | window.addEventListener("DOMContentLoaded", init); | ||||||
|  |  | ||||||
| @@ -52,6 +52,9 @@ class UserCard extends HTMLElement { | |||||||
| 				* { | 				* { | ||||||
| 					margin: 0; | 					margin: 0; | ||||||
| 				} | 				} | ||||||
|  | 				p { | ||||||
|  | 					width: 100%; | ||||||
|  | 				} | ||||||
| 			</style> | 			</style> | ||||||
| 			<div class="w3-row" style="margin-top: 1em; margin-bottom: 1em;"> | 			<div class="w3-row" style="margin-top: 1em; margin-bottom: 1em;"> | ||||||
| 				<p class="w3-col l4 m4 s6" id="user-name"> | 				<p class="w3-col l4 m4 s6" id="user-name"> | ||||||
| @@ -95,7 +98,12 @@ class UserCard extends HTMLElement { | |||||||
| 		nameParagraph.innerText = this.username; | 		nameParagraph.innerText = this.username; | ||||||
|  |  | ||||||
| 		const groupsParagraph = this.shadowRoot.querySelector("#user-groups"); | 		const groupsParagraph = this.shadowRoot.querySelector("#user-groups"); | ||||||
| 		groupsParagraph.innerText = `${this.groups.toString()}`; | 		if (isEmpty(this.groups)) { | ||||||
|  | 			groupsParagraph.innerHTML = " "; | ||||||
|  | 		} | ||||||
|  | 		else { | ||||||
|  | 			groupsParagraph.innerText = this.groups.toString(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		const adminParagraph = this.shadowRoot.querySelector("#user-admin"); | 		const adminParagraph = this.shadowRoot.querySelector("#user-admin"); | ||||||
| 		adminParagraph.innerText = this.admin; | 		adminParagraph.innerText = this.admin; | ||||||
|   | |||||||
							
								
								
									
										496
									
								
								scripts/user.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										496
									
								
								scripts/user.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,496 @@ | |||||||
|  | import { goToPage, getURIData, setTitleAndHeader, setAppearance, requestAPI, resourcesConfig, mergeDeep, addResourceLine, setSVGAlt, setSVGSrc } from "./utils.js"; | ||||||
|  | import { alert, dialog } from "./dialog.js"; | ||||||
|  |  | ||||||
|  | window.addEventListener("DOMContentLoaded", init); | ||||||
|  |  | ||||||
|  | let username; | ||||||
|  | let userData; | ||||||
|  | let allGroups; | ||||||
|  | let allNodes; | ||||||
|  | let allPools; | ||||||
|  | let clusterResourceConfig; | ||||||
|  |  | ||||||
|  | const resourceInputTypes = { // input types for each resource for config page | ||||||
|  | 	cpu: { | ||||||
|  | 		element: "interactive-list", | ||||||
|  | 		align: "start" | ||||||
|  | 	}, | ||||||
|  | 	cores: { | ||||||
|  | 		element: "input", | ||||||
|  | 		attributes: { | ||||||
|  | 			type: "number" | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
|  | 	memory: { | ||||||
|  | 		element: "input", | ||||||
|  | 		attributes: { | ||||||
|  | 			type: "number" | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
|  | 	swap: { | ||||||
|  | 		element: "input", | ||||||
|  | 		attributes: { | ||||||
|  | 			type: "number" | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
|  | 	network: { | ||||||
|  | 		element: "input", | ||||||
|  | 		attributes: { | ||||||
|  | 			type: "number" | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
|  | 	pci: { | ||||||
|  | 		element: "interactive-list", | ||||||
|  | 		align: "start" | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | class InteractiveList extends HTMLElement { | ||||||
|  | 	#name; | ||||||
|  | 	#addText; | ||||||
|  |  | ||||||
|  | 	constructor () { | ||||||
|  | 		super(); | ||||||
|  | 		this.attachShadow({ mode: "open" }); | ||||||
|  | 		this.shadowRoot.innerHTML = ` | ||||||
|  | 			<link rel="stylesheet" href="modules/w3.css"> | ||||||
|  | 			<link rel="stylesheet" href="https://www.w3schools.com/w3css/4/w3.css"> | ||||||
|  | 			<link rel="stylesheet" href="css/style.css"> | ||||||
|  | 			<style> | ||||||
|  | 				#wrapper { | ||||||
|  | 					border: 1px dotted var(--main-text-color); | ||||||
|  | 					padding: 8px; | ||||||
|  | 				} | ||||||
|  | 			</style> | ||||||
|  | 			<div class="w3-center" id="wrapper"> | ||||||
|  | 				<div id="container"></div> | ||||||
|  | 				<svg id="add-btn" class="clickable" tabindex="0" role="button"><use></use></svg> | ||||||
|  | 			</div> | ||||||
|  | 		`; | ||||||
|  | 		this.addBtn = this.shadowRoot.querySelector("#add-btn"); | ||||||
|  | 		this.addBtn.onclick = this.#handleAdd.bind(this); | ||||||
|  | 		this.container = this.shadowRoot.querySelector("#container"); | ||||||
|  | 		setSVGSrc(this.addBtn, "images/common/add.svg"); | ||||||
|  | 		setSVGAlt(this.addBtn, "Add Item"); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	get name () { | ||||||
|  | 		return this.#name; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	set name (name) { | ||||||
|  | 		this.#name = name; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	get addText () { | ||||||
|  | 		return this.#addText; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	set addText (addText) { | ||||||
|  | 		this.#addText = addText; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	get value () { | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	set value (value) { | ||||||
|  | 		for (const item of value) { | ||||||
|  | 			this.#addItem(item); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	#addItem (item) { | ||||||
|  | 		const itemElem = document.createElement("interactive-list-match-item"); | ||||||
|  | 		itemElem.name = item.name; | ||||||
|  | 		itemElem.match = item.match; | ||||||
|  | 		itemElem.max = item.max; | ||||||
|  | 		this.container.appendChild(itemElem); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	#handleAdd () { | ||||||
|  | 		const header = `Add New ${this.#name} Rule`; | ||||||
|  |  | ||||||
|  | 		const body = ` | ||||||
|  | 			<form method="dialog" class="input-grid" style="grid-template-columns: auto 1fr;" id="form"> | ||||||
|  | 				<label for="name">Rule Name</label> | ||||||
|  | 				<input class="w3-input w3-border" name="name" id="name" type="text" required> | ||||||
|  | 				<label for="match">Matching Pattern</label> | ||||||
|  | 				<input class="w3-input w3-border" name="match" id="match" type="text" required> | ||||||
|  | 				<label for="max">Max Resource</label> | ||||||
|  | 				<input class="w3-input w3-border" name="max" id="max" type="number" required> | ||||||
|  | 			</form> | ||||||
|  | 		`; | ||||||
|  |  | ||||||
|  | 		dialog(header, body, (result, form) => { | ||||||
|  | 			if (result === "confirm") { | ||||||
|  | 				const newItem = { | ||||||
|  | 					name: form.get("name"), | ||||||
|  | 					match: form.get("match"), | ||||||
|  | 					max: form.get("max") | ||||||
|  | 				}; | ||||||
|  | 				this.#addItem(newItem); | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class InteractiveListMatchItem extends HTMLElement { | ||||||
|  | 	#name; | ||||||
|  | 	#match; | ||||||
|  | 	#max; | ||||||
|  |  | ||||||
|  | 	#nameElem; | ||||||
|  | 	#matchElem; | ||||||
|  | 	#maxElem; | ||||||
|  |  | ||||||
|  | 	constructor () { | ||||||
|  | 		super(); | ||||||
|  | 		this.attachShadow({ mode: "open" }); | ||||||
|  | 		this.shadowRoot.innerHTML = ` | ||||||
|  | 			<link rel="stylesheet" href="modules/w3.css"> | ||||||
|  | 			<link rel="stylesheet" href="https://www.w3schools.com/w3css/4/w3.css"> | ||||||
|  | 			<link rel="stylesheet" href="css/style.css"> | ||||||
|  | 			<style> | ||||||
|  | 				#container { | ||||||
|  | 					text-align: left; | ||||||
|  | 				} | ||||||
|  | 				p { | ||||||
|  | 					margin: 0; | ||||||
|  | 					padding: 0; | ||||||
|  | 				} | ||||||
|  | 				svg { | ||||||
|  | 					margin-top: calc(-0.5em + 0.5lh); | ||||||
|  | 				} | ||||||
|  | 			</style> | ||||||
|  | 			<div id="container" class="w3-row"> | ||||||
|  | 				<p class="w3-col l2 w3-hide-medium w3-hide-small"><span id="name"></span></p> | ||||||
|  | 				<p class="w3-col l7 m9 s11">match="<span id="match"></span>"</p> | ||||||
|  | 				<p class="w3-col l2 m2 w3-hide-small">max=<span id="max"></span></p> | ||||||
|  | 				<svg id="delete-btn" class="w3-col l1 m1 s1 clickable" tabindex="0" role="button"><use></use></svg> | ||||||
|  | 			</div> | ||||||
|  | 		`; | ||||||
|  | 		this.#nameElem = this.shadowRoot.querySelector("#name"); | ||||||
|  | 		this.#matchElem = this.shadowRoot.querySelector("#match"); | ||||||
|  | 		this.#maxElem = this.shadowRoot.querySelector("#max"); | ||||||
|  |  | ||||||
|  | 		this.deleteBtn = this.shadowRoot.querySelector("#delete-btn"); | ||||||
|  | 		this.deleteBtn.onclick = this.#handleDelete.bind(this); | ||||||
|  | 		setSVGSrc(this.deleteBtn, "images/actions/delete-active.svg"); | ||||||
|  | 		setSVGAlt(this.deleteBtn, "Delete Item"); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	#update () { | ||||||
|  | 		this.#nameElem.innerText = this.#name; | ||||||
|  | 		this.#matchElem.innerText = this.#match; | ||||||
|  | 		this.#maxElem.innerText = this.#max; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	get name () { | ||||||
|  | 		return this.#name; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	set name (name) { | ||||||
|  | 		this.#name = name; | ||||||
|  | 		this.#update(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	get match () { | ||||||
|  | 		return this.#match; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	set match (match) { | ||||||
|  | 		this.#match = match; | ||||||
|  | 		this.#update(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	get max () { | ||||||
|  | 		return this.#max; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	set max (max) { | ||||||
|  | 		this.#max = max; | ||||||
|  | 		this.#update(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	#handleDelete () { | ||||||
|  | 		const header = `Delete ${this.name}`; | ||||||
|  | 		const body = `<p>Are you sure you want to <strong>delete</strong> ${this.name}</p>`; | ||||||
|  |  | ||||||
|  | 		dialog(header, body, async (result, form) => { | ||||||
|  | 			if (result === "confirm") { | ||||||
|  | 				if (this.parentElement) { | ||||||
|  | 					this.parentElement.removeChild(this); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | customElements.define("interactive-list", InteractiveList); | ||||||
|  | customElements.define("interactive-list-match-item", InteractiveListMatchItem); | ||||||
|  |  | ||||||
|  | const resourcesConfigPage = mergeDeep({}, resourcesConfig, resourceInputTypes); | ||||||
|  |  | ||||||
|  | async function init () { | ||||||
|  | 	setAppearance(); | ||||||
|  | 	setTitleAndHeader(); | ||||||
|  | 	const cookie = document.cookie; | ||||||
|  | 	if (cookie === "") { | ||||||
|  | 		goToPage("login.html"); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	const uriData = getURIData(); | ||||||
|  | 	username = uriData.username; | ||||||
|  |  | ||||||
|  | 	document.querySelector("#name").innerHTML = document.querySelector("#name").innerHTML.replace("%{username}", username); | ||||||
|  |  | ||||||
|  | 	await getUser(); | ||||||
|  | 	await populateGroups(); | ||||||
|  | 	await populateResources(); | ||||||
|  | 	await populateCluster(); | ||||||
|  |  | ||||||
|  | 	clusterResourceConfig = (await requestAPI("/global/config/resources")).resources; | ||||||
|  |  | ||||||
|  | 	document.querySelector("#exit").addEventListener("click", handleFormExit); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function getUser () { | ||||||
|  | 	userData = (await requestAPI(`/access/users/${username}`)).user; | ||||||
|  | 	allGroups = (await requestAPI("/access/groups")).groups; | ||||||
|  | 	allNodes = (await requestAPI("/cluster/nodes")).nodes; | ||||||
|  | 	allPools = (await requestAPI("/cluster/pools")).pools; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function populateGroups () { | ||||||
|  | 	const groupsDisabled = document.querySelector("#groups-disabled"); | ||||||
|  | 	const groupsEnabled = document.querySelector("#groups-enabled"); | ||||||
|  | 	// for each group in cluster | ||||||
|  | 	for (const groupName of Object.keys(allGroups)) { | ||||||
|  | 		const group = allGroups[groupName]; | ||||||
|  | 		const item = document.createElement("draggable-item"); | ||||||
|  | 		item.data = group; | ||||||
|  | 		item.innerHTML = ` | ||||||
|  | 			<div style="display: grid; grid-template-columns: auto 1fr; column-gap: 10px; align-items: center;"> | ||||||
|  | 				<svg id="drag" role="application" aria-label="drag icon"><title>drag icon</title><use href="images/actions/drag.svg#symb"></use></svg> | ||||||
|  | 				<p style="margin: 0px;">${group.attributes.cn}</p> | ||||||
|  | 			</div> | ||||||
|  | 		`; | ||||||
|  | 		// if user in group | ||||||
|  | 		if (userData.attributes.memberOf.indexOf(group.dn) !== -1) { | ||||||
|  | 			groupsEnabled.append(item); | ||||||
|  | 		} | ||||||
|  | 		// user is not in group | ||||||
|  | 		else { | ||||||
|  | 			groupsDisabled.append(item); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function populateResources () { | ||||||
|  | 	const field = document.querySelector("#resources"); | ||||||
|  | 	for (const resourceName of Object.keys(userData.resources)) { | ||||||
|  | 		const resource = userData.resources[resourceName]; | ||||||
|  | 		if (resourcesConfigPage[resourceName]) { | ||||||
|  | 			const resourceConfig = resourcesConfigPage[resourceName]; | ||||||
|  | 			let resourceLine; | ||||||
|  |  | ||||||
|  | 			if (resourceName === "cpu" || resourceName === "pci") { | ||||||
|  | 				resourceLine = addResourceLine(resourcesConfigPage, field, resourceName, { value: resource.global }, "(Global)"); | ||||||
|  | 			} | ||||||
|  | 			else { | ||||||
|  | 				resourceLine = addResourceLine(resourcesConfigPage, field, resourceName, { value: resource.global.max }, "(Global)"); | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			postPopulateResourceLine(field, resourceName, "global", resourceConfig, resourceLine); | ||||||
|  |  | ||||||
|  | 			for (const nodeSpecificName of Object.keys(resource.nodes)) { // for each node specific, add a line with the node name as a prefix | ||||||
|  | 				if (resourceName === "cpu" || resourceName === "pci") { | ||||||
|  | 					resourceLine = addResourceLine(resourcesConfigPage, field, resourceName, { value: resource.nodes[nodeSpecificName] }, `(${nodeSpecificName})`); | ||||||
|  | 				} | ||||||
|  | 				else { | ||||||
|  | 					resourceLine = addResourceLine(resourcesConfigPage, field, resourceName, { value: resource.nodes[nodeSpecificName].max }, `(${nodeSpecificName})`); | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				postPopulateResourceLine(field, resourceName, nodeSpecificName, resourceConfig, resourceLine); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	document.querySelector("#resource-add").addEventListener("click", handleResourceAdd); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function postPopulateResourceLine (field, resourceName, resourceScope, resourceConfig, resourceLine) { | ||||||
|  | 	const deleteBtn = document.createElementNS("http://www.w3.org/2000/svg", "svg"); | ||||||
|  | 	deleteBtn.classList.add("clickable"); | ||||||
|  | 	setSVGSrc(deleteBtn, "images/actions/delete-active.svg"); | ||||||
|  | 	setSVGAlt(deleteBtn, "Delete Rule"); | ||||||
|  | 	field.appendChild(deleteBtn); | ||||||
|  |  | ||||||
|  | 	resourceLine.field = field; | ||||||
|  | 	resourceLine.deleteBtn = deleteBtn; | ||||||
|  | 	deleteBtn.onclick = handleResourceDelete.bind(resourceLine); | ||||||
|  |  | ||||||
|  | 	if (resourceConfig.align && resourceConfig.align === "start") { | ||||||
|  | 		resourceLine.icon.style.alignSelf = "start"; | ||||||
|  | 		resourceLine.icon.style.marginTop = "calc(8px + (0.5lh - 0.5em))"; | ||||||
|  | 		resourceLine.label.style.alignSelf = "start"; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	resourceLine.resourceName = resourceName; | ||||||
|  | 	resourceLine.resourceScope = resourceScope; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function handleResourceAdd () { | ||||||
|  | 	const header = "Add New Resource Constraint"; | ||||||
|  | 	const body = ` | ||||||
|  | 		<form method="dialog" class="input-grid" style="grid-template-columns: auto 1fr;" id="form"> | ||||||
|  | 			<label for="name">PVE Resource Name</label> | ||||||
|  | 			<select class="w3-select w3-border" name="name" id="name" required></select> | ||||||
|  | 			<label for="scope">Constraint Scope</label> | ||||||
|  | 			<select class="w3-select w3-border" name="scope" id="scope" required> | ||||||
|  | 				<option value="global">Global</option> | ||||||
|  | 			</select> | ||||||
|  | 		</form> | ||||||
|  | 	`; | ||||||
|  |  | ||||||
|  | 	const d = dialog(header, body, async (result, form) => { | ||||||
|  | 		if (result === "confirm") { | ||||||
|  | 			const name = form.get("name"); | ||||||
|  | 			const type = clusterResourceConfig[name].type; | ||||||
|  | 			const scope = form.get("scope"); | ||||||
|  |  | ||||||
|  | 			console.log(name, type, scope); | ||||||
|  |  | ||||||
|  | 			// check if the resource name is not in the cluster config resources | ||||||
|  | 			if (!clusterResourceConfig[name]) { | ||||||
|  | 				alert(`${name} is not an allowed resource name`); | ||||||
|  | 			} | ||||||
|  | 			// check if a global scope rule already exists in the user's resource config | ||||||
|  | 			else if (scope === "global" && userData.resources[name] && userData.resources[name].global) { | ||||||
|  | 				alert(`${name} (${scope}) is already a rule`); | ||||||
|  | 			} | ||||||
|  | 			// check if node specific rule already exists in the user's resource config | ||||||
|  | 			else if (scope !== "global" && userData.resources[name] && userData.resources[name].nodes[scope]) { | ||||||
|  | 				alert(`${name} (${scope}) is already a rule`); | ||||||
|  | 			} | ||||||
|  | 			// no existing rule exists, add a new resource rule line and add a the rule to userData | ||||||
|  | 			else { | ||||||
|  | 				// if the rule does not exist at all, add a temporary filler to mark that a new rule has been created | ||||||
|  | 				if (!userData.resources[name]) { | ||||||
|  | 					userData.resources[name] = { | ||||||
|  | 						global: null, | ||||||
|  | 						node: {} | ||||||
|  | 					}; | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				const field = document.querySelector("#resources"); | ||||||
|  | 				let resourceLine; | ||||||
|  |  | ||||||
|  | 				if (scope === "global" && type === "numeric") { | ||||||
|  | 					userData.resources[name].global = { max: 0 }; | ||||||
|  | 					resourceLine = addResourceLine(resourcesConfigPage, field, name, { value: userData.resources[name].global.max }, "(Global)"); | ||||||
|  | 				} | ||||||
|  | 				else if (scope === "global" && type === "list") { | ||||||
|  | 					userData.resources[name].global = []; | ||||||
|  | 					resourceLine = addResourceLine(resourcesConfigPage, field, name, { value: userData.resources[name].global }, "(Global)"); | ||||||
|  | 				} | ||||||
|  | 				else if (scope !== "global" && type === "numeric") { | ||||||
|  | 					userData.resources[name].nodes[scope] = { max: 0 }; | ||||||
|  | 					resourceLine = addResourceLine(resourcesConfigPage, field, name, { value: userData.resources[name].nodes[scope].max }, `(${scope})`); | ||||||
|  | 				} | ||||||
|  | 				else if (scope !== "global" && type === "list") { | ||||||
|  | 					userData.resources[name].nodes[scope] = []; | ||||||
|  | 					resourceLine = addResourceLine(resourcesConfigPage, field, name, { value: userData.resources[name].nodes[scope] }, `(${scope})`); | ||||||
|  | 				} | ||||||
|  |  | ||||||
|  | 				postPopulateResourceLine(field, name, scope, resourcesConfigPage[name], resourceLine); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	}); | ||||||
|  |  | ||||||
|  | 	const nameSelect = d.querySelector("#name"); | ||||||
|  | 	for (const resourceName of Object.keys(clusterResourceConfig)) { | ||||||
|  | 		nameSelect.add(new Option(resourceName, resourceName)); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	const scopeSelect = d.querySelector("#scope"); | ||||||
|  | 	for (const node of allNodes) { | ||||||
|  | 		scopeSelect.add(new Option(node, node)); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function handleResourceDelete () { | ||||||
|  | 	const header = `Delete Resource Constraint ${this.label.innerText}`; | ||||||
|  | 	const body = `<p>Are you sure you want to <strong>delete</strong> VM ${this.label.innerText}</p>`; | ||||||
|  |  | ||||||
|  | 	dialog(header, body, async (result, form) => { | ||||||
|  | 		if (result === "confirm") { | ||||||
|  | 			this.icon.parentElement.removeChild(this.icon); | ||||||
|  | 			this.label.parentElement.removeChild(this.label); | ||||||
|  | 			this.element.parentElement.removeChild(this.element); | ||||||
|  | 			this.unit.parentElement.removeChild(this.unit); | ||||||
|  | 			this.deleteBtn.parentElement.removeChild(this.deleteBtn); | ||||||
|  |  | ||||||
|  | 			if (this.resourceScope === "global") { | ||||||
|  | 				userData.resources[this.resourceName].global = false; | ||||||
|  | 			} | ||||||
|  | 			else { | ||||||
|  | 				userData.resources[this.resourceName].nodes[this.resourceScope] = false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	}); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function populateCluster () { | ||||||
|  | 	const nodesEnabled = document.querySelector("#nodes-enabled"); | ||||||
|  | 	const nodesDisabled = document.querySelector("#nodes-disabled"); | ||||||
|  | 	const poolsEnabled = document.querySelector("#pools-enabled"); | ||||||
|  | 	const poolsDisabled = document.querySelector("#pools-disabled"); | ||||||
|  |  | ||||||
|  | 	for (const node of allNodes) { // for each node of all cluster nodes | ||||||
|  | 		const item = document.createElement("draggable-item"); | ||||||
|  | 		item.data = node; | ||||||
|  | 		item.innerHTML = ` | ||||||
|  | 			<div style="display: grid; grid-template-columns: auto 1fr; column-gap: 10px; align-items: center;"> | ||||||
|  | 				<svg id="drag" role="application" aria-label="drag icon"><title>drag icon</title><use href="images/actions/drag.svg#symb"></use></svg> | ||||||
|  | 				<p style="margin: 0px;">${node}</p> | ||||||
|  | 			</div> | ||||||
|  | 		`; | ||||||
|  | 		if (userData.cluster.nodes[node] === true) { | ||||||
|  | 			nodesEnabled.append(item); | ||||||
|  | 		} | ||||||
|  | 		else { | ||||||
|  | 			nodesDisabled.append(item); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for (const pool of allPools) { // for each pool of all cluster pools | ||||||
|  | 		const item = document.createElement("draggable-item"); | ||||||
|  | 		item.data = pool; | ||||||
|  | 		item.innerHTML = ` | ||||||
|  | 			<div style="display: grid; grid-template-columns: auto 1fr; column-gap: 10px; align-items: center;"> | ||||||
|  | 				<svg id="drag" role="application" aria-label="drag icon"><title>drag icon</title><use href="images/actions/drag.svg#symb"></use></svg> | ||||||
|  | 				<p style="margin: 0px;">${pool}</p> | ||||||
|  | 			</div> | ||||||
|  | 		`; | ||||||
|  | 		if (userData.cluster.pools[pool] === true) { | ||||||
|  | 			poolsEnabled.append(item); | ||||||
|  | 		} | ||||||
|  | 		else { | ||||||
|  | 			poolsDisabled.append(item); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	const vmidMin = document.querySelector("#vmid-min"); | ||||||
|  | 	const vmidMax = document.querySelector("#vmid-max"); | ||||||
|  |  | ||||||
|  | 	vmidMin.value = userData.cluster.vmid.min; | ||||||
|  | 	vmidMax.value = userData.cluster.vmid.max; | ||||||
|  |  | ||||||
|  | 	const adminCheckbox = document.querySelector("#admin"); | ||||||
|  | 	adminCheckbox.checked = userData.cluster.admin === true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function handleFormExit () { | ||||||
|  | 	// TODO | ||||||
|  | } | ||||||
| @@ -408,6 +408,25 @@ export function mergeDeep (target, ...sources) { | |||||||
| 	return mergeDeep(target, ...sources); | 	return mergeDeep(target, ...sources); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Checks if object or array is empty | ||||||
|  |  * @param {*} obj | ||||||
|  |  * @returns | ||||||
|  |  */ | ||||||
|  | export function isEmpty (obj) { | ||||||
|  | 	if (obj instanceof Array) { | ||||||
|  | 		return obj.length === 0; | ||||||
|  | 	} | ||||||
|  | 	else { | ||||||
|  | 		for (const prop in obj) { | ||||||
|  | 			if (Object.hasOwn(obj, prop)) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| export function addResourceLine (config, field, resourceType, attributesOverride, labelPrefix = null) { | export function addResourceLine (config, field, resourceType, attributesOverride, labelPrefix = null) { | ||||||
| 	const resourceConfig = config[resourceType]; | 	const resourceConfig = config[resourceType]; | ||||||
| 	const iconHref = resourceConfig.icon; | 	const iconHref = resourceConfig.icon; | ||||||
| @@ -427,56 +446,62 @@ export function addResourceLine (config, field, resourceType, attributesOverride | |||||||
| 	label.htmlFor = id; | 	label.htmlFor = id; | ||||||
| 	field.append(label); | 	field.append(label); | ||||||
|  |  | ||||||
|  | 	let element; | ||||||
|  |  | ||||||
| 	if (elementType === "input") { | 	if (elementType === "input") { | ||||||
| 		const input = document.createElement("input"); | 		element = document.createElement("input"); | ||||||
| 		for (const k in attributes) { | 		for (const k in attributes) { | ||||||
| 			input.setAttribute(k, attributes[k]); | 			element.setAttribute(k, attributes[k]); | ||||||
| 		} | 		} | ||||||
| 		input.id = id; | 		element.id = id; | ||||||
| 		input.name = id; | 		element.name = id; | ||||||
| 		input.required = true; | 		element.required = true; | ||||||
| 		input.classList.add("w3-input"); | 		element.classList.add("w3-input"); | ||||||
| 		input.classList.add("w3-border"); | 		element.classList.add("w3-border"); | ||||||
| 		field.append(input); | 		field.append(element); | ||||||
| 	} | 	} | ||||||
| 	else if (elementType === "select" || elementType === "multi-select") { | 	else if (elementType === "select" || elementType === "multi-select") { | ||||||
| 		const select = document.createElement("select"); | 		element = document.createElement("select"); | ||||||
| 		for (const option of attributes.options) { | 		for (const option of attributes.options) { | ||||||
| 			select.append(new Option(option)); | 			element.append(new Option(option)); | ||||||
| 		} | 		} | ||||||
| 		select.value = attributes.value; | 		element.value = attributes.value; | ||||||
| 		select.id = id; | 		element.id = id; | ||||||
| 		select.name = id; | 		element.name = id; | ||||||
| 		select.required = true; | 		element.required = true; | ||||||
| 		select.classList.add("w3-select"); | 		element.classList.add("w3-select"); | ||||||
| 		select.classList.add("w3-border"); | 		element.classList.add("w3-border"); | ||||||
| 		if (elementType === "multi-select") { | 		if (elementType === "multi-select") { | ||||||
| 			select.setAttribute("multiple", true); | 			element.setAttribute("multiple", true); | ||||||
| 		} | 		} | ||||||
| 		field.append(select); | 		field.append(element); | ||||||
| 	} | 	} | ||||||
| 	else if (customElements.get(elementType)) { | 	else if (customElements.get(elementType)) { | ||||||
| 		const elem = document.createElement(elementType); | 		element = document.createElement(elementType); | ||||||
| 		if (attributes.options) { | 		if (attributes.options) { | ||||||
| 			for (const option of attributes.options) { | 			for (const option of attributes.options) { | ||||||
| 				elem.append(new Option(option)); | 				element.append(new Option(option)); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		elem.value = attributes.value; | 		element.value = attributes.value; | ||||||
| 		elem.id = id; | 		element.id = id; | ||||||
| 		elem.name = id; | 		element.name = id; | ||||||
| 		elem.required = true; | 		element.required = true; | ||||||
| 		field.append(elem); | 		field.append(element); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	let unit; | ||||||
|  |  | ||||||
| 	if (unitText) { | 	if (unitText) { | ||||||
| 		const unit = document.createElement("p"); | 		unit = document.createElement("p"); | ||||||
| 		unit.innerText = unitText; | 		unit.innerText = unitText; | ||||||
| 		field.append(unit); | 		field.append(unit); | ||||||
| 	} | 	} | ||||||
| 	else { | 	else { | ||||||
| 		const unit = document.createElement("div"); | 		unit = document.createElement("div"); | ||||||
| 		unit.classList.add("hidden"); | 		unit.classList.add("hidden"); | ||||||
| 		field.append(unit); | 		field.append(unit); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	return { icon, label, element, unit }; | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										102
									
								
								user.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								user.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | |||||||
|  | <!DOCTYPE html> | ||||||
|  | <html lang="en"> | ||||||
|  | 	<head> | ||||||
|  | 		<meta charset="utf-8"> | ||||||
|  | 		<meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|  | 		<title>proxmox - dashboard</title> | ||||||
|  | 		<link rel="icon" href="images/favicon.svg" sizes="any" type="image/svg+xml"> | ||||||
|  | 		<link rel="stylesheet" href="modules/w3.css"> | ||||||
|  | 		<link rel="stylesheet" href="https://www.w3schools.com/w3css/4/w3.css"> | ||||||
|  | 		<link rel="stylesheet" href="css/style.css"> | ||||||
|  | 		<link rel="stylesheet" href="css/nav.css"> | ||||||
|  | 		<link rel="stylesheet" href="css/form.css"> | ||||||
|  | 		<script src="scripts/user.js" type="module"></script> | ||||||
|  | 		<script src="scripts/draggable.js" type="module"></script> | ||||||
|  | 		<script src="modules/Sortable.min.js"></script> | ||||||
|  | 		<script src="https://cdn.jsdelivr.net/npm/sortablejs@latest/Sortable.min.js"></script> | ||||||
|  | 		<style> | ||||||
|  | 			div.two-columns { | ||||||
|  | 				display: grid; | ||||||
|  | 				grid-template-columns: 50% 50%; | ||||||
|  | 			} | ||||||
|  | 			div.two-columns * { | ||||||
|  | 				padding: 0; | ||||||
|  | 				margin: 0; | ||||||
|  | 			} | ||||||
|  | 			draggable-container { | ||||||
|  | 				height: 100% | ||||||
|  | 			} | ||||||
|  | 		</style> | ||||||
|  | 	</head> | ||||||
|  | 	<body> | ||||||
|  | 		<header> | ||||||
|  | 			<h1>proxmox</h1> | ||||||
|  | 			<label for="navtoggle">☰</label> | ||||||
|  | 			<input type="checkbox" id="navtoggle"> | ||||||
|  | 			<nav id="navigation"> | ||||||
|  | 				<a href="index.html">Instances</a> | ||||||
|  | 				<a href="account.html">Account</a> | ||||||
|  | 				<a href="settings.html">Settings</a> | ||||||
|  | 				<a id="admin-link" href="admin.html" aria-current="page">Admin</a> | ||||||
|  | 				<a href="login.html">Logout</a> | ||||||
|  | 			</nav> | ||||||
|  | 		</header> | ||||||
|  | 		<main> | ||||||
|  | 			<section> | ||||||
|  | 				<h2 id="name"><a href="admin.html">Admin</a> / Users / %{username}</h2> | ||||||
|  | 				<form> | ||||||
|  | 					<fieldset class="w3-card w3-padding"> | ||||||
|  | 						<legend>Groups</legend> | ||||||
|  | 						<div class="input-grid two-columns"> | ||||||
|  | 							<p>Member Of</p> | ||||||
|  | 							<p>Not Member Of</p> | ||||||
|  | 							<draggable-container id="groups-enabled"></draggable-container> | ||||||
|  | 							<draggable-container id="groups-disabled"></draggable-container> | ||||||
|  | 						</div> | ||||||
|  | 					</fieldset> | ||||||
|  | 					<fieldset class="w3-card w3-padding"> | ||||||
|  | 						<legend>Resources</legend> | ||||||
|  | 						<div id="resources" class="input-grid" style="grid-template-columns: auto auto auto auto 1fr;"></div> | ||||||
|  | 						<div class="w3-container w3-center"> | ||||||
|  | 							<button type="button" id="resource-add" class="w3-button" aria-label="Add New Resource Constraint"> | ||||||
|  | 								<span class="large" style="margin: 0;">Add Resource</span> | ||||||
|  | 								<svg class="small" role="img" style="height: 1lh; width: 1lh;" aria-label="Add New Resource Constraint"><use href="images/actions/user/add.svg#symb"></use></svg> | ||||||
|  | 							</button> | ||||||
|  | 						</div> | ||||||
|  | 					</fieldset> | ||||||
|  | 					<fieldset class="w3-card w3-padding"> | ||||||
|  | 						<legend>Cluster</legend> | ||||||
|  | 						<div class="input-grid" style="grid-template-columns: auto auto 1fr;"> | ||||||
|  | 							<label for="admin">Admin</label> | ||||||
|  | 							<input class="w3-check w3-border" type="checkbox" id="admin" name="admin"> | ||||||
|  | 							<div></div> | ||||||
|  | 							<label style="align-self: start;">Nodes</label> | ||||||
|  | 							<div class="grid two-columns" style="grid-column: 2 / span 2"> | ||||||
|  | 								<p>Allowed</p> | ||||||
|  | 								<p>Not Allowed</p> | ||||||
|  | 								<draggable-container id="nodes-enabled"></draggable-container> | ||||||
|  | 								<draggable-container id="nodes-disabled"></draggable-container> | ||||||
|  | 							</div> | ||||||
|  | 							<label style="align-self: start;">Pools</label> | ||||||
|  | 							<div class="grid two-columns" style="grid-column: 2 / span 2"> | ||||||
|  | 								<p>Allowed</p> | ||||||
|  | 								<p>Not Allowed</p> | ||||||
|  | 								<draggable-container id="pools-enabled"></draggable-container> | ||||||
|  | 								<draggable-container id="pools-disabled"></draggable-container> | ||||||
|  | 							</div> | ||||||
|  | 							<label for="vmid-min">VMID Min</label> | ||||||
|  | 							<input class="w3-input w3-border" type="number" id="vmid-min" name="vmid-min"> | ||||||
|  | 							<div></div> | ||||||
|  | 							<label for="vmid-max">VMID Max</label> | ||||||
|  | 							<input class="w3-input w3-border" type="number" id="vmid-max" name="vmid-max"> | ||||||
|  | 							<div></div> | ||||||
|  | 						</div> | ||||||
|  | 					</fieldset> | ||||||
|  | 					<div class="w3-container w3-center" id="form-actions"> | ||||||
|  | 						<button class="w3-button w3-margin" id="exit" type="button">EXIT</button> | ||||||
|  | 					</div> | ||||||
|  | 				</form> | ||||||
|  | 			</section> | ||||||
|  | 		</main> | ||||||
|  | 	</body> | ||||||
|  | </html> | ||||||
		Reference in New Issue
	
	Block a user