class CCPMethodEditorController extends HTMLElement { #boot; #rootdoc; #serviceurl; #infrastructures; #locked = false; #isupdate = false; #cloneornew_dialog = null; #dragging_method = null; #tmp_inputs = [] #tmp_outputs = [] #current = null; #method_template = { id: "", title: "New Method", description: "New empty method", version: "1.0.0", jobControlOptions: "async-execute", metadata: [], inputs: {}, outputs: {}, additionalParameters: { parameters: [ { name: "deploy-script", value: [] }, { name: "execute-script", value: [] }, { name: "undeploy-script", value: [] } ] }, links: [] } #messages = { "en" : { "clone" : "Clone", "clone_help" : "All setting will be cloned but no relation to cloned method will be kept", "derive" : "Derive", "derive_help" : "All settings will be cloned and a relation to the originating method will be kept", "edit" : "Edit", "edit_help" : "Edit this method", "edit_clone_derive_help" : "Choose whether you want to clone this method or edit it", "save_help" : "Save this method definition", "reset_help" : "Reset all fields", "delete_help" : "Permanently delete this method", "title" : "Title", "title_help" : "The title for the method", "version" : "Version", "version_help" : "The version of the method", "description" : "Description", "description_help" : "The description of this method", "keywords" : "Keywords", "keywords_help" : "Keywords that can be used to search for the method. Write and add by hitting enter.", "categories" : "Categories", "categories_help" : "Categories to classify the method. Select one from the list or type a new one. Add by pressing enter.", "compatible_infrastructures" : "Compatible infrastructures", "compatible_infrastructures_help" : "Select the infrastructures that are able to execute your method", "inputs" : "Inputs", "outputs" : "Outputs", "scripts" : "Scripts", "scripts_help" : "Deploy and execute scripts configure the deployment and execution life cycle of the method on the selected infrastructures. Use {{input_id}} notation to expand the inputs with their default values.", "add-input_help" : "Add an input to this method", "add-output_help" : "Add an output to this method", "add_ccpnote_help" : "Add ccpnote annotation for this method. This helps in identifying executions.", "add_stdout_help" : "Add standard outout", "add_stderr_help" : "Add standard error", "preview" : "Preview", "err_fetch_infrastructures" : "Unable to fetch insfrastructures", "err_save_method" : "Unable to save method", "err_load_method" : "Unable to load method", "err_delete_method" : "Unable to delete method", "confirm_reset" : "All unsaved data will be lost. Proceed?", "confirm_save" : "Confirm updating of method", "confirm_delete" : "Confirm deletion of method", "execute-script" : "Execute script", "deploy-script" : "Deploy script", } } #style = ` ` #erase_icon = ` ` #plus_icon = ` ` #disc_icon = ` ` #output_icon = ` ` #delete_icon = ` ` #annotation_input_icon = ` ` getLabel(key, localehint){ const locale = localehint ? localehint : navigator.language const actlocale = this.#messages[locale] ? locale : "en" const msg = this.#messages[actlocale][key] return msg == null || msg == undefined ? key : this.#messages[actlocale][key] } constructor() { super(); this.#boot = document.querySelector("d4s-boot-2") this.#rootdoc = this.attachShadow({ "mode": "open" }) } connectedCallback() { this.#serviceurl = this.getAttribute("serviceurl") this.initMethod() this.fetchInfrastructures() this.connectNewEditRequest() } connectNewEditRequest() { document.addEventListener("neweditrequest", ev => { this.cloneOrEditMethod(ev.detail) }) } isInfrastructureRegistered(infra) { return (this.#infrastructures.filter(i => i.id === infra.id)).length > 0 } fetchInfrastructures() { this.#boot.secureFetch(this.#serviceurl + "/infrastructures"). then(resp => { if (resp.status !== 200) throw this.getLabel("err_fetch_infrastructures"); return resp.json() }).then(data => { this.#infrastructures = data this.render() }).catch(err => { alert(err) }) } saveMethod() { if (this.#locked) return; if (this.#current != null) { this.adoptTemporaries() const text = this.getLabel("confirm_save") + ` ${this.#current.title} v. ${this.#current.version}` if (window.confirm(text)) { this.lockRender() const url = this.#serviceurl + "/methods" const args = { body: JSON.stringify(this.#current), method: this.#isupdate ? "PUT" : "POST", headers: { "Content-type": "application/json" } } this.#boot.secureFetch(url, args).then( (resp) => { if (resp.ok) { return resp.text() } else throw this.getLabel("err_save_method") + resp.status }).then(data => { if (!this.#isupdate) { this.#current = JSON.parse(data) this.#isupdate = true this.#tmp_inputs = Object.keys(this.#current.inputs).map(k => this.#current.inputs[k]) this.#tmp_outputs = Object.keys(this.#current.outputs).map(k => this.#current.outputs[k]) } this.unlockRender() }).catch(err => { alert(err) this.unlockRender() }) } } } deleteMethod() { if (this.#locked) return; if (this.#current != null) { const text = this.getLabel("confirm_delete") + ` ${this.#current.title} v. ${this.#current.version}` if (window.confirm(text)) { this.lockRender() const url = this.#serviceurl + "/methods/" + this.#current.id const args = { method: "DELETE" } this.#boot.secureFetch(url, args).then( (resp) => { if (resp.status === 404 || resp.status === 204) { return null } else throw this.getLabel("err_delete_method") + ": " + resp.status }).then(data => { this.#isupdate = false this.initMethod() this.unlockRender() }).catch(err => { alert(err) this.unlockRender() }) } } } initMethod() { this.#current = JSON.parse(JSON.stringify(this.#method_template)) this.#current.id = Math.abs((Math.random() * 10e11) | 0) this.#current.metadata = [] this.#tmp_inputs = [ { id: "ccpimage", title: "Runtime", description: "The image of the runtime to use for method execution. This depends on the infrastructure specific protocol for interacting with registries.", minOccurs: 1, maxOccurs: 1, schema: { type: "string", format: "url", contentMediaType: "text/plain", default: "", readOnly: true, } } ] this.#tmp_outputs = [] } resetMethod() { this.initMethod() this.render() } cloneOrEditMethod(method) { const subject = this.#boot.subject const matchingauthors = method.metadata ? method.metadata.filter(md => md.role === "author" && md.href.endsWith("/" + subject)).length : 0 if (matchingauthors === 0) this.cloneMethod(method.id, true); else { this.#dragging_method = method.id this.#cloneornew_dialog.style.display = "block" this.#cloneornew_dialog.classList.add("show") } } cloneMethod(method, derivation) { if (this.#locked) return; this.lockRender() const derive = derivation ? "?derive=true" : "" this.#boot.secureFetch(this.#serviceurl + "/methods/" + method + "/clone" + derive).then( (resp) => { if (resp.status === 200) { return resp.json() } else throw this.getLabel("err_load_method") + ": " + resp.status } ).then(data => { this.#current = data this.#isupdate = false this.#tmp_inputs = Object.keys(this.#current.inputs).map(k => this.#current.inputs[k]) this.#tmp_outputs = Object.keys(this.#current.outputs).map(k => this.#current.outputs[k]) this.unlockRender() }).catch(err => { this.unlockRender() }) } editMethod(method) { if (this.#locked) return; this.lockRender() this.#boot.secureFetch(this.#serviceurl + "/methods/" + method + "/updatable").then( (resp) => { if (resp.status === 200) { return resp.json() } else throw this.getLabel("err_load_method") + ": " + resp.status } ).then(data => { this.#current = data this.#isupdate = true this.#tmp_inputs = Object.keys(this.#current.inputs).map(k => this.#current.inputs[k]) this.#tmp_outputs = Object.keys(this.#current.outputs).map(k => this.#current.outputs[k]) this.unlockRender() }).catch(err => { this.unlockRender() }) } adoptTemporaries() { this.#current.inputs = {} this.#tmp_inputs.forEach(t => { this.#current.inputs[t.id] = t }) this.#current.outputs = {} this.#tmp_outputs.forEach(t => { this.#current.outputs[t.id] = t }) } getInputAt(i) { return this.#tmp_inputs[i] } deleteTmpInputAt(i) { this.#tmp_inputs.splice(i, 1) } deleteTmpOutputAt(i) { this.#tmp_outputs.splice(i, 1) } lockRender() { this.#locked = true this.#rootdoc.querySelector(".plexiglass").classList.toggle("d-none") } unlockRender() { this.#rootdoc.querySelector(".plexiglass").classList.toggle("d-none") this.render() this.#locked = false if (this.parentElement) this.parentElement.scrollIntoViewIfNeeded(); } render() { this.#rootdoc.innerHTML = `
${this.#style}
${this.#current.title}
${this.renderSaveButton()} ${this.renderResetButton()} ${this.#isupdate ? this.renderDeleteButton() : ""}
${this.renderAuthors()}
${this.renderKeywords()}
${this.renderCategoryHints()}
${this.renderCategories()}
${this.renderInfrastructures()}
${this.getLabel("inputs")}
${this.renderStandardInputButtons()} ${this.renderPlusButton("add-input")}
${this.getLabel("outputs")}
${this.renderStandardOutputButtons()} ${this.renderPlusButton("add-output")}
${this.renderScripts()}
` this.renderInputs() this.renderOutputs() this.#cloneornew_dialog = this.#rootdoc.querySelector("#cloneornew") this.#rootdoc.querySelector("#cloneornew button[name=clone]").addEventListener("click", ev => { this.#cloneornew_dialog.classList.remove("show") this.#cloneornew_dialog.style.display = "none" this.cloneMethod(this.#dragging_method) this.#dragging_method = null }) this.#rootdoc.querySelector("#cloneornew button[name=derive]").addEventListener("click", ev => { this.#cloneornew_dialog.classList.remove("show") this.#cloneornew_dialog.style.display = "none" this.cloneMethod(this.#dragging_method, true) this.#dragging_method = null }) this.#rootdoc.querySelector("#cloneornew button[name=edit]").addEventListener("click", ev => { this.#cloneornew_dialog.classList.remove("show") this.#cloneornew_dialog.style.display = "none" this.editMethod(this.#dragging_method) this.#dragging_method = null }) this.#rootdoc.querySelector("input[name=title]").addEventListener("input", ev => { this.#current.title = ev.currentTarget.value this.#rootdoc.querySelector("span[name=header]").innerText = this.#current.title }) this.#rootdoc.querySelector("input[name=version]").addEventListener("input", ev => { this.#current.version = ev.currentTarget.value }) this.#rootdoc.querySelector("textarea[name=description]").addEventListener("input", ev => { this.#current.description = ev.currentTarget.value }) this.#rootdoc.addEventListener("drop", ev => { if (ev.dataTransfer && ev.dataTransfer.getData('text/plain+ccpmethod')) { const method = JSON.parse(ev.dataTransfer.getData('application/json+ccpmethod')) ev.stopImmediatePropagation() ev.preventDefault() ev.stopPropagation() this.cloneOrEditMethod(method) } }) this.#rootdoc.addEventListener("dragover", ev => { ev.preventDefault() }) this.#rootdoc.querySelector("button[name=reset]").addEventListener("click", ev => { if (window.confirm(this.getLabel("confirm_reset"))) { this.resetMethod() } ev.preventDefault() ev.stopPropagation() }) this.#rootdoc.querySelector("button[name=save]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() this.saveMethod() }) if (this.#isupdate) { this.#rootdoc.querySelector("button[name=delete]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() this.deleteMethod() }) } this.#rootdoc.querySelector("input[name=keyword-input]").addEventListener("keypress", ev => { if (ev.key === "Enter" || ev.which === 13) { ev.preventDefault() ev.stopPropagation() const val = ev.target.value ev.target.value = null if (!this.#current.keywords) { this.#current.keywords = [val] } else { this.#current.keywords.push(val) } this.reRenderKeywords() } }) this.#rootdoc.querySelector("div[name=keyword-list]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() if (ev.target.getAttribute('name') === "delete-keyword") { const index = ev.target.getAttribute("data-index") this.#current.keywords.splice(index, 1) this.reRenderKeywords() this.#rootdoc.querySelector("input[name=keyword-input]").focus() } }) this.#rootdoc.querySelector("input[name=category-input]").addEventListener("keypress", ev => { if (ev.key === "Enter" || ev.which === 13) { ev.preventDefault() ev.stopPropagation() const val = ev.target.value ev.target.value = null if (this.#current.metadata.filter(md => md.role === "category" && md.title === val).length === 0) { this.#current.metadata.push({ role: "category", title: val }) this.reRenderCategories() } } }) this.#rootdoc.querySelector("div[name=category-list]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() if (ev.target.getAttribute('name') === "delete-category") { const val = ev.target.parentElement.title this.#current.metadata = this.#current.metadata.filter(md => md.role !== "category" || md.title !== val) this.reRenderCategories() this.#rootdoc.querySelector("input[name=category-input]").focus() } }) this.#rootdoc.querySelector("div[name=infrastructures]").addEventListener("change", ev => { if (ev.target.getAttribute("name") === "infrastructure-input" && ev.target.value) { ev.preventDefault() ev.stopPropagation() const id = ev.target.value const display = ev.target.options[ev.target.selectedIndex].text let link = { rel: "compatibleWith", title: display, href: "infrastructures/" + id } if (!this.#current.links) { this.#current.links = [link] } else { this.#current.links.push(link) } this.reRenderInfrastructures(ev.currentTarget) } }) this.#rootdoc.querySelector("div[name=infrastructure-list]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() if (ev.target.getAttribute('name') === "delete-infrastructure") { const index = ev.target.getAttribute("data-index") this.#current.links.splice(index, 1) this.reRenderInfrastructures() } }) this.#rootdoc.querySelector("button[name=add-input]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() this.#tmp_inputs.push( { id: "new_input", title: "New input", description: "A new input field", minOccurs: 1, maxOccurs: 1, schema: { type: "string", format: null, contentMediaType: "text/plain", default: "" } } ) this.renderInputs() this.reRenderScripts() }) this.#rootdoc.querySelector("button[name=add-ccpannotation]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() this.#tmp_inputs.push( { id: "ccpnote", title: "Annotations for execution", description: "The value of this parameter will be associated as annotation to the execution", minOccurs: 1, maxOccurs: 1, schema: { type: "string", format: null, contentMediaType: "text/plain", default: "" } } ) this.renderInputs() this.reRenderScripts() }) this.#rootdoc.querySelector("div[name=input-list]").addEventListener("click", ev => { const evname = ev.target.getAttribute('name') if (evname === "delete-input") { ev.preventDefault() ev.stopPropagation() const index = Number(ev.target.getAttribute("data-index")) console.log("deleting input at index", index) this.deleteTmpInputAt(index) this.renderInputs() this.reRenderScripts() } }) //when any input changes update scripts so that possibly variable get re-expanded this.#rootdoc.querySelector("div[name=input-list]").addEventListener("input", ev => { this.reRenderScripts() }) this.#rootdoc.querySelector("div[name=input-list]").addEventListener("change", ev => { this.reRenderScripts() }) this.#rootdoc.querySelector("div[name=input-list]").addEventListener("click", ev => { const src = ev.target.getAttribute("name") if (src === "plus" || src === "minus" || src === "maxOccurs" || src === "minOccurs") { this.reRenderScripts() } }) this.#rootdoc.querySelector("button[name=add-output]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() this.#tmp_outputs.push( { id: "new_output", title: "New ouput", description: "A new output field", minOccurs: 1, maxOccurs: 1, metadata: [ { "role": "file", "title": "newoutput.txt", "href": "newoutput.txt" } ], schema: { type: "string", contentMediaType: "text/plain" } } ) this.renderOutputs() }) this.#rootdoc.querySelector("button[name=add-stdout]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() this.#tmp_outputs.push( { id: "stdout", title: "Standard output", description: "Standard output channel", minOccurs: 1, maxOccurs: 1, metadata: [ { "role": "file", "title": "stdout", "href": "stdout" } ], schema: { type: "string", contentMediaType: "text/plain" } } ) this.renderOutputs() }) this.#rootdoc.querySelector("button[name=add-stderr]").addEventListener("click", ev => { ev.preventDefault() ev.stopPropagation() this.#tmp_outputs.push( { id: "stderr", title: "Standard error", description: "Standard error channel", minOccurs: 1, maxOccurs: 1, metadata: [ { "role": "file", "title": "stderr", "href": "stderr" } ], schema: { type: "string", contentMediaType: "text/plain" } } ) this.renderOutputs() }) this.#rootdoc.querySelector("div[name=output-list]").addEventListener("click", ev => { const evname = ev.target.getAttribute('name') if (evname === "delete-output") { ev.preventDefault() ev.stopPropagation() const index = Number(ev.target.getAttribute("data-index")) console.log("deleting output at index", index) this.deleteTmpOutputAt(index) this.renderOutputs() } }) this.#rootdoc.querySelector("div[name=scripts_container]").addEventListener("input", ev => { ev.preventDefault() ev.stopPropagation() const scriptname = ev.target.getAttribute("name") const script = this.#current.additionalParameters.parameters.find(p => p.name === scriptname) if (script) script.value = ev.target.value.split(/\r?\n/); this.reRenderScripts() }) } renderAuthors() { return ` ` } renderCategoryHints() { const ml = document.querySelector("d4s-ccp-methodlist") const cats = ml.getCategoryHints() if (cats.indexOf("Uncategorised") === -1) cats.push("Uncategorised") return ` ${cats.map(c => ``)} ` } renderContexts() { return ` ` } renderSaveButton() { return ` ` } renderDeleteButton() { return ` ` } renderResetButton() { return ` ` } renderPlusButton(name) { return ` ` } renderStandardInputButtons() { return ` ` } renderStandardOutputButtons() { return ` ` } reRenderKeywords() { this.#rootdoc.querySelector("div[name=keyword-list]").innerHTML = this.renderKeywords() } renderKeywords() { if (this.#current.keywords) { return this.#current.keywords.map((k, i) => { return `
${k} x
` }).join("\n") } else { return "" } } reRenderCategories() { this.#rootdoc.querySelector("div[name=category-list]").innerHTML = this.renderCategories() } renderCategories() { if (this.#current.metadata) { return this.#current.metadata.filter(md => md.role === "category").map((k, i) => { return `
${k.title} x
` }).join("\n") } else { return "" } } reRenderInfrastructures() { this.#rootdoc.querySelector("select[name=infrastructure-input]").innerHTML = this.renderInfrastructureOptions() this.#rootdoc.querySelector("div[name=infrastructure-list]").innerHTML = this.renderInfrastructures() } renderInfrastructureOptions() { const selectedinfras = this.#current.links.filter(l => l.rel === "compatibleWith").map(i => i.href.split("/")[1]) const available = this.#infrastructures.filter(i => { return selectedinfras.indexOf(i.id) === -1 }) return ` ${available.map(infra => { return ` ` }).join("\n") } ` } renderInfrastructures() { return this.#current.links.filter(l => l.rel === "compatibleWith").map((l, i) => { return `
${l.title} x
` }).join("\n") } renderInputs() { const parent = this.#rootdoc.querySelector("div[name=input-list]") parent.innerHTML = "" return this.#tmp_inputs.map((inp, i) => { const c = document.createElement("d4s-ccp-input-editor"); parent.appendChild(c) c.render(inp, i) }) } renderOutputs() { const parent = this.#rootdoc.querySelector("div[name=output-list]") parent.innerHTML = "" return this.#tmp_outputs.map((outp, i) => { const c = document.createElement("d4s-ccp-output-editor"); parent.appendChild(c) c.render(outp, i) }) } codeToPreview(code) { return this.#tmp_inputs.reduce((acc, i) => { const r = new RegExp(`{{\\s*${i.id}\\s*}}`, 'g'); var def = i.schema.default if(i.schema.format === "secret"){ def = Array.isArray(def) ? def : [def] def = def.map(v=>"***") } if(i.schema.format === "file"){ def = Array.isArray(def) ? def : [def] def = def.map(v=>v.length > 15 ? v.substring(0,5) + "..." + v.substring(v.length-5) : v) } return acc.replaceAll(r, def) }, code) } reRenderScripts(scriptname) { const container = this.#rootdoc.querySelector("div[name=scripts_container]") const scripts = ["deploy-script", "execute-script"] scripts.forEach(sname=>{ const script = this.#current.additionalParameters.parameters.find(s => s.name === sname) const code = script.value && script.value.length ? script.value.join("\r\n") : "" const preview = container.querySelector(`details[name=${sname}] textarea[name=preview]`) preview.value = this.codeToPreview(code) }) } renderScriptContent(scriptname) { const script = this.#current.additionalParameters.parameters.find(s => s.name === scriptname) if (!script) return ''; const code = script.value && script.value.length ? script.value.join("\r\n") : "" return `
Preview
` } renderScripts(){ const scripts = ["deploy-script", "execute-script"] const html = scripts.map(sname=>{ return `
${this.getLabel(sname)}
${this.renderScriptContent(sname)}
` }).join("") return html } } window.customElements.define('d4s-ccp-methodeditor', CCPMethodEditorController);