0
0
Fork 0
mirror of https://github.com/VueTubeApp/VueTube synced 2024-11-25 04:35:17 +00:00
VueTube/NUXT/plugins/innertube.js

459 lines
14 KiB
JavaScript
Raw Normal View History

//⚠️🚧 WARNING: THIS FILE IS IN MAINTENANCE MODE 🚧⚠️
// DO NOT ADD NEW FEATURES TO THIS FILE. INNERTUBE.JS IS NOW A SEPARATE LIBRARY
// contribute to the library here: https://github.com/VueTubeApp/Vuetube-Extractor
// Code specific to working with the innertube API
// https://www.youtube.com/youtubei/v1
2022-03-21 23:47:11 +00:00
import { Http } from "@capacitor-community/http";
2022-04-09 03:06:35 +00:00
import { getBetweenStrings, delay } from "./utils";
import rendererUtils from "./renderers";
2022-03-21 23:47:11 +00:00
import constants from "./constants";
class Innertube {
2022-03-21 23:47:11 +00:00
//--- Initiation ---//
constructor(ErrorCallback) {
this.ErrorCallback = ErrorCallback || undefined;
this.retry_count = 0;
}
checkErrorCallback() {
return typeof this.ErrorCallback === "function";
}
async initAsync() {
const html = await Http.get({
url: constants.URLS.YT_URL,
params: { hl: "en" },
}).catch((error) => error);
try {
if (html instanceof Error && this.checkErrorCallback)
this.ErrorCallback(html.message, true);
try {
const data = JSON.parse(
2023-05-03 12:03:39 +00:00
"{" + getBetweenStrings(html.data, "ytcfg.set({", ");")
2022-03-21 23:47:11 +00:00
);
if (data.INNERTUBE_CONTEXT) {
this.key = data.INNERTUBE_API_KEY;
this.context = data.INNERTUBE_CONTEXT;
this.logged_in = data.LOGGED_IN;
this.context.client = constants.INNERTUBE_CLIENT(this.context.client);
this.header = constants.INNERTUBE_HEADER(this.context.client);
}
2022-03-21 23:47:11 +00:00
} catch (err) {
console.log(err);
2022-04-09 02:28:08 +00:00
if (this.checkErrorCallback) {
this.ErrorCallback(html.data, true);
this.ErrorCallback(err, true);
}
2022-04-09 03:06:35 +00:00
if (this.retry_count < 10) {
this.retry_count += 1;
if (this.checkErrorCallback)
this.ErrorCallback(
`retry count: ${this.retry_count}`,
false,
`An error occurred while trying to init the innertube API. Retrial number: ${this.retry_count}/10`
);
await delay(5000);
await this.initAsync();
2022-03-21 23:47:11 +00:00
} else {
if (this.checkErrorCallback)
2022-04-09 03:06:35 +00:00
this.ErrorCallback(
"Failed to retrieve Innertube session",
true,
"An error occurred while retrieving the innertube session. Check the Logs for more information."
);
}
2022-03-21 23:47:11 +00:00
}
} catch (error) {
this.ErrorCallback(error, true);
}
2022-03-21 23:47:11 +00:00
}
static async createAsync(ErrorCallback) {
const created = new Innertube(ErrorCallback);
await created.initAsync();
return created;
}
//--- API Calls ---//
2022-05-05 03:57:37 +00:00
async browseAsync(action_type, args = {}) {
2023-05-03 12:03:39 +00:00
let data = {
context: {
client: constants.INNERTUBE_CLIENT(this.context.client),
},
};
2022-03-21 23:47:11 +00:00
switch (action_type) {
case "recommendations":
2022-05-04 05:21:14 +00:00
args.browseId = "FEwhat_to_watch";
2022-03-21 23:47:11 +00:00
break;
case "playlist":
2022-05-04 05:21:14 +00:00
case "channel":
if (args && args.browseId) {
break;
} else {
throw new ReferenceError("No browseId provided");
}
2022-03-21 23:47:11 +00:00
default:
}
2022-05-05 03:57:37 +00:00
data = { ...data, ...args };
2022-03-21 23:47:11 +00:00
console.log(data);
const response = await Http.post({
url: `${constants.URLS.YT_BASE_API}/browse?key=${this.key}`,
data: data,
headers: { "Content-Type": "application/json" },
}).catch((error) => error);
if (response instanceof Error)
return {
success: false,
status_code: response.status,
message: response.message,
};
return {
success: true,
status_code: response.status,
data: response.data,
};
}
async getContinuationsAsync(continuation, type, contextAdditional = {}) {
let data = {
context: { ...this.context, ...contextAdditional },
continuation: continuation,
};
let url;
2022-04-19 14:03:46 +00:00
switch (type.toLowerCase()) {
2022-03-31 02:22:22 +00:00
case "browse":
url = `${constants.URLS.YT_BASE_API}/browse?key=${this.key}`;
break;
2022-03-31 02:22:22 +00:00
case "search":
url = `${constants.URLS.YT_BASE_API}/search?key=${this.key}`;
break;
2022-03-31 02:22:22 +00:00
case "next":
url = `${constants.URLS.YT_BASE_API}/next?key=${this.key}`;
break;
2022-03-31 02:22:22 +00:00
default:
throw "Invalid type";
2022-03-31 02:22:22 +00:00
}
const response = await Http.post({
url: url,
data: data,
headers: { "Content-Type": "application/json" },
}).catch((error) => error);
if (response instanceof Error) {
return {
success: false,
status_code: response.status,
message: response.message,
};
}
return {
success: true,
status_code: response.status,
data: response.data,
};
}
2022-03-21 23:47:11 +00:00
async getVidAsync(id) {
2023-05-03 12:03:39 +00:00
let data = {
context: {
client: constants.INNERTUBE_VIDEO(this.context.client),
},
videoId: id,
};
const responseNext = await Http.post({
url: `${constants.URLS.YT_BASE_API}/next?key=${this.key}`,
data: {
...data,
...{
context: {
client: {
clientName: constants.YT_API_VALUES.CLIENT_WEB_M,
clientVersion: constants.YT_API_VALUES.VERSION_WEB,
},
},
},
},
headers: constants.INNERTUBE_HEADER(this.context.client),
2022-03-21 23:47:11 +00:00
}).catch((error) => error);
const response = await Http.post({
2022-03-21 23:47:11 +00:00
url: `${constants.URLS.YT_BASE_API}/player?key=${this.key}`,
2023-05-03 12:03:39 +00:00
data: {
...data,
...{
contentCheckOk: false,
mwebCapabilities: {
mobileClientSupportsLivestream: true,
},
playbackContext: {
contentPlaybackContext: {
currentUrl: "/watch?v=" + id,
vis: 0,
splay: false,
autoCaptionsDefaultOn: false,
autonavState: "STATE_NONE",
html5Preference: "HTML5_PREF_WANTS",
signatureTimestamp: 19473,
referer: "https://m.youtube.com/",
lactMilliseconds: "-1",
watchAmbientModeContext: {
watchAmbientModeEnabled: true,
},
},
},
},
},
// headers: constants.INNERTUBE_HEADER(this.context.client),
headers: constants.INNERTUBE_NEW_HEADER(this.context.client),
2022-03-21 23:47:11 +00:00
}).catch((error) => error);
if (response.error)
2022-03-21 23:47:11 +00:00
return {
success: false,
status_code: response.status,
2022-03-21 23:47:11 +00:00
message: response.message,
};
else if (responseNext.error)
return {
success: false,
status_code: responseNext.status,
message: responseNext.message,
};
2022-03-21 23:47:11 +00:00
return {
success: true,
status_code: response.status,
data: { output: response.data, outputNext: responseNext.data },
2022-03-21 23:47:11 +00:00
};
}
async searchAsync(query) {
let data = { context: this.context, query: query };
const response = await Http.post({
url: `${constants.URLS.YT_BASE_API}/search?key=${this.key}`,
data: data,
headers: { "Content-Type": "application/json" },
}).catch((error) => error);
if (response instanceof Error)
return {
success: false,
status_code: response.status,
message: response.message,
};
return {
success: true,
status_code: response.status,
data: response.data,
};
}
2022-05-04 05:21:14 +00:00
async getEndPoint(url) {
let data = { context: this.context, url: url };
const response = await Http.post({
url: `${constants.URLS.YT_BASE_API}/navigation/resolve_url?key=${this.key}`,
data: data,
headers: { "Content-Type": "application/json" },
}).catch((error) => error);
if (response instanceof Error)
return {
success: false,
status_code: response.status,
message: response.message,
};
return {
success: true,
status_code: response.status,
data: response.data,
};
}
// WARNING: This is tracking the user's activity, but is required for recommendations to properly work
async apiStats(params, url) {
console.log(params);
await Http.get({
url: url,
params: {
...params,
...{
ver: 2,
c: constants.YT_API_VALUES.CLIENTNAME.toLowerCase(),
cbrver: constants.YT_API_VALUES.VERSION,
cver: constants.YT_API_VALUES.VERSION,
},
},
headers: this.header,
});
}
// Static methods
static getThumbnail(id, resolution) {
if (resolution == "max") {
const url = `https://img.youtube.com/vi/${id}/maxresdefault.jpg`;
let img = new Image();
img.src = url;
img.onload = function () {
if (img.height !== 120) return url;
};
}
return `https://img.youtube.com/vi/${id}/mqdefault.jpg`;
}
2022-03-21 23:47:11 +00:00
// Simple Wrappers
async getRecommendationsAsync() {
const rec = await this.browseAsync("recommendations");
return rec;
}
2022-05-04 05:21:14 +00:00
async getChannelAsync(url) {
const channelEndpoint = await this.getEndPoint(url);
if (
channelEndpoint.success &&
channelEndpoint.data.endpoint?.browseEndpoint
) {
return await this.browseAsync(
"channel",
channelEndpoint.data.endpoint?.browseEndpoint
);
} else {
throw new ReferenceError("Cannot find channel");
}
}
2022-03-21 23:47:11 +00:00
async VidInfoAsync(id) {
let response = await this.getVidAsync(id);
if (
response.success == false ||
response.data.output?.playabilityStatus?.status == ("ERROR" || undefined)
2022-03-21 23:47:11 +00:00
)
throw new Error(
`Could not get information for video: ${
response.status_code ||
response.data.output?.playabilityStatus?.status
} - ${
response.message || response.data.output?.playabilityStatus?.reason
}`
2022-03-21 23:47:11 +00:00
);
const responseInfo = response.data.output;
const responseNext = response.data.outputNext;
const details = responseInfo.videoDetails;
// const columnUI =
// responseInfo[3].response?.contents.singleColumnWatchNextResults?.results
// ?.results;
const resolutions = responseInfo.streamingData;
const columnUI =
responseNext.contents.singleColumnWatchNextResults.results.results;
const vidMetadata = columnUI.contents.find(
(content) => content.slimVideoMetadataSectionRenderer
).slimVideoMetadataSectionRenderer;
const recommendations = columnUI?.contents.find(
2022-04-09 06:20:51 +00:00
(content) => content?.itemSectionRenderer?.targetId == "watch-next-feed"
).itemSectionRenderer;
const ownerData = vidMetadata.contents.find(
(content) => content.slimOwnerRenderer
)?.slimOwnerRenderer;
const vidData = {
2022-03-21 23:47:11 +00:00
id: details.videoId,
title: details.title,
isLive: details.isLiveContent,
channelName: details.author,
channelSubs: ownerData?.collapsedSubtitle?.runs[0]?.text,
2022-05-02 04:42:56 +00:00
channelUrl: rendererUtils.getNavigationEndpoints(
ownerData.navigationEndpoint
),
channelImg: ownerData?.thumbnail?.thumbnails[0].url,
2022-03-21 23:47:11 +00:00
availableResolutions: resolutions?.formats,
availableResolutionsAdaptive: resolutions?.adaptiveFormats,
metadata: {
contents: vidMetadata.contents,
description: details.shortDescription,
thumbnails: details.thumbnails?.thumbnails,
2022-03-21 23:47:11 +00:00
isPrivate: details.isPrivate,
viewCount: details.viewCount,
lengthSeconds: details.lengthSeconds,
2023-01-06 19:12:55 +00:00
// likes: parseInt(
// vidMetadata.contents
// .find((content) => content.slimVideoActionBarRenderer)
// .slimVideoActionBarRenderer.buttons.find(
// (button) => button.slimMetadataToggleButtonRenderer.isLike
// )
// .slimMetadataToggleButtonRenderer.button.toggleButtonRenderer.defaultText.accessibility.accessibilityData.label.replace(
// /\D/g,
// ""
// )
// ), // Yes. I know.
likes: "broken",
// NOTE: likes are pulled from RYD for now untill extractor is fixed
2022-03-21 23:47:11 +00:00
},
renderedData: {
description: responseNext.engagementPanels
.find(
(panel) =>
panel.engagementPanelSectionListRenderer.panelIdentifier ==
"video-description-ep-identifier"
)
.engagementPanelSectionListRenderer.content.structuredDescriptionContentRenderer.items.find(
(item) => item.expandableVideoDescriptionBodyRenderer
).expandableVideoDescriptionBodyRenderer,
recommendations: recommendations,
recommendationsContinuation:
recommendations.contents[recommendations.contents.length - 1]
.continuationItemRenderer?.continuationEndpoint.continuationCommand
.token,
2022-03-21 23:47:11 +00:00
},
engagementPanels: responseNext.engagementPanels,
2022-04-09 06:20:51 +00:00
commentData: columnUI.contents
.find((content) => content.itemSectionRenderer?.contents)
?.itemSectionRenderer.contents.find(
(content) => content.commentsEntryPointHeaderRenderer
)?.commentsEntryPointHeaderRenderer,
playbackTracking: responseInfo.playbackTracking,
2022-04-19 14:03:46 +00:00
commentContinuation: responseNext.engagementPanels
.find(
(panel) =>
panel.engagementPanelSectionListRenderer.panelIdentifier ==
"engagement-panel-comments-section"
)
?.engagementPanelSectionListRenderer.content.sectionListRenderer.contents.find(
(content) => content.itemSectionRenderer
)
?.itemSectionRenderer.contents.find(
(content) => content.continuationItemRenderer
)?.continuationItemRenderer.continuationEndpoint.continuationCommand
.token,
2022-03-21 23:47:11 +00:00
};
return vidData;
2022-03-21 23:47:11 +00:00
}
async getSearchAsync(query) {
const search = await this.searchAsync(query);
if (search.success == false)
throw new Error(
`Could not get search results: ${search.status_code} - ${search.message}`
);
console.log(search.data);
2022-03-24 11:47:13 +00:00
return search.data;
}
}
2022-03-21 23:47:11 +00:00
export default Innertube;