CyberChef/src/core/operations/SeqUtils.js

258 lines
6.3 KiB
JavaScript
Raw Normal View History

import Utils from "../Utils.js";
2016-11-28 11:42:58 +01:00
/**
* Sequence utility operations.
*
* @author n1474335 [n1474335@gmail.com]
* @copyright Crown Copyright 2016
* @license Apache-2.0
*
* @namespace
*/
const SeqUtils = {
2016-11-28 11:42:58 +01:00
/**
* @constant
* @default
*/
DELIMITER_OPTIONS: ["Line feed", "CRLF", "Space", "Comma", "Semi-colon", "Colon", "Nothing (separate chars)"],
/**
* @constant
* @default
*/
SORT_REVERSE: false,
/**
* @constant
* @default
*/
2017-05-29 14:59:06 +02:00
SORT_ORDER: ["Alphabetical (case sensitive)", "Alphabetical (case insensitive)", "IP address", "Numeric"],
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Sort operation.
*
* @param {string} input
* @param {Object[]} args
* @returns {string}
*/
runSort: function (input, args) {
2017-04-13 19:08:50 +02:00
let delim = Utils.charRep[args[0]],
sortReverse = args[1],
2016-11-28 11:42:58 +01:00
order = args[2],
sorted = input.split(delim);
2017-02-09 16:09:33 +01:00
2016-12-14 17:39:17 +01:00
if (order === "Alphabetical (case sensitive)") {
2016-11-28 11:42:58 +01:00
sorted = sorted.sort();
2016-12-14 17:39:17 +01:00
} else if (order === "Alphabetical (case insensitive)") {
sorted = sorted.sort(SeqUtils._caseInsensitiveSort);
2016-12-14 17:39:17 +01:00
} else if (order === "IP address") {
sorted = sorted.sort(SeqUtils._ipSort);
2017-05-29 14:59:06 +02:00
} else if (order === "Numeric") {
sorted = sorted.sort(SeqUtils._numericSort);
2016-11-28 11:42:58 +01:00
}
2017-02-09 16:09:33 +01:00
if (sortReverse) sorted.reverse();
2016-11-28 11:42:58 +01:00
return sorted.join(delim);
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Unique operation.
*
* @param {string} input
* @param {Object[]} args
* @returns {string}
*/
runUnique: function (input, args) {
2017-04-13 19:08:50 +02:00
const delim = Utils.charRep[args[0]];
2016-11-28 11:42:58 +01:00
return input.split(delim).unique().join(delim);
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* @constant
* @default
*/
SEARCH_TYPE: ["Regex", "Extended (\\n, \\t, \\x...)", "Simple string"],
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Count occurrences operation.
*
* @param {string} input
* @param {Object[]} args
* @returns {number}
*/
runCount: function(input, args) {
2017-04-13 19:08:50 +02:00
let search = args[0].string,
2016-11-28 11:42:58 +01:00
type = args[0].option;
2017-02-09 16:09:33 +01:00
2016-12-14 17:39:17 +01:00
if (type === "Regex" && search) {
2016-11-28 11:42:58 +01:00
try {
2017-04-13 19:08:50 +02:00
let regex = new RegExp(search, "gi"),
2016-11-28 11:42:58 +01:00
matches = input.match(regex);
return matches.length;
2017-02-09 16:09:33 +01:00
} catch (err) {
2016-11-28 11:42:58 +01:00
return 0;
}
} else if (search) {
if (type.indexOf("Extended") === 0) {
search = Utils.parseEscapedChars(search);
2016-11-28 11:42:58 +01:00
}
return input.count(search);
} else {
return 0;
}
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* @constant
* @default
*/
REVERSE_BY: ["Character", "Line"],
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Reverse operation.
*
* @param {byteArray} input
2016-11-28 11:42:58 +01:00
* @param {Object[]} args
* @returns {byteArray}
2016-11-28 11:42:58 +01:00
*/
runReverse: function (input, args) {
2017-04-13 19:31:26 +02:00
let i;
2016-12-14 17:39:17 +01:00
if (args[0] === "Line") {
2017-04-13 19:08:50 +02:00
let lines = [],
2016-11-28 11:42:58 +01:00
line = [],
result = [];
2017-04-13 19:31:26 +02:00
for (i = 0; i < input.length; i++) {
2016-12-14 17:39:17 +01:00
if (input[i] === 0x0a) {
2016-11-28 11:42:58 +01:00
lines.push(line);
line = [];
} else {
line.push(input[i]);
}
}
lines.push(line);
lines.reverse();
for (i = 0; i < lines.length; i++) {
result = result.concat(lines[i]);
result.push(0x0a);
}
return result.slice(0, input.length);
} else {
return input.reverse();
}
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Add line numbers operation.
*
* @param {string} input
* @param {Object[]} args
* @returns {string}
*/
runAddLineNumbers: function(input, args) {
2017-04-13 19:08:50 +02:00
let lines = input.split("\n"),
2016-11-28 11:42:58 +01:00
output = "",
width = lines.length.toString().length;
2017-02-09 16:09:33 +01:00
2017-04-13 19:08:50 +02:00
for (let n = 0; n < lines.length; n++) {
2016-11-28 11:42:58 +01:00
output += Utils.pad((n+1).toString(), width, " ") + " " + lines[n] + "\n";
}
return output.slice(0, output.length-1);
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Remove line numbers operation.
*
* @param {string} input
* @param {Object[]} args
* @returns {string}
*/
runRemoveLineNumbers: function(input, args) {
2016-11-28 11:42:58 +01:00
return input.replace(/^[ \t]{0,5}\d+[\s:|\-,.)\]]/gm, "");
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Expand alphabet range operation.
*
* @param {string} input
* @param {Object[]} args
* @returns {string}
*/
runExpandAlphRange: function(input, args) {
return Utils.expandAlphRange(input).join(args[0]);
2016-11-28 11:42:58 +01:00
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Comparison operation for sorting of strings ignoring case.
*
* @private
* @param {string} a
* @param {string} b
* @returns {number}
*/
_caseInsensitiveSort: function(a, b) {
2016-11-28 11:42:58 +01:00
return a.toLowerCase().localeCompare(b.toLowerCase());
},
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
/**
* Comparison operation for sorting of IPv4 addresses.
*
* @private
* @param {string} a
* @param {string} b
* @returns {number}
*/
_ipSort: function(a, b) {
2017-04-13 19:08:50 +02:00
let a_ = a.split("."),
2016-11-28 11:42:58 +01:00
b_ = b.split(".");
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
a_ = a_[0] * 0x1000000 + a_[1] * 0x10000 + a_[2] * 0x100 + a_[3] * 1;
b_ = b_[0] * 0x1000000 + b_[1] * 0x10000 + b_[2] * 0x100 + b_[3] * 1;
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
if (isNaN(a_) && !isNaN(b_)) return 1;
if (!isNaN(a_) && isNaN(b_)) return -1;
if (isNaN(a_) && isNaN(b_)) return a.localeCompare(b);
2017-02-09 16:09:33 +01:00
2016-11-28 11:42:58 +01:00
return a_ - b_;
},
2017-02-09 16:09:33 +01:00
2017-05-29 14:59:06 +02:00
/**
* Comparison operation for sorting of numeric values.
*
* @author Chris van Marle
2017-05-29 14:59:06 +02:00
* @private
* @param {string} a
* @param {string} b
* @returns {number}
*/
_numericSort: function _numericSort(a, b) {
let a_ = a.split(/([^\d]+)/),
b_ = b.split(/([^\d]+)/);
for (let i = 0; i < a_.length && i < b.length; ++i) {
2017-05-29 14:59:06 +02:00
if (isNaN(a_[i]) && !isNaN(b_[i])) return 1; // Numbers after non-numbers
if (!isNaN(a_[i]) && isNaN(b_[i])) return -1;
if (isNaN(a_[i]) && isNaN(b_[i])) {
let ret = a_[i].localeCompare(b_[i]); // Compare strings
if (ret !== 0) return ret;
}
if (!isNaN(a_[i]) && !isNaN(a_[i])) { // Compare numbers
if (a_[i] - b_[i] !== 0) return a_[i] - b_[i];
}
}
return 0;
},
2016-11-28 11:42:58 +01:00
};
export default SeqUtils;