mirror of
https://github.com/LuanRT/YouTube.js.git
synced 2026-06-19 12:31:17 +00:00
* refactor: remove dependancies removes node-forge and uuid in favor of Web APIs * refactor!: commonjs to es6 To aid with #93 I will make all my changes in TypeScript instead. This is the first step into making that happen. Used: https://github.com/wessberg/cjstoesm * refactor!: NToken and Signature TS files Bring this PR up to speed with #93 * feat: cross platform cache (WIP) this is untested! should remove idb as dependecy. * feat: EventEmitter polyfill * refactor: remove events * feat: HTTPClient based on Fetch API (WIP) * refactor!: parsers refactor (WIP) Initial TS support for parsers as per #93 This adds several type safety checks to the parser which'll help to ensure valid data is returned by the parser. * refactor!: parsers refactor (WIP) Bring more in line with the existing implementations & make less verbose * refactor!: parser refactor I was overcomplicating things, this is much simpler and compatible with the existing JS API * fix: some missed parsers while refactoring * fix: better type inferance for parseResponse * feat(TS): typesafe YTNode casts * feat: more type safety in YTNode and Parser * refactor: VideoInfo download with fetch & TS (WIP) Again, this also does some work for #93 * fix: LiveChat in VideoInfo * refactor!: more typesafety in parser * refactor!: VideoInfo almost completed * refactor!: player and session refactors - Remove the Player class' dependance on Session. - Add additional context to the Session. * refactor!: move auth logic to Session (WIP) * refactor: TS port for Actions and Innertube My fingers hurt from typing out all those types :-P * refactor: NavigationEndpoint TS this is still a WIP and should be improved. NavigationEndpoint should probably be refactored further. * refactor!: VideoInfo compiles without errors * chore: delete old player * fix: import errors It compiles and runs!! * fix: Utils import fixes * fix: several runtime errors * fix: video streaming * chore: remove console.log debugging Whoops, forgot to remove these before I pushed the previous commit * chore: remove old unused dependencies * fix: typescript errors Now emitting declarations and source maps * refactor: TS feed * chore: delete old Feed * refactor: move streamToIterable into Utils * refactor: AccountManager TS * refactor: FilterableFeed to TS * refactor: InteractionManager to TS * refactor: PlaylistManager to TS * refactor: TabbedFeed to TS * refactor: Music to TS (WIP) more work to be done, see TODO comments * fix: getting the tests to pass (6/12) YouTube.js Tests Search ✓ Should search on YouTube (1152 ms) ✕ Should search on YouTube Music (705 ms) ✕ Should retrieve YouTube search suggestions (722 ms) ✓ Should retrieve YouTube Music search suggestions (233 ms) Comments ✓ Should retrieve comments (585 ms) ✕ Should retrieve next batch of comments (221 ms) ✕ Should retrieve comment replies (1 ms) General ✕ Should retrieve playlist with YouTube (732 ms) ✓ Should retrieve home feed (838 ms) ✓ Should retrieve trending content (543 ms) ✓ Should retrieve video info (639 ms) ✕ Should download video (5 ms) * fix: tests (7/12) YouTube.js Tests Search ✓ Should search on YouTube (1984 ms) ✕ Should search on YouTube Music (1139 ms) ✕ Should retrieve YouTube search suggestions (1433 ms) ✓ Should retrieve YouTube Music search suggestions (529 ms) Comments ✓ Should retrieve comments (324 ms) ✓ Should retrieve next batch of comments (395 ms) ✕ Should retrieve comment replies General ✕ Should retrieve playlist with YouTube (653 ms) ✓ Should retrieve home feed (1085 ms) ✓ Should retrieve trending content (513 ms) ✓ Should retrieve video info (921 ms) ✕ Should download video (3 ms) * fix: download tests (8/12) YouTube.js Tests Search ✓ Should search on YouTube (1293 ms) ✕ Should search on YouTube Music (927 ms) ✕ Should retrieve YouTube search suggestions (1250 ms) ✓ Should retrieve YouTube Music search suggestions (258 ms) Comments ✓ Should retrieve comments (803 ms) ✓ Should retrieve next batch of comments (511 ms) ✕ Should retrieve comment replies General ✕ Should retrieve playlist with YouTube (528 ms) ✓ Should retrieve home feed (1047 ms) ✓ Should retrieve trending content (548 ms) ✓ Should retrieve video info (825 ms) ✓ Should download video (1779 ms) * fix: tests (9/12) YouTube.js Tests Search ✓ Should search on YouTube (1276 ms) ✕ Should search on YouTube Music (955 ms) ✓ Should retrieve YouTube search suggestions (661 ms) ✓ Should retrieve YouTube Music search suggestions (491 ms) Comments ✓ Should retrieve comments (624 ms) ✓ Should retrieve next batch of comments (353 ms) ✕ Should retrieve comment replies General ✕ Should retrieve playlist with YouTube (672 ms) ✓ Should retrieve home feed (1277 ms) ✓ Should retrieve trending content (999 ms) ✓ Should retrieve video info (1106 ms) ✓ Should download video (2514 ms) * feat: key based type validation for parsers * fix: comments tests pass (10/12) YouTube.js Tests Search ✓ Should search on YouTube (938 ms) ✕ Should search on YouTube Music (850 ms) ✓ Should retrieve YouTube search suggestions (528 ms) ✓ Should retrieve YouTube Music search suggestions (224 ms) Comments ✓ Should retrieve comments (518 ms) ✓ Should retrieve next batch of comments (337 ms) ✓ Should retrieve comment replies (358 ms) General ✕ Should retrieve playlist with YouTube (466 ms) ✓ Should retrieve home feed (1051 ms) ✓ Should retrieve trending content (623 ms) ✓ Should retrieve video info (863 ms) ✓ Should download video (2656 ms) * refactor: type safety checks removing @ts-ignore * fix: playlist tests pass (11/12) YouTube.js Tests Search ✓ Should search on YouTube (991 ms) ✕ Should search on YouTube Music (924 ms) ✓ Should retrieve YouTube search suggestions (606 ms) ✓ Should retrieve YouTube Music search suggestions (225 ms) Comments ✓ Should retrieve comments (393 ms) ✓ Should retrieve next batch of comments (284 ms) ✓ Should retrieve comment replies (252 ms) General ✓ Should retrieve playlist with YouTube (578 ms) ✓ Should retrieve home feed (1148 ms) ✓ Should retrieve trending content (541 ms) ✓ Should retrieve video info (799 ms) ✓ Should download video (1419 ms) * fix: all tests pass for node 🎉 YouTube.js Tests Search ✓ Should search on YouTube (1053 ms) ✓ Should search on YouTube Music (761 ms) ✓ Should retrieve YouTube search suggestions (453 ms) ✓ Should retrieve YouTube Music search suggestions (221 ms) Comments ✓ Should retrieve comments (627 ms) ✓ Should retrieve next batch of comments (412 ms) ✓ Should retrieve comment replies (268 ms) General ✓ Should retrieve playlist with YouTube (565 ms) ✓ Should retrieve home feed (775 ms) ✓ Should retrieve trending content (498 ms) ✓ Should retrieve video info (875 ms) ✓ Should download video (1364 ms) * build: working Deno bundle Still need to test whether this bundle works in the browser * docs: update deno example to download video * refactor: MusicResponsiveListItem to TS * docs: TSDoc for Parser helpers * docs: Parser documentation for TS * docs: add note about parseItem and parseArray * test: remove browser tests since they're identical * feat: browser support and proxy example * fix: PlaylistManager TS after merge * feat: in-browser video streaming * refactor: cleanup the Dash example * feat: allow custom fetch implementations * feat: fetch debugger * fix: OAuth login * refactor: remove file extensions from imports * refactor: build scripts * fix: CustomEvent on node * fix: LiveChat * fix: linting * fix: liniting in build-parser-json * chore: update test workflow * fix: NToken errors after lint fixes * fix: codacy complaints * docs: update to reflect changes Definitly needs more work but its a start * refactor: cleanup imports/exports * fix: browser example - Remove user-agent before making request. - Fix cache on browsers * fix: cache on node * fix: stupid mistake * refactor: Session#signIn to wait untill success This also splits the 'auth' event up into 3 distinct events: - 'auth' -> fired on success - 'auth-pending' -> fired when pending authentication - 'auth-error' -> fired when an error occurred * refactor: freeze Constants * refactor: cleanup HTTPClient Request * refactor: debugFetch readability * chore: lint * refactor: replace jsdoc with tsdoc eslint plugin remove @param annotations without descriptions * fix: bunch of liniting warnings * refactor: better inference on YTNode#is As suggested by @MasterOfBob777 * fix: linting warnings * revert: undici import * refactor: rename `list_type` to `item_type`
235 lines
8.1 KiB
TypeScript
235 lines
8.1 KiB
TypeScript
import Parser, { ParsedResponse } from '../index';
|
|
|
|
// TODO: refactor this
|
|
import { YTNode } from '../helpers';
|
|
import Actions, { ActionsResponse } from '../../core/Actions';
|
|
|
|
class NavigationEndpoint extends YTNode {
|
|
payload;
|
|
dialog;
|
|
metadata: {
|
|
url?: string;
|
|
api_url?: string;
|
|
page_type?: string;
|
|
send_post?: boolean; // TODO: is this boolean?
|
|
};
|
|
// TODO: these should be given proper types, currently infered
|
|
browse;
|
|
watch;
|
|
search;
|
|
subscribe;
|
|
unsubscribe;
|
|
like;
|
|
perform_comment_action;
|
|
offline_video;
|
|
continuation;
|
|
feedback;
|
|
watch_playlist;
|
|
playlist_edit;
|
|
add_to_playlist;
|
|
get_report_form;
|
|
live_chat_item_context_menu;
|
|
send_live_chat_vote;
|
|
static type = 'NavigationEndpoint';
|
|
constructor(data: any) {
|
|
super();
|
|
const name = Object.keys(data || {})
|
|
.find((item) => item.endsWith('Endpoint') || item.endsWith('Command'));
|
|
this.payload = name ? Reflect.get(data, name) : {};
|
|
if (Reflect.has(this.payload, 'dialog')) {
|
|
this.dialog = Parser.parse(this.payload.dialog);
|
|
}
|
|
if (data?.serviceEndpoint) {
|
|
data = data.serviceEndpoint;
|
|
}
|
|
this.metadata = {};
|
|
if (data?.commandMetadata?.webCommandMetadata?.url) {
|
|
this.metadata.url = data.commandMetadata.webCommandMetadata.url;
|
|
}
|
|
if (data?.commandMetadata?.webCommandMetadata?.webPageType) {
|
|
this.metadata.page_type = data.commandMetadata.webCommandMetadata.webPageType;
|
|
}
|
|
if (data?.commandMetadata?.webCommandMetadata?.apiUrl) {
|
|
this.metadata.api_url = data.commandMetadata.webCommandMetadata.apiUrl.replace('/youtubei/v1/', '');
|
|
}
|
|
if (data?.commandMetadata?.webCommandMetadata?.sendPost) {
|
|
this.metadata.send_post = data.commandMetadata.webCommandMetadata.sendPost;
|
|
}
|
|
if (data?.browseEndpoint) {
|
|
const configs = data?.browseEndpoint?.browseEndpointContextSupportedConfigs?.browseEndpointContextMusicConfig;
|
|
this.browse = {
|
|
id: data?.browseEndpoint?.browseId || null,
|
|
params: data?.browseEndpoint.params || null,
|
|
base_url: data?.browseEndpoint?.canonicalBaseUrl || null,
|
|
page_type: configs?.pageType || null
|
|
};
|
|
}
|
|
if (data?.watchEndpoint) {
|
|
const configs = data?.watchEndpoint?.watchEndpointMusicSupportedConfigs?.watchEndpointMusicConfig;
|
|
this.watch = {
|
|
video_id: data?.watchEndpoint?.videoId,
|
|
playlist_id: data?.watchEndpoint.playlistId || null,
|
|
params: data?.watchEndpoint.params || null,
|
|
index: data?.watchEndpoint.index || null,
|
|
supported_onesie_config: data?.watchEndpoint?.watchEndpointSupportedOnesieConfig,
|
|
music_video_type: configs?.musicVideoType || null
|
|
};
|
|
}
|
|
if (data?.searchEndpoint) {
|
|
this.search = {
|
|
query: data.searchEndpoint.query,
|
|
params: data.searchEndpoint.params
|
|
};
|
|
}
|
|
if (data?.subscribeEndpoint) {
|
|
this.subscribe = {
|
|
channel_ids: data.subscribeEndpoint.channelIds,
|
|
params: data.subscribeEndpoint.params
|
|
};
|
|
}
|
|
if (data?.unsubscribeEndpoint) {
|
|
this.unsubscribe = {
|
|
channel_ids: data.unsubscribeEndpoint.channelIds,
|
|
params: data.unsubscribeEndpoint.params
|
|
};
|
|
}
|
|
if (data?.likeEndpoint) {
|
|
this.like = {
|
|
status: data.likeEndpoint.status,
|
|
target: {
|
|
video_id: data.likeEndpoint.target.videoId,
|
|
playlist_id: data.likeEndpoint.target.playlistId
|
|
},
|
|
params: data.likeEndpoint?.removeLikeParams ||
|
|
data.likeEndpoint?.likeParams ||
|
|
data.likeEndpoint?.dislikeParams
|
|
};
|
|
}
|
|
if (data?.performCommentActionEndpoint) {
|
|
this.perform_comment_action = {
|
|
action: data?.performCommentActionEndpoint.action
|
|
};
|
|
}
|
|
if (data?.offlineVideoEndpoint) {
|
|
this.offline_video = {
|
|
video_id: data.offlineVideoEndpoint.videoId,
|
|
on_add_command: {
|
|
get_download_action: {
|
|
video_id: data.offlineVideoEndpoint.videoId,
|
|
params: data.offlineVideoEndpoint.onAddCommand.getDownloadActionCommand.params
|
|
}
|
|
}
|
|
};
|
|
}
|
|
if (data?.continuationCommand) {
|
|
this.continuation = {
|
|
request: data?.continuationCommand?.request || null,
|
|
token: data?.continuationCommand?.token || null
|
|
};
|
|
}
|
|
if (data?.feedbackEndpoint) {
|
|
this.feedback = {
|
|
token: data.feedbackEndpoint.feedbackToken
|
|
};
|
|
}
|
|
if (data?.watchPlaylistEndpoint) {
|
|
this.watch_playlist = {
|
|
playlist_id: data.watchPlaylistEndpoint?.playlistId
|
|
};
|
|
}
|
|
if (data?.playlistEditEndpoint) {
|
|
this.playlist_edit = {
|
|
playlist_id: data.playlistEditEndpoint.playlistId,
|
|
actions: data.playlistEditEndpoint.actions.map((item: any) => ({
|
|
action: item.action,
|
|
removed_video_id: item.removedVideoId
|
|
}))
|
|
};
|
|
}
|
|
if (data?.addToPlaylistEndpoint) {
|
|
this.add_to_playlist = {
|
|
video_id: data.addToPlaylistEndpoint.videoId
|
|
};
|
|
}
|
|
if (data?.addToPlaylistServiceEndpoint) {
|
|
this.add_to_playlist = {
|
|
video_id: data.addToPlaylistServiceEndpoint.videoId
|
|
};
|
|
}
|
|
if (data?.getReportFormEndpoint) {
|
|
this.get_report_form = {
|
|
params: data.getReportFormEndpoint.params
|
|
};
|
|
}
|
|
if (data?.liveChatItemContextMenuEndpoint) {
|
|
this.live_chat_item_context_menu = {
|
|
params: data?.liveChatItemContextMenuEndpoint?.params
|
|
};
|
|
}
|
|
if (data?.sendLiveChatVoteEndpoint) {
|
|
this.send_live_chat_vote = {
|
|
params: data.sendLiveChatVoteEndpoint.params
|
|
};
|
|
}
|
|
if (data?.liveChatItemContextMenuEndpoint) {
|
|
this.live_chat_item_context_menu = {
|
|
params: data.liveChatItemContextMenuEndpoint.params
|
|
};
|
|
}
|
|
}
|
|
/**
|
|
* Calls the endpoint. (This is an experiment and may replace {@link call} in the future.).
|
|
*/
|
|
async callTest(actions: Actions, args = { parse: true, params: {} }) {
|
|
if (!actions)
|
|
throw new Error('An active caller must be provided');
|
|
if (!this.metadata.api_url)
|
|
throw new Error('Expected an api_url, but none was found, this is a bug.');
|
|
const response = await actions.execute(this.metadata.api_url, { ...this.payload, ...args.params, parse: args.parse });
|
|
return response;
|
|
}
|
|
|
|
// TODO: replace client with an enum or something
|
|
async #call(actions: Actions, client?: string) {
|
|
if (!actions)
|
|
throw new Error('An active caller must be provided');
|
|
if (this.continuation) {
|
|
switch (this.continuation.request) {
|
|
case 'CONTINUATION_REQUEST_TYPE_BROWSE': {
|
|
return await actions.browse(this.continuation.token, { is_ctoken: true });
|
|
}
|
|
case 'CONTINUATION_REQUEST_TYPE_SEARCH': {
|
|
return await actions.search({ ctoken: this.continuation.token });
|
|
}
|
|
case 'CONTINUATION_REQUEST_TYPE_WATCH_NEXT': {
|
|
return await actions.next({ ctoken: this.continuation.token });
|
|
}
|
|
default:
|
|
throw new Error(`${this.continuation.request} not implemented`);
|
|
}
|
|
}
|
|
if (this.search) {
|
|
return await actions.search({ query: this.search.query, params: this.search.params, client });
|
|
}
|
|
if (this.browse) {
|
|
return await actions.browse(this.browse.id, { ...this.browse, client });
|
|
}
|
|
if (this.like) {
|
|
if (!this.metadata.api_url)
|
|
throw new Error('Like endpoint requires an api_url, but was not parsed from the response.');
|
|
const response = await actions.engage(this.metadata.api_url, { video_id: this.like.target.video_id, params: this.like.params });
|
|
return response;
|
|
}
|
|
}
|
|
|
|
async call(actions: Actions, client: string | undefined, parse: true) : Promise<ParsedResponse | undefined>;
|
|
async call(actions: Actions, client?: string, parse?: false) : Promise<ActionsResponse | undefined>;
|
|
async call(actions: Actions, client?: string, parse?: boolean): Promise<ParsedResponse | ActionsResponse | undefined> {
|
|
const result = await this.#call(actions, client);
|
|
if (parse && result)
|
|
return Parser.parseResponse(result.data);
|
|
return this.#call(actions, client);
|
|
}
|
|
}
|
|
export default NavigationEndpoint;
|