2018-12-14 17:43:03 +01:00
|
|
|
/**
|
|
|
|
* @author n1474335 [n1474335@gmail.com]
|
|
|
|
* @copyright Crown Copyright 2018
|
|
|
|
* @license Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
import Operation from "../Operation";
|
2019-03-02 17:12:21 +01:00
|
|
|
import OperationError from "../errors/OperationError";
|
2018-12-14 17:43:03 +01:00
|
|
|
import Utils from "../Utils";
|
2019-01-01 16:12:01 +01:00
|
|
|
import {scanForFileTypes, extractFile} from "../lib/FileType";
|
2019-01-14 19:55:10 +01:00
|
|
|
import {FILE_SIGNATURES} from "../lib/FileSignatures";
|
2018-12-14 17:43:03 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Extract Files operation
|
|
|
|
*/
|
|
|
|
class ExtractFiles extends Operation {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ExtractFiles constructor
|
|
|
|
*/
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.name = "Extract Files";
|
|
|
|
this.module = "Default";
|
|
|
|
this.description = "TODO";
|
|
|
|
this.infoURL = "https://forensicswiki.org/wiki/File_Carving";
|
|
|
|
this.inputType = "ArrayBuffer";
|
|
|
|
this.outputType = "List<File>";
|
|
|
|
this.presentType = "html";
|
2019-01-14 19:55:10 +01:00
|
|
|
this.args = Object.keys(FILE_SIGNATURES).map(cat => {
|
|
|
|
return {
|
|
|
|
name: cat,
|
|
|
|
type: "boolean",
|
|
|
|
value: cat === "Miscellaneous" ? false : true
|
|
|
|
};
|
2019-03-02 17:12:21 +01:00
|
|
|
}).concat([
|
|
|
|
{
|
|
|
|
name: "Ignore failed extractions",
|
|
|
|
type: "boolean",
|
|
|
|
value: "true"
|
|
|
|
}
|
|
|
|
]);
|
2018-12-14 17:43:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {ArrayBuffer} input
|
|
|
|
* @param {Object[]} args
|
|
|
|
* @returns {List<File>}
|
|
|
|
*/
|
|
|
|
run(input, args) {
|
2019-01-14 19:55:10 +01:00
|
|
|
const bytes = new Uint8Array(input),
|
2019-03-02 17:12:21 +01:00
|
|
|
categories = [],
|
|
|
|
ignoreFailedExtractions = args.pop(1);
|
2019-01-14 19:55:10 +01:00
|
|
|
|
|
|
|
args.forEach((cat, i) => {
|
|
|
|
if (cat) categories.push(Object.keys(FILE_SIGNATURES)[i]);
|
|
|
|
});
|
2018-12-14 17:43:03 +01:00
|
|
|
|
|
|
|
// Scan for embedded files
|
2019-01-14 19:55:10 +01:00
|
|
|
const detectedFiles = scanForFileTypes(bytes, categories);
|
2018-12-14 17:43:03 +01:00
|
|
|
|
|
|
|
// Extract each file that we support
|
|
|
|
const files = [];
|
2018-12-18 18:44:42 +01:00
|
|
|
detectedFiles.forEach(detectedFile => {
|
2018-12-14 17:43:03 +01:00
|
|
|
try {
|
2018-12-18 18:44:42 +01:00
|
|
|
files.push(extractFile(bytes, detectedFile.fileDetails, detectedFile.offset));
|
2019-01-11 18:44:13 +01:00
|
|
|
} catch (err) {
|
2019-03-02 17:12:21 +01:00
|
|
|
if (!ignoreFailedExtractions && err.message.indexOf("No extraction algorithm available") < 0) {
|
|
|
|
throw new OperationError(
|
|
|
|
`Error while attempting to extract ${detectedFile.fileDetails.name} ` +
|
|
|
|
`at offset ${detectedFile.offset}:\n` +
|
|
|
|
`${err.message}`
|
|
|
|
);
|
|
|
|
}
|
2019-01-11 18:44:13 +01:00
|
|
|
}
|
2018-12-14 17:43:03 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
return files;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Displays the files in HTML for web apps.
|
|
|
|
*
|
|
|
|
* @param {File[]} files
|
|
|
|
* @returns {html}
|
|
|
|
*/
|
|
|
|
async present(files) {
|
|
|
|
return await Utils.displayFilesAsHTML(files);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
export default ExtractFiles;
|