@gmod/bbi 1.0.30 → 1.0.33

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/blockView.js CHANGED
@@ -1,82 +1,148 @@
1
1
  "use strict";
2
-
3
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
-
5
- Object.defineProperty(exports, "__esModule", {
6
- value: true
7
- });
2
+ var __assign = (this && this.__assign) || function () {
3
+ __assign = Object.assign || function(t) {
4
+ for (var s, i = 1, n = arguments.length; i < n; i++) {
5
+ s = arguments[i];
6
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
7
+ t[p] = s[p];
8
+ }
9
+ return t;
10
+ };
11
+ return __assign.apply(this, arguments);
12
+ };
13
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
14
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
15
+ return new (P || (P = Promise))(function (resolve, reject) {
16
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
17
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
18
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
19
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
20
+ });
21
+ };
22
+ var __generator = (this && this.__generator) || function (thisArg, body) {
23
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
24
+ return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
25
+ function verb(n) { return function (v) { return step([n, v]); }; }
26
+ function step(op) {
27
+ if (f) throw new TypeError("Generator is already executing.");
28
+ while (_) try {
29
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
30
+ if (y = 0, t) op = [op[0] & 2, t.value];
31
+ switch (op[0]) {
32
+ case 0: case 1: t = op; break;
33
+ case 4: _.label++; return { value: op[1], done: false };
34
+ case 5: _.label++; y = op[1]; op = [0]; continue;
35
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
36
+ default:
37
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
38
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
39
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
40
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
41
+ if (t[2]) _.ops.pop();
42
+ _.trys.pop(); continue;
43
+ }
44
+ op = body.call(thisArg, _);
45
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
46
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
47
+ }
48
+ };
49
+ var __importDefault = (this && this.__importDefault) || function (mod) {
50
+ return (mod && mod.__esModule) ? mod : { "default": mod };
51
+ };
52
+ Object.defineProperty(exports, "__esModule", { value: true });
8
53
  exports.BlockView = void 0;
9
-
10
- var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator"));
11
-
12
- var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator"));
13
-
14
- var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
15
-
16
- var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
17
-
18
- var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
19
-
20
- var _binaryParser = require("@gmod/binary-parser");
21
-
22
- var _abortablePromiseCache = _interopRequireDefault(require("abortable-promise-cache"));
23
-
24
- var _zlib = _interopRequireDefault(require("zlib"));
25
-
26
- var _quickLru = _interopRequireDefault(require("quick-lru"));
27
-
28
- var _range = _interopRequireDefault(require("./range"));
29
-
30
- var _util = require("./util");
31
-
32
- function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
33
-
34
- function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2.default)(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
35
-
54
+ var binary_parser_1 = require("@gmod/binary-parser");
55
+ var abortable_promise_cache_1 = __importDefault(require("abortable-promise-cache"));
56
+ var unzip_1 = require("./unzip");
57
+ var quick_lru_1 = __importDefault(require("quick-lru"));
58
+ var range_1 = __importDefault(require("./range"));
59
+ var util_1 = require("./util");
36
60
  var BIG_WIG_TYPE_GRAPH = 1;
37
61
  var BIG_WIG_TYPE_VSTEP = 2;
38
62
  var BIG_WIG_TYPE_FSTEP = 3;
39
-
40
63
  function getParsers(isBigEndian) {
41
- var _choices;
42
-
43
- var le = isBigEndian ? 'big' : 'little';
44
- var summaryParser = new _binaryParser.Parser().endianess(le).uint32('chromId').uint32('start').uint32('end').uint32('validCnt').float('minScore').float('maxScore').float('sumData').float('sumSqData');
45
- var leafParser = new _binaryParser.Parser().endianess(le).uint8('isLeaf').skip(1).uint16('cnt').choice({
46
- tag: 'isLeaf',
47
- choices: {
48
- 1: new _binaryParser.Parser().array('blocksToFetch', {
49
- length: 'cnt',
50
- type: new _binaryParser.Parser().uint32('startChrom').uint32('startBase').uint32('endChrom').uint32('endBase').uint64('blockOffset').uint64('blockSize')
51
- }),
52
- 0: new _binaryParser.Parser().array('recurOffsets', {
53
- length: 'cnt',
54
- type: new _binaryParser.Parser().uint32('startChrom').uint32('startBase').uint32('endChrom').uint32('endBase').uint64('blockOffset')
55
- })
56
- }
57
- });
58
- var bigBedParser = new _binaryParser.Parser().endianess(le).uint32('chromId').int32('start').int32('end').string('rest', {
59
- zeroTerminated: true
60
- });
61
- var bigWigParser = new _binaryParser.Parser().endianess(le).skip(4).int32('blockStart').skip(4).uint32('itemStep').uint32('itemSpan').uint8('blockType').skip(1).uint16('itemCount').choice({
62
- tag: 'blockType',
63
- choices: (_choices = {}, (0, _defineProperty2.default)(_choices, BIG_WIG_TYPE_FSTEP, new _binaryParser.Parser().array('items', {
64
- length: 'itemCount',
65
- type: new _binaryParser.Parser().float('score')
66
- })), (0, _defineProperty2.default)(_choices, BIG_WIG_TYPE_VSTEP, new _binaryParser.Parser().array('items', {
67
- length: 'itemCount',
68
- type: new _binaryParser.Parser().int32('start').float('score')
69
- })), (0, _defineProperty2.default)(_choices, BIG_WIG_TYPE_GRAPH, new _binaryParser.Parser().array('items', {
70
- length: 'itemCount',
71
- type: new _binaryParser.Parser().int32('start').int32('end').float('score')
72
- })), _choices)
73
- });
74
- return {
75
- bigWigParser: bigWigParser,
76
- bigBedParser: bigBedParser,
77
- summaryParser: summaryParser,
78
- leafParser: leafParser
79
- };
64
+ var _a;
65
+ var le = isBigEndian ? 'big' : 'little';
66
+ var summaryParser = new binary_parser_1.Parser()
67
+ .endianess(le)
68
+ .uint32('chromId')
69
+ .uint32('start')
70
+ .uint32('end')
71
+ .uint32('validCnt')
72
+ .float('minScore')
73
+ .float('maxScore')
74
+ .float('sumData')
75
+ .float('sumSqData');
76
+ var leafParser = new binary_parser_1.Parser()
77
+ .endianess(le)
78
+ .uint8('isLeaf')
79
+ .skip(1)
80
+ .uint16('cnt')
81
+ .choice({
82
+ tag: 'isLeaf',
83
+ choices: {
84
+ 1: new binary_parser_1.Parser().array('blocksToFetch', {
85
+ length: 'cnt',
86
+ type: new binary_parser_1.Parser()
87
+ .uint32('startChrom')
88
+ .uint32('startBase')
89
+ .uint32('endChrom')
90
+ .uint32('endBase')
91
+ .uint64('blockOffset')
92
+ .uint64('blockSize'),
93
+ }),
94
+ 0: new binary_parser_1.Parser().array('recurOffsets', {
95
+ length: 'cnt',
96
+ type: new binary_parser_1.Parser()
97
+ .uint32('startChrom')
98
+ .uint32('startBase')
99
+ .uint32('endChrom')
100
+ .uint32('endBase')
101
+ .uint64('blockOffset'),
102
+ }),
103
+ },
104
+ });
105
+ var bigBedParser = new binary_parser_1.Parser()
106
+ .endianess(le)
107
+ .uint32('chromId')
108
+ .int32('start')
109
+ .int32('end')
110
+ .string('rest', {
111
+ zeroTerminated: true,
112
+ });
113
+ var bigWigParser = new binary_parser_1.Parser()
114
+ .endianess(le)
115
+ .skip(4)
116
+ .int32('blockStart')
117
+ .skip(4)
118
+ .uint32('itemStep')
119
+ .uint32('itemSpan')
120
+ .uint8('blockType')
121
+ .skip(1)
122
+ .uint16('itemCount')
123
+ .choice({
124
+ tag: 'blockType',
125
+ choices: (_a = {},
126
+ _a[BIG_WIG_TYPE_FSTEP] = new binary_parser_1.Parser().array('items', {
127
+ length: 'itemCount',
128
+ type: new binary_parser_1.Parser().float('score'),
129
+ }),
130
+ _a[BIG_WIG_TYPE_VSTEP] = new binary_parser_1.Parser().array('items', {
131
+ length: 'itemCount',
132
+ type: new binary_parser_1.Parser().int32('start').float('score'),
133
+ }),
134
+ _a[BIG_WIG_TYPE_GRAPH] = new binary_parser_1.Parser().array('items', {
135
+ length: 'itemCount',
136
+ type: new binary_parser_1.Parser().int32('start').int32('end').float('score'),
137
+ }),
138
+ _a),
139
+ });
140
+ return {
141
+ bigWigParser: bigWigParser,
142
+ bigBedParser: bigBedParser,
143
+ summaryParser: summaryParser,
144
+ leafParser: leafParser,
145
+ };
80
146
  }
81
147
  /**
82
148
  * View into a subset of the data in a BigWig file.
@@ -85,432 +151,275 @@ function getParsers(isBigEndian) {
85
151
  * Explorer by Thomas Down.
86
152
  * @constructs
87
153
  */
88
-
89
-
90
- var BlockView = /*#__PURE__*/function () {
91
- function BlockView(bbi, refsByName, cirTreeOffset, cirTreeLength, isBigEndian, isCompressed, blockType) {
92
- var _this = this;
93
-
94
- (0, _classCallCheck2.default)(this, BlockView);
95
- (0, _defineProperty2.default)(this, "cirTreeOffset", void 0);
96
- (0, _defineProperty2.default)(this, "cirTreeLength", void 0);
97
- (0, _defineProperty2.default)(this, "bbi", void 0);
98
- (0, _defineProperty2.default)(this, "isCompressed", void 0);
99
- (0, _defineProperty2.default)(this, "isBigEndian", void 0);
100
- (0, _defineProperty2.default)(this, "refsByName", void 0);
101
- (0, _defineProperty2.default)(this, "blockType", void 0);
102
- (0, _defineProperty2.default)(this, "cirTreePromise", void 0);
103
- (0, _defineProperty2.default)(this, "featureCache", new _abortablePromiseCache.default({
104
- cache: new _quickLru.default({
105
- maxSize: 1000
106
- }),
107
- fill: function () {
108
- var _fill = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee(requestData, signal) {
109
- var length, offset, _yield$_this$bbi$read, buffer;
110
-
111
- return _regenerator.default.wrap(function _callee$(_context) {
112
- while (1) {
113
- switch (_context.prev = _context.next) {
114
- case 0:
115
- length = requestData.length, offset = requestData.offset;
116
- _context.next = 3;
117
- return _this.bbi.read(Buffer.alloc(length), 0, length, offset, {
118
- signal: signal
119
- });
120
-
121
- case 3:
122
- _yield$_this$bbi$read = _context.sent;
123
- buffer = _yield$_this$bbi$read.buffer;
124
- return _context.abrupt("return", buffer);
125
-
126
- case 6:
127
- case "end":
128
- return _context.stop();
129
- }
130
- }
131
- }, _callee);
132
- }));
133
-
134
- function fill(_x, _x2) {
135
- return _fill.apply(this, arguments);
136
- }
137
-
138
- return fill;
139
- }()
140
- }));
141
- (0, _defineProperty2.default)(this, "leafParser", void 0);
142
- (0, _defineProperty2.default)(this, "bigWigParser", void 0);
143
- (0, _defineProperty2.default)(this, "bigBedParser", void 0);
144
- (0, _defineProperty2.default)(this, "summaryParser", void 0);
145
- if (!(cirTreeOffset >= 0)) throw new Error('invalid cirTreeOffset!');
146
- if (!(cirTreeLength > 0)) throw new Error('invalid cirTreeLength!');
147
- this.cirTreeOffset = cirTreeOffset;
148
- this.cirTreeLength = cirTreeLength;
149
- this.isCompressed = isCompressed;
150
- this.refsByName = refsByName;
151
- this.isBigEndian = isBigEndian;
152
- this.bbi = bbi;
153
- this.blockType = blockType;
154
- Object.assign(this, getParsers(isBigEndian));
155
- }
156
-
157
- (0, _createClass2.default)(BlockView, [{
158
- key: "readWigData",
159
- value: function () {
160
- var _readWigData = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee3(chrName, start, end, observer, opts) {
161
- var _this2 = this;
162
-
163
- var refsByName, bbi, cirTreeOffset, isBigEndian, signal, chrId, request, _yield$this$cirTreePr, buffer, cirBlockSize, blocksToFetch, outstanding, cirFobRecur2, filterFeats, cirFobStartFetch, cirFobRecur;
164
-
165
- return _regenerator.default.wrap(function _callee3$(_context3) {
166
- while (1) {
167
- switch (_context3.prev = _context3.next) {
168
- case 0:
169
- _context3.prev = 0;
170
- refsByName = this.refsByName, bbi = this.bbi, cirTreeOffset = this.cirTreeOffset, isBigEndian = this.isBigEndian;
171
- signal = opts.signal;
172
- chrId = refsByName[chrName];
173
-
174
- if (chrId === undefined) {
175
- observer.complete();
176
- }
177
-
178
- request = {
179
- chrId: chrId,
180
- start: start,
181
- end: end
182
- };
183
-
184
- if (!this.cirTreePromise) {
185
- this.cirTreePromise = bbi.read(Buffer.alloc(48), 0, 48, cirTreeOffset, {
186
- signal: signal
187
- });
188
- }
189
-
190
- _context3.next = 9;
191
- return this.cirTreePromise;
192
-
193
- case 9:
194
- _yield$this$cirTreePr = _context3.sent;
195
- buffer = _yield$this$cirTreePr.buffer;
196
- cirBlockSize = isBigEndian ? buffer.readUInt32BE(4) : buffer.readUInt32LE(4);
197
- blocksToFetch = [];
198
- outstanding = 0; //eslint-disable-next-line prefer-const
199
-
200
- filterFeats = function filterFeats(b) {
201
- return (b.startChrom < chrId || b.startChrom === chrId && b.startBase <= end) && (b.endChrom > chrId || b.endChrom === chrId && b.endBase >= start);
202
- };
203
-
204
- cirFobStartFetch = /*#__PURE__*/function () {
205
- var _ref = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee2(off, fr, level) {
206
- var length, offset, resultBuffer, i;
207
- return _regenerator.default.wrap(function _callee2$(_context2) {
208
- while (1) {
209
- switch (_context2.prev = _context2.next) {
210
- case 0:
211
- _context2.prev = 0;
212
- length = fr.max() - fr.min();
213
- offset = fr.min();
214
- _context2.next = 5;
215
- return _this2.featureCache.get("".concat(length, "_").concat(offset), {
216
- length: length,
217
- offset: offset
218
- }, signal);
219
-
220
- case 5:
221
- resultBuffer = _context2.sent;
222
-
223
- for (i = 0; i < off.length; i += 1) {
224
- if (fr.contains(off[i])) {
225
- cirFobRecur2(resultBuffer, off[i] - offset, level, observer, opts);
226
- outstanding -= 1;
227
-
228
- if (outstanding === 0) {
229
- _this2.readFeatures(observer, blocksToFetch, _objectSpread(_objectSpread({}, opts), {}, {
230
- request: request
231
- }));
232
- }
233
- }
234
- }
235
-
236
- _context2.next = 12;
237
- break;
238
-
239
- case 9:
240
- _context2.prev = 9;
241
- _context2.t0 = _context2["catch"](0);
242
- observer.error(_context2.t0);
243
-
244
- case 12:
245
- case "end":
246
- return _context2.stop();
247
- }
248
- }
249
- }, _callee2, null, [[0, 9]]);
250
- }));
251
-
252
- return function cirFobStartFetch(_x8, _x9, _x10) {
253
- return _ref.apply(this, arguments);
254
- };
255
- }();
256
-
257
- cirFobRecur = function cirFobRecur(offset, level) {
258
- try {
259
- outstanding += offset.length;
260
- var maxCirBlockSpan = 4 + cirBlockSize * 32; // Upper bound on size, based on a completely full leaf node.
261
-
262
- var spans = new _range.default(offset[0], offset[0] + maxCirBlockSpan);
263
-
264
- for (var i = 1; i < offset.length; i += 1) {
265
- var blockSpan = new _range.default(offset[i], offset[i] + maxCirBlockSpan);
266
- spans = spans.union(blockSpan);
267
- }
268
-
269
- spans.getRanges().map(function (fr) {
270
- return cirFobStartFetch(offset, fr, level);
271
- });
272
- } catch (e) {
273
- observer.error(e);
274
- }
275
- };
276
-
277
- cirFobRecur2 = function cirFobRecur2(cirBlockData, offset, level) {
278
- try {
279
- var data = cirBlockData.slice(offset);
280
-
281
- var p = _this2.leafParser.parse(data).result;
282
-
283
- if (p.blocksToFetch) {
284
- blocksToFetch = blocksToFetch.concat(p.blocksToFetch.filter(filterFeats).map(function (l) {
285
- return {
286
- offset: l.blockOffset,
287
- length: l.blockSize
288
- };
289
- }));
154
+ var BlockView = /** @class */ (function () {
155
+ function BlockView(bbi, refsByName, cirTreeOffset, cirTreeLength, isBigEndian, isCompressed, blockType) {
156
+ var _this = this;
157
+ this.featureCache = new abortable_promise_cache_1.default({
158
+ cache: new quick_lru_1.default({ maxSize: 1000 }),
159
+ fill: function (requestData, signal) { return __awaiter(_this, void 0, void 0, function () {
160
+ var length, offset, buffer;
161
+ return __generator(this, function (_a) {
162
+ switch (_a.label) {
163
+ case 0:
164
+ length = requestData.length, offset = requestData.offset;
165
+ return [4 /*yield*/, this.bbi.read(Buffer.alloc(length), 0, length, offset, { signal: signal })];
166
+ case 1:
167
+ buffer = (_a.sent()).buffer;
168
+ return [2 /*return*/, buffer];
290
169
  }
291
-
292
- if (p.recurOffsets) {
293
- var recurOffsets = p.recurOffsets.filter(filterFeats).map(function (l) {
294
- return l.blockOffset;
295
- });
296
-
297
- if (recurOffsets.length > 0) {
298
- cirFobRecur(recurOffsets, level + 1);
299
- }
300
- }
301
- } catch (e) {
302
- observer.error(e);
303
- }
304
- };
305
-
306
- return _context3.abrupt("return", cirFobRecur([cirTreeOffset + 48], 1));
307
-
308
- case 21:
309
- _context3.prev = 21;
310
- _context3.t0 = _context3["catch"](0);
311
- observer.error(_context3.t0);
312
-
313
- case 24:
314
- case "end":
315
- return _context3.stop();
316
- }
317
- }
318
- }, _callee3, this, [[0, 21]]);
319
- }));
320
-
321
- function readWigData(_x3, _x4, _x5, _x6, _x7) {
322
- return _readWigData.apply(this, arguments);
323
- }
324
-
325
- return readWigData;
326
- }()
327
- }, {
328
- key: "parseSummaryBlock",
329
- value: function parseSummaryBlock(data, startOffset, request) {
330
- var features = [];
331
- var currOffset = startOffset;
332
-
333
- while (currOffset < data.byteLength) {
334
- var res = this.summaryParser.parse(data.slice(currOffset));
335
- features.push(res.result);
336
- currOffset += res.offset;
337
- }
338
-
339
- var items = features;
340
- if (request) items = items.filter(function (elt) {
341
- return elt.chromId === request.chrId;
342
- });
343
- items = items.map(function (elt) {
344
- return {
345
- start: elt.start,
346
- end: elt.end,
347
- maxScore: elt.maxScore,
348
- minScore: elt.minScore,
349
- score: elt.sumData / (elt.validCnt || 1),
350
- summary: true
351
- };
352
- });
353
- return request ? items.filter(function (f) {
354
- return BlockView.coordFilter(f, request);
355
- }) : items;
356
- }
357
- }, {
358
- key: "parseBigBedBlock",
359
- value: function parseBigBedBlock(data, startOffset, offset, request) {
360
- var items = [];
361
- var currOffset = startOffset;
362
-
363
- while (currOffset < data.byteLength) {
364
- var res = this.bigBedParser.parse(data.slice(currOffset));
365
- res.result.uniqueId = "bb-".concat(offset + currOffset);
366
- items.push(res.result);
367
- currOffset += res.offset;
368
- }
369
-
370
- return request ? items.filter(function (f) {
371
- return BlockView.coordFilter(f, request);
372
- }) : items;
373
- }
374
- }, {
375
- key: "parseBigWigBlock",
376
- value: function parseBigWigBlock(bytes, startOffset, request) {
377
- var data = bytes.slice(startOffset);
378
- var results = this.bigWigParser.parse(data).result;
379
- var items = results.items,
380
- itemSpan = results.itemSpan,
381
- itemStep = results.itemStep,
382
- blockStart = results.blockStart,
383
- blockType = results.blockType;
384
-
385
- if (blockType === BIG_WIG_TYPE_FSTEP) {
386
- for (var i = 0; i < items.length; i++) {
387
- items[i].start = blockStart + i * itemStep;
388
- items[i].end = blockStart + i * itemStep + itemSpan;
170
+ });
171
+ }); },
172
+ });
173
+ if (!(cirTreeOffset >= 0)) {
174
+ throw new Error('invalid cirTreeOffset!');
389
175
  }
390
- } else if (blockType === BIG_WIG_TYPE_VSTEP) {
391
- for (var _i = 0; _i < items.length; _i++) {
392
- items[_i].end = items[_i].start + itemSpan;
176
+ if (!(cirTreeLength > 0)) {
177
+ throw new Error('invalid cirTreeLength!');
393
178
  }
394
- }
395
-
396
- return request ? items.filter(function (f) {
397
- return BlockView.coordFilter(f, request);
398
- }) : items;
179
+ this.cirTreeOffset = cirTreeOffset;
180
+ this.cirTreeLength = cirTreeLength;
181
+ this.isCompressed = isCompressed;
182
+ this.refsByName = refsByName;
183
+ this.isBigEndian = isBigEndian;
184
+ this.bbi = bbi;
185
+ this.blockType = blockType;
186
+ Object.assign(this, getParsers(isBigEndian));
399
187
  }
400
- }, {
401
- key: "readFeatures",
402
- value: function () {
403
- var _readFeatures = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee5(observer, blocks) {
404
- var _this3 = this;
405
-
406
- var opts,
407
- blockType,
408
- isCompressed,
409
- signal,
410
- request,
411
- blockGroupsToFetch,
412
- _args5 = arguments;
413
- return _regenerator.default.wrap(function _callee5$(_context5) {
414
- while (1) {
415
- switch (_context5.prev = _context5.next) {
416
- case 0:
417
- opts = _args5.length > 2 && _args5[2] !== undefined ? _args5[2] : {};
418
- _context5.prev = 1;
419
- blockType = this.blockType, isCompressed = this.isCompressed;
420
- signal = opts.signal, request = opts.request;
421
- blockGroupsToFetch = (0, _util.groupBlocks)(blocks);
422
- (0, _util.checkAbortSignal)(signal);
423
- _context5.next = 8;
424
- return Promise.all(blockGroupsToFetch.map( /*#__PURE__*/function () {
425
- var _ref2 = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee4(blockGroup) {
426
- var length, offset, data;
427
- return _regenerator.default.wrap(function _callee4$(_context4) {
428
- while (1) {
429
- switch (_context4.prev = _context4.next) {
430
- case 0:
431
- (0, _util.checkAbortSignal)(signal);
432
- length = blockGroup.length, offset = blockGroup.offset;
433
- _context4.next = 4;
434
- return _this3.featureCache.get("".concat(length, "_").concat(offset), blockGroup, signal);
435
-
436
- case 4:
437
- data = _context4.sent;
438
- blockGroup.blocks.forEach(function (block) {
439
- (0, _util.checkAbortSignal)(signal);
440
- var blockOffset = block.offset - blockGroup.offset;
441
- var resultData = data;
442
-
443
- if (isCompressed) {
444
- resultData = _zlib.default.inflateSync(data.slice(blockOffset));
445
- blockOffset = 0;
446
- }
447
-
448
- (0, _util.checkAbortSignal)(signal);
449
-
450
- switch (blockType) {
451
- case 'summary':
452
- observer.next(_this3.parseSummaryBlock(resultData, blockOffset, request));
453
- break;
454
-
455
- case 'bigwig':
456
- observer.next(_this3.parseBigWigBlock(resultData, blockOffset, request));
457
- break;
458
-
459
- case 'bigbed':
460
- observer.next( // eslint-disable-next-line no-bitwise
461
- _this3.parseBigBedBlock(resultData, blockOffset, block.offset * (1 << 8), request));
462
- break;
463
-
464
- default:
465
- console.warn("Don't know what to do with ".concat(blockType));
466
- }
188
+ BlockView.prototype.readWigData = function (chrName, start, end, observer, opts) {
189
+ return __awaiter(this, void 0, void 0, function () {
190
+ var _a, refsByName, bbi, cirTreeOffset, isBigEndian, signal_1, chrId_1, request_1, buffer, cirBlockSize_1, blocksToFetch_1, outstanding_1, cirFobRecur2_1, filterFeats_1, cirFobStartFetch_1, cirFobRecur_1, e_1;
191
+ var _this = this;
192
+ return __generator(this, function (_b) {
193
+ switch (_b.label) {
194
+ case 0:
195
+ _b.trys.push([0, 2, , 3]);
196
+ _a = this, refsByName = _a.refsByName, bbi = _a.bbi, cirTreeOffset = _a.cirTreeOffset, isBigEndian = _a.isBigEndian;
197
+ signal_1 = opts.signal;
198
+ chrId_1 = refsByName[chrName];
199
+ if (chrId_1 === undefined) {
200
+ observer.complete();
201
+ }
202
+ request_1 = { chrId: chrId_1, start: start, end: end };
203
+ if (!this.cirTreePromise) {
204
+ this.cirTreePromise = bbi.read(Buffer.alloc(48), 0, 48, cirTreeOffset, {
205
+ signal: signal_1,
467
206
  });
468
-
469
- case 6:
470
- case "end":
471
- return _context4.stop();
472
207
  }
473
- }
474
- }, _callee4);
475
- }));
476
-
477
- return function (_x13) {
478
- return _ref2.apply(this, arguments);
479
- };
480
- }()));
481
-
482
- case 8:
483
- observer.complete();
484
- _context5.next = 14;
485
- break;
486
-
487
- case 11:
488
- _context5.prev = 11;
489
- _context5.t0 = _context5["catch"](1);
490
- observer.error(_context5.t0);
491
-
492
- case 14:
493
- case "end":
494
- return _context5.stop();
208
+ return [4 /*yield*/, this.cirTreePromise];
209
+ case 1:
210
+ buffer = (_b.sent()).buffer;
211
+ cirBlockSize_1 = isBigEndian
212
+ ? buffer.readUInt32BE(4)
213
+ : buffer.readUInt32LE(4);
214
+ blocksToFetch_1 = [];
215
+ outstanding_1 = 0;
216
+ cirFobRecur2_1 = function (cirBlockData, offset, level) {
217
+ try {
218
+ var data = cirBlockData.subarray(offset);
219
+ var p = _this.leafParser.parse(data).result;
220
+ if (p.blocksToFetch) {
221
+ blocksToFetch_1 = blocksToFetch_1.concat(p.blocksToFetch.filter(filterFeats_1).map(function (l) { return ({
222
+ offset: l.blockOffset,
223
+ length: l.blockSize,
224
+ }); }));
225
+ }
226
+ if (p.recurOffsets) {
227
+ var recurOffsets = p.recurOffsets
228
+ .filter(filterFeats_1)
229
+ .map(function (l) { return l.blockOffset; });
230
+ if (recurOffsets.length > 0) {
231
+ cirFobRecur_1(recurOffsets, level + 1);
232
+ }
233
+ }
234
+ }
235
+ catch (e) {
236
+ observer.error(e);
237
+ }
238
+ };
239
+ filterFeats_1 = function (b) {
240
+ var startChrom = b.startChrom, startBase = b.startBase, endChrom = b.endChrom, endBase = b.endBase;
241
+ return ((startChrom < chrId_1 || (startChrom === chrId_1 && startBase <= end)) &&
242
+ (endChrom > chrId_1 || (endChrom === chrId_1 && endBase >= start)));
243
+ };
244
+ cirFobStartFetch_1 = function (off, fr, level) { return __awaiter(_this, void 0, void 0, function () {
245
+ var length_1, offset, resultBuffer, i, e_2;
246
+ return __generator(this, function (_a) {
247
+ switch (_a.label) {
248
+ case 0:
249
+ _a.trys.push([0, 2, , 3]);
250
+ length_1 = fr.max() - fr.min();
251
+ offset = fr.min();
252
+ return [4 /*yield*/, this.featureCache.get("".concat(length_1, "_").concat(offset), { length: length_1, offset: offset }, signal_1)];
253
+ case 1:
254
+ resultBuffer = _a.sent();
255
+ for (i = 0; i < off.length; i += 1) {
256
+ if (fr.contains(off[i])) {
257
+ cirFobRecur2_1(resultBuffer, off[i] - offset, level);
258
+ outstanding_1 -= 1;
259
+ if (outstanding_1 === 0) {
260
+ this.readFeatures(observer, blocksToFetch_1, __assign(__assign({}, opts), { request: request_1 }));
261
+ }
262
+ }
263
+ }
264
+ return [3 /*break*/, 3];
265
+ case 2:
266
+ e_2 = _a.sent();
267
+ observer.error(e_2);
268
+ return [3 /*break*/, 3];
269
+ case 3: return [2 /*return*/];
270
+ }
271
+ });
272
+ }); };
273
+ cirFobRecur_1 = function (offset, level) {
274
+ try {
275
+ outstanding_1 += offset.length;
276
+ var maxCirBlockSpan = 4 + cirBlockSize_1 * 32; // Upper bound on size, based on a completely full leaf node.
277
+ var spans = new range_1.default(offset[0], offset[0] + maxCirBlockSpan);
278
+ for (var i = 1; i < offset.length; i += 1) {
279
+ var blockSpan = new range_1.default(offset[i], offset[i] + maxCirBlockSpan);
280
+ spans = spans.union(blockSpan);
281
+ }
282
+ spans.getRanges().map(function (fr) { return cirFobStartFetch_1(offset, fr, level); });
283
+ }
284
+ catch (e) {
285
+ observer.error(e);
286
+ }
287
+ };
288
+ return [2 /*return*/, cirFobRecur_1([cirTreeOffset + 48], 1)];
289
+ case 2:
290
+ e_1 = _b.sent();
291
+ observer.error(e_1);
292
+ return [3 /*break*/, 3];
293
+ case 3: return [2 /*return*/];
294
+ }
295
+ });
296
+ });
297
+ };
298
+ BlockView.prototype.parseSummaryBlock = function (data, startOffset, request) {
299
+ var features = [];
300
+ var currOffset = startOffset;
301
+ while (currOffset < data.byteLength) {
302
+ var res = this.summaryParser.parse(data.subarray(currOffset));
303
+ features.push(res.result);
304
+ currOffset += res.offset;
305
+ }
306
+ var items = features;
307
+ if (request) {
308
+ items = items.filter(function (elt) { return elt.chromId === request.chrId; });
309
+ }
310
+ var feats = items.map(function (elt) { return ({
311
+ start: elt.start,
312
+ end: elt.end,
313
+ maxScore: elt.maxScore,
314
+ minScore: elt.minScore,
315
+ score: elt.sumData / (elt.validCnt || 1),
316
+ summary: true,
317
+ }); });
318
+ return request
319
+ ? feats.filter(function (f) { return BlockView.coordFilter(f, request); })
320
+ : feats;
321
+ };
322
+ BlockView.prototype.parseBigBedBlock = function (data, startOffset, offset, request) {
323
+ var items = [];
324
+ var currOffset = startOffset;
325
+ while (currOffset < data.byteLength) {
326
+ var res = this.bigBedParser.parse(data.subarray(currOffset));
327
+ res.result.uniqueId = "bb-".concat(offset + currOffset);
328
+ items.push(res.result);
329
+ currOffset += res.offset;
330
+ }
331
+ return request
332
+ ? items.filter(function (f) { return BlockView.coordFilter(f, request); })
333
+ : items;
334
+ };
335
+ BlockView.prototype.parseBigWigBlock = function (bytes, startOffset, request) {
336
+ var data = bytes.subarray(startOffset);
337
+ var results = this.bigWigParser.parse(data).result;
338
+ var items = results.items, itemSpan = results.itemSpan, itemStep = results.itemStep, blockStart = results.blockStart, blockType = results.blockType;
339
+ if (blockType === BIG_WIG_TYPE_FSTEP) {
340
+ for (var i = 0; i < items.length; i++) {
341
+ items[i].start = blockStart + i * itemStep;
342
+ items[i].end = blockStart + i * itemStep + itemSpan;
495
343
  }
496
- }
497
- }, _callee5, this, [[1, 11]]);
498
- }));
499
-
500
- function readFeatures(_x11, _x12) {
501
- return _readFeatures.apply(this, arguments);
502
- }
503
-
504
- return readFeatures;
505
- }()
506
- }], [{
507
- key: "coordFilter",
508
- value: function coordFilter(f, range) {
509
- return f.start < range.end && f.end >= range.start;
510
- }
511
- }]);
512
- return BlockView;
513
- }();
514
-
344
+ }
345
+ else if (blockType === BIG_WIG_TYPE_VSTEP) {
346
+ for (var i = 0; i < items.length; i++) {
347
+ items[i].end = items[i].start + itemSpan;
348
+ }
349
+ }
350
+ return request
351
+ ? items.filter(function (f) { return BlockView.coordFilter(f, request); })
352
+ : items;
353
+ };
354
+ BlockView.coordFilter = function (f, range) {
355
+ return f.start < range.end && f.end >= range.start;
356
+ };
357
+ BlockView.prototype.readFeatures = function (observer, blocks, opts) {
358
+ if (opts === void 0) { opts = {}; }
359
+ return __awaiter(this, void 0, void 0, function () {
360
+ var _a, blockType_1, isCompressed_1, signal_2, request_2, blockGroupsToFetch, e_3;
361
+ var _this = this;
362
+ return __generator(this, function (_b) {
363
+ switch (_b.label) {
364
+ case 0:
365
+ _b.trys.push([0, 2, , 3]);
366
+ _a = this, blockType_1 = _a.blockType, isCompressed_1 = _a.isCompressed;
367
+ signal_2 = opts.signal, request_2 = opts.request;
368
+ blockGroupsToFetch = (0, util_1.groupBlocks)(blocks);
369
+ (0, util_1.checkAbortSignal)(signal_2);
370
+ return [4 /*yield*/, Promise.all(blockGroupsToFetch.map(function (blockGroup) { return __awaiter(_this, void 0, void 0, function () {
371
+ var length, offset, data;
372
+ var _this = this;
373
+ return __generator(this, function (_a) {
374
+ switch (_a.label) {
375
+ case 0:
376
+ (0, util_1.checkAbortSignal)(signal_2);
377
+ length = blockGroup.length, offset = blockGroup.offset;
378
+ return [4 /*yield*/, this.featureCache.get("".concat(length, "_").concat(offset), blockGroup, signal_2)];
379
+ case 1:
380
+ data = _a.sent();
381
+ blockGroup.blocks.forEach(function (block) {
382
+ (0, util_1.checkAbortSignal)(signal_2);
383
+ var blockOffset = block.offset - blockGroup.offset;
384
+ var resultData = data;
385
+ if (isCompressed_1) {
386
+ resultData = (0, unzip_1.unzip)(data.subarray(blockOffset));
387
+ blockOffset = 0;
388
+ }
389
+ (0, util_1.checkAbortSignal)(signal_2);
390
+ switch (blockType_1) {
391
+ case 'summary':
392
+ observer.next(_this.parseSummaryBlock(resultData, blockOffset, request_2));
393
+ break;
394
+ case 'bigwig':
395
+ observer.next(_this.parseBigWigBlock(resultData, blockOffset, request_2));
396
+ break;
397
+ case 'bigbed':
398
+ observer.next(_this.parseBigBedBlock(resultData, blockOffset,
399
+ // eslint-disable-next-line no-bitwise
400
+ block.offset * (1 << 8), request_2));
401
+ break;
402
+ default:
403
+ console.warn("Don't know what to do with ".concat(blockType_1));
404
+ }
405
+ });
406
+ return [2 /*return*/];
407
+ }
408
+ });
409
+ }); }))];
410
+ case 1:
411
+ _b.sent();
412
+ observer.complete();
413
+ return [3 /*break*/, 3];
414
+ case 2:
415
+ e_3 = _b.sent();
416
+ observer.error(e_3);
417
+ return [3 /*break*/, 3];
418
+ case 3: return [2 /*return*/];
419
+ }
420
+ });
421
+ });
422
+ };
423
+ return BlockView;
424
+ }());
515
425
  exports.BlockView = BlockView;
