import { from_string, to_string } from 'libsodium-wrappers-sumo'; import { isArray, isEmpty, isNumber, isPlainObject, isString, toNumber } from 'lodash'; import { StringUtils } from '.'; const e = 'e'; // end of whatever was before const l = 'l'; // list of values const i = 'i'; // start of integer const d = 'd'; // start of dictionary const colon = ':'; const eCode = e.charCodeAt(0); // end of whatever was before const lCode = l.charCodeAt(0); // list of values const iCode = i.charCodeAt(0); // start of integer const dCode = d.charCodeAt(0); // start of dictionary const colonCode = colon.charCodeAt(0); interface BencodeDictType { [key: string]: BencodeElementType; } type BencodeArrayType = Array; type BencodeElementType = number | string | BencodeDictType | BencodeArrayType; const NUMBERS = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9']; export class BDecode { private readonly content: Uint8Array; private currentParsingIndex = 0; private readonly parsedContent: BencodeElementType; constructor(content: Uint8Array | string) { this.content = isString(content) ? from_string(content) : content; this.parsedContent = this.parseContent(); } public getParsedContent() { return this.parsedContent; } /** * Decode an int from a byte array starting with charCode of `i` and ending with charCode `e` */ private parseInt(): number { if (this.currentParsingIndex >= this.content.length) { throw new Error('parseInt: out of bounds'); } if (this.content[this.currentParsingIndex] !== iCode) { throw new Error('parseInt: not the start of an int'); } this.currentParsingIndex++; // drop `i` const startIntStr = this.currentParsingIndex; // save the start of the int const nextEndSeparator = this.content.indexOf(eCode, this.currentParsingIndex); if (nextEndSeparator === -1) { throw new Error('parseInt: not an int to be parsed here: no end separator'); } const parsed = toNumber(to_string(this.content.slice(startIntStr, nextEndSeparator))); if (!isFinite(parsed)) { throw new Error(`parseInt: could not parse number ${parsed}`); } this.currentParsingIndex = nextEndSeparator; this.currentParsingIndex++; // drop the 'e' return parsed; } private parseList(): BencodeArrayType { const parsed: BencodeArrayType = []; if (this.currentParsingIndex >= this.content.length) { throw new Error('parseList: out of bounds'); } if (this.content[this.currentParsingIndex] !== lCode) { throw new Error('parseList: not the start of a list'); } this.currentParsingIndex++; // drop `l` while ( this.currentParsingIndex < this.content.length && this.content[this.currentParsingIndex] !== eCode ) { parsed.push(this.parseBlock()); } this.currentParsingIndex++; // drop the 'e' return parsed; } private parseDict() { const parsed: BencodeDictType = {}; if (this.currentParsingIndex >= this.content.length) { throw new Error('parseDict: out of bounds'); } if (this.content[this.currentParsingIndex] !== dCode) { throw new Error('parseDict: not the start of a dict'); } this.currentParsingIndex++; // drop `d` while ( this.currentParsingIndex < this.content.length && this.content[this.currentParsingIndex] !== eCode ) { const key = this.parseString(); const value = this.parseBlock(); parsed[key] = value; } this.currentParsingIndex++; // drop the 'e' return parsed; } /** * Decode a string element from iterator assumed to have structure `length:data` */ private parseString(): string { if (this.currentParsingIndex >= this.content.length) { throw new Error('parseString: out of bounds'); } // this.currentParsingIndex++; const separatorIndex = this.content.indexOf(colonCode, this.currentParsingIndex); if (separatorIndex === -1) { throw new Error('parseString: cannot parse string without separator'); } const strLength = toNumber( to_string(this.content.slice(this.currentParsingIndex, separatorIndex)) ); if (!isFinite(strLength)) { throw new Error('parseString: cannot parse string without length'); } if (strLength === 0) { return ''; } if (strLength > this.content.length - separatorIndex - 1) { throw new Error( 'parseString: length is too long considering what we have left on this string' ); } const strContent = this.content.slice(separatorIndex + 1, separatorIndex + 1 + strLength); this.currentParsingIndex = separatorIndex + 1 + strLength; return StringUtils.decode(strContent, 'utf8'); } private parseContent() { return this.parseBlock(); } private parseBlock() { let parsed: BencodeElementType; if (this.content.length < this.currentParsingIndex) { throw new Error('Out of bounds'); } if (this.content[this.currentParsingIndex] === lCode) { parsed = this.parseList(); } else if (this.content[this.currentParsingIndex] === dCode) { parsed = this.parseDict(); } else if (this.content[this.currentParsingIndex] === iCode) { parsed = this.parseInt(); } else if (NUMBERS.some(num => this.content[this.currentParsingIndex] === num.charCodeAt(0))) { parsed = this.parseString(); } else { throw new Error( `parseBlock: Could not parse charCode at ${this.currentParsingIndex}: ${ this.content[this.currentParsingIndex] }. Length: ${this.content.length}` ); } return parsed; } } export class BEncode { private readonly input: BencodeElementType; private readonly bencodedContent: Uint8Array; constructor(content: BencodeElementType) { this.input = content; this.bencodedContent = this.encodeContent(); } public getBencodedContent() { return this.bencodedContent; } private encodeItem(item: BencodeElementType): Uint8Array { if (isNumber(item) && isFinite(item)) { return from_string(`i${item}e`); } if (isNumber(item)) { throw new Error('encodeItem not finite number'); } if (isString(item)) { const content = new Uint8Array(StringUtils.encode(item, 'utf8')); const contentLengthLength = `${content.length}`.length; const toReturn = new Uint8Array(content.length + 1 + contentLengthLength); toReturn.set(from_string(`${content.length}`)); toReturn.set([colonCode], contentLengthLength); toReturn.set(content, contentLengthLength + 1); return toReturn; } if (isArray(item)) { let content = new Uint8Array(); for (let index = 0; index < item.length; index++) { const encodedItem = this.encodeItem(item[index]); const encodedItemLength = encodedItem.length; const existingContentLength = content.length; const newContent = new Uint8Array(existingContentLength + encodedItemLength); newContent.set(content); newContent.set(encodedItem, content.length); content = newContent; } const toReturn = new Uint8Array(content.length + 2); toReturn.set([lCode]); toReturn.set(content, 1); toReturn.set([eCode], content.length + 1); return toReturn; } if (isPlainObject(item)) { // bencoded objects keys must be sorted lexicographically const sortedKeys = Object.keys(item).sort(); let content = new Uint8Array(); sortedKeys.forEach(key => { const value = item[key]; const encodedKey = this.encodeItem(key); const encodedValue = this.encodeItem(value); const newContent = new Uint8Array(content.length + encodedKey.length + encodedValue.length); newContent.set(content); newContent.set(encodedKey, content.length); newContent.set(encodedValue, content.length + encodedKey.length); content = newContent; }); const toReturn = new Uint8Array(content.length + 2); toReturn.set([dCode]); toReturn.set(content, 1); toReturn.set([eCode], content.length + 1); return toReturn; } throw new Error(`encodeItem: unknown type to encode ${typeof item}`); } private encodeContent(): Uint8Array { if (!this.input || (isEmpty(this.input) && !isNumber(this.input))) { return new Uint8Array(); } return this.encodeItem(this.input); } }