Skip to content

Commit 9e22a44

Browse files
committed
version bump 0.13.2
- file writing includes `fs` as appropriate - CLI modify and write options - infrastructure update
1 parent e6d1c8b commit 9e22a44

25 files changed

+306
-144
lines changed

.flowconfig

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
.*/dist/.*
44
.*/test_files/.*
55
.*/test_files_pres/.*
6-
.*/test.js
76

87
.*/bits/.*
98
.*/ctest/.*
@@ -23,6 +22,7 @@
2322
cfb.flow.js
2423
xlscfb.flow.js
2524
.*/bin/.*.njs
25+
test.js
2626

2727
[libs]
2828
bits/10_types.js

.spelling

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
# cfb.js (C) 2013-present SheetJS -- http://sheetjs.com
2+
SheetJS
3+
js-xlsx
4+
5+
# CFB-related terms
6+
CFB
7+
storages
8+
9+
# Third-party
10+
NPM
11+
nodejs
12+
npm
13+
14+
# Other terms
15+
Base64
16+
metadata

Makefile

+16-2
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,11 @@ XLSDEPS=misc/suppress_export.js $(filter-out $(XLSSKIP),$(DEPS))
6666
xlscfb.flow.js: $(XLSDEPS) ## Build support library
6767
cat $^ | tr -d '\15\32' > $@
6868

69+
BYTEFILE=dist/cfb.min.js dist/xlscfb.js
70+
.PHONY: bytes
71+
bytes: ## Display minified and gzipped file sizes
72+
for i in $(BYTEFILE); do printj "%-30s %7d %10d" $$i $$(wc -c < $$i) $$(gzip --best --stdout $$i | wc -c); done
73+
6974

7075
## Testing
7176

@@ -74,6 +79,7 @@ test mocha: test.js $(TARGET) ## Run test suite
7479
mocha -R spec -t 20000
7580

7681
#* To run tests for one format, make test_<fmt>
82+
#* To run the core test suite, make test_misc
7783
TESTFMT=$(patsubst %,test_%,$(FMT))
7884
.PHONY: $(TESTFMT)
7985
$(TESTFMT): test_%:
@@ -82,6 +88,9 @@ $(TESTFMT): test_%:
8288

8389
## Code Checking
8490

91+
.PHONY: fullint
92+
fullint: lint old-lint tslint flow mdlint ## Run all checks
93+
8594
.PHONY: lint
8695
lint: $(TARGET) $(AUXTARGETS) ## Run eslint checks
8796
@eslint --ext .js,.njs,.json,.html,.htm $(TARGET) $(AUXTARGETS) $(CMDS) $(HTMLLINT) package.json
@@ -91,9 +100,9 @@ lint: $(TARGET) $(AUXTARGETS) ## Run eslint checks
91100
old-lint: $(TARGET) $(AUXTARGETS) ## Run jshint and jscs checks
92101
@jshint --show-non-errors $(TARGET) $(AUXTARGETS)
93102
@jshint --show-non-errors $(CMDS)
94-
@jshint --show-non-errors package.json
103+
@jshint --show-non-errors package.json test.js
95104
@jshint --show-non-errors --extract=always $(HTMLLINT)
96-
@jscs $(TARGET) $(AUXTARGETS)
105+
@jscs $(TARGET) $(AUXTARGETS) test.js
97106
if [ -e $(CLOSURE) ]; then java -jar $(CLOSURE) $(REQS) $(FLOWTARGET) --jscomp_warning=reportUnknownTypes >/dev/null; fi
98107

99108
.PHONY: tslint
@@ -115,6 +124,11 @@ misc/coverage.html: $(TARGET) test.js
115124
coveralls: ## Coverage Test + Send to coveralls.io
116125
mocha --require blanket --reporter mocha-lcov-reporter -t 20000 | node ./node_modules/coveralls/bin/coveralls.js
117126

127+
MDLINT=README.md
128+
.PHONY: mdlint
129+
mdlint: $(MDLINT) ## Check markdown documents
130+
alex $^
131+
mdspell -a -n -x -r --en-us $^
118132

119133
.PHONY: help
120134
help:

README.md

+29-29
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
# Compound File Binary Format
22

3-
Pure-JS implementation of MS-CFB: Compound File Binary File Format, a container
3+
Pure JS implementation of MS-CFB: Compound File Binary File Format, a container
44
format used in many Microsoft file types (XLS, DOC, VBA blobs in XLSX and XLSB)
55

