Removed dependency on external package by bringing it in-line

This commit is contained in:
jg42526 2025-05-16 09:35:46 +00:00
parent d8d594a493
commit 41ce2eceff
4 changed files with 278 additions and 8 deletions

1
package-lock.json generated
View file

@ -19,7 +19,6 @@
"arrive": "^2.4.1",
"avsc": "^5.7.7",
"bcryptjs": "^2.4.3",
"bencodec": "^3.0.1",
"bignumber.js": "^9.1.2",
"blakejs": "^1.2.1",
"bootstrap": "4.6.2",

View file

@ -105,7 +105,6 @@
"arrive": "^2.4.1",
"avsc": "^5.7.7",
"bcryptjs": "^2.4.3",
"bencodec": "^3.0.1",
"bignumber.js": "^9.1.2",
"blakejs": "^1.2.1",
"bootstrap": "4.6.2",

View file

@ -5,7 +5,6 @@
*/
import Operation from "../Operation.mjs";
import bencodec from "bencodec";
/**
* URL Decode operation
@ -13,7 +12,7 @@ import bencodec from "bencodec";
class BencodeDecode extends Operation {
/**
* URLDecode constructor
* URL Decode constructor
*/
constructor() {
super();
@ -33,7 +32,10 @@ class BencodeDecode extends Operation {
* @returns {string}
*/
run(input, args) {
if (input) return toStringRepresentation(bencodec.decode(input, { stringify: true }));
if (input) {
const decoder = new BencodeDecoder(input, {stringify: true}).decode();
return toStringRepresentation(decoder);
}
return "";
}
@ -61,3 +63,140 @@ function toStringRepresentation(value) {
// For other types (undefined, null), handle as you see fit, e.g.:
return String(value);
}
const FLAG = {
INTEGER: 0x69, // 'i'
STR_DELIMITER: 0x3a, // ':'
LIST: 0x6c, // 'l'
DICTIONARY: 0x64, // 'd'
END: 0x65, // 'e'
MINUS: 0x2d, // '-'
PLUS: 0x2b, // '+'
DOT: 0x2e, // '.'
};
/**
* Class for decoding data from the Bencode format.
*/
class BencodeDecoder {
/**
* Creates an instance of BencodeDecoder.
* @param {Buffer|string} data - The bencoded data to decode.
* @param {Object} [options={}] - Optional decoding options.
* @param {boolean} [options.stringify=false] - Whether to return strings instead of Buffers.
*/
constructor(data, options = {}) {
if (!data) throw new Error("Nothing to decode");
this._index = 0;
this._options = options;
this._buffer = typeof data === "string" ? Buffer.from(data) : data;
}
/**
* Checks if a character code represents a digit (09).
* @param {number} char - The character code to check.
* @returns {boolean} - True if the character is a digit.
*/
static _isInteger(char) {
return char >= 0x30 && char <= 0x39;
}
/**
* Returns the current character code in the buffer.
* @returns {number} - The current character code.
*/
_currentChar() {
return this._buffer[this._index];
}
/**
* Returns the next character code in the buffer and advances the index.
* @returns {number} - The next character code.
*/
_next() {
return this._buffer[this._index++];
}
/**
* Decodes the bencoded data.
* @returns {*} - The decoded value (string, number, list, or dictionary).
*/
decode() {
const char = this._currentChar();
if (BencodeDecoder._isInteger(char)) return this._decodeString();
if (char === FLAG.INTEGER) return this._decodeInteger();
if (char === FLAG.LIST) return this._decodeList();
if (char === FLAG.DICTIONARY) return this._decodeDictionary();
throw new Error("Invalid bencode data");
}
/**
* Decodes a bencoded string.
* @returns {Buffer|string} - The decoded string or Buffer.
*/
_decodeString() {
const length = this._decodeInteger();
const acc = [];
for (let i = 0; i < length; i++) acc.push(this._next());
const result = Buffer.from(acc);
return this._options.stringify ? result.toString("utf8") : result;
}
/**
* Decodes a bencoded integer.
* @returns {number} - The decoded integer.
*/
_decodeInteger() {
let sign = 1;
let integer = 0;
if (this._currentChar() === FLAG.INTEGER) this._index++;
if (this._currentChar() === FLAG.PLUS) this._index++;
if (this._currentChar() === FLAG.MINUS) {
this._index++;
sign = -1;
}
while (BencodeDecoder._isInteger(this._currentChar()) || this._currentChar() === FLAG.DOT) {
if (this._currentChar() === FLAG.DOT) {
this._index++; // Skip dot (float not supported)
} else {
integer = integer * 10 + (this._next() - 0x30);
}
}
if (this._currentChar() === FLAG.END) this._index++;
if (this._currentChar() === FLAG.STR_DELIMITER) this._index++;
return integer * sign;
}
/**
* Decodes a bencoded list.
* @returns {Array} - The decoded list.
*/
_decodeList() {
const acc = [];
this._next(); // Skip 'l'
while (this._currentChar() !== FLAG.END) {
acc.push(this.decode());
}
this._next(); // Skip 'e'
return acc;
}
/**
* Decodes a bencoded dictionary.
* @returns {Object} - The decoded dictionary.
*/
_decodeDictionary() {
const acc = {};
this._next(); // Skip 'd'
while (this._currentChar() !== FLAG.END) {
const key = this._decodeString();
acc[key.toString()] = this.decode();
}
this._next(); // Skip 'e'
return acc;
}
}

View file

@ -5,7 +5,6 @@
*/
import Operation from "../Operation.mjs";
import bencodec from "bencodec";
/**
* URL Decode operation
@ -33,7 +32,8 @@ class BencodeEncode extends Operation {
* @returns {string}
*/
run(input, args) {
return bencodec.encode(parseValue(input), { stringify: true });
const encoder = new BencodeEncoder({ stringify: true });
return encoder.encode(parseValue(input));
}
}
@ -41,7 +41,7 @@ class BencodeEncode extends Operation {
export default BencodeEncode;
/**
* Parses string, returns appropraite data structure
* Parses string, returns appropriate data structure
*/
function parseValue(str) {
const trimmed = str.trim();
@ -53,3 +53,136 @@ function parseValue(str) {
return trimmed;
}
}
const FLAG = {
INTEGER: 0x69, // 'i'
STR_DELIMITER: 0x3a, // ':'
LIST: 0x6c, // 'l'
DICTIONARY: 0x64, // 'd'
END: 0x65, // 'e'
};
/**
* BencodeEncoder class for encoding data into bencode format.
*/
class BencodeEncoder {
/**
*
*/
constructor(options = {}) {
this._integerIdentifier = Buffer.from([FLAG.INTEGER]);
this._stringDelimiterIdentifier = Buffer.from([FLAG.STR_DELIMITER]);
this._listIdentifier = Buffer.from([FLAG.LIST]);
this._dictionaryIdentifier = Buffer.from([FLAG.DICTIONARY]);
this._endIdentifier = Buffer.from([FLAG.END]);
this._buffer = [];
this._options = options;
}
/**
* Encodes the given data into bencode format.
* @param {*} data - The data to encode.
* @returns {Buffer|string} - The encoded data as a Buffer or string.
*/
encode(data) {
this._encodeType(data);
const result = Buffer.concat(this._buffer);
return this._options.stringify ? result.toString("utf8") : result;
}
/**
* Determines the type of data and encodes it accordingly.
* @param {*} data - The data to encode.
*/
_encodeType(data) {
if (Buffer.isBuffer(data)) {
return this._encodeBuffer(data);
}
if (Array.isArray(data)) {
return this._encodeList(data);
}
if (ArrayBuffer.isView(data)) {
return this._encodeBuffer(Buffer.from(data.buffer, data.byteOffset, data.byteLength));
}
if (data instanceof ArrayBuffer) {
return this._encodeBuffer(Buffer.from(data));
}
if (typeof data === "boolean") {
return this._encodeInteger(data ? 1 : 0);
}
if (typeof data === "number") {
return this._encodeInteger(data);
}
if (typeof data === "string") {
return this._encodeString(data);
}
if (typeof data === "object") {
return this._encodeDictionary(data);
}
throw new Error(`${typeof data} is unsupported type.`);
}
/**
* Buffer into bencode format.
* @param {Buffer} data - The buffer to encode.
*/
_encodeBuffer(data) {
this._buffer.push(
Buffer.from(String(data.length)),
this._stringDelimiterIdentifier,
data
);
}
/**
* Encodes a string into bencode format.
* @param {string} data - The string to encode.
*/
_encodeString(data) {
this._buffer.push(
Buffer.from(String(Buffer.byteLength(data))),
this._stringDelimiterIdentifier,
Buffer.from(data)
);
}
/**
* Encodes an integer into bencode format.
* @param {number} data - The integer to encode.
*/
_encodeInteger(data) {
this._buffer.push(
this._integerIdentifier,
Buffer.from(String(Math.round(data))),
this._endIdentifier
);
}
/**
* Encodes a list (array) into bencode format.
* @param {Array} data - The list to encode.
*/
_encodeList(data) {
this._buffer.push(this._listIdentifier);
for (const item of data) {
if (item === null || item === undefined) continue;
this._encodeType(item);
}
this._buffer.push(this._endIdentifier);
}
/**
* Encodes a dictionary (object) into bencode format.
* @param {Object} data - The dictionary to encode.
*/
_encodeDictionary(data) {
this._buffer.push(this._dictionaryIdentifier);
const keys = Object.keys(data).sort();
for (const key of keys) {
if (data[key] === null || data[key] === undefined) continue;
this._encodeString(key);
this._encodeType(data[key]);
}
this._buffer.push(this._endIdentifier);
}
}