securityos/node_modules/music-metadata/lib/wavpack/WavPackParser.js

100 lines
5.6 KiB
JavaScript

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.WavPackParser = void 0;
const Token = require("token-types");
const APEv2Parser_1 = require("../apev2/APEv2Parser");
const FourCC_1 = require("../common/FourCC");
const BasicParser_1 = require("../common/BasicParser");
const WavPackToken_1 = require("./WavPackToken");
const debug_1 = require("debug");
const debug = (0, debug_1.default)('music-metadata:parser:WavPack');
/**
* WavPack Parser
*/
class WavPackParser extends BasicParser_1.BasicParser {
async parse() {
this.audioDataSize = 0;
// First parse all WavPack blocks
await this.parseWavPackBlocks();
// try to parse APEv2 header
return APEv2Parser_1.APEv2Parser.tryParseApeHeader(this.metadata, this.tokenizer, this.options);
}
async parseWavPackBlocks() {
do {
const blockId = await this.tokenizer.peekToken(FourCC_1.FourCcToken);
if (blockId !== 'wvpk')
break;
const header = await this.tokenizer.readToken(WavPackToken_1.WavPack.BlockHeaderToken);
if (header.BlockID !== 'wvpk')
throw new Error('Invalid WavPack Block-ID');
debug(`WavPack header blockIndex=${header.blockIndex}, len=${WavPackToken_1.WavPack.BlockHeaderToken.len}`);
if (header.blockIndex === 0 && !this.metadata.format.container) {
this.metadata.setFormat('container', 'WavPack');
this.metadata.setFormat('lossless', !header.flags.isHybrid);
// tagTypes: this.type,
this.metadata.setFormat('bitsPerSample', header.flags.bitsPerSample);
if (!header.flags.isDSD) {
// In case isDSD, these values will ne set in ID_DSD_BLOCK
this.metadata.setFormat('sampleRate', header.flags.samplingRate);
this.metadata.setFormat('duration', header.totalSamples / header.flags.samplingRate);
}
this.metadata.setFormat('numberOfChannels', header.flags.isMono ? 1 : 2);
this.metadata.setFormat('numberOfSamples', header.totalSamples);
this.metadata.setFormat('codec', header.flags.isDSD ? 'DSD' : 'PCM');
}
const ignoreBytes = header.blockSize - (WavPackToken_1.WavPack.BlockHeaderToken.len - 8);
await (header.blockIndex === 0 ? this.parseMetadataSubBlock(header, ignoreBytes) : this.tokenizer.ignore(ignoreBytes));
if (header.blockSamples > 0) {
this.audioDataSize += header.blockSize; // Count audio data for bit-rate calculation
}
} while (!this.tokenizer.fileInfo.size || this.tokenizer.fileInfo.size - this.tokenizer.position >= WavPackToken_1.WavPack.BlockHeaderToken.len);
this.metadata.setFormat('bitrate', this.audioDataSize * 8 / this.metadata.format.duration);
}
/**
* Ref: http://www.wavpack.com/WavPack5FileFormat.pdf, 3.0 Metadata Sub-blocks
* @param remainingLength
*/
async parseMetadataSubBlock(header, remainingLength) {
while (remainingLength > WavPackToken_1.WavPack.MetadataIdToken.len) {
const id = await this.tokenizer.readToken(WavPackToken_1.WavPack.MetadataIdToken);
const dataSizeInWords = await this.tokenizer.readNumber(id.largeBlock ? Token.UINT24_LE : Token.UINT8);
const data = Buffer.alloc(dataSizeInWords * 2 - (id.isOddSize ? 1 : 0));
await this.tokenizer.readBuffer(data);
debug(`Metadata Sub-Blocks functionId=0x${id.functionId.toString(16)}, id.largeBlock=${id.largeBlock},data-size=${data.length}`);
switch (id.functionId) {
case 0x0: // ID_DUMMY: could be used to pad WavPack blocks
break;
case 0xe: // ID_DSD_BLOCK
debug('ID_DSD_BLOCK');
// https://github.com/dbry/WavPack/issues/71#issuecomment-483094813
const mp = 1 << data.readUInt8(0);
const samplingRate = header.flags.samplingRate * mp * 8; // ToDo: second factor should be read from DSD-metadata block https://github.com/dbry/WavPack/issues/71#issuecomment-483094813
if (!header.flags.isDSD)
throw new Error('Only expect DSD block if DSD-flag is set');
this.metadata.setFormat('sampleRate', samplingRate);
this.metadata.setFormat('duration', header.totalSamples / samplingRate);
break;
case 0x24: // ID_ALT_TRAILER: maybe used to embed original ID3 tag header
debug('ID_ALT_TRAILER: trailer for non-wav files');
break;
case 0x26: // ID_MD5_CHECKSUM
this.metadata.setFormat('audioMD5', data);
break;
case 0x2f: // ID_BLOCK_CHECKSUM
debug(`ID_BLOCK_CHECKSUM: checksum=${data.toString('hex')}`);
break;
default:
debug(`Ignore unsupported meta-sub-block-id functionId=0x${id.functionId.toString(16)}`);
break;
}
remainingLength -= WavPackToken_1.WavPack.MetadataIdToken.len + (id.largeBlock ? Token.UINT24_LE.len : Token.UINT8.len) + dataSizeInWords * 2;
debug(`remainingLength=${remainingLength}`);
if (id.isOddSize)
this.tokenizer.ignore(1);
}
if (remainingLength !== 0)
throw new Error('metadata-sub-block should fit it remaining length');
}
}
exports.WavPackParser = WavPackParser;