66
[![Build Status](https://travis-ci.org/SheetJS/js-cfb.svg?branch=master)](https://travis-ci.org/SheetJS/js-cfb)
@@ -35,7 +35,7 @@ In node:
3535
var CFB = require('cfb');
3636
```
3737

38-
For example, to get the Workbook content from an XLS file:
38+
For example, to get the Workbook content from an Excel 2003 XLS file:
3939

4040
```js
4141
var cfb = CFB.read(filename, {type: 'file'});
@@ -52,15 +52,23 @@ It is preferable to install the library globally with npm:
5252
$ npm install -g cfb
5353
```
5454

55-
The global installation adds a command `cfb` which can work with existing files:
55+
The global installation adds a command `cfb` which can work with files:
5656

57-
- `cfb file` will extract the contents of the file to the current directory.
58-
It will make the corresponding subdirectories.
59-
- `cfb --list-files file` will show a listing of the contained files.
60-
The format follows the `unzip -l` "short format".
61-
- `cfb --repair file` will attempt to repair by reading and re-writing the file.
57+
- `cfb file [names...]` extracts the contents of the file. If additional names
58+
are supplied, only the listed files will be extracted.
59+
60+
- `cfb -l file` lists the contained files (following `unzip -l` "short format")
61+
62+
- `cfb -r file` attempts to repair by reading and re-writing the file.
6263
This fixes some issues with files generated by non-standard tools.
6364

65+
- `cfb -c file [files...]` creates a new file containing the listed files.
66+
The default root entry name is `Root Entry`.
67+
68+
- `cfb -a file [files...]` adds the listed files to the original file.
69+
70+
- `cfb -d file [files...]` deletes the listed files from the original file.
71+
6472

6573
## JS API
6674

@@ -73,20 +81,24 @@ parsed representation of the data.
7381

7482
`CFB.read(blob, opts)` wraps `parse`. `opts.type` controls the behavior:
7583

76-
- `file`: `blob` is interpreted as a file name that will be read
77-
- `base64`: `blob` is interpreted as base64 string
78-
- `binary`: `blob` is interpreted as binary string
79-
- default: `blob` is interpreted as nodejs buffer or array of bytes
84+
| `type` | expected input |
85+
|------------|-----------------------------------------------------------------|
86+
| `"base64"` | string: Base64 encoding of the file |
87+
| `"binary"` | string: binary string (byte `n` is `data.charCodeAt(n)`) |
88+
| `"file"` | string: path of file that will be read (nodejs only) |
89+
| (default) | buffer or array of 8-bit unsigned int (byte `n` is `data[n]`) |
8090

8191
`CFB.find(cfb, path)` performs a case-insensitive match for the path (or file
8292
name, if there are no slashes) and returns an entry object or null if not found.
8393

8494
`CFB.write(cfb, opts)` generates a file based on the container. `opts.type`
8595
controls the behavior:
8696

87-
- `base64`: returns a base64 string
88-
- `binary`: returns a binary string
89-
- default: returns a nodejs buffer or array of bytes
97+
| `type` | output |
98+
|------------|-----------------------------------------------------------------|
99+
| `"base64"` | string: Base64 encoding of the file |
100+
| `"binary"` | string: binary string (byte `n` is `data.charCodeAt(n)`) |
101+
| (default) | buffer if available, array of 8-bit unsigned int otherwise |
90102

91103
`CFB.writeFile(cfb, filename, opts)` creates a file with the specified name.
92104

@@ -110,19 +122,8 @@ The objects returned by `parse` and `read` have the following properties:
110122
storages (directories) in the container. The paths are properly prefixed from
111123
the root entry (so the entries are unique)
112124

113-
- `.FullPathDir` is an object whose keys are entries in `.FullPaths` and whose
114-
values are objects with metadata and content (described below)
115-
116-
- `.FileIndex` is an array of the objects from `.FullPathDir`, in the same order
117-
as `.FullPaths`.
118-
119-
- `.raw` contains the raw header and sectors
120-
121-
122-
## Entry Object Description
123-
124-
The entry objects are available from `FullPathDir` and `FileIndex` elements of
125-
the container object:
125+
- `.FileIndex` is an array, in the same order as `.FullPaths`, whose values are
126+
objects following the schema:
126127

127128
```typescript
128129
interface CFBEntry {
@@ -151,4 +152,3 @@ granted by the Apache 2.0 License are reserved by the Original Author.
151152

152153
</details>
153154

154-

bin/cfb.njs

+77-29
Original file line numberDiff line numberDiff line change
@@ -2,57 +2,67 @@
22
/* cfb.js (C) 2013-present SheetJS -- http://sheetjs.com */
33
/* eslint-env node */
44
/* vim: set ts=2 ft=javascript: */
5+
var n = "cfb";
56
var X = require('../');
67
var fs = require('fs');
78
var program = require('commander');
89
var PRINTJ = require("printj");
10+
var sprintf = PRINTJ.sprintf;
911
program
1012
.version(X.version)
11-
.usage('[options] <file>')
12-
.option('-q, --quiet', 'process but do not report')
13+
.usage('[options] <file> [subfiles...]')
1314
.option('-l, --list-files', 'list files')
14-
.option('-d, --dump', 'dump internal representation but do not extract')
1515
.option('-r, --repair', 'attempt to repair and garbage-collect archive')
16+
.option('-c, --create', 'create file')
17+
.option('-a, --append', 'add files to CFB (overwrite existing data)')
18+
.option('-d, --delete', 'delete files from CFB')
19+
.option('-z, --dump', 'dump internal representation but do not extract')
20+
.option('-q, --quiet', 'process but do not report')
1621
.option('--dev', 'development mode')
1722
.option('--read', 'read but do not print out contents');
1823

1924
program.parse(process.argv);
2025

21-
if(program.args.length === 0 || !fs.existsSync(program.args[0])) {
22-
console.error("Usage: " + process.argv[1] + " [-q] <cfb_file>");
23-
process.exit(1);
26+
var exit = process.exit;
27+
var die = function(errno/*:number*/, msg/*:string*/) { console.error(n + ": " + msg); exit(errno); };
28+
var logit = function(cmd/*:string*/, f/*:string*/) { console.error(sprintf("%-6s %s", cmd, f)); };
29+
30+
if(program.args.length === 0) die(1, "must specify a filename");
31+
32+
if(program.create) {
33+
logit("create", program.args[0]);
34+
var newcfb = X.utils.cfb_new();
35+
X.writeFile(newcfb, program.args[0]);
2436
}
2537

38+
if(!fs.existsSync(program.args[0])) die(1, "must specify a filename");
39+
2640
var opts = ({type:'file'}/*:any*/);
2741
if(program.dev) opts.WTF = true;
2842

2943
var cfb = X.read(program.args[0], opts);
30-
if(program.quiet) process.exit(0);
44+
if(program.quiet) exit(0);
3145

3246
if(program.dump) {
3347
console.log("Full Paths:");
34-
console.log(cfb.FullPaths.map(function(x) { return " " + x; }).join("\n"));
35-
console.log("Full Path Directory:");
36-
console.log(cfb.FullPathDir);
37-
process.exit(0);
38-
}
39-
if(program.repair) {
40-
X.writeFile(cfb, program.args[0]);
41-
process.exit(0);
48+
console.log(cfb.FullPaths.map(function(x/*:string*/) { return " " + x; }).join("\n"));
49+
console.log("File Index:");
50+
console.log(cfb.FileIndex);
51+
exit(0);
4252
}
53+
if(program.repair) { X.writeFile(cfb, program.args[0]); exit(0); }
4354

44-
var sprintf = PRINTJ.sprintf;
4555
function fix_string(x/*:string*/)/*:string*/ { return x.replace(/[\u0000-\u001f]/, function($$) { return sprintf("\\u%04X", $$.charCodeAt(0)); }); }
46-
if(program.listFiles) {
47-
var format_date = function(date/*:Date*/)/*:string*/ {
48-
return sprintf("%02u-%02u-%02u %02u:%02u", date.getUTCMonth()+1, date.getUTCDate(), date.getUTCFullYear()%100, date.getUTCHours(), date.getUTCMinutes());
49-
};
56+
var format_date = function(date/*:Date*/)/*:string*/ {
57+
return sprintf("%02u-%02u-%02u %02u:%02u", date.getUTCMonth()+1, date.getUTCDate(), date.getUTCFullYear()%100, date.getUTCHours(), date.getUTCMinutes());
58+
};
5059

60+
if(program.listFiles) {
5161
var basetime = new Date(1980,0,1);
5262
var cnt = 0, rootsize = 0, filesize = 0;
5363
console.log(" Length Date Time Name");
5464
console.log(" -------- ---- ---- ----");
55-
cfb.FileIndex.forEach(function(file, i/*:number*/) {
65+
cfb.FileIndex.forEach(function(file/*:CFBEntry*/, i/*:number*/) {
5666
switch(file.type) {
5767
case 5:
5868
basetime = file.ct || file.mt || basetime;
@@ -67,14 +77,52 @@ if(program.listFiles) {
6777
console.log(" -------- -------");
6878
console.log(sprintf("%9lu %lu file%s", rootsize || filesize, cnt, (cnt !== 1 ? "s" : "")));
6979

70-
process.exit(0);
80+
exit(0);
81+
}
82+
83+
function mkdirp(path/*:string*/) { path.split("/").reduce(function(acc/*:string*/, p/*:string*/) {
84+
acc += p + "/";
85+
if(!fs.existsSync(acc)) { logit("mkdir", acc); fs.mkdirSync(acc); }
86+
return acc;
87+
}, ""); }
88+
89+
function write(path/*:string*/, data/*:CFBEntry*/) {
90+
logit("write", fix_string(path));
91+
fs.writeFileSync(path, /*::new Buffer((*/data.content/*:: :any))*/);
7192
}
93+
94+
if(program.create || program.append) {
95+
program.args.slice(1).forEach(function(x/*:string*/) {
96+
logit("append", x);
97+
X.utils.cfb_add(cfb, "/" + x, fs.readFileSync(x));
98+
});
99+
X.writeFile(cfb, program.args[0]);
100+
exit(0);
101+
}
102+
103+
if(program.delete) {
104+
program.args.slice(1).forEach(function(x/*:string*/) {
105+
logit("delete", x);
106+
X.utils.cfb_del(cfb, "/" + x);
107+
});
108+
X.writeFile(cfb, program.args[0]);
109+
exit(0);
110+
}
111+
112+
if(program.args.length > 1) {
113+
program.args.slice(1).forEach(function(x/*:string*/) {
114+
var data/*:?CFBEntry*/ = X.find(cfb, x);
115+
if(!data) { console.error(x + ": file not found"); return; }
116+
if(data.type !== 2) { console.error(x + ": not a file"); return; }
117+
var idx = cfb.FileIndex.indexOf(data), path = cfb.FullPaths[idx];
118+
mkdirp(path.slice(0, path.lastIndexOf("/")));
119+
write(path, data);
120+
});
121+
exit(0);
122+
}
123+
72124
for(var i=0; i!==cfb.FullPaths.length; ++i) {
73-
if(cfb.FullPaths[i].slice(-1) === "/") {
74-
console.error("mkdir " + fix_string(cfb.FullPaths[i]));
75-
fs.mkdirSync(cfb.FullPaths[i]);
76-
} else {
77-
console.error("write " + fix_string(cfb.FullPaths[i]));
78-
fs.writeFileSync(cfb.FullPaths[i], /*::new Buffer((*/cfb.FileIndex[i].content/*:: :any))*/);
79-
}
125+
if(!cfb.FileIndex[i].name) continue;
126+
if(cfb.FullPaths[i].slice(-1) === "/") mkdirp(cfb.FullPaths[i]);
127+
else write(cfb.FullPaths[i], cfb.FileIndex[i]);
80128
}

bits/31_version.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
exports.version = '0.13.1';
1+
exports.version = '0.13.2';

bits/39_fs.js

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
var fs/*:: = require('fs'); */;
2+
function get_fs() { return fs || (fs = require('fs')); }

bits/49_readutils.js

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
1-
var fs/*:: = require('fs'); */;
21
function read_file(filename/*:string*/, options/*:CFBReadOpts*/) {
3-
if(fs == null) fs = require('fs');
2+
get_fs();
43
return parse(fs.readFileSync(filename), options);
54
}
65

bits/77_writeutils.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
function write_file(cfb/*:CFBContainer*/, filename/*:string*/, options/*:CFBWriteOpts*/)/*:void*/ {
2+
get_fs();
23
var o = _write(cfb, options);
34
/*:: if(typeof Buffer == 'undefined' || !Buffer.isBuffer(o) || !(o instanceof Buffer)) throw new Error("unreachable"); */
45
fs.writeFileSync(filename, o);
@@ -13,7 +14,7 @@ function a2s(o/*:RawBytes*/)/*:string*/ {
1314
function write(cfb/*:CFBContainer*/, options/*:CFBWriteOpts*/)/*:RawBytes|string*/ {
1415
var o = _write(cfb, options);
1516
switch(options && options.type) {
16-
case "file": fs.writeFileSync(options.filename, (o/*:any*/)); return o;
17+
case "file": get_fs(); fs.writeFileSync(options.filename, (o/*:any*/)); return o;
1718
case "binary": return a2s(o);
1819
case "base64": return Base64.encode(a2s(o));
1920
}

bits/85_api.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,13 @@ function cfb_add(cfb/*:CFBContainer*/, name/*:string*/, content/*:?RawBytes*/, o
88
init_cfb(cfb);
99
var file = CFB.find(cfb, name);
1010
if(!file) {
11-
var fpath = cfb.FullPaths[0];
11+
var fpath/*:string*/ = cfb.FullPaths[0];
1212
if(name.slice(0, fpath.length) == fpath) fpath = name;
1313
else {
1414
if(fpath.slice(-1) != "/") fpath += "/";
1515
fpath = (fpath + name).replace("//","/");
1616
}
17-
file = ({name: filename(name)}/*:any*/);
17+
file = ({name: filename(name), type: 2}/*:any*/);
1818
cfb.FileIndex.push(file);
1919
cfb.FullPaths.push(fpath);
2020
CFB.utils.cfb_gc(cfb);

0 commit comments

Comments
 (0)