diff --git a/src/core/errors/ExcludedOperationError.mjs b/src/core/errors/ExcludedOperationError.mjs new file mode 100644 index 00000000..2972c31d --- /dev/null +++ b/src/core/errors/ExcludedOperationError.mjs @@ -0,0 +1,25 @@ +/** + * Custom error type for handling operation that isnt included in node.js API + * + * @author d98762625 [d98762625@gmail.com] + * @copyright Crown Copyright 2018 + * @license Apache-2.0 + */ +class ExcludedOperationError extends Error { + /** + * Standard error constructor. Adds no new behaviour. + * + * @param args - Standard error args + */ + constructor(...args) { + super(...args); + + this.type = "ExcludedOperationError"; + + if (Error.captureStackTrace) { + Error.captureStackTrace(this, ExcludedOperationError); + } + } +} + +export default ExcludedOperationError; diff --git a/src/node/api.mjs b/src/node/api.mjs index 1af668ac..7c0551e6 100644 --- a/src/node/api.mjs +++ b/src/node/api.mjs @@ -11,6 +11,7 @@ import SyncDish from "./SyncDish"; import Recipe from "./Recipe"; import OperationConfig from "./config/OperationConfig.json"; import { sanitise } from "./apiUtils"; +import ExludedOperationError from "../core/errors/ExcludedOperationError"; /** @@ -246,3 +247,19 @@ export function bake(operations){ return recipe.execute(dish); }; } + +/** + * Explain that the given operation is not included in the Node.js version. + * @param {String} name - name of operation + */ +export function explainExludedFunction(name) { + /** + * Throw new error type with useful message. + */ + const func = () => { + throw new ExludedOperationError(`Sorry, the ${name} operation is not available in the Node.js version of CyberChef.`); + }; + // Add opName prop so Recipe can handle it, just like wrap does. + func.opName = name; + return func; +} diff --git a/src/node/config/excludedOperations.mjs b/src/node/config/excludedOperations.mjs index aede6808..6344215d 100644 --- a/src/node/config/excludedOperations.mjs +++ b/src/node/config/excludedOperations.mjs @@ -22,7 +22,7 @@ export default [ // esprima doesn't work in .mjs "JavaScriptBeautify", "JavaScriptMinify", - "JavaScriptParse", + "JavaScriptParser", // Relies on state of recipe. // "Magic", diff --git a/src/node/config/scripts/generateNodeIndex.mjs b/src/node/config/scripts/generateNodeIndex.mjs index 5acbbf60..3b68028b 100644 --- a/src/node/config/scripts/generateNodeIndex.mjs +++ b/src/node/config/scripts/generateNodeIndex.mjs @@ -40,8 +40,9 @@ let code = `/** import "babel-polyfill"; import SyncDish from "./SyncDish"; -import { wrap, help, bake } from "./api"; +import { wrap, help, bake, explainExludedFunction } from "./api"; import { + // import as core_ to avoid name clashes after wrap. `; includedOperations.forEach((op) => { @@ -76,23 +77,32 @@ includedOperations.forEach((op) => { code += ` "${decapitalise(op)}": wrap(core_${op}),\n`; }); +excludedOperations.forEach((op) => { + code += ` "${decapitalise(op)}": explainExludedFunction("${op}"),\n`; +}); + code += ` }; } const chef = generateChef(); +// Add some additional features to chef object. chef.help = help; chef.dish = SyncDish; + +// Define consts here so we can add to top-level export - wont allow +// export of chef property. `; -includedOperations.forEach((op) => { +Object.keys(operations).forEach((op) => { code += `const ${decapitalise(op)} = chef.${decapitalise(op)};\n`; }); code +=` +// Define array of all operations to create register for bake. const operations = [\n`; -includedOperations.forEach((op) => { +Object.keys(operations).forEach((op) => { code += ` ${decapitalise(op)},\n`; }); @@ -100,11 +110,13 @@ code += `]; chef.bake = bake(operations); export default chef; + +// Operations as top level exports. export { operations, `; -includedOperations.forEach((op) => { +Object.keys(operations).forEach((op) => { code += ` ${decapitalise(op)},\n`; }); diff --git a/src/node/index.mjs b/src/node/index.mjs index d803485f..c879614e 100644 --- a/src/node/index.mjs +++ b/src/node/index.mjs @@ -11,8 +11,9 @@ import "babel-polyfill"; import SyncDish from "./SyncDish"; -import { wrap, help, bake } from "./api"; +import { wrap, help, bake, explainExludedFunction } from "./api"; import { + // import as core_ to avoid name clashes after wrap. ADD as core_ADD, AESDecrypt as core_AESDecrypt, AESEncrypt as core_AESEncrypt, @@ -122,7 +123,6 @@ import { JPathExpression as core_JPathExpression, JSONBeautify as core_JSONBeautify, JSONMinify as core_JSONMinify, - JavaScriptParser as core_JavaScriptParser, Keccak as core_Keccak, MD2 as core_MD2, MD4 as core_MD4, @@ -381,7 +381,6 @@ function generateChef() { "JPathExpression": wrap(core_JPathExpression), "JSONBeautify": wrap(core_JSONBeautify), "JSONMinify": wrap(core_JSONMinify), - "javaScriptParser": wrap(core_JavaScriptParser), "keccak": wrap(core_Keccak), "MD2": wrap(core_MD2), "MD4": wrap(core_MD4), @@ -510,12 +509,31 @@ function generateChef() { "XPathExpression": wrap(core_XPathExpression), "zlibDeflate": wrap(core_ZlibDeflate), "zlibInflate": wrap(core_ZlibInflate), + "fork": explainExludedFunction("Fork"), + "merge": explainExludedFunction("Merge"), + "jump": explainExludedFunction("Jump"), + "conditionalJump": explainExludedFunction("ConditionalJump"), + "label": explainExludedFunction("Label"), + "comment": explainExludedFunction("Comment"), + "tar": explainExludedFunction("Tar"), + "untar": explainExludedFunction("Untar"), + "unzip": explainExludedFunction("Unzip"), + "zip": explainExludedFunction("Zip"), + "javaScriptBeautify": explainExludedFunction("JavaScriptBeautify"), + "javaScriptMinify": explainExludedFunction("JavaScriptMinify"), + "javaScriptParser": explainExludedFunction("JavaScriptParser"), + "renderImage": explainExludedFunction("RenderImage"), + "syntaxHighlighter": explainExludedFunction("SyntaxHighlighter"), }; } const chef = generateChef(); +// Add some additional features to chef object. chef.help = help; chef.dish = SyncDish; + +// Define consts here so we can add to top-level export - wont allow +// export of chef property. const ADD = chef.ADD; const AESDecrypt = chef.AESDecrypt; const AESEncrypt = chef.AESEncrypt; @@ -547,8 +565,10 @@ const CTPH = chef.CTPH; const cartesianProduct = chef.cartesianProduct; const changeIPFormat = chef.changeIPFormat; const chiSquare = chef.chiSquare; +const comment = chef.comment; const compareCTPHHashes = chef.compareCTPHHashes; const compareSSDEEPHashes = chef.compareSSDEEPHashes; +const conditionalJump = chef.conditionalJump; const convertArea = chef.convertArea; const convertDataUnits = chef.convertDataUnits; const convertDistance = chef.convertDistance; @@ -586,6 +606,7 @@ const fletcher16Checksum = chef.fletcher16Checksum; const fletcher32Checksum = chef.fletcher32Checksum; const fletcher64Checksum = chef.fletcher64Checksum; const fletcher8Checksum = chef.fletcher8Checksum; +const fork = chef.fork; const formatMACAddresses = chef.formatMACAddresses; const frequencyDistribution = chef.frequencyDistribution; const fromBCD = chef.fromBCD; @@ -625,8 +646,12 @@ const hexToPEM = chef.hexToPEM; const JPathExpression = chef.JPathExpression; const JSONBeautify = chef.JSONBeautify; const JSONMinify = chef.JSONMinify; +const javaScriptBeautify = chef.javaScriptBeautify; +const javaScriptMinify = chef.javaScriptMinify; const javaScriptParser = chef.javaScriptParser; +const jump = chef.jump; const keccak = chef.keccak; +const label = chef.label; const MD2 = chef.MD2; const MD4 = chef.MD4; const MD5 = chef.MD5; @@ -634,6 +659,7 @@ const MD6 = chef.MD6; const magic = chef.magic; const mean = chef.mean; const median = chef.median; +const merge = chef.merge; const microsoftScriptDecoder = chef.microsoftScriptDecoder; const multiply = chef.multiply; const NOT = chef.NOT; @@ -675,6 +701,7 @@ const removeEXIF = chef.removeEXIF; const removeLineNumbers = chef.removeLineNumbers; const removeNullBytes = chef.removeNullBytes; const removeWhitespace = chef.removeWhitespace; +const renderImage = chef.renderImage; const Return = chef.Return; const reverse = chef.reverse; const rotateLeft = chef.rotateLeft; @@ -707,9 +734,11 @@ const subtract = chef.subtract; const sum = chef.sum; const swapEndianness = chef.swapEndianness; const symmetricDifference = chef.symmetricDifference; +const syntaxHighlighter = chef.syntaxHighlighter; const TCPIPChecksum = chef.TCPIPChecksum; const tail = chef.tail; const takeBytes = chef.takeBytes; +const tar = chef.tar; const toBCD = chef.toBCD; const toBase = chef.toBase; const toBase32 = chef.toBase32; @@ -742,6 +771,8 @@ const URLEncode = chef.URLEncode; const unescapeString = chef.unescapeString; const unescapeUnicodeCharacters = chef.unescapeUnicodeCharacters; const unique = chef.unique; +const untar = chef.untar; +const unzip = chef.unzip; const vigenèreDecode = chef.vigenèreDecode; const vigenèreEncode = chef.vigenèreEncode; const whirlpool = chef.whirlpool; @@ -752,10 +783,12 @@ const XMLMinify = chef.XMLMinify; const XOR = chef.XOR; const XORBruteForce = chef.XORBruteForce; const XPathExpression = chef.XPathExpression; +const zip = chef.zip; const zlibDeflate = chef.zlibDeflate; const zlibInflate = chef.zlibInflate; +// Define array of all operations to create register for bake. const operations = [ ADD, AESDecrypt, @@ -788,8 +821,10 @@ const operations = [ cartesianProduct, changeIPFormat, chiSquare, + comment, compareCTPHHashes, compareSSDEEPHashes, + conditionalJump, convertArea, convertDataUnits, convertDistance, @@ -827,6 +862,7 @@ const operations = [ fletcher32Checksum, fletcher64Checksum, fletcher8Checksum, + fork, formatMACAddresses, frequencyDistribution, fromBCD, @@ -866,8 +902,12 @@ const operations = [ JPathExpression, JSONBeautify, JSONMinify, + javaScriptBeautify, + javaScriptMinify, javaScriptParser, + jump, keccak, + label, MD2, MD4, MD5, @@ -875,6 +915,7 @@ const operations = [ magic, mean, median, + merge, microsoftScriptDecoder, multiply, NOT, @@ -916,6 +957,7 @@ const operations = [ removeLineNumbers, removeNullBytes, removeWhitespace, + renderImage, Return, reverse, rotateLeft, @@ -948,9 +990,11 @@ const operations = [ sum, swapEndianness, symmetricDifference, + syntaxHighlighter, TCPIPChecksum, tail, takeBytes, + tar, toBCD, toBase, toBase32, @@ -983,6 +1027,8 @@ const operations = [ unescapeString, unescapeUnicodeCharacters, unique, + untar, + unzip, vigenèreDecode, vigenèreEncode, whirlpool, @@ -993,12 +1039,15 @@ const operations = [ XOR, XORBruteForce, XPathExpression, + zip, zlibDeflate, zlibInflate, ]; chef.bake = bake(operations); export default chef; + +// Operations as top level exports. export { operations, ADD, @@ -1032,8 +1081,10 @@ export { cartesianProduct, changeIPFormat, chiSquare, + comment, compareCTPHHashes, compareSSDEEPHashes, + conditionalJump, convertArea, convertDataUnits, convertDistance, @@ -1071,6 +1122,7 @@ export { fletcher32Checksum, fletcher64Checksum, fletcher8Checksum, + fork, formatMACAddresses, frequencyDistribution, fromBCD, @@ -1110,8 +1162,12 @@ export { JPathExpression, JSONBeautify, JSONMinify, + javaScriptBeautify, + javaScriptMinify, javaScriptParser, + jump, keccak, + label, MD2, MD4, MD5, @@ -1119,6 +1175,7 @@ export { magic, mean, median, + merge, microsoftScriptDecoder, multiply, NOT, @@ -1160,6 +1217,7 @@ export { removeLineNumbers, removeNullBytes, removeWhitespace, + renderImage, Return, reverse, rotateLeft, @@ -1192,9 +1250,11 @@ export { sum, swapEndianness, symmetricDifference, + syntaxHighlighter, TCPIPChecksum, tail, takeBytes, + tar, toBCD, toBase, toBase32, @@ -1227,6 +1287,8 @@ export { unescapeString, unescapeUnicodeCharacters, unique, + untar, + unzip, vigenèreDecode, vigenèreEncode, whirlpool, @@ -1237,6 +1299,7 @@ export { XOR, XORBruteForce, XPathExpression, + zip, zlibDeflate, zlibInflate, SyncDish as Dish diff --git a/test/tests/nodeApi/nodeApi.mjs b/test/tests/nodeApi/nodeApi.mjs index 88a285b2..c7bfa046 100644 --- a/test/tests/nodeApi/nodeApi.mjs +++ b/test/tests/nodeApi/nodeApi.mjs @@ -318,4 +318,22 @@ TestRegister.addApiTests([ assert.strictEqual(JSONDish.type, 6); }), + it("Excluded operations: throw a sensible error when you try and call one", () => { + try { + chef.fork(); + } catch (e) { + assert.strictEqual(e.type, "ExcludedOperationError"); + assert.strictEqual(e.message, "Sorry, the Fork operation is not available in the Node.js version of CyberChef."); + } + }), + + it("Excluded operations: throw a sensible error when you try and call one", () => { + try { + chef.renderImage(); + } catch (e) { + assert.strictEqual(e.type, "ExcludedOperationError"); + assert.strictEqual(e.message, "Sorry, the RenderImage operation is not available in the Node.js version of CyberChef."); + } + }) + ]);