/** * @author n1474335 [n1474335@gmail.com] * @author j433866 [j433866@gmail.com] * @copyright Crown Copyright 2019 * @license Apache-2.0 */ import ChefWorker from "worker-loader?inline&fallback=false!../core/ChefWorker"; /** * Waiter to handle conversations with the ChefWorker */ class WorkerWaiter { /** * WorkerWaiter constructor * * @param {App} app - The main view object for CyberChef * @param {Manager} manager - The CyberChef event manager */ constructor(app, manager) { this.app = app; this.manager = manager; this.chefWorkers = []; this.maxWorkers = navigator.hardwareConcurrency || 4; this.inputs = []; this.totalOutputs = 0; } /** * Terminates any existing ChefWorkers and sets up a new worker */ setupChefWorker() { for (let i = 0; i < this.chefWorkers.length; i++) { const worker = this.chefWorkers.pop(); worker.terminate(); } this.addChefWorker(); } /** * Adds a new ChefWorker * * @returns {number} The index of the created worker */ addChefWorker() { // First find if there are any inactive workers, as this will be // more efficient than creating a new one for (let i = 0; i < this.chefWorkers.length; i++) { if (!this.chefWorkers[i].active) { return i; } } if (this.chefWorkers.length === this.maxWorkers) { // Can't create any more workers return -1; } log.debug("Adding new ChefWorker"); // Create a new ChefWorker and send it the docURL const newWorker = new ChefWorker(); newWorker.addEventListener("message", this.handleChefMessage.bind(this)); let docURL = document.location.href.split(/[#?]/)[0]; const index = docURL.lastIndexOf("/"); if (index > 0) { docURL = docURL.substring(0, index); } newWorker.postMessage({"action": "docURL", "data": docURL}); // Store the worker, whether or not it's active, and the inputNum as an object const newWorkerObj = { worker: newWorker, active: false, inputNum: this.manager.input.getActiveTab() }; this.chefWorkers.push(newWorkerObj); return this.chefWorkers.indexOf(newWorkerObj); } /** * Removes a ChefWorker * * @param {Object} workerObj */ removeChefWorker(workerObj) { const index = this.chefWorkers.indexOf(workerObj); if (index === -1) { return; } if (this.chefWorkers.length > 1 || this.chefWorkers[index].active) { this.chefWorkers[index].worker.terminate(); this.chefWorkers.splice(index, 1); } // There should always be a ChefWorker loaded if (this.chefWorkers.length === 0) { this.addChefWorker(); } } /** * Finds and returns the object for the ChefWorker of a given inputNum * * @param {number} inputNum */ getChefWorker(inputNum) { for (let i = 0; i < this.chefWorkers.length; i++) { if (this.chefWorkers[i].inputNum === inputNum) { return this.chefWorkers[i]; } } } /** * Handler for messages sent back by the ChefWorkers * * @param {MessageEvent} e */ handleChefMessage(e) { const r = e.data; let inputNum = 0; log.debug(`Receiving ${r.action} from ChefWorker.`); if (r.data.hasOwnProperty("inputNum")) { inputNum = r.data.inputNum; } const currentWorker = this.getChefWorker(inputNum); switch (r.action) { case "bakeComplete": log.debug(`Bake ${inputNum} complete.`); this.updateOutput(r.data, r.data.inputNum); if (this.inputs.length > 0) { this.bakeNextInput(this.chefWorkers.indexOf(currentWorker)); } else { // The ChefWorker is no longer needed log.debug("No more inputs to bake. Closing ChefWorker."); currentWorker.active = false; this.removeChefWorker(currentWorker); const progress = this.getBakeProgress(); if (progress.total === progress.baked) { this.bakingComplete(); } } break; case "BakeError": this.manager.output.updateOutputError(r.data, inputNum); // do more here break; case "dishReturned": this.callbacks[r.data.id](r.data); break; case "silentBakeComplete": break; case "workerLoaded": this.app.workerLoaded = true; log.debug("ChefWorker loaded."); this.app.loaded(); break; case "statusMessage": // Status message should be done per output // log.error(r); this.manager.output.updateOutputMessage(r.data.message, r.data.inputNum); break; case "optionUpdate": log.debug(`Setting ${r.data.option} to ${r.data.value}`); this.app.options[r.data.option] = r.data.value; break; case "setRegisters": // Should this update with the tabs? this.manager.recipe.setRegisters(r.data.opIndex, r.data.numPrevRegisters, r.data.registers); break; case "highlightsCalculated": this.manager.highlighter.displayHighlights(r.data.pos, r.data.direction); break; default: log.error("Unrecognised message from ChefWorker", e); break; } } /** * Update the value of an output * * @param {Object} data * @param {number} inputNum */ updateOutput(data, inputNum) { this.manager.output.updateOutputValue(data, inputNum); this.manager.output.updateOutputStatus("baked", inputNum); // this.manager.recipe.updateBreakpointIndicator(this.app.progress); } /** * Updates the UI to show if baking is in process or not. * * @param {boolean} bakingStatus */ setBakingStatus(bakingStatus) { this.app.baking = bakingStatus; this.manager.controls.toggleBakeButtonFunction(bakingStatus); } /** * Get the progress of the ChefWorkers */ getBakeProgress() { const pendingInputs = this.inputs.length; let bakingInputs = 0; for (let i = 0; i < this.chefWorkers.length; i++) { if (this.chefWorkers[i].active) { bakingInputs++; } } const total = this.totalOutputs; const bakedInputs = total - pendingInputs - bakingInputs; return { total: total, pending: pendingInputs, baking: bakingInputs, baked: bakedInputs }; } /** * Cancels the current bake by terminating and removing all ChefWorkers */ cancelBake() { for (let i = this.chefWorkers.length - 1; i >= 0; i--) { this.removeChefWorker(this.chefWorkers[i]); } this.setBakingStatus(false); this.inputs = []; this.totalOutputs = 0; this.manager.controls.showStaleIndicator(); } /** * Handler for completed bakes */ bakingComplete() { this.setBakingStatus(false); let duration = new Date().getTime() - this.bakeStartTime; duration = duration.toString() + "ms"; const progress = this.getBakeProgress(); let width = progress.total.toString().length; if (duration.length > width) { width = duration.length; } width = width < 2 ? 2 : width; const totalStr = progress.total.toString().padStart(width, " ").replace(/ /g, " "); const durationStr = duration.padStart(width, " ").replace(/ /g, " "); const msg = `Total: ${totalStr}
Time: ${durationStr}`; document.getElementById("bake-info").innerHTML = msg; // look into changing this to something better // for (let i = 0; i < this.outputs.length; i++) { // if (this.outputs[i].data.error) { // this.app.handleError(this.outputs[i].error); // } // } // Should be a value for each input, not just one // Get store the progress for every output // When we run set(), update the breakpoint indicator then // this.app.progress = this.outputs[0].data.progress; // this.app.dish = this.outputs[0].data.dish; // this.manager.recipe.updateBreakpointIndicator(this.app.progress); // Don't need to update the output here as updateOutput() will take care of that document.getElementById("bake").style.background = ""; this.totalOutputs = 0; // Reset for next time log.debug("--- Bake complete ---"); } /** * Bakes the next input * * @param {number} workerIdx */ bakeNextInput(workerIdx) { if (this.inputs.length === 0) return; if (workerIdx === -1) return; if (!this.chefWorkers[workerIdx]) return; const nextInput = this.inputs.splice(0, 1)[0]; log.debug(`Baking input ${nextInput.inputNum}.`); this.manager.output.updateOutputStatus("baking", nextInput.inputNum); this.manager.output.updateOutputMessage("Baking...", nextInput.inputNum); this.chefWorkers[workerIdx].inputNum = nextInput.inputNum; this.chefWorkers[workerIdx].active = true; const input = nextInput.input; if (typeof input === "string") { this.chefWorkers[workerIdx].worker.postMessage({ action: "bake", data: { input: input, recipeConfig: this.recipeConfig, options: this.options, progress: this.progress, step: this.step, inputNum: nextInput.inputNum } }); } else { this.chefWorkers[workerIdx].worker.postMessage({ action: "bake", data: { input: input, recipeConfig: this.recipeConfig, options: this.options, progress: this.progress, step: this.step, inputNum: nextInput.inputNum } }, [nextInput.input]); } } /** * Bakes the current input using the current recipe. * * @param {Object[]} recipeConfig * @param {Object} options * @param {number} progress * @param {boolean} step */ bake(recipeConfig, options, progress, step) { this.setBakingStatus(true); this.bakeStartTime = new Date().getTime(); this.recipeConfig = recipeConfig; this.options = options; this.progress = progress; this.step = step; for (let i = 0; i < this.maxWorkers; i++) { const workerIdx = this.addChefWorker(); if (workerIdx === -1) break; this.bakeNextInput(workerIdx); } this.displayProgress(); } /** * Queues an input ready to be baked * * @param {object} inputData * @param {string | ArrayBuffer} inputData.input * @param {number} inputData.inputNum */ queueInput(inputData) { this.manager.output.updateOutputStatus("pending", inputData.inputNum); this.totalOutputs++; this.inputs.push(inputData); } /** * Asks the ChefWorker to run a silent bake, forcing the browser to load and cache all the relevant * JavaScript code needed to do a real bake. * * @param {Object[]} [recipeConfig] */ silentBake(recipeConfig) { // If there aren't any active ChefWorkers, addChefWorker will // return an inactive worker instead of creating a new one const workerId = this.addChefWorker(); if (workerId === -1) return; this.chefWorkers[workerId].worker.postMessage({ action: "silentBake", data: { recipeConfig: recipeConfig } }); } /** * Asks the ChefWorker to return the dish as the specified type * * @param {Dish} dish * @param {string} type * @param {Function} callback */ getDishAs(dish, type, callback) { const id = this.callbackID++; const workerId = this.addChefWorker(); if (workerId === -1) return; this.callbacks[id] = callback; this.chefWorkers[workerId].worker.postMessage({ action: "getDishAs", data: { dish: dish, type: type, id: id } }); } /** * Sets the console log level in the workers. * * @param {string} level */ setLogLevel(level) { for (let i = 0; i < this.chefWorkers.length; i++) { this.chefWorkers[i].worker.postMessage({ action: "setLogLevel", data: log.getLevel() }); } } /** * Display the bake progress in the output bar and bake button */ displayProgress() { const progress = this.getBakeProgress(); if (progress.total === progress.baked) return; const percentComplete = ((progress.pending + progress.baking) / progress.total) * 100; const bakeButton = document.getElementById("bake"); if (this.app.baking) { if (percentComplete < 100) { document.getElementById("bake").style.background = `linear-gradient(to left, #fea79a ${percentComplete}%, #f44336 ${percentComplete}%)`; } else { bakeButton.style.background = ""; } } else { // not baking bakeButton.style.background = ""; } const bakeInfo = document.getElementById("bake-info"); let width = progress.total.toString().length; width = width < 2 ? 2 : width; const totalStr = progress.total.toString().padStart(width, " ").replace(/ /g, " "); const bakedStr = progress.baked.toString().padStart(width, " ").replace(/ /g, " "); const pendingStr = progress.pending.toString().padStart(width, " ").replace(/ /g, " "); const bakingStr = progress.baking.toString().padStart(width, " ").replace(/ /g, " "); let msg = "Total: " + totalStr; msg += "
Baked: " + bakedStr; if (progress.pending > 0) { msg += "
Pending: " + pendingStr; } else if (progress.baking > 0) { msg += "
Baking: " + bakingStr; } bakeInfo.innerHTML = msg; if (progress.total !== progress.baked) { setTimeout(function() { this.displayProgress(); }.bind(this), 100); } } } export default WorkerWaiter;