Skip to content
Open
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion src/core/config/Categories.json
Original file line number Diff line number Diff line change
Expand Up @@ -337,7 +337,8 @@
"Sleep",
"File Tree",
"Take nth bytes",
"Drop nth bytes"
"Drop nth bytes",
"Word Count"
]
},
{
Expand Down
116 changes: 116 additions & 0 deletions src/core/operations/WordCount.mjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,116 @@
/**
* @author sw5678
* @copyright Crown Copyright 2016
* @license Apache-2.0
*/

import Operation from "../Operation.mjs";
import Utils from "../Utils.mjs";
import {LETTER_DELIM_OPTIONS} from "../lib/Delim.mjs";
import {caseInsensitiveSort} from "../lib/Sort.mjs";


/**
* Word Count operation
*/
class WordCount extends Operation {

/**
* Word Count constructor
*/
constructor() {
super();

this.name = "Word Count";
this.module = "Default";
this.description = "Provides a count of each word in a given text";
this.inputType = "string";
this.outputType = "string";
this.args = [
{
name: "Delimiter",
type: "option",
value: LETTER_DELIM_OPTIONS
},
{
"name": "Include Total",
"type": "boolean",
"value": true
},
{
"name": "Order",
"type": "option",
"value": ["Alphabetical", "Count"]
}
];
}

/**
* @param {string} input
* @param {Object[]} args
* @returns {string}
*/
run(input, args) {

const delimiter = Utils.charRep(args[0]);

// Lower case and split
const inputArray = input.replace(/(?:\r\n|\r|\n)/g, delimiter).toLowerCase().split(delimiter);

// Count up the words
const counter = {};
let total = 0;
for (let j = 0; j < inputArray.length; j++) {

// Trim whitespace and replace punctuation
const word = inputArray[j].replace(/(?:!|"|#|\$|%|&|\(|\)|\*|\+|,|-|\.|\/|:|;|<|=|>|\?|@|\[|\\|\]|\^|_|`|\{|\||\}|~|£)/g, "").trim();
Comment thread
sw5678 marked this conversation as resolved.
Outdated

// If empty string or ', then skip
if (word === "" || /[']+/.test(word)) {
continue;
} else if (word in counter) {
counter[word]++;
total++;
} else {
counter[word] = 1;
total++;
}
}

// Sort results
let order;
if (args[2] === "Alphabetical") {
// Sort alphabetically
order = Object.keys(counter).sort(caseInsensitiveSort);
} else if (args[2] === "Count") {
// Sort by count
// Create the array of key-value pairs
order = Object.keys(counter).map((key) => {
return [key, counter[key]];
});
// Sort the array based on the second element (i.e. the value)
order.sort((first, second) => {
return second[1] - first[1];
});
// Obtain the list of keys in sorted order of the values.
order = order.map((e) => {
return e[0];
});
}

// Process output to string
let output = "WORD,COUNT\n";
for (let k = 0; k < order.length; k++) {
Comment thread
sw5678 marked this conversation as resolved.
Outdated
output = output + order[k] + "," + counter[order[k]] + "\n";
}

// Add total counter at the bottom
if (args[1]) {
output = output + "TOTAL," + total;
}

return output;
}
}

export default WordCount;
1 change: 1 addition & 0 deletions tests/operations/index.mjs
Original file line number Diff line number Diff line change
Expand Up @@ -170,6 +170,7 @@ import "./tests/JA3Fingerprint.mjs";
import "./tests/JA3SFingerprint.mjs";
import "./tests/HASSH.mjs";
import "./tests/JSONtoYAML.mjs";
import "./tests/WordCount.mjs";

// Cannot test operations that use the File type yet
// import "./tests/SplitColourChannels.mjs";
Expand Down
117 changes: 117 additions & 0 deletions tests/operations/tests/WordCount.mjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,117 @@
/**
* @author sw5678
* @copyright Crown Copyright 2023
* @license Apache-2.0
*/
import TestRegister from "../../lib/TestRegister.mjs";

TestRegister.addTests([
{
"name": "Word Count: Empty test 1",
"input": "",
"expectedOutput": "WORD,COUNT\nTOTAL,0",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", true, "Alphabetical"],
},
],
},
{
"name": "Word Count: Empty test 2",
"input": "",
"expectedOutput": "WORD,COUNT\nTOTAL,0",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", true, "Count"],
},
],
},
{
"name": "Word Count: Empty test 3",
"input": "",
"expectedOutput": "WORD,COUNT\n",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", false, "Alphabetical"],
},
],
},
{
"name": "Word Count: Empty test 4",
"input": "",
"expectedOutput": "WORD,COUNT\n",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", false, "Count"],
},
],
},
{
"name": "Word Count: Count test 1",
"input": "Hello world. Hello. \n\n World, ''!@£$%^&*()_+=-[]{};'|:/.,<>? world",
"expectedOutput": "WORD,COUNT\nhello,2\nworld,3\nTOTAL,5",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", true, "Alphabetical"],
},
],
},
{
"name": "Word Count: Count test 2",
"input": "Hello world. Hello. \n\n World, ''!@£$%^&*()_+=-[]{};'|:/.,<>? world",
"expectedOutput": "WORD,COUNT\nworld,3\nhello,2\nTOTAL,5",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", true, "Count"],
},
],
},
{
"name": "Word Count: Count test 3",
"input": "Hello world. Hello. \n\n World, ''!@£$%^&*()_+=-[]{};'|:/.,<>? world",
"expectedOutput": "WORD,COUNT\nhello,2\nworld,3\n",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", false, "Alphabetical"],
},
],
},
{
"name": "Word Count: Count test 4",
"input": "Hello world. Hello. \n\n World, ''!@£$%^&*()_+=-[]{};'|:/.,<>? world",
"expectedOutput": "WORD,COUNT\nworld,3\nhello,2\n",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Space", false, "Count"],
},
],
},
{
"name": "Word Count: Different delimiter test",
"input": "Hello, World\nhello, world \n''!@£$%^&*()_+=-[]{};'|:/.,<>? world",
"expectedOutput": "WORD,COUNT\nworld,3\nhello,2\n",

"recipeConfig": [
{
"op": "Word Count",
"args": ["Comma", false, "Count"],
},
],
}
]);
Loading