import Utils from "../core/Utils.js"; import Chef from "../core/Chef.js"; import Manager from "./Manager.js"; import HTMLCategory from "./HTMLCategory.js"; import HTMLOperation from "./HTMLOperation.js"; import Split from "split.js"; /** * HTML view for CyberChef responsible for building the web page and dealing with all user * interactions. * * @author n1474335 [n1474335@gmail.com] * @copyright Crown Copyright 2016 * @license Apache-2.0 * * @constructor * @param {CatConf[]} categories - The list of categories and operations to be populated. * @param {Object.} operations - The list of operation configuration objects. * @param {String[]} defaultFavourites - A list of default favourite operations. * @param {Object} options - Default setting for app options. */ const App = function(categories, operations, defaultFavourites, defaultOptions) { this.categories = categories; this.operations = operations; this.dfavourites = defaultFavourites; this.doptions = defaultOptions; this.options = Utils.extend({}, defaultOptions); this.chef = new Chef(); this.manager = new Manager(this); this.baking = false; this.autoBake_ = false; this.progress = 0; this.ingId = 0; window.chef = this.chef; }; /** * This function sets up the stage and creates listeners for all events. * * @fires Manager#appstart */ App.prototype.setup = function() { document.dispatchEvent(this.manager.appstart); this.initialiseSplitter(); this.loadLocalStorage(); this.populateOperationsList(); this.manager.setup(); this.resetLayout(); this.setCompileMessage(); this.loadURIParams(); }; /** * An error handler for displaying the error to the user. * * @param {Error} err */ App.prototype.handleError = function(err) { console.error(err); const msg = err.displayStr || err.toString(); this.alert(msg, "danger", this.options.errorTimeout, !this.options.showErrors); }; /** * Updates the UI to show if baking is in process or not. * * @param {bakingStatus} */ App.prototype.setBakingStatus = function(bakingStatus) { this.baking = bakingStatus; var inputLoadingIcon = document.querySelector("#input .title .loading-icon"), outputLoadingIcon = document.querySelector("#output .title .loading-icon"), outputElement = document.querySelector("#output-text"); if (bakingStatus) { inputLoadingIcon.style.display = "inline-block"; outputLoadingIcon.style.display = "inline-block"; outputElement.classList.add("disabled"); outputElement.disabled = true; } else { inputLoadingIcon.style.display = "none"; outputLoadingIcon.style.display = "none"; outputElement.classList.remove("disabled"); outputElement.disabled = false; } }; /** * Calls the Chef to bake the current input using the current recipe. * * @param {boolean} [step] - Set to true if we should only execute one operation instead of the * whole recipe. */ App.prototype.bake = async function(step) { let response; if (this.baking) return; this.setBakingStatus(true); try { response = await this.chef.bake( this.getInput(), // The user's input this.getRecipeConfig(), // The configuration of the recipe this.options, // Options set by the user this.progress, // The current position in the recipe step // Whether or not to take one step or execute the whole recipe ); } catch (err) { this.handleError(err); } this.setBakingStatus(false); if (!response) return; if (response.error) { this.handleError(response.error); } this.options = response.options; this.dishStr = response.type === "html" ? Utils.stripHtmlTags(response.result, true) : response.result; this.progress = response.progress; this.manager.recipe.updateBreakpointIndicator(response.progress); this.manager.output.set(response.result, response.type, response.duration); // If baking took too long, disable auto-bake if (response.duration > this.options.autoBakeThreshold && this.autoBake_) { this.manager.controls.setAutoBake(false); this.alert("Baking took longer than " + this.options.autoBakeThreshold + "ms, Auto Bake has been disabled.", "warning", 5000); } }; /** * Runs Auto Bake if it is set. */ App.prototype.autoBake = function() { if (this.autoBake_) { this.bake(); } }; /** * Runs a silent bake forcing the browser to load and cache all the relevant JavaScript code needed * to do a real bake. * * The output will not be modified (hence "silent" bake). This will only actually execute the * recipe if auto-bake is enabled, otherwise it will just load the recipe, ingredients and dish. * * @returns {number} - The number of miliseconds it took to run the silent bake. */ App.prototype.silentBake = function() { let startTime = new Date().getTime(), recipeConfig = this.getRecipeConfig(); if (this.autoBake_) { this.chef.silentBake(recipeConfig); } return new Date().getTime() - startTime; }; /** * Gets the user's input data. * * @returns {string} */ App.prototype.getInput = function() { const input = this.manager.input.get(); // Save to session storage in case we need to restore it later sessionStorage.setItem("inputLength", input.length); sessionStorage.setItem("input", input); return input; }; /** * Sets the user's input data. * * @param {string} input - The string to set the input to */ App.prototype.setInput = function(input) { sessionStorage.setItem("inputLength", input.length); sessionStorage.setItem("input", input); this.manager.input.set(input); }; /** * Populates the operations accordion list with the categories and operations specified in the * view constructor. * * @fires Manager#oplistcreate */ App.prototype.populateOperationsList = function() { // Move edit button away before we overwrite it document.body.appendChild(document.getElementById("edit-favourites")); let html = ""; let i; for (i = 0; i < this.categories.length; i++) { let catConf = this.categories[i], selected = i === 0, cat = new HTMLCategory(catConf.name, selected); for (let j = 0; j < catConf.ops.length; j++) { let opName = catConf.ops[j], op = new HTMLOperation(opName, this.operations[opName], this, this.manager); cat.addOperation(op); } html += cat.toHtml(); } document.getElementById("categories").innerHTML = html; const opLists = document.querySelectorAll("#categories .op-list"); for (i = 0; i < opLists.length; i++) { opLists[i].dispatchEvent(this.manager.oplistcreate); } // Add edit button to first category (Favourites) document.querySelector("#categories a").appendChild(document.getElementById("edit-favourites")); }; /** * Sets up the adjustable splitter to allow the user to resize areas of the page. */ App.prototype.initialiseSplitter = function() { this.columnSplitter = Split(["#operations", "#recipe", "#IO"], { sizes: [20, 30, 50], minSize: [240, 325, 440], gutterSize: 4, onDrag: function() { this.manager.controls.adjustWidth(); this.manager.output.adjustWidth(); }.bind(this) }); this.ioSplitter = Split(["#input", "#output"], { direction: "vertical", gutterSize: 4, }); this.resetLayout(); }; /** * Loads the information previously saved to the HTML5 local storage object so that user options * and favourites can be restored. */ App.prototype.loadLocalStorage = function() { // Load options let lOptions; if (localStorage.options !== undefined) { lOptions = JSON.parse(localStorage.options); } this.manager.options.load(lOptions); // Load favourites this.loadFavourites(); }; /** * Loads the user's favourite operations from the HTML5 local storage object and populates the * Favourites category with them. * If the user currently has no saved favourites, the defaults from the view constructor are used. */ App.prototype.loadFavourites = function() { let favourites = localStorage.favourites && localStorage.favourites.length > 2 ? JSON.parse(localStorage.favourites) : this.dfavourites; favourites = this.validFavourites(favourites); this.saveFavourites(favourites); const favCat = this.categories.filter(function(c) { return c.name === "Favourites"; })[0]; if (favCat) { favCat.ops = favourites; } else { this.categories.unshift({ name: "Favourites", ops: favourites }); } }; /** * Filters the list of favourite operations that the user had stored and removes any that are no * longer available. The user is notified if this is the case. * @param {string[]} favourites - A list of the user's favourite operations * @returns {string[]} A list of the valid favourites */ App.prototype.validFavourites = function(favourites) { const validFavs = []; for (let i = 0; i < favourites.length; i++) { if (this.operations.hasOwnProperty(favourites[i])) { validFavs.push(favourites[i]); } else { this.alert("The operation \"" + Utils.escapeHtml(favourites[i]) + "\" is no longer available. It has been removed from your favourites.", "info"); } } return validFavs; }; /** * Saves a list of favourite operations to the HTML5 local storage object. * * @param {string[]} favourites - A list of the user's favourite operations */ App.prototype.saveFavourites = function(favourites) { localStorage.setItem("favourites", JSON.stringify(this.validFavourites(favourites))); }; /** * Resets favourite operations back to the default as specified in the view constructor and * refreshes the operation list. */ App.prototype.resetFavourites = function() { this.saveFavourites(this.dfavourites); this.loadFavourites(); this.populateOperationsList(); this.manager.recipe.initialiseOperationDragNDrop(); }; /** * Adds an operation to the user's favourites. * * @param {string} name - The name of the operation */ App.prototype.addFavourite = function(name) { const favourites = JSON.parse(localStorage.favourites); if (favourites.indexOf(name) >= 0) { this.alert("'" + name + "' is already in your favourites", "info", 2000); return; } favourites.push(name); this.saveFavourites(favourites); this.loadFavourites(); this.populateOperationsList(); this.manager.recipe.initialiseOperationDragNDrop(); }; /** * Checks for input and recipe in the URI parameters and loads them if present. */ App.prototype.loadURIParams = function() { // Load query string from URI this.queryString = (function(a) { if (a === "") return {}; const b = {}; for (let i = 0; i < a.length; i++) { const p = a[i].split("="); if (p.length !== 2) { b[a[i]] = true; } else { b[p[0]] = decodeURIComponent(p[1].replace(/\+/g, " ")); } } return b; })(window.location.search.substr(1).split("&")); // Turn off auto-bake while loading const autoBakeVal = this.autoBake_; this.autoBake_ = false; // Read in recipe from query string if (this.queryString.recipe) { try { const recipeConfig = JSON.parse(this.queryString.recipe); this.setRecipeConfig(recipeConfig); } catch (err) {} } else if (this.queryString.op) { // If there's no recipe, look for single operations this.manager.recipe.clearRecipe(); try { this.manager.recipe.addOperation(this.queryString.op); } catch (err) { // If no exact match, search for nearest match and add that const matchedOps = this.manager.ops.filterOperations(this.queryString.op, false); if (matchedOps.length) { this.manager.recipe.addOperation(matchedOps[0].name); } // Populate search with the string const search = document.getElementById("search"); search.value = this.queryString.op; search.dispatchEvent(new Event("search")); } } // Read in input data from query string if (this.queryString.input) { try { const inputData = Utils.fromBase64(this.queryString.input); this.setInput(inputData); } catch (err) {} } // Restore auto-bake state this.autoBake_ = autoBakeVal; this.autoBake(); }; /** * Returns the next ingredient ID and increments it for next time. * * @returns {number} */ App.prototype.nextIngId = function() { return this.ingId++; }; /** * Gets the current recipe configuration. * * @returns {Object[]} */ App.prototype.getRecipeConfig = function() { const recipeConfig = this.manager.recipe.getConfig(); sessionStorage.setItem("recipeConfig", JSON.stringify(recipeConfig)); return recipeConfig; }; /** * Given a recipe configuration, sets the recipe to that configuration. * * @param {Object[]} recipeConfig - The recipe configuration */ App.prototype.setRecipeConfig = function(recipeConfig) { sessionStorage.setItem("recipeConfig", JSON.stringify(recipeConfig)); document.getElementById("rec-list").innerHTML = null; for (let i = 0; i < recipeConfig.length; i++) { const item = this.manager.recipe.addOperation(recipeConfig[i].op); // Populate arguments const args = item.querySelectorAll(".arg"); for (let j = 0; j < args.length; j++) { if (args[j].getAttribute("type") === "checkbox") { // checkbox args[j].checked = recipeConfig[i].args[j]; } else if (args[j].classList.contains("toggle-string")) { // toggleString args[j].value = recipeConfig[i].args[j].string; args[j].previousSibling.children[0].innerHTML = Utils.escapeHtml(recipeConfig[i].args[j].option) + " "; } else { // all others args[j].value = recipeConfig[i].args[j]; } } // Set disabled and breakpoint if (recipeConfig[i].disabled) { item.querySelector(".disable-icon").click(); } if (recipeConfig[i].breakpoint) { item.querySelector(".breakpoint").click(); } this.progress = 0; } }; /** * Resets the splitter positions to default. */ App.prototype.resetLayout = function() { this.columnSplitter.setSizes([20, 30, 50]); this.ioSplitter.setSizes([50, 50]); this.manager.controls.adjustWidth(); this.manager.output.adjustWidth(); }; /** * Sets the compile message. */ App.prototype.setCompileMessage = function() { // Display time since last build and compile message let now = new Date(), timeSinceCompile = Utils.fuzzyTime(now.getTime() - window.compileTime), compileInfo = "Last build: " + timeSinceCompile.substr(0, 1).toUpperCase() + timeSinceCompile.substr(1) + " ago"; if (window.compileMessage !== "") { compileInfo += " - " + window.compileMessage; } compileInfo += ""; document.getElementById("notice").innerHTML = compileInfo; }; /** * Pops up a message to the user and writes it to the console log. * * @param {string} str - The message to display (HTML supported) * @param {string} style - The colour of the popup * "danger" = red * "warning" = amber * "info" = blue * "success" = green * @param {number} timeout - The number of milliseconds before the popup closes automatically * 0 for never (until the user closes it) * @param {boolean} [silent=false] - Don't show the message in the popup, only print it to the * console * * @example * // Pops up a red box with the message "[current time] Error: Something has gone wrong!" * // that will need to be dismissed by the user. * this.alert("Error: Something has gone wrong!", "danger", 0); * * // Pops up a blue information box with the message "[current time] Happy Christmas!" * // that will disappear after 5 seconds. * this.alert("Happy Christmas!", "info", 5000); */ App.prototype.alert = function(str, style, timeout, silent) { const time = new Date(); console.log("[" + time.toLocaleString() + "] " + str); if (silent) return; style = style || "danger"; timeout = timeout || 0; let alertEl = document.getElementById("alert"), alertContent = document.getElementById("alert-content"); alertEl.classList.remove("alert-danger"); alertEl.classList.remove("alert-warning"); alertEl.classList.remove("alert-info"); alertEl.classList.remove("alert-success"); alertEl.classList.add("alert-" + style); // If the box hasn't been closed, append to it rather than replacing if (alertEl.style.display === "block") { alertContent.innerHTML += "

[" + time.toLocaleTimeString() + "] " + str; } else { alertContent.innerHTML = "[" + time.toLocaleTimeString() + "] " + str; } // Stop the animation if it is in progress $("#alert").stop(); alertEl.style.display = "block"; alertEl.style.opacity = 1; if (timeout > 0) { clearTimeout(this.alertTimeout); this.alertTimeout = setTimeout(function(){ $("#alert").slideUp(100); }, timeout); } }; /** * Pops up a box asking the user a question and sending the answer to a specified callback function. * * @param {string} title - The title of the box * @param {string} body - The question (HTML supported) * @param {function} callback - A function accepting one boolean argument which handles the * response e.g. function(answer) {...} * @param {Object} [scope=this] - The object to bind to the callback function * * @example * // Pops up a box asking if the user would like a cookie. Prints the answer to the console. * this.confirm("Question", "Would you like a cookie?", function(answer) {console.log(answer);}); */ App.prototype.confirm = function(title, body, callback, scope) { scope = scope || this; document.getElementById("confirm-title").innerHTML = title; document.getElementById("confirm-body").innerHTML = body; document.getElementById("confirm-modal").style.display = "block"; this.confirmClosed = false; $("#confirm-modal").modal() .one("show.bs.modal", function(e) { this.confirmClosed = false; }.bind(this)) .one("click", "#confirm-yes", function() { this.confirmClosed = true; callback.bind(scope)(true); $("#confirm-modal").modal("hide"); }.bind(this)) .one("hide.bs.modal", function(e) { if (!this.confirmClosed) callback.bind(scope)(false); this.confirmClosed = true; }.bind(this)); }; /** * Handler for the alert close button click event. * Closes the alert box. */ App.prototype.alertCloseClick = function() { document.getElementById("alert").style.display = "none"; }; /** * Handler for CyerChef statechange events. * Fires whenever the input or recipe changes in any way. * * @listens Manager#statechange * @param {event} e */ App.prototype.stateChange = function(e) { this.autoBake(); // Update the current history state (not creating a new one) if (this.options.updateUrl) { this.lastStateUrl = this.manager.controls.generateStateUrl(true, true); window.history.replaceState({}, "CyberChef", this.lastStateUrl); } }; /** * Handler for the history popstate event. * Reloads parameters from the URL. * * @param {event} e */ App.prototype.popState = function(e) { if (window.location.href.split("#")[0] !== this.lastStateUrl) { this.loadURIParams(); } }; /** * Function to call an external API from this view. */ App.prototype.callApi = function(url, type, data, dataType, contentType) { type = type || "POST"; data = data || {}; dataType = dataType || undefined; contentType = contentType || "application/json"; let response = null, success = false; $.ajax({ url: url, async: false, type: type, data: data, dataType: dataType, contentType: contentType, success: function(data) { success = true; response = data; }, error: function(data) { success = false; response = data; }, }); return { success: success, response: response }; }; export default App;