This commit is contained in:
Windham Wong 2017-12-27 01:34:47 +00:00 committed by GitHub
commit f398c98637
15 changed files with 706 additions and 18 deletions

View file

@ -6,7 +6,30 @@
![](https://reposs.herokuapp.com/?path=gchq/CyberChef&color=blue)
[![](https://img.shields.io/badge/license-Apache%202.0-blue.svg)](https://github.com/gchq/CyberChef/blob/master/LICENSE)
[![Gitter](https://badges.gitter.im/gchq/CyberChef.svg)](https://gitter.im/gchq/CyberChef?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge)
This is a fork of original CyberChef by GCHQ. This project has implemented my own operations for the daily SoC tasks. Please feel free to clone/test/share/contribute it.
## Features
- opeartion **nTcpdump**: tcpdump hexdump convert
- operation **From 0x[Hex]**: e.g. <code>0x217e21</code> to <code>!~!</code>
- operation **From char(hex)**:
- e.g. <code>chr(33)</code> to <code>!</code>
- This operation supports char() and chr()
- Combining the usage of **From 0x[Hex]** and **From char(hex)** can decode <code>chr(0x3333)</code> to <code>!</code>
- operation **HTTP gzip decrypt**: Decrypt gzip payload in HTTP
## Todo
- Operation SQL comment strip function
- Adding more support to char(hex, hex)
- Adding support of themes
---
## Original project info
####*The Cyber Swiss Army Knife*
#### *The Cyber Swiss Army Knife*

26
prod_image_list.txt Normal file
View file

@ -0,0 +1,26 @@
./src/web/static/images/bug-16x16.png
./src/web/static/images/clean-16x16.png
./src/web/static/images/cook_male-32x32.png
./src/web/static/images/cyberchef-128x128.png
./src/web/static/images/download-24x24.png
./src/web/static/images/erase-16x16.png
./src/web/static/images/favourite-16x16.png
./src/web/static/images/favourite-24x24.png
./src/web/static/images/fork_me.png
./src/web/static/images/gitter-badge.svg
./src/web/static/images/help-16x16.png
./src/web/static/images/help-22x22.png
./src/web/static/images/layout-16x16.png
./src/web/static/images/maximise-16x16.png
./src/web/static/images/open_yellow-16x16.png
./src/web/static/images/open_yellow-24x24.png
./src/web/static/images/recycle-16x16.png
./src/web/static/images/save-22x22.png
./src/web/static/images/save_as-16x16.png
./src/web/static/images/settings-22x22.png
./src/web/static/images/speech-16x16.png
./src/web/static/images/step-16x16.png
./src/web/static/images/switch-16x16.png
./src/web/static/images/thumb_down-16x16.png
./src/web/static/images/thumb_up-16x16.png
./src/web/static/images/undo-16x16.png

16
sample_recipe.md Normal file
View file

@ -0,0 +1,16 @@
# Ready-To-Go Sample Recipe
## tcpdump (no ASCII) Daily Use
1. tcpdump (no ASCII) convert to plaintext (e.g. <code>0x0000: 0020 0A0D</code>)
2. URL decode loop (e.g. <code>%3d</code> to <code>=</code>)
3. From HTML Entity (e.g. <code>&amp;</code> to <code>&</code>)
4. From 0x[Hex] (e.g. <code>0x33</code> to <code>!</code>)
5. From Char(Hex) (e.g. <code>char(33)</code> to <code>!</code>)
[{"op":"From nTcpdump","args":[]},
{"op":"URL Decode","args":[]},
{"op":"Conditional Jump","args":["\\%([0-9a-fA-F]{2,})","-1","45"]},
{"op":"From HTML Entity","args":[]},
{"op":"From 0x[Hex]","args":[]},
{"op":"From Char(Hex)","args":[]}]

View file

@ -340,6 +340,68 @@ const Utils = {
},
/**
* Translates a hex string into an array of bytes.
*
* @param {string} hexStr
* @returns {byteArray}
*
* @example
* // returns [0xfe, 0x09, 0xa7]
* Utils.hexToByteArray("fe09a7");
*/
hexToByteArray: function(hexStr) {
// TODO: Handle errors i.e. input string is not hex
if (!hexStr) return [];
hexStr = hexStr.replace(/\s+/g, "");
const byteArray = [];
for (let i = 0; i < hexStr.length; i += 2) {
byteArray.push(parseInt(hexStr.substr(i, 2), 16));
}
return byteArray;
},
/**
* Translates an array of bytes to a hex string.
*
* @param {byteArray} byteArray
* @returns {string}
*
* @example
* // returns "fe09a7"
* Utils.byteArrayToHex([0xfe, 0x09, 0xa7]);
*/
byteArrayToHex: function(byteArray) {
if (!byteArray) return "";
let hexStr = "";
for (let i = 0; i < byteArray.length; i++) {
hexStr += Utils.hex(byteArray[i]) + " ";
}
return hexStr.slice(0, hexStr.length-1);
},
/**
* Translates an array of bytes to a hex string.
*
* @param {byteArray} byteArray
* @returns {string}
*
* @example
* // returns "fe09a7"
* Utils.byteArrayToHex([0xfe, 0x09, 0xa7]);
*/
byteArrayToHexNoSpace: function(byteArray) {
if (!byteArray) return "";
let hexStr = "";
for (let i = 0; i < byteArray.length; i++) {
hexStr += Utils.hex(byteArray[i]);
}
return hexStr.slice(0, hexStr.length-1);
},
/**
* Converts a string to a byte array.
* Treats the string as UTF-8 if any values are over 255.

View file

@ -27,6 +27,8 @@ const Categories = [
ops: [
"To Hexdump",
"From Hexdump",
"From 0x[Hex]",
"From Char(Hex)",
"To Hex",
"From Hex",
"To Charcode",
@ -304,6 +306,14 @@ const Categories = [
"To Kebab case",
]
},
{
name: "Packets",
ops: [
"From Tcpdump",
"HTTP gzip Decrypt",
"Strip TCP Headers",
]
},
{
name: "Other",
ops: [

View file

@ -37,7 +37,7 @@ import StrUtils from "../operations/StrUtils.js";
import Tidy from "../operations/Tidy.js";
import Unicode from "../operations/Unicode.js";
import URL_ from "../operations/URL.js";
import Packets from "../operations/Packets.js";
/**
* Type definition for an OpConf.
@ -520,6 +520,24 @@ const OperationConfig = {
}
]
},
"From 0x[Hex]": {
module: "Default",
description: "Converts a hexadecimal byte string back into a its raw value.<br><br>e.g. <code>0x217e21</code> becomes the UTF-8 encoded string <code>!~!</code>",
highlight: ByteRepr.highlightFrom,
highlightReverse: ByteRepr.highlightTo,
inputType: "string",
outputType: "string",
args: []
},
"From Char(Hex)": {
module: "Default",
description: "Converts a hexadecimal byte string back into a its raw value.<br><br>e.g. <code>chr(33)</code> becomes the UTF-8 encoded string <code>!</code>",
highlight: ByteRepr.highlightFrom,
highlightReverse: ByteRepr.highlightTo,
inputType: "string",
outputType: "string",
args: []
},
"Sum": {
module: "Default",
description: "Adds together a list of numbers. If an item in the string is not a number it is excluded from the list.<br><br>e.g. <code>0x0a 8 .5</code> becomes <code>18.5</code>",
@ -1851,6 +1869,24 @@ const OperationConfig = {
outputType: "string",
args: []
},
"HTTP gzip Decrypt": {
module: "Compression",
description: "Decrypts Gzip payload from a request or response and returning plaintext of the header and decrypted payload.<br><br>Arguments:<br>Library: The library used for decoding GZIP data.<br>Threshold: The threshold value for searching non-GZIP data. It has to be at least 8.",
inputType: "byteArray",
outputType: "byteArray",
args: [
{
name: "Library",
type: "option",
value: Compress.HTTP_GZIP_OPTION
},
{
name: "Threshold",
type: "number",
value: Compress.HTTP_GZIP_THRESHOLD
},
]
},
"Parse User Agent": {
module: "HTTP",
description: "Attempts to identify and categorise information contained in a user-agent string.",
@ -3964,6 +4000,36 @@ const OperationConfig = {
}
]
},
"From Tcpdump": {
module: "Packets",
description: "[DEPRECATED] Converts Tcpdump hex to string",
inputType: "string",
outputType: "byteArray",
args: []
},
"Strip TCP Headers": {
module: "Packets",
description: "Remove selected TCP headers from hexstream using Regular Expressions.<br /><br />Ethernet Header: <code>/^(([0-9a-f]{4} ){6,8}0800 )/igm</code><br />IP Header: <code>/^((45[0-9a-f]{2} ([0-9a-f]{4} ){9}))/igm</code><br />TCP Header: <code>/^([0-9a-f]{4} ){6}((80[0-9a-f]{2} ([0-9a-f]{4} ?){9})|(50[0-9a-f]{2} ([0-9a-f]{4} ?){3}))/igm</code>",
inputType: "string",
outputType: "string",
args: [
{
name: "Ethernet Header",
type: "boolean",
value: Packets.STRIP_ETHERNET_HEADER,
},
{
name: "IP Header",
type: "boolean",
value: Packets.STRIP_IP_HEADER,
},
{
name: "Ethernet Header",
type: "boolean",
value: Packets.STRIP_TCP_HEADER,
},
]
},
"PHP Deserialize": {
module: "Default",
description: "Deserializes PHP serialized data, outputting keyed arrays as JSON.<br><br>This function does not support <code>object</code> tags.<br><br>Example:<br><code>a:2:{s:1:&quot;a&quot;;i:10;i:0;a:1:{s:2:&quot;ab&quot;;b:1;}}</code><br>becomes<br><code>{&quot;a&quot;: 10,0: {&quot;ab&quot;: true}}</code><br><br><u>Output valid JSON:</u> JSON doesn't support integers as keys, whereas PHP serialization does. Enabling this will cast these integers to strings. This will also escape backslashes.",

View file

@ -15,17 +15,18 @@ import Compress from "../../operations/Compress.js";
let OpModules = typeof self === "undefined" ? {} : self.OpModules || {};
OpModules.Compression = {
"Raw Deflate": Compress.runRawDeflate,
"Raw Inflate": Compress.runRawInflate,
"Zlib Deflate": Compress.runZlibDeflate,
"Zlib Inflate": Compress.runZlibInflate,
"Gzip": Compress.runGzip,
"Gunzip": Compress.runGunzip,
"Zip": Compress.runPkzip,
"Unzip": Compress.runPkunzip,
"Bzip2 Decompress": Compress.runBzip2Decompress,
"Tar": Compress.runTar,
"Untar": Compress.runUntar,
"Raw Deflate": Compress.runRawDeflate,
"Raw Inflate": Compress.runRawInflate,
"Zlib Deflate": Compress.runZlibDeflate,
"Zlib Inflate": Compress.runZlibInflate,
"Gzip": Compress.runGzip,
"Gunzip": Compress.runGunzip,
"Zip": Compress.runPkzip,
"Unzip": Compress.runPkunzip,
"Bzip2 Decompress": Compress.runBzip2Decompress,
"Tar": Compress.runTar,
"Untar": Compress.runUntar,
"HTTP gzip Decrypt": Compress.runHttpGzip,
};

View file

@ -53,6 +53,8 @@ OpModules.Default = {
"From Hexdump": Hexdump.runFrom,
"To Hex": ByteRepr.runToHex,
"From Hex": ByteRepr.runFromHex,
"From 0x[Hex]": ByteRepr.runFrom0xHex,
"From Char(Hex)": ByteRepr.runFromCharHex,
"To Octal": ByteRepr.runToOct,
"From Octal": ByteRepr.runFromOct,
"To Charcode": ByteRepr.runToCharcode,

View file

@ -20,6 +20,7 @@ import JSBNModule from "./JSBN.js";
import PublicKeyModule from "./PublicKey.js";
import ShellcodeModule from "./Shellcode.js";
import URLModule from "./URL.js";
import PacketsModule from "./Packets.js";
Object.assign(
OpModules,
@ -35,7 +36,8 @@ Object.assign(
JSBNModule,
PublicKeyModule,
ShellcodeModule,
URLModule
URLModule,
PacketsModule
);
export default OpModules;

View file

@ -0,0 +1,20 @@
import Packets from "../../operations/Packets.js";
/**
* Packets module.
*
* Libraries:
* - Utils.js
*
* @author drkna [whytho@email]
* @copyright Crown Copyright 2017
* @license Apache-2.0
*/
let OpModules = typeof self === "undefined" ? {} : self.OpModules || {};
OpModules.Packets = {
"From Tcpdump": Packets.runFromTcpdump,
"Strip TCP Headers": Packets.stripPacketHeader
};
export default OpModules;

View file

@ -54,6 +54,48 @@ const ByteRepr = {
},
/**
* From 0xHex operation.
*
* @param {string} input (Starting with 0x only in the raw input)
* @param {Object[]} args
* @returns {byteArray}
*/
runFrom0xHex: function(input, args) {
let data = input.replace(/0x([0-9a-f]{2,})/ig,
function(match, p1) {
if (p1) {
return Utils.byteArrayToChars(Utils.fromHex(p1));
}
});
return data;
},
/**
* From char(hex) operation.
*
* @param {string} input (Starting with chr or char only in the raw input)
* @param {Object[]} args
* @returns {byteArray}
*/
runFromCharHex: function(input, args) {
let re = /cha?r\((((\d{1,3})(,\s?)?)+)\)/ig;
let innerRe = /(\d{1,3}),?/g;
let data = input.replace(re,
function(match, p1) {
if (p1) {
let result = "", innerMatch;
while ((innerMatch = innerRe.exec(p1)) != null) {
result += Utils.byteArrayToChars([parseInt(innerMatch[1], 10)]);
}
return result;
}
});
return data;
},
/**
* To Octal operation.
*

View file

@ -5,6 +5,7 @@ import zlibAndGzip from "zlibjs/bin/zlib_and_gzip.min";
import zip from "zlibjs/bin/zip.min";
import unzip from "zlibjs/bin/unzip.min";
import bzip2 from "exports-loader?bzip2!../lib/bzip2.js";
import pako from "pako/index.js";
const Zlib = {
RawDeflate: rawdeflate.Zlib.RawDeflate,
@ -254,6 +255,69 @@ const Compress = {
},
/**
* @constant
* @default
*/
HTTP_GZIP_OPTION: ["pako.js", "zlib.js"],
/**
* @constant
* @default
*/
HTTP_GZIP_THRESHOLD: 8,
/**
* HTTP Gzip operation.
*
* @param {byteArray} input
* @param {Object[]} args
* @returns {byteArray}
*/
runHttpGzip: function(input, args) {
const library = Compress.HTTP_GZIP_OPTION.indexOf(args[0]);
let threshold = Compress.HTTP_GZIP_THRESHOLD;
if (args[1] > 8) {
threshold = args[1];
}
input = Utils.byteArrayToHexNoSpace(input);
let output = input;
let regexStr = /1f8b080[0-8][0-9a-f]{12}/;
let gzipPos = input.search(regexStr);
if (gzipPos === -1) {
return Utils.hexToByteArray(input);
}
while (gzipPos !== -1) {
output = input;
let plainData = output.substr(0, gzipPos);
let gzipData = output.substr(gzipPos);
let httpDataAfter = "";
let httpDataPosRegex = new RegExp("/((3[0-9])|(6[0-9a-f])|(7[0-9a])|(4[0-9a-f])|(5[0-9a])|(2[e-f])|(2b)|(20)){" + threshold + "}/");
let httpDataPos = gzipData.search(httpDataPosRegex);
if (httpDataPos !== -1) {
httpDataAfter = gzipData.substr(httpDataPos);
gzipData = gzipData.substr(0, httpDataPos);
}
console.log(httpDataPos);
gzipData = Utils.hexToByteArray(gzipData);
if (library === 0) {
output = Utils.hexToByteArray(plainData).concat(Array.prototype.slice.call(pako.inflate(gzipData))).concat(Utils.hexToByteArray(httpDataAfter));
} else if (library === 1) {
let gzipDataRaw = new Zlib.Gunzip(gzipData);
output = Utils.hexToByteArray(plainData).concat(Array.prototype.slice.call(gzipDataRaw.decompress())).concat(Utils.hexToByteArray("0d0a 0d0a")).concat(Utils.hexToByteArray(httpDataAfter));
}
input = Utils.byteArrayToHexNoSpace(output);
gzipPos = input.search(regexStr);
}
return output;
},
/**
* @constant
* @default

View file

@ -0,0 +1,81 @@
import Utils from "../Utils.js";
/**
* Packets operations.
*
* @author drkna [whytho@email]
* @copyright Crown Copyright 2016
* @license Apache-2.0
*
* @namespace
*/
const Packets = {
/**
* From Tcpdump Hexstring operation.
*
* @param {string} input
* @param {Object[]} args
* @returns {byteArray}
*/
runFromTcpdump: function(input, args) {
let output = [];
let regex = /^\s*(?:0x[\dA-F]{4}:?)?\s*((?:[\dA-F]{4}\s?){1,8})/igm;
let block = regex.exec(input);
while (block) {
let line = Utils.fromHex(block[1].replace(/-/g, " "));
for (let i = 0; i < line.length; i++) {
output.push(line[i]);
}
block = regex.exec(input);
}
return output;
},
/**
* @constant
* @default
*/
STRIP_ETHERNET_HEADER: true,
/**
* @constant
* @default
*/
STRIP_IP_HEADER: true,
/**
* @constant
* @default
*/
STRIP_TCP_HEADER: true,
/**
* Strip TCP Headersoperation.
*
* @param {string} input
* @param {Object[]} args
* @returns {string}
*/
stripPacketHeader: function(input, args) {
let output = input,
stripEthernet = args[0],
stripIP = args[1],
stripTCP = args[2];
if (stripEthernet) {
output = output.replace(/^(([0-9a-f]{4} ?){6,8}0800 ?)/igm, "");
}
if (stripIP) {
output = output.replace(/^((45[0-9a-f]{2} ?([0-9a-f]{4} ?){9}))/igm, "");
}
if (stripTCP) {
output = output.replace(/^([0-9a-f]{4} ?){6}((80[0-9a-f]{2} ?([0-9a-f]{4} ?){9})|(50[0-9a-f]{2} ?([0-9a-f]{4} ?){3}))/igm, "");
}
return output;
},
};
export default Packets;

278
src/css/themes/orange.css Normal file
View file

@ -0,0 +1,278 @@
#banner {
border-bottom: 1px solid #ddd;
}
.title {
border-bottom: 1px solid #ddd;
font-weight: bold;
color: #424242;
background-color: #fafafa;
}
.gutter {
background-color: #eee;
background-repeat: no-repeat;
background-position: 50%;
}
.gutter.gutter-horizontal {
background-image: url('');
cursor: ew-resize;
}
.gutter.gutter-vertical {
background-image: url('');
cursor: ns-resize;
}
.operation {
border: 1px solid #999;
border-top-width: 0;
}
.op-list .operation { /*blue*/
color: #3a87ad;
background-color: #d9edf7;
border-color: #bce8f1;
}
#rec-list .no-select { /*Overriding*/
color: orange;
background-color: black;
border-color: orange;
}
#rec-list .operation { /*green*/
color: orange;
background-color: whitesmoke;
border-color: orange;
}
#controls {
border-top: 1px solid #ddd;
background-color: #fafafa;
}
.textarea-wrapper textarea,
.textarea-wrapper div {
font-family: Consolas, monospace;
font-size: inherit;
}
.io-info {
font-family: Consolas, monospace;
font-weight: normal;
font-size: 8pt;
}
.arg-title {
font-weight: bold;
}
.arg-input {
height: 34px;
font-size: 15px;
line-height: 1.428571429;
color: #424242;
background-color: #fff;
border: 1px solid #ddd;
font-family: Consolas, monospace;
}
select {
padding: 6px 8px;
height: 34px;
border: 1px solid #ddd;
background-color: #fff;
color: #424242;
}
.arg[disabled] {
background-color: #eee;
}
textarea.arg {
color: #424242;
font-family: Consolas, monospace;
}
.break {
color: #b94a48 !important;
background-color: #f2dede !important;
border-color: #eed3d7 !important;
}
.category-title {
background-color: #fafafa;
border-bottom: 1px solid #eee;
font-weight: bold;
}
.category-title[href='#catFavourites'] {
border-bottom-color: #ddd;
}
.category-title[aria-expanded=true] {
border-bottom-color: #ddd;
}
.category-title.collapsed {
border-bottom-color: #eee;
}
.category-title:hover {
color: #3a87ad;
}
#search {
border-bottom: 1px solid #e3e3e3;
}
.dropping-file {
border: 5px dashed #3a87ad !important;
}
.selected-op {
color: #c09853 !important;
background-color: #fcf8e3 !important;
border-color: #fbeed5 !important;
}
.option-item input[type=number] {
font-size: 14px;
line-height: 1.428571429;
color: #555;
background-color: #fff;
border: 1px solid #ccc;
}
.favourites-hover {
color: #468847;
background-color: #dff0d8;
border: 2px dashed #468847 !important;
padding: 8px 8px 9px 8px;
}
#edit-favourites-list {
border: 1px solid #bce8f1;
}
#edit-favourites-list .operation {
border-left: none;
border-right: none;
}
#edit-favourites-list .operation:last-child {
border-bottom: none;
}
.subtext {
font-style: italic;
font-size: 13px;
color: #999;
}
#save-footer {
border-bottom: 1px solid #e5e5e5;
}
.flow-control-op {
color: #396f3a !important;
background-color: #c7e4ba !important;
border-color: #b3dba2 !important;
}
.flow-control-op.break {
color: #94312f !important;
background-color: #eabfbf !important;
border-color: #e2aeb5 !important;
}
#support-modal textarea {
font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
}
#save-text,
#load-text {
font-family: Consolas, monospace;
}
button.dropdown-toggle {
background-color: #f4f4f4;
}
::-webkit-scrollbar {
width: 10px;
height: 10px;
}
::-webkit-scrollbar-track {
background-color: #fafafa;
}
::-webkit-scrollbar-thumb {
background-color: #ccc;
}
::-webkit-scrollbar-thumb:hover {
background-color: #bbb;
}
::-webkit-scrollbar-corner {
background-color: #fafafa;
}
.disabled {
color: #999 !important;
background-color: #dfdfdf !important;
border-color: #cdcdcd !important;
}
.grey {
color: #333;
background-color: #f5f5f5;
border-color: #ddd;
}
.dark-blue {
color: #fff;
background-color: #428bca;
border-color: #428bca;
}
.red {
color: #b94a48;
background-color: #f2dede;
border-color: #eed3d7;
}
.amber {
color: #c09853;
background-color: #fcf8e3;
border-color: #fbeed5;
}
.green {
color: #468847;
background-color: #dff0d8;
border-color: #d6e9c6;
}
.blue {
color: #3a87ad;
background-color: #d9edf7;
border-color: #bce8f1;
}
#input-text,
#output-text,
#output-html {
position: relative;
border-width: 0px;
margin: 0;
resize: none;
background-color: transparent;
white-space: pre-wrap;
word-wrap: break-word;
color: orange;
background-color: black;
}

View file

@ -1,11 +1,6 @@
<!-- Begin Google Analytics -->
<script>
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
})(window,document,'script','https://www.google-analytics.com/analytics.js','ga');
ga('create', 'UA-85682716-2', 'auto');