mirror of
https://github.com/LuanRT/YouTube.js.git
synced 2026-06-16 19:12:24 +00:00
* dev: add response types * dev: refactor `Parser#parseResponse()` * dev: update YouTube parsers * dev: update YouTube Music classes * dev: update YouTube Kids classes * dev: update core classes * dev(Parser): fix some inconsistencies * chore: update docs * chore: update docs x2 * fix: export response types * chore(docs): update parser example
482 lines
12 KiB
TypeScript
482 lines
12 KiB
TypeScript
import { deepCompare, ParsingError } from '../utils/Utils.js';
|
|
|
|
const isObserved = Symbol('ObservedArray.isObserved');
|
|
|
|
export class YTNode {
|
|
static readonly type: string = 'YTNode';
|
|
readonly type: string;
|
|
|
|
constructor() {
|
|
this.type = (this.constructor as YTNodeConstructor).type;
|
|
}
|
|
|
|
/**
|
|
* Check if the node is of the given type.
|
|
* @param type - The type to check
|
|
* @returns whether the node is of the given type
|
|
*/
|
|
#is<T extends YTNode>(type: YTNodeConstructor<T>): this is T {
|
|
return this.type === type.type;
|
|
}
|
|
|
|
/**
|
|
* Check if the node is of the given type.
|
|
* @param types - The type to check
|
|
* @returns whether the node is of the given type
|
|
*/
|
|
is<T extends YTNode, K extends YTNodeConstructor<T>[]>(...types: K): this is InstanceType<K[number]> {
|
|
return types.some((type) => this.#is(type));
|
|
}
|
|
|
|
/**
|
|
* Cast to one of the given types.
|
|
*/
|
|
as<T extends YTNode, K extends YTNodeConstructor<T>[]>(...types: K): InstanceType<K[number]> {
|
|
if (!this.is(...types)) {
|
|
throw new ParsingError(`Cannot cast ${this.type} to one of ${types.map((t) => t.type).join(', ')}`);
|
|
}
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Check for a key without asserting the type.
|
|
* @param key - The key to check
|
|
* @returns Whether the node has the key
|
|
*/
|
|
hasKey<T extends string, R = any>(key: T): this is this & { [k in T]: R } {
|
|
return Reflect.has(this, key);
|
|
}
|
|
|
|
/**
|
|
* Assert that the node has the given key and return it.
|
|
* @param key - The key to check
|
|
* @returns The value of the key wrapped in a Maybe
|
|
* @throws If the node does not have the key
|
|
*/
|
|
key<T extends string, R = any>(key: T) {
|
|
if (!this.hasKey<T, R>(key)) {
|
|
throw new ParsingError(`Missing key ${key}`);
|
|
}
|
|
return new Maybe(this[key]);
|
|
}
|
|
}
|
|
|
|
export class Maybe {
|
|
#value;
|
|
|
|
constructor (value: any) {
|
|
this.#value = value;
|
|
}
|
|
|
|
#checkPrimative(type: 'string' | 'number' | 'bigint' | 'boolean' | 'symbol' | 'undefined' | 'object' | 'function') {
|
|
if (typeof this.#value !== type) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
#assertPrimative(type: 'string' | 'number' | 'bigint' | 'boolean' | 'symbol' | 'undefined' | 'object' | 'function') {
|
|
if (!this.#checkPrimative(type)) {
|
|
throw new TypeError(`Expected ${type}, got ${this.typeof}`);
|
|
}
|
|
return this.#value;
|
|
}
|
|
|
|
get typeof() {
|
|
return typeof this.#value;
|
|
}
|
|
|
|
string(): string {
|
|
return this.#assertPrimative('string');
|
|
}
|
|
|
|
isString() {
|
|
return this.#checkPrimative('string');
|
|
}
|
|
|
|
number(): number {
|
|
return this.#assertPrimative('number');
|
|
}
|
|
|
|
isNumber() {
|
|
return this.#checkPrimative('number');
|
|
}
|
|
|
|
bigint(): bigint {
|
|
return this.#assertPrimative('bigint');
|
|
}
|
|
|
|
isBigint() {
|
|
return this.#checkPrimative('bigint');
|
|
}
|
|
|
|
boolean(): boolean {
|
|
return this.#assertPrimative('boolean');
|
|
}
|
|
|
|
isBoolean() {
|
|
return this.#checkPrimative('boolean');
|
|
}
|
|
|
|
symbol(): symbol {
|
|
return this.#assertPrimative('symbol');
|
|
}
|
|
|
|
isSymbol() {
|
|
return this.#checkPrimative('symbol');
|
|
}
|
|
|
|
undefined(): undefined {
|
|
return this.#assertPrimative('undefined');
|
|
}
|
|
|
|
isUndefined() {
|
|
return this.#checkPrimative('undefined');
|
|
}
|
|
|
|
null(): null {
|
|
if (this.#value !== null)
|
|
throw new TypeError(`Expected null, got ${typeof this.#value}`);
|
|
return this.#value;
|
|
}
|
|
|
|
isNull() {
|
|
return this.#value === null;
|
|
}
|
|
|
|
object(): object {
|
|
return this.#assertPrimative('object');
|
|
}
|
|
|
|
isObject() {
|
|
return this.#checkPrimative('object');
|
|
}
|
|
|
|
/* eslint-ignore */
|
|
function(): Function {
|
|
return this.#assertPrimative('function');
|
|
}
|
|
|
|
isFunction() {
|
|
return this.#checkPrimative('function');
|
|
}
|
|
|
|
/**
|
|
* Get the value as an array.
|
|
* @returns the value as any[]
|
|
* @throws If the value is not an array
|
|
*/
|
|
array(): any[] {
|
|
if (!Array.isArray(this.#value)) {
|
|
throw new TypeError(`Expected array, got ${typeof this.#value}`);
|
|
}
|
|
return this.#value;
|
|
}
|
|
|
|
/**
|
|
* More typesafe variant of {@link Maybe#array}.
|
|
* @returns a proxied array which returns all the values as {@link Maybe}
|
|
* @throws If the value is not an array
|
|
*/
|
|
arrayOfMaybe(): Maybe[] {
|
|
const arrayProps: any[] = [];
|
|
return new Proxy(this.array(), {
|
|
get(target, prop) {
|
|
if (Reflect.has(arrayProps, prop)) {
|
|
return Reflect.get(target, prop);
|
|
}
|
|
return new Maybe(Reflect.get(target, prop));
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Check whether the value is an array.
|
|
* @returns whether the value is an array
|
|
*/
|
|
isArray() {
|
|
return Array.isArray(this.#value);
|
|
}
|
|
|
|
/**
|
|
* Get the value as a YTNode
|
|
* @returns the value as a YTNode
|
|
* @throws If the value is not a YTNode
|
|
*/
|
|
node() {
|
|
if (!(this.#value instanceof YTNode)) {
|
|
throw new TypeError(`Expected YTNode, got ${this.#value.constructor.name}`);
|
|
}
|
|
return this.#value;
|
|
}
|
|
|
|
/**
|
|
* Check if the value is a YTNode
|
|
* @returns Whether the value is a YTNode
|
|
*/
|
|
isNode() {
|
|
return this.#value instanceof YTNode;
|
|
}
|
|
|
|
/**
|
|
* Get the value as a YTNode of the given type.
|
|
* @param type - The type to cast to
|
|
* @returns The node casted to the given type
|
|
* @throws If the node is not of the given type
|
|
*/
|
|
nodeOfType<T extends YTNode, K extends YTNodeConstructor<T>[]>(...types: K) {
|
|
return this.node().as(...types);
|
|
}
|
|
|
|
/**
|
|
* Check if the value is a YTNode of the given type.
|
|
* @param type - the type to check
|
|
* @returns Whether the value is a YTNode of the given type
|
|
*/
|
|
isNodeOfType<T extends YTNode, K extends YTNodeConstructor<T>[]>(...types: K) {
|
|
return this.isNode() && this.node().is(...types);
|
|
}
|
|
|
|
/**
|
|
* Get the value as an ObservedArray.
|
|
* @returns the value of the Maybe as a ObservedArray
|
|
*/
|
|
observed(): ObservedArray<YTNode> {
|
|
if (!this.isObserved()) {
|
|
throw new TypeError(`Expected ObservedArray, got ${typeof this.#value}`);
|
|
}
|
|
return this.#value;
|
|
}
|
|
|
|
/**
|
|
* Check if the value is an ObservedArray.
|
|
*/
|
|
isObserved() {
|
|
return this.#value?.[isObserved];
|
|
}
|
|
|
|
/**
|
|
* Get the value of the Maybe as a SuperParsedResult.
|
|
* @returns the value as a SuperParsedResult
|
|
* @throws If the value is not a SuperParsedResult
|
|
*/
|
|
parsed(): SuperParsedResult {
|
|
if (!(this.#value instanceof SuperParsedResult)) {
|
|
throw new TypeError(`Expected SuperParsedResult, got ${typeof this.#value}`);
|
|
}
|
|
return this.#value;
|
|
}
|
|
|
|
/**
|
|
* Is the result a SuperParsedResult?
|
|
*/
|
|
isParsed() {
|
|
return this.#value instanceof SuperParsedResult;
|
|
}
|
|
|
|
/**
|
|
* @deprecated This call is not meant to be used outside of debugging. Please use the specific type getter instead.
|
|
*/
|
|
any(): any {
|
|
console.warn('This call is not meant to be used outside of debugging. Please use the specific type getter instead.');
|
|
return this.#value;
|
|
}
|
|
|
|
/**
|
|
* Get the node as an instance of the given class.
|
|
* @param type - The type to check
|
|
* @returns the value as the given type
|
|
* @throws If the node is not of the given type
|
|
*/
|
|
instanceof<T extends object>(type: Constructor<T>): T {
|
|
if (!this.isInstanceof(type)) {
|
|
throw new TypeError(`Expected instance of ${type.name}, got ${this.#value.constructor.name}`);
|
|
}
|
|
return this.#value;
|
|
}
|
|
|
|
/**
|
|
* Check if the node is an instance of the given class.
|
|
* @param type - The type to check
|
|
* @returns Whether the node is an instance of the given type
|
|
*/
|
|
isInstanceof<T extends object>(type: Constructor<T>): this is this & T {
|
|
return this.#value instanceof type;
|
|
}
|
|
}
|
|
|
|
export interface Constructor<T> {
|
|
new (...args: any[]): T;
|
|
}
|
|
|
|
export interface YTNodeConstructor<T extends YTNode = YTNode> {
|
|
new(data: any): T;
|
|
readonly type: string;
|
|
}
|
|
|
|
/**
|
|
* Represents a parsed response in an unknown state. Either a YTNode or a YTNode[] or null.
|
|
*/
|
|
export class SuperParsedResult<T extends YTNode = YTNode> {
|
|
#result;
|
|
|
|
constructor(result: T | ObservedArray<T> | null) {
|
|
this.#result = result;
|
|
}
|
|
|
|
get is_null() {
|
|
return this.#result === null;
|
|
}
|
|
get is_array() {
|
|
return !this.is_null && Array.isArray(this.#result);
|
|
}
|
|
get is_node() {
|
|
return !this.is_array;
|
|
}
|
|
|
|
array() {
|
|
if (!this.is_array) {
|
|
throw new TypeError('Expected an array, got a node');
|
|
}
|
|
return this.#result as ObservedArray<T>;
|
|
}
|
|
|
|
item() {
|
|
if (!this.is_node) {
|
|
throw new TypeError('Expected a node, got an array');
|
|
}
|
|
return this.#result as T;
|
|
}
|
|
}
|
|
|
|
|
|
export type ObservedArray<T extends YTNode = YTNode> = Array<T> & {
|
|
/**
|
|
* Returns the first object to match the rule.
|
|
*/
|
|
get: (rule: object, del_item?: boolean) => T | undefined;
|
|
/**
|
|
* Returns all objects that match the rule.
|
|
*/
|
|
getAll: (rule: object, del_items?: boolean) => T[];
|
|
/**
|
|
* Returns the first object to match the condition.
|
|
*/
|
|
matchCondition: (condition: (node: T) => boolean) => T | undefined;
|
|
/**
|
|
* Removes the item at the given index.
|
|
*/
|
|
remove: (index: number) => T[];
|
|
/**
|
|
* Get all items of a specific type
|
|
*/
|
|
filterType<R extends YTNode, K extends YTNodeConstructor<R>[]>(...types: K): ObservedArray<InstanceType<K[number]>>;
|
|
/**
|
|
* Get the first of a specific type
|
|
*/
|
|
firstOfType<R extends YTNode, K extends YTNodeConstructor<R>[]>(...types: K): InstanceType<K[number]> | undefined;
|
|
/**
|
|
* Get the first item
|
|
*/
|
|
first: () => T;
|
|
/**
|
|
* This is similar to filter but throws if there's a type mismatch.
|
|
*/
|
|
as<R extends YTNode, K extends YTNodeConstructor<R>[]>(...types: K): ObservedArray<InstanceType<K[number]>>;
|
|
};
|
|
|
|
/**
|
|
* Creates a trap to intercept property access
|
|
* and add utilities to an object.
|
|
*/
|
|
export function observe<T extends YTNode>(obj: Array<T>): ObservedArray<T> {
|
|
return new Proxy(obj, {
|
|
get(target, prop) {
|
|
if (prop == 'get') {
|
|
return (rule: object, del_item?: boolean) => (
|
|
target.find((obj, index) => {
|
|
const match = deepCompare(rule, obj);
|
|
if (match && del_item) {
|
|
target.splice(index, 1);
|
|
}
|
|
return match;
|
|
})
|
|
);
|
|
}
|
|
|
|
if (prop == isObserved) {
|
|
return true;
|
|
}
|
|
|
|
if (prop == 'getAll') {
|
|
return (rule: object, del_items: boolean) => (
|
|
target.filter((obj, index) => {
|
|
const match = deepCompare(rule, obj);
|
|
if (match && del_items) {
|
|
target.splice(index, 1);
|
|
}
|
|
return match;
|
|
})
|
|
);
|
|
}
|
|
|
|
if (prop == 'matchCondition') {
|
|
return (condition: (node: T) => boolean) => (
|
|
target.find((obj) => {
|
|
return condition(obj);
|
|
})
|
|
);
|
|
}
|
|
|
|
if (prop == 'filterType') {
|
|
return (...types: YTNodeConstructor<YTNode>[]) => {
|
|
return observe(target.filter((node: YTNode) => {
|
|
if (node.is(...types))
|
|
return true;
|
|
return false;
|
|
|
|
}));
|
|
};
|
|
}
|
|
|
|
|
|
if (prop == 'firstOfType') {
|
|
return (...types: YTNodeConstructor<YTNode>[]) => {
|
|
return target.find((node: YTNode) => {
|
|
if (node.is(...types))
|
|
return true;
|
|
return false;
|
|
});
|
|
};
|
|
}
|
|
|
|
if (prop == 'first') {
|
|
return () => target[0];
|
|
}
|
|
|
|
if (prop == 'as') {
|
|
return (...types: YTNodeConstructor<YTNode>[]) => {
|
|
return observe(target.map((node: YTNode) => {
|
|
if (node.is(...types))
|
|
return node;
|
|
throw new ParsingError(`Expected node of any type ${types.map((type) => type.type).join(', ')}, got ${(node as YTNode).type}`);
|
|
}));
|
|
};
|
|
}
|
|
|
|
if (prop == 'remove') {
|
|
return (index: number): any => target.splice(index, 1);
|
|
}
|
|
|
|
return Reflect.get(target, prop);
|
|
}
|
|
}) as ObservedArray<T>;
|
|
}
|
|
|
|
export class Memo extends Map<string, YTNode[]> {
|
|
getType<T extends YTNode>(type: YTNodeConstructor<T> | YTNodeConstructor<T>[]) {
|
|
if (Array.isArray(type))
|
|
return observe(type.flatMap((type) => (this.get(type.type) || []) as T[]));
|
|
return observe((this.get(type.type) || []) as T[]);
|
|
}
|
|
} |