2017-07-28 17:53:08 +00:00
|
|
|
function parse(file/*:RawBytes*/, options/*:CFBReadOpts*/)/*:CFBContainer*/ {
|
2018-09-04 07:14:20 +00:00
|
|
|
if(file[0] == 0x50 && file[1] == 0x4b) return parse_zip(file, options);
|
2018-03-05 03:49:40 +00:00
|
|
|
if(file.length < 512) throw new Error("CFB file size " + file.length + " < 512");
|
2017-09-14 21:14:22 +00:00
|
|
|
var mver = 3;
|
|
|
|
var ssz = 512;
|
2014-06-24 04:00:39 +00:00
|
|
|
var nmfs = 0; // number of mini FAT sectors
|
2017-09-14 21:14:22 +00:00
|
|
|
var difat_sec_cnt = 0;
|
|
|
|
var dir_start = 0;
|
|
|
|
var minifat_start = 0;
|
|
|
|
var difat_start = 0;
|
2014-06-24 04:00:39 +00:00
|
|
|
|
2017-07-28 17:53:08 +00:00
|
|
|
var fat_addrs/*:Array<number>*/ = []; // locations of FAT sectors
|
2014-06-24 04:00:39 +00:00
|
|
|
|
|
|
|
/* [MS-CFB] 2.2 Compound File Header */
|
2017-07-28 17:53:08 +00:00
|
|
|
var blob/*:CFBlob*/ = /*::(*/file.slice(0,512)/*:: :any)*/;
|
2014-06-24 04:00:39 +00:00
|
|
|
prep_blob(blob, 0);
|
|
|
|
|
|
|
|
/* major version */
|
2014-11-03 04:02:42 +00:00
|
|
|
var mv = check_get_mver(blob);
|
|
|
|
mver = mv[0];
|
2014-06-24 04:00:39 +00:00
|
|
|
switch(mver) {
|
|
|
|
case 3: ssz = 512; break; case 4: ssz = 4096; break;
|
2018-09-04 07:14:20 +00:00
|
|
|
case 0: if(mv[1] == 0) return parse_zip(file, options);
|
|
|
|
/* falls through */
|
2017-03-30 21:34:37 +00:00
|
|
|
default: throw new Error("Major Version: Expected 3 or 4 saw " + mver);
|
2014-06-24 04:00:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* reprocess header */
|
2017-07-28 17:53:08 +00:00
|
|
|
if(ssz !== 512) { blob = /*::(*/file.slice(0,ssz)/*:: :any)*/; prep_blob(blob, 28 /* blob.l */); }
|
2014-06-24 04:00:39 +00:00
|
|
|
/* Save header for final object */
|
2017-07-28 17:53:08 +00:00
|
|
|
var header/*:RawBytes*/ = file.slice(0,ssz);
|
2014-06-24 04:00:39 +00:00
|
|
|
|
|
|
|
check_shifts(blob, mver);
|
|
|
|
|
|
|
|
// Number of Directory Sectors
|
2017-09-14 21:14:22 +00:00
|
|
|
var dir_cnt/*:number*/ = blob.read_shift(4, 'i');
|
|
|
|
if(mver === 3 && dir_cnt !== 0) throw new Error('# Directory Sectors: Expected 0 saw ' + dir_cnt);
|
2014-06-24 04:00:39 +00:00
|
|
|
|
|
|
|
// Number of FAT Sectors
|
|
|
|
blob.l += 4;
|
|
|
|
|
|
|
|
// First Directory Sector Location
|
|
|
|
dir_start = blob.read_shift(4, 'i');
|
|
|
|
|
|
|
|
// Transaction Signature
|
|
|
|
blob.l += 4;
|
|
|
|
|
|
|
|
// Mini Stream Cutoff Size
|
|
|
|
blob.chk('00100000', 'Mini Stream Cutoff Size: ');
|
|
|
|
|
|
|
|
// First Mini FAT Sector Location
|
|
|
|
minifat_start = blob.read_shift(4, 'i');
|
|
|
|
|
|
|
|
// Number of Mini FAT Sectors
|
|
|
|
nmfs = blob.read_shift(4, 'i');
|
|
|
|
|
|
|
|
// First DIFAT sector location
|
|
|
|
difat_start = blob.read_shift(4, 'i');
|
|
|
|
|
|
|
|
// Number of DIFAT Sectors
|
2017-09-14 21:14:22 +00:00
|
|
|
difat_sec_cnt = blob.read_shift(4, 'i');
|
2014-06-24 04:00:39 +00:00
|
|
|
|
|
|
|
// Grab FAT Sector Locations
|
2017-07-28 17:53:08 +00:00
|
|
|
for(var q = -1, j = 0; j < 109; ++j) { /* 109 = (512 - blob.l)>>>2; */
|
2014-06-24 04:00:39 +00:00
|
|
|
q = blob.read_shift(4, 'i');
|
|
|
|
if(q<0) break;
|
|
|
|
fat_addrs[j] = q;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Break the file up into sectors */
|
2017-07-28 17:53:08 +00:00
|
|
|
var sectors/*:Array<RawBytes>*/ = sectorify(file, ssz);
|
2014-06-24 04:00:39 +00:00
|
|
|
|
2017-09-14 21:14:22 +00:00
|
|
|
sleuth_fat(difat_start, difat_sec_cnt, sectors, ssz, fat_addrs);
|
2014-06-24 04:00:39 +00:00
|
|
|
|
|
|
|
/** Chains */
|
2017-02-24 05:11:45 +00:00
|
|
|
var sector_list/*:SectorList*/ = make_sector_list(sectors, dir_start, fat_addrs, ssz);
|
2014-06-24 04:00:39 +00:00
|
|
|
|
|
|
|
sector_list[dir_start].name = "!Directory";
|
|
|
|
if(nmfs > 0 && minifat_start !== ENDOFCHAIN) sector_list[minifat_start].name = "!MiniFAT";
|
|
|
|
sector_list[fat_addrs[0]].name = "!FAT";
|
2014-11-03 04:02:42 +00:00
|
|
|
sector_list.fat_addrs = fat_addrs;
|
|
|
|
sector_list.ssz = ssz;
|
2014-06-24 04:00:39 +00:00
|
|
|
|
|
|
|
/* [MS-CFB] 2.6.1 Compound File Directory Entry */
|
2017-11-05 00:15:50 +00:00
|
|
|
var files/*:CFBFiles*/ = {}, Paths/*:Array<string>*/ = [], FileIndex/*:CFBFileIndex*/ = [], FullPaths/*:Array<string>*/ = [];
|
2017-11-05 16:54:11 +00:00
|
|
|
read_directory(dir_start, sector_list, sectors, Paths, nmfs, files, FileIndex, minifat_start);
|
2014-06-24 04:00:39 +00:00
|
|
|
|
2017-11-05 00:15:50 +00:00
|
|
|
build_full_paths(FileIndex, FullPaths, Paths);
|
2017-09-14 21:14:22 +00:00
|
|
|
Paths.shift();
|
2014-06-24 04:00:39 +00:00
|
|
|
|
2017-09-14 21:14:22 +00:00
|
|
|
var o = {
|
2014-06-24 04:00:39 +00:00
|
|
|
FileIndex: FileIndex,
|
2017-11-05 00:15:50 +00:00
|
|
|
FullPaths: FullPaths
|
2014-06-24 04:00:39 +00:00
|
|
|
};
|
2017-09-14 21:14:22 +00:00
|
|
|
|
|
|
|
// $FlowIgnore
|
|
|
|
if(options && options.raw) o.raw = {header: header, sectors: sectors};
|
|
|
|
return o;
|
2014-06-24 04:00:39 +00:00
|
|
|
} // parse
|
|
|
|
|