securityos/node_modules/music-metadata/lib/ogg/OggParser.js

127 lines
5.5 KiB
JavaScript

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.OggParser = exports.SegmentTable = void 0;
const Token = require("token-types");
const core_1 = require("strtok3/lib/core");
const debug_1 = require("debug");
const util = require("../common/Util");
const FourCC_1 = require("../common/FourCC");
const BasicParser_1 = require("../common/BasicParser");
const VorbisParser_1 = require("./vorbis/VorbisParser");
const OpusParser_1 = require("./opus/OpusParser");
const SpeexParser_1 = require("./speex/SpeexParser");
const TheoraParser_1 = require("./theora/TheoraParser");
const debug = (0, debug_1.default)('music-metadata:parser:ogg');
class SegmentTable {
static sum(buf, off, len) {
let s = 0;
for (let i = off; i < off + len; ++i) {
s += buf[i];
}
return s;
}
constructor(header) {
this.len = header.page_segments;
}
get(buf, off) {
return {
totalPageSize: SegmentTable.sum(buf, off, this.len)
};
}
}
exports.SegmentTable = SegmentTable;
/**
* Parser for Ogg logical bitstream framing
*/
class OggParser extends BasicParser_1.BasicParser {
/**
* Parse page
* @returns {Promise<void>}
*/
async parse() {
debug('pos=%s, parsePage()', this.tokenizer.position);
try {
let header;
do {
header = await this.tokenizer.readToken(OggParser.Header);
if (header.capturePattern !== 'OggS')
throw new Error('Invalid Ogg capture pattern');
this.metadata.setFormat('container', 'Ogg');
this.header = header;
this.pageNumber = header.pageSequenceNo;
debug('page#=%s, Ogg.id=%s', header.pageSequenceNo, header.capturePattern);
const segmentTable = await this.tokenizer.readToken(new SegmentTable(header));
debug('totalPageSize=%s', segmentTable.totalPageSize);
const pageData = await this.tokenizer.readToken(new Token.Uint8ArrayType(segmentTable.totalPageSize));
debug('firstPage=%s, lastPage=%s, continued=%s', header.headerType.firstPage, header.headerType.lastPage, header.headerType.continued);
if (header.headerType.firstPage) {
const id = new Token.StringType(7, 'ascii').get(Buffer.from(pageData), 0);
switch (id) {
case '\x01vorbis': // Ogg/Vorbis
debug('Set page consumer to Ogg/Vorbis');
this.pageConsumer = new VorbisParser_1.VorbisParser(this.metadata, this.options);
break;
case 'OpusHea': // Ogg/Opus
debug('Set page consumer to Ogg/Opus');
this.pageConsumer = new OpusParser_1.OpusParser(this.metadata, this.options, this.tokenizer);
break;
case 'Speex ': // Ogg/Speex
debug('Set page consumer to Ogg/Speex');
this.pageConsumer = new SpeexParser_1.SpeexParser(this.metadata, this.options, this.tokenizer);
break;
case 'fishead':
case '\x00theora': // Ogg/Theora
debug('Set page consumer to Ogg/Theora');
this.pageConsumer = new TheoraParser_1.TheoraParser(this.metadata, this.options, this.tokenizer);
break;
default:
throw new Error('gg audio-codec not recognized (id=' + id + ')');
}
}
this.pageConsumer.parsePage(header, pageData);
} while (!header.headerType.lastPage);
}
catch (err) {
if (err instanceof core_1.EndOfStreamError) {
this.metadata.addWarning('Last OGG-page is not marked with last-page flag');
debug(`End-of-stream`);
this.metadata.addWarning('Last OGG-page is not marked with last-page flag');
if (this.header) {
this.pageConsumer.calculateDuration(this.header);
}
}
else if (err.message.startsWith('FourCC')) {
if (this.pageNumber > 0) {
// ignore this error: work-around if last OGG-page is not marked with last-page flag
this.metadata.addWarning('Invalid FourCC ID, maybe last OGG-page is not marked with last-page flag');
this.pageConsumer.flush();
}
}
else {
throw err;
}
}
}
}
OggParser.Header = {
len: 27,
get: (buf, off) => {
return {
capturePattern: FourCC_1.FourCcToken.get(buf, off),
version: Token.UINT8.get(buf, off + 4),
headerType: {
continued: util.getBit(buf, off + 5, 0),
firstPage: util.getBit(buf, off + 5, 1),
lastPage: util.getBit(buf, off + 5, 2)
},
// packet_flag: buf.readUInt8(off + 5),
absoluteGranulePosition: Number(Token.UINT64_LE.get(buf, off + 6)),
streamSerialNumber: Token.UINT32_LE.get(buf, off + 14),
pageSequenceNo: Token.UINT32_LE.get(buf, off + 18),
pageChecksum: Token.UINT32_LE.get(buf, off + 22),
page_segments: Token.UINT8.get(buf, off + 26)
};
}
};
exports.OggParser = OggParser;