516
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/blockView.ts"],"names":["BIG_WIG_TYPE_GRAPH","BIG_WIG_TYPE_VSTEP","BIG_WIG_TYPE_FSTEP","getParsers","isBigEndian","le","summaryParser","Parser","endianess","uint32","float","leafParser","uint8","skip","uint16","choice","tag","choices","array","length","type","uint64","bigBedParser","int32","string","zeroTerminated","bigWigParser","BlockView","bbi","refsByName","cirTreeOffset","cirTreeLength","isCompressed","blockType","AbortablePromiseCache","cache","QuickLRU","maxSize","fill","requestData","signal","offset","read","Buffer","alloc","buffer","Error","Object","assign","chrName","start","end","observer","opts","chrId","undefined","complete","request","cirTreePromise","cirBlockSize","readUInt32BE","readUInt32LE","blocksToFetch","outstanding","filterFeats","b","startChrom","startBase","endChrom","endBase","cirFobStartFetch","off","fr","level","max","min","featureCache","get","resultBuffer","i","contains","cirFobRecur2","readFeatures","error","cirFobRecur","maxCirBlockSpan","spans","Range","blockSpan","union","getRanges","map","e","cirBlockData","data","slice","p","parse","result","concat","filter","l","blockOffset","blockSize","recurOffsets","startOffset","features","currOffset","byteLength","res","push","items","elt","chromId","maxScore","minScore","score","sumData","validCnt","summary","f","coordFilter","uniqueId","bytes","results","itemSpan","itemStep","blockStart","blocks","blockGroupsToFetch","Promise","all","blockGroup","forEach","block","resultData","zlib","inflateSync","next","parseSummaryBlock","parseBigWigBlock","parseBigBedBlock","console","warn","range"],"mappings":"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","sourcesContent":["/* eslint no-bitwise: [\"error\", { \"allow\": [\"|\"] }] */\nimport { Observer } from 'rxjs'\nimport { Parser } from '@gmod/binary-parser'\nimport AbortablePromiseCache from 'abortable-promise-cache'\nimport { GenericFilehandle } from 'generic-filehandle'\nimport zlib from 'zlib'\nimport QuickLRU from 'quick-lru'\nimport { Feature } from './bbi'\nimport Range from './range'\nimport { groupBlocks, checkAbortSignal } from './util'\n\ninterface CoordRequest {\n  chrId: number\n  start: number\n  end: number\n}\ninterface DataBlock {\n  startChrom: number\n  endChrom: number\n  startBase: number\n  endBase: number\n  validCnt: number\n  minVal: number\n  maxVal: number\n  sumData: number\n  sumSqData: number\n}\ninterface ReadData {\n  offset: number\n  length: number\n}\n\ninterface SummaryBlock {\n  chromId: number\n  start: number\n  end: number\n  validCnt: number\n  minScore: number\n  maxScore: number\n  sumData: number\n  sumSqData: number\n}\ninterface Options {\n  signal?: AbortSignal\n  request?: CoordRequest\n}\n\nconst BIG_WIG_TYPE_GRAPH = 1\nconst BIG_WIG_TYPE_VSTEP = 2\nconst BIG_WIG_TYPE_FSTEP = 3\n\nfunction getParsers(isBigEndian: boolean): any {\n  const le = isBigEndian ? 'big' : 'little'\n  const summaryParser = new Parser()\n    .endianess(le)\n    .uint32('chromId')\n    .uint32('start')\n    .uint32('end')\n    .uint32('validCnt')\n    .float('minScore')\n    .float('maxScore')\n    .float('sumData')\n    .float('sumSqData')\n\n  const leafParser = new Parser()\n    .endianess(le)\n    .uint8('isLeaf')\n    .skip(1)\n    .uint16('cnt')\n    .choice({\n      tag: 'isLeaf',\n      choices: {\n        1: new Parser().array('blocksToFetch', {\n          length: 'cnt',\n          type: new Parser()\n            .uint32('startChrom')\n            .uint32('startBase')\n            .uint32('endChrom')\n            .uint32('endBase')\n            .uint64('blockOffset')\n            .uint64('blockSize'),\n        }),\n        0: new Parser().array('recurOffsets', {\n          length: 'cnt',\n          type: new Parser()\n            .uint32('startChrom')\n            .uint32('startBase')\n            .uint32('endChrom')\n            .uint32('endBase')\n            .uint64('blockOffset'),\n        }),\n      },\n    })\n  const bigBedParser = new Parser()\n    .endianess(le)\n    .uint32('chromId')\n    .int32('start')\n    .int32('end')\n    .string('rest', {\n      zeroTerminated: true,\n    })\n\n  const bigWigParser = new Parser()\n    .endianess(le)\n    .skip(4)\n    .int32('blockStart')\n    .skip(4)\n    .uint32('itemStep')\n    .uint32('itemSpan')\n    .uint8('blockType')\n    .skip(1)\n    .uint16('itemCount')\n    .choice({\n      tag: 'blockType',\n      choices: {\n        [BIG_WIG_TYPE_FSTEP]: new Parser().array('items', {\n          length: 'itemCount',\n          type: new Parser().float('score'),\n        }),\n        [BIG_WIG_TYPE_VSTEP]: new Parser().array('items', {\n          length: 'itemCount',\n          type: new Parser().int32('start').float('score'),\n        }),\n        [BIG_WIG_TYPE_GRAPH]: new Parser().array('items', {\n          length: 'itemCount',\n          type: new Parser()\n            .int32('start')\n            .int32('end')\n            .float('score'),\n        }),\n      },\n    })\n  return {\n    bigWigParser,\n    bigBedParser,\n    summaryParser,\n    leafParser,\n  }\n}\n\n/**\n * View into a subset of the data in a BigWig file.\n *\n * Adapted by Robert Buels and Colin Diesh from bigwig.js in the Dalliance Genome\n * Explorer by Thomas Down.\n * @constructs\n */\n\nexport class BlockView {\n  private cirTreeOffset: number\n\n  private cirTreeLength: number\n\n  private bbi: GenericFilehandle\n\n  private isCompressed: boolean\n\n  private isBigEndian: boolean\n\n  private refsByName: any\n\n  private blockType: string\n\n  private cirTreePromise?: Promise<{ bytesRead: number; buffer: Buffer }>\n\n  private featureCache = new AbortablePromiseCache({\n    cache: new QuickLRU({ maxSize: 1000 }),\n\n    fill: async (requestData: ReadData, signal: AbortSignal) => {\n      const { length, offset } = requestData\n      const { buffer } = await this.bbi.read(Buffer.alloc(length), 0, length, offset, { signal })\n      return buffer\n    },\n  })\n\n  private leafParser: any\n\n  private bigWigParser: any\n\n  private bigBedParser: any\n\n  private summaryParser: any\n\n  public constructor(\n    bbi: GenericFilehandle,\n    refsByName: any,\n    cirTreeOffset: number,\n    cirTreeLength: number,\n    isBigEndian: boolean,\n    isCompressed: boolean,\n    blockType: string,\n  ) {\n    if (!(cirTreeOffset >= 0)) throw new Error('invalid cirTreeOffset!')\n    if (!(cirTreeLength > 0)) throw new Error('invalid cirTreeLength!')\n\n    this.cirTreeOffset = cirTreeOffset\n    this.cirTreeLength = cirTreeLength\n    this.isCompressed = isCompressed\n    this.refsByName = refsByName\n    this.isBigEndian = isBigEndian\n    this.bbi = bbi\n    this.blockType = blockType\n    Object.assign(this, getParsers(isBigEndian))\n  }\n\n  public async readWigData(\n    chrName: string,\n    start: number,\n    end: number,\n    observer: Observer<Feature[]>,\n    opts: Options,\n  ): Promise<void> {\n    try {\n      const { refsByName, bbi, cirTreeOffset, isBigEndian } = this\n      const { signal } = opts\n      const chrId = refsByName[chrName]\n      if (chrId === undefined) {\n        observer.complete()\n      }\n      const request = { chrId, start, end }\n      if (!this.cirTreePromise) {\n        this.cirTreePromise = bbi.read(Buffer.alloc(48), 0, 48, cirTreeOffset, { signal })\n      }\n      const { buffer } = await this.cirTreePromise\n      const cirBlockSize = isBigEndian ? buffer.readUInt32BE(4) : buffer.readUInt32LE(4)\n      let blocksToFetch: any[] = []\n      let outstanding = 0\n\n      //eslint-disable-next-line prefer-const\n      let cirFobRecur2: Function\n\n      const filterFeats = (b: DataBlock): boolean =>\n        (b.startChrom < chrId || (b.startChrom === chrId && b.startBase <= end)) &&\n        (b.endChrom > chrId || (b.endChrom === chrId && b.endBase >= start))\n\n      const cirFobStartFetch = async (off: any, fr: any, level: number): Promise<void> => {\n        try {\n          const length = fr.max() - fr.min()\n          const offset = fr.min()\n          const resultBuffer = await this.featureCache.get(\n            `${length}_${offset}`,\n            { length, offset },\n            signal,\n          )\n          for (let i = 0; i < off.length; i += 1) {\n            if (fr.contains(off[i])) {\n              cirFobRecur2(resultBuffer, off[i] - offset, level, observer, opts)\n              outstanding -= 1\n              if (outstanding === 0) {\n                this.readFeatures(observer, blocksToFetch, { ...opts, request })\n              }\n            }\n          }\n        } catch (e) {\n          observer.error(e)\n        }\n      }\n      const cirFobRecur = (offset: any, level: number): void => {\n        try {\n          outstanding += offset.length\n\n          const maxCirBlockSpan = 4 + cirBlockSize * 32 // Upper bound on size, based on a completely full leaf node.\n          let spans = new Range(offset[0], offset[0] + maxCirBlockSpan)\n          for (let i = 1; i < offset.length; i += 1) {\n            const blockSpan = new Range(offset[i], offset[i] + maxCirBlockSpan)\n            spans = spans.union(blockSpan)\n          }\n          spans.getRanges().map((fr: Range) => cirFobStartFetch(offset, fr, level))\n        } catch (e) {\n          observer.error(e)\n        }\n      }\n\n      cirFobRecur2 = (cirBlockData: Buffer, offset: number, level: number): void => {\n        try {\n          const data = cirBlockData.slice(offset)\n\n          const p = this.leafParser.parse(data).result\n          if (p.blocksToFetch) {\n            blocksToFetch = blocksToFetch.concat(\n              p.blocksToFetch\n                .filter(filterFeats)\n                .map((l: any): any => ({ offset: l.blockOffset, length: l.blockSize })),\n            )\n          }\n          if (p.recurOffsets) {\n            const recurOffsets = p.recurOffsets\n              .filter(filterFeats)\n              .map((l: any): any => l.blockOffset)\n            if (recurOffsets.length > 0) {\n              cirFobRecur(recurOffsets, level + 1)\n            }\n          }\n        } catch (e) {\n          observer.error(e)\n        }\n      }\n\n      return cirFobRecur([cirTreeOffset + 48], 1)\n    } catch (e) {\n      observer.error(e)\n    }\n  }\n\n  private parseSummaryBlock(data: Buffer, startOffset: number, request?: CoordRequest): Feature[] {\n    const features = []\n    let currOffset = startOffset\n    while (currOffset < data.byteLength) {\n      const res = this.summaryParser.parse(data.slice(currOffset))\n      features.push(res.result)\n      currOffset += res.offset\n    }\n    let items = features\n    if (request) items = items.filter((elt: SummaryBlock): boolean => elt.chromId === request.chrId)\n    items = items.map(\n      (elt: SummaryBlock): Feature => ({\n        start: elt.start,\n        end: elt.end,\n        maxScore: elt.maxScore,\n        minScore: elt.minScore,\n        score: elt.sumData / (elt.validCnt || 1),\n        summary: true,\n      }),\n    )\n    return request ? items.filter(f => BlockView.coordFilter(f, request)) : items\n  }\n\n  private parseBigBedBlock(\n    data: Buffer,\n    startOffset: number,\n    offset: number,\n    request?: CoordRequest,\n  ): Feature[] {\n    const items = []\n    let currOffset = startOffset\n    while (currOffset < data.byteLength) {\n      const res = this.bigBedParser.parse(data.slice(currOffset))\n      res.result.uniqueId = `bb-${offset + currOffset}`\n      items.push(res.result)\n      currOffset += res.offset\n    }\n\n    return request ? items.filter((f: any) => BlockView.coordFilter(f, request)) : items\n  }\n\n  private parseBigWigBlock(bytes: Buffer, startOffset: number, request?: CoordRequest): Feature[] {\n    const data = bytes.slice(startOffset)\n    const results = this.bigWigParser.parse(data).result\n    const { items, itemSpan, itemStep, blockStart, blockType } = results\n    if (blockType === BIG_WIG_TYPE_FSTEP) {\n      for (let i = 0; i < items.length; i++) {\n        items[i].start = blockStart + i * itemStep\n        items[i].end = blockStart + i * itemStep + itemSpan\n      }\n    } else if (blockType === BIG_WIG_TYPE_VSTEP) {\n      for (let i = 0; i < items.length; i++) {\n        items[i].end = items[i].start + itemSpan\n      }\n    }\n    return request ? items.filter((f: any) => BlockView.coordFilter(f, request)) : items\n  }\n\n  private static coordFilter(f: Feature, range: CoordRequest): boolean {\n    return f.start < range.end && f.end >= range.start\n  }\n\n  public async readFeatures(\n    observer: Observer<Feature[]>,\n    blocks: any,\n    opts: Options = {},\n  ): Promise<void> {\n    try {\n      const { blockType, isCompressed } = this\n      const { signal, request } = opts\n      const blockGroupsToFetch = groupBlocks(blocks)\n      checkAbortSignal(signal)\n      await Promise.all(\n        blockGroupsToFetch.map(async (blockGroup: any) => {\n          checkAbortSignal(signal)\n          const { length, offset } = blockGroup\n          const data = await this.featureCache.get(`${length}_${offset}`, blockGroup, signal)\n          blockGroup.blocks.forEach((block: any) => {\n            checkAbortSignal(signal)\n            let blockOffset = block.offset - blockGroup.offset\n            let resultData = data\n            if (isCompressed) {\n              resultData = zlib.inflateSync(data.slice(blockOffset))\n              blockOffset = 0\n            }\n            checkAbortSignal(signal)\n\n            switch (blockType) {\n              case 'summary':\n                observer.next(this.parseSummaryBlock(resultData, blockOffset, request))\n                break\n              case 'bigwig':\n                observer.next(this.parseBigWigBlock(resultData, blockOffset, request))\n                break\n              case 'bigbed':\n                observer.next(\n                  // eslint-disable-next-line no-bitwise\n                  this.parseBigBedBlock(resultData, blockOffset, block.offset * (1 << 8), request),\n                )\n                break\n              default:\n                console.warn(`Don't know what to do with ${blockType}`)\n            }\n          })\n        }),\n      )\n      observer.complete()\n    } catch (e) {\n      observer.error(e)\n    }\n  }\n}\n"]}