mirror of
https://github.com/gchq/CyberChef.git
synced 2024-11-16 08:58:30 +01:00
108 lines
2.9 KiB
JavaScript
108 lines
2.9 KiB
JavaScript
/**
|
|
* @author Ge0rg3 [georgeomnet+cyberchef@gmail.com]
|
|
* @copyright Crown Copyright 2019
|
|
* @license Apache-2.0
|
|
*/
|
|
|
|
import Operation from "../Operation.mjs";
|
|
import OperationError from "../errors/OperationError.mjs";
|
|
import Utils from "../Utils.mjs";
|
|
import { isImage } from "../lib/FileType.mjs";
|
|
import { toBase64 } from "../lib/Base64.mjs";
|
|
import jimplib from "jimp/es/index.js";
|
|
const jimp = jimplib.default ? jimplib.default : jimplib;
|
|
|
|
/**
|
|
* View Bit Plane operation
|
|
*/
|
|
class ViewBitPlane extends Operation {
|
|
|
|
/**
|
|
* ViewBitPlane constructor
|
|
*/
|
|
constructor() {
|
|
super();
|
|
|
|
this.name = "View Bit Plane";
|
|
this.module = "Image";
|
|
this.description = "Extracts and displays a bit plane of any given image. These show only a single bit from each pixel, and can be used to hide messages in Steganography.";
|
|
this.infoURL = "https://wikipedia.org/wiki/Bit_plane";
|
|
this.inputType = "ArrayBuffer";
|
|
this.outputType = "ArrayBuffer";
|
|
this.presentType = "html";
|
|
this.args = [
|
|
{
|
|
name: "Colour",
|
|
type: "option",
|
|
value: COLOUR_OPTIONS
|
|
},
|
|
{
|
|
name: "Bit",
|
|
type: "number",
|
|
value: 0
|
|
}
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @param {ArrayBuffer} input
|
|
* @param {Object[]} args
|
|
* @returns {ArrayBuffer}
|
|
*/
|
|
async run(input, args) {
|
|
if (!isImage(input)) throw new OperationError("Please enter a valid image file.");
|
|
|
|
const [colour, bit] = args,
|
|
parsedImage = await jimp.read(input),
|
|
width = parsedImage.bitmap.width,
|
|
height = parsedImage.bitmap.height,
|
|
colourIndex = COLOUR_OPTIONS.indexOf(colour),
|
|
bitIndex = 7-bit;
|
|
|
|
if (bit < 0 || bit > 7) {
|
|
throw new OperationError("Error: Bit argument must be between 0 and 7");
|
|
}
|
|
|
|
let pixel, bin, newPixelValue;
|
|
|
|
parsedImage.scan(0, 0, width, height, function(x, y, idx) {
|
|
pixel = this.bitmap.data[idx + colourIndex];
|
|
bin = Utils.bin(pixel);
|
|
newPixelValue = 255;
|
|
|
|
if (bin.charAt(bitIndex) === "1") newPixelValue = 0;
|
|
|
|
for (let i=0; i < 3; i++) {
|
|
this.bitmap.data[idx + i] = newPixelValue;
|
|
}
|
|
this.bitmap.data[idx + 3] = 255;
|
|
|
|
});
|
|
|
|
const imageBuffer = await parsedImage.getBufferAsync(jimp.AUTO);
|
|
|
|
return new Uint8Array(imageBuffer).buffer;
|
|
}
|
|
|
|
/**
|
|
* Displays the extracted data as an image for web apps.
|
|
* @param {ArrayBuffer} data
|
|
* @returns {html}
|
|
*/
|
|
present(data) {
|
|
if (!data.length) return "";
|
|
const type = isImage(data);
|
|
|
|
return `<img src="data:${type};base64,${toBase64(data)}">`;
|
|
}
|
|
|
|
}
|
|
|
|
const COLOUR_OPTIONS = [
|
|
"Red",
|
|
"Green",
|
|
"Blue",
|
|
"Alpha"
|
|
];
|
|
|
|
export default ViewBitPlane;
|