Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
sync upstream (api@0a7cf75)
  • Loading branch information
Sticks committed Mar 22, 2025
1 parent 918fdaa commit 9d629ce
Show file tree
Hide file tree
Showing 11 changed files with 203 additions and 118 deletions.
5 changes: 2 additions & 3 deletions api/package.json
@@ -1,7 +1,7 @@
{
"name": "@imput/cobalt-api",
"description": "save what you love",
"version": "10.7.7",
"version": "10.8.2",
"author": "imput",
"exports": "./src/cobalt.js",
"type": "module",
Expand All @@ -11,7 +11,6 @@
"scripts": {
"start": "node src/cobalt",
"test": "node src/util/test",
"token:youtube": "node src/util/generate-youtube-tokens",
"token:jwt": "node src/util/generate-jwt-secret"
},
"repository": {
Expand Down Expand Up @@ -39,7 +38,7 @@
"set-cookie-parser": "2.6.0",
"undici": "^5.19.1",
"url-pattern": "1.0.3",
"youtubei.js": "^13.1.0",
"youtubei.js": "^13.2.0",
"zod": "^3.23.8"
},
"optionalDependencies": {
Expand Down
12 changes: 12 additions & 0 deletions api/src/config.js
@@ -1,3 +1,4 @@
import { Constants } from "youtubei.js";
import { getVersion } from "@imput/version-info";
import { services } from "./processing/service-config.js";
import { supportsReusePort } from "./misc/cluster.js";
Expand Down Expand Up @@ -52,6 +53,11 @@ const env = {
keyReloadInterval: 900,

enabledServices,

customInnertubeClient: process.env.CUSTOM_INNERTUBE_CLIENT,
ytSessionServer: process.env.YOUTUBE_SESSION_SERVER,
ytSessionReloadInterval: 300,
ytSessionInnertubeClient: process.env.YOUTUBE_SESSION_INNERTUBE_CLIENT,
}

const genericUserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36";
Expand All @@ -74,6 +80,12 @@ if (env.instanceCount > 1 && !env.redisURL) {
throw new Error('SO_REUSEPORT is not supported');
}

if (env.customInnertubeClient && !Constants.SUPPORTED_CLIENTS.includes(env.customInnertubeClient)) {
console.error("CUSTOM_INNERTUBE_CLIENT is invalid. Provided client is not supported.");
console.error(`Supported clients are: ${Constants.SUPPORTED_CLIENTS.join(', ')}\n`);
throw new Error("Invalid CUSTOM_INNERTUBE_CLIENT");
}

export {
env,
genericUserAgent,
Expand Down
8 changes: 7 additions & 1 deletion api/src/core/api.js
Expand Up @@ -18,8 +18,10 @@ import { verifyTurnstileToken } from "../security/turnstile.js";
import { friendlyServiceName } from "../processing/service-alias.js";
import { verifyStream, getInternalStream } from "../stream/manage.js";
import { createResponse, normalizeRequest, getIP } from "../processing/request.js";

import * as APIKeys from "../security/api-keys.js";
import * as Cookies from "../processing/cookie/manager.js";
import * as YouTubeSession from "../processing/helpers/youtube-session.js";

const git = {
branch: await getBranch(),
Expand Down Expand Up @@ -354,7 +356,7 @@ export const runAPI = async (express, app, __dirname, isPrimary = true) => {
}, () => {
if (isPrimary) {
console.log(`\n` +
Bright(Cyan("cobalt ")) + Bright("API ^ω^") + "\n" +
Bright(Cyan("cobalt ")) + Bright("API ^ω^") + "\n" +

"~~~~~~\n" +
Bright("version: ") + version + "\n" +
Expand All @@ -376,6 +378,10 @@ export const runAPI = async (express, app, __dirname, isPrimary = true) => {
if (env.cookiePath) {
Cookies.setup(env.cookiePath);
}

if (env.ytSessionServer) {
YouTubeSession.setup();
}
});

if (isCluster) {
Expand Down
1 change: 0 additions & 1 deletion api/src/processing/cookie/manager.js
Expand Up @@ -13,7 +13,6 @@ const VALID_SERVICES = new Set([
'reddit',
'twitter',
'youtube',
'youtube_oauth'
]);

const invalidCookies = {};
Expand Down
2 changes: 1 addition & 1 deletion api/src/processing/match.js
Expand Up @@ -109,7 +109,7 @@ export default async function({ host, patternMatch, params }) {
}

if (url.hostname === "music.youtube.com" || isAudioOnly) {
fetchInfo.quality = "max";
fetchInfo.quality = "1080";
fetchInfo.format = "vp9";
fetchInfo.isAudioOnly = true;
fetchInfo.isAudioMuted = false;
Expand Down
2 changes: 1 addition & 1 deletion api/src/processing/services/pinterest.js
Expand Up @@ -23,7 +23,7 @@ export default async function(o) {

const videoLink = [...html.matchAll(videoRegex)]
.map(([, link]) => link)
.find(a => a.endsWith('.mp4') && a.includes('720p'));
.find(a => a.endsWith('.mp4'));

if (videoLink) return {
urls: videoLink,
Expand Down
130 changes: 102 additions & 28 deletions api/src/processing/services/twitter.js
Expand Up @@ -24,6 +24,11 @@ const badContainerEnd = new Date(1702605600000);

function needsFixing(media) {
const representativeId = media.source_status_id_str ?? media.id_str;

// syndication api doesn't have media ids in its response,
// so we just assume it's all good
if (!representativeId) return false;

const mediaTimestamp = new Date(
Number((BigInt(representativeId) >> 22n) + TWITTER_EPOCH)
);
Expand Down Expand Up @@ -53,6 +58,25 @@ const getGuestToken = async (dispatcher, forceReload = false) => {
}
}

const requestSyndication = async(dispatcher, tweetId) => {
// thank you
// https://github.com/yt-dlp/yt-dlp/blob/05c8023a27dd37c49163c0498bf98e3e3c1cb4b9/yt_dlp/extractor/twitter.py#L1334
const token = (id) => ((Number(id) / 1e15) * Math.PI).toString(36).replace(/(0+|\.)/g, '');
const syndicationUrl = new URL("https://cdn.syndication.twimg.com/tweet-result");

syndicationUrl.searchParams.set("id", tweetId);
syndicationUrl.searchParams.set("token", token(tweetId));

const result = await fetch(syndicationUrl, {
headers: {
"user-agent": genericUserAgent
},
dispatcher
});

return result;
}

const requestTweet = async(dispatcher, tweetId, token, cookie) => {
const graphqlTweetURL = new URL(graphqlURL);

Expand Down Expand Up @@ -87,36 +111,24 @@ const requestTweet = async(dispatcher, tweetId, token, cookie) => {
let result = await fetch(graphqlTweetURL, { headers, dispatcher });
updateCookie(cookie, result.headers);

// we might have been missing the `ct0` cookie, retry
// we might have been missing the ct0 cookie, retry
if (result.status === 403 && result.headers.get('set-cookie')) {
result = await fetch(graphqlTweetURL, {
headers: {
...headers,
'x-csrf-token': cookie.values().ct0
},
dispatcher
});
const cookieValues = cookie?.values();
if (cookieValues?.ct0) {
result = await fetch(graphqlTweetURL, {
headers: {
...headers,
'x-csrf-token': cookieValues.ct0
},
dispatcher
});
}
}

return result
}

export default async function({ id, index, toGif, dispatcher, alwaysProxy }) {
const cookie = await getCookie('twitter');

let guestToken = await getGuestToken(dispatcher);
if (!guestToken) return { error: "fetch.fail" };

let tweet = await requestTweet(dispatcher, id, guestToken);

// get new token & retry if old one expired
if ([403, 429].includes(tweet.status)) {
guestToken = await getGuestToken(dispatcher, true);
tweet = await requestTweet(dispatcher, id, guestToken)
}

tweet = await tweet.json();

const extractGraphqlMedia = async (tweet, dispatcher, id, guestToken, cookie) => {
let tweetTypename = tweet?.data?.tweetResult?.result?.__typename;

if (!tweetTypename) {
Expand All @@ -127,13 +139,13 @@ export default async function({ id, index, toGif, dispatcher, alwaysProxy }) {
const reason = tweet?.data?.tweetResult?.result?.reason;
switch(reason) {
case "Protected":
return { error: "content.post.private" }
return { error: "content.post.private" };
case "NsfwLoggedOut":
if (cookie) {
tweet = await requestTweet(dispatcher, id, guestToken, cookie);
tweet = await tweet.json();
tweetTypename = tweet?.data?.tweetResult?.result?.__typename;
} else return { error: "content.post.age" }
} else return { error: "content.post.age" };
}
}

Expand All @@ -150,7 +162,69 @@ export default async function({ id, index, toGif, dispatcher, alwaysProxy }) {
repostedTweet = baseTweet?.retweeted_status_result?.result.tweet.legacy.extended_entities;
}

let media = (repostedTweet?.media || baseTweet?.extended_entities?.media);
return (repostedTweet?.media || baseTweet?.extended_entities?.media);
}

const testResponse = (result) => {
const contentLength = result.headers.get("content-length");

if (!contentLength || contentLength === '0') {
return false;
}

if (!result.headers.get("content-type").startsWith("application/json")) {
return false;
}

return true;
}

export default async function({ id, index, toGif, dispatcher, alwaysProxy }) {
const cookie = await getCookie('twitter');

let syndication = false;

let guestToken = await getGuestToken(dispatcher);
if (!guestToken) return { error: "fetch.fail" };

// for now we assume that graphql api will come back after some time,
// so we try it first

let tweet = await requestTweet(dispatcher, id, guestToken);

// get new token & retry if old one expired
if ([403, 429].includes(tweet.status)) {
guestToken = await getGuestToken(dispatcher, true);
if (cookie) {
tweet = await requestTweet(dispatcher, id, guestToken, cookie);
} else {
tweet = await requestTweet(dispatcher, id, guestToken);
}
}

const testGraphql = testResponse(tweet);

// if graphql requests fail, then resort to tweet embed api
if (!testGraphql) {
syndication = true;
tweet = await requestSyndication(dispatcher, id);

const testSyndication = testResponse(tweet);

// if even syndication request failed, then cry out loud
if (!testSyndication) {
return { error: "fetch.fail" };
}
}

tweet = await tweet.json();

let media =
syndication
? tweet.mediaDetails
: await extractGraphqlMedia(tweet, dispatcher, id, guestToken, cookie);

if (!media) return { error: "fetch.empty" };

// check if there's a video at given index (/video/<index>)
if (index >= 0 && index < media?.length) {
Expand All @@ -163,7 +237,7 @@ export default async function({ id, index, toGif, dispatcher, alwaysProxy }) {
service: "twitter",
type: "proxy",
url, filename,
})
});

switch (media?.length) {
case undefined:
Expand Down

0 comments on commit 9d629ce

Please sign in to comment.