diff --git a/.deepsource.toml b/.deepsource.toml
index e8c615cb..d6a19e95 100644
--- a/.deepsource.toml
+++ b/.deepsource.toml
@@ -1,5 +1,11 @@
version = 1
+test_patterns = [
+ "src/test/test.js"
+]
+
[[analyzers]]
name = "javascript"
-enabled = true
\ No newline at end of file
+enabled = true
+ [analyzers.meta]
+ environment = ["nodejs"]
diff --git a/.gitignore b/.gitignore
index 01d4a695..97df23a8 100644
--- a/.gitignore
+++ b/.gitignore
@@ -6,4 +6,4 @@ package-lock.json
.env
# esbuild
-min
\ No newline at end of file
+min
diff --git a/package.json b/package.json
index f0772552..791b8a41 100644
--- a/package.json
+++ b/package.json
@@ -1,7 +1,7 @@
{
"name": "cobalt",
"description": "save what you love",
- "version": "4.8",
+ "version": "5.0",
"author": "wukko",
"exports": "./src/cobalt.js",
"type": "module",
@@ -10,7 +10,8 @@
},
"scripts": {
"start": "node src/cobalt",
- "setup": "node src/modules/setup"
+ "setup": "node src/modules/setup",
+ "test": "node src/test/test"
},
"repository": {
"type": "git",
diff --git a/src/cobalt.js b/src/cobalt.js
index 74004f91..f21835a4 100644
--- a/src/cobalt.js
+++ b/src/cobalt.js
@@ -1,12 +1,12 @@
-import "dotenv/config"
+import "dotenv/config";
import express from "express";
import cors from "cors";
import * as fs from "fs";
import rateLimit from "express-rate-limit";
-import { shortCommit } from "./modules/sub/currentCommit.js";
-import { appName, genericUserAgent, version, internetExplorerRedirect } from "./modules/config.js";
+import { getCurrentBranch, shortCommit } from "./modules/sub/currentCommit.js";
+import { appName, genericUserAgent, version } from "./modules/config.js";
import { getJSON } from "./modules/api.js";
import renderPage from "./modules/pageRender/page.js";
import { apiJSON, checkJSONPost, languageCode } from "./modules/sub/utils.js";
@@ -18,13 +18,14 @@ import { changelogHistory } from "./modules/pageRender/onDemand.js";
import { sha256 } from "./modules/sub/crypto.js";
const commitHash = shortCommit();
+const branch = getCurrentBranch();
const app = express();
app.disable('x-powered-by');
if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt && process.env.port) {
const apiLimiter = rateLimit({
- windowMs: 1 * 60 * 1000,
+ windowMs: 60000,
max: 12,
standardHeaders: true,
legacyHeaders: false,
@@ -33,7 +34,7 @@ if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt &&
}
});
const apiLimiterStream = rateLimit({
- windowMs: 1 * 60 * 1000,
+ windowMs: 60000,
max: 12,
standardHeaders: true,
legacyHeaders: false,
@@ -57,18 +58,25 @@ if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt &&
}
next();
});
+ app.use((req, res, next) => {
+ if (req.header("user-agent") && req.header("user-agent").includes("Trident")) {
+ res.destroy()
+ }
+ next();
+ });
app.use('/api/json', express.json({
verify: (req, res, buf) => {
try {
JSON.parse(buf);
if (buf.length > 720) throw new Error();
- if (req.header('Content-Type') != "application/json") res.status(500).json({ 'status': 'error', 'text': 'invalid content type header' })
- if (req.header('Accept') != "application/json") res.status(500).json({ 'status': 'error', 'text': 'invalid accept header' })
+ if (String(req.header('Content-Type')) !== "application/json") res.status(500).json({ 'status': 'error', 'text': 'invalid content type header' })
+ if (String(req.header('Accept')) !== "application/json") res.status(500).json({ 'status': 'error', 'text': 'invalid accept header' })
} catch(e) {
res.status(500).json({ 'status': 'error', 'text': 'invalid json body.' })
}
}
}));
+
app.post('/api/:type', cors({ origin: process.env.selfURL, optionsSuccessStatus: 200 }), async (req, res) => {
try {
let ip = sha256(req.header('x-forwarded-for') ? req.header('x-forwarded-for') : req.ip.replace('::ffff:', ''), process.env.streamSalt);
@@ -82,10 +90,10 @@ if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt &&
let j = await getJSON(chck["url"], languageCode(req), chck)
res.status(j.status).json(j.body);
} else if (request.url && !chck) {
- let j = apiJSON(3, { t: loc(languageCode(req), 'ErrorCouldntFetch') });
+ let j = apiJSON(0, { t: loc(languageCode(req), 'ErrorCouldntFetch') });
res.status(j.status).json(j.body);
} else {
- let j = apiJSON(3, { t: loc(languageCode(req), 'ErrorNoLink') })
+ let j = apiJSON(0, { t: loc(languageCode(req), 'ErrorNoLink') })
res.status(j.status).json(j.body);
}
} catch (e) {
@@ -101,12 +109,16 @@ if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt &&
res.status(500).json({ 'status': 'error', 'text': loc(languageCode(req), 'ErrorCantProcess') })
}
});
+
app.get('/api/:type', cors({ origin: process.env.selfURL, optionsSuccessStatus: 200 }), (req, res) => {
try {
let ip = sha256(req.header('x-forwarded-for') ? req.header('x-forwarded-for') : req.ip.replace('::ffff:', ''), process.env.streamSalt);
switch (req.params.type) {
case 'json':
- res.status(405).json({ 'status': 'error', 'text': 'GET method for this request has been deprecated. see https://github.com/wukko/cobalt/blob/current/docs/API.md for up-to-date API documentation.' });
+ res.status(405).json({
+ 'status': 'error',
+ 'text': 'GET method for this endpoint has been deprecated. see https://github.com/wukko/cobalt/blob/current/docs/API.md for up-to-date API documentation.'
+ });
break;
case 'stream':
if (req.query.p) {
@@ -146,24 +158,18 @@ if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt &&
res.status(500).json({ 'status': 'error', 'text': loc(languageCode(req), 'ErrorCantProcess') })
}
});
+
app.get("/api", (req, res) => {
res.redirect('/api/json')
});
app.get("/", (req, res) => {
- if (req.header("user-agent") && req.header("user-agent").includes("Trident")) {
- if (internetExplorerRedirect.newNT.includes(req.header("user-agent").split('NT ')[1].split(';')[0])) {
- res.redirect(internetExplorerRedirect.new)
- } else {
- res.redirect(internetExplorerRedirect.old)
- }
- } else {
- res.send(renderPage({
- "hash": commitHash,
- "type": "default",
- "lang": languageCode(req),
- "useragent": req.header('user-agent') ? req.header('user-agent') : genericUserAgent
- }))
- }
+ res.send(renderPage({
+ "hash": commitHash,
+ "type": "default",
+ "lang": languageCode(req),
+ "useragent": req.header('user-agent') ? req.header('user-agent') : genericUserAgent,
+ "branch": branch
+ }))
});
app.get("/favicon.ico", (req, res) => {
res.redirect('/icons/favicon.ico');
@@ -171,9 +177,10 @@ if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt &&
app.get("/*", (req, res) => {
res.redirect('/')
});
+
app.listen(process.env.port, () => {
let startTime = new Date();
- console.log(`\n${Cyan(appName)} ${Bright(`v.${version}-${commitHash}`)}\nStart time: ${Bright(`${startTime.toUTCString()} (${Math.floor(new Date().getTime())})`)}\n\nURL: ${Cyan(`${process.env.selfURL}`)}\nPort: ${process.env.port}\n`)
+ console.log(`\n${Cyan(appName)} ${Bright(`v.${version}-${commitHash} (${branch})`)}\nStart time: ${Bright(`${startTime.toUTCString()} (${Math.floor(new Date().getTime())})`)}\n\nURL: ${Cyan(`${process.env.selfURL}`)}\nPort: ${process.env.port}\n`)
});
} else {
console.log(Red(`cobalt hasn't been configured yet or configuration is invalid.\n`) + Bright(`please run the setup script to fix this: `) + Green(`npm run setup`))
diff --git a/src/config.json b/src/config.json
index 4ce8c81f..fe59d601 100644
--- a/src/config.json
+++ b/src/config.json
@@ -2,7 +2,7 @@
"streamLifespan": 120000,
"maxVideoDuration": 7500000,
"maxAudioDuration": 7500000,
- "genericUserAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 Safari/537.36",
+ "genericUserAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36",
"authorInfo": {
"name": "wukko",
"link": "https://wukko.me/",
@@ -18,11 +18,6 @@
}
}
},
- "internetExplorerRedirect": {
- "newNT": ["6.1", "6.2", "6.3", "10.0"],
- "old": "https://mypal-browser.org/",
- "new": "https://www.mozilla.org/firefox/new/"
- },
"donations": {
"crypto": {
"bitcoin": "bc1q59jyyjvrzj4c22rkk3ljeecq6jmpyscgz9spnd",
diff --git a/src/front/cobalt.css b/src/front/cobalt.css
index eb10b61c..88c5b133 100644
--- a/src/front/cobalt.css
+++ b/src/front/cobalt.css
@@ -278,7 +278,7 @@ input[type="checkbox"] {
.italic {
font-style: italic;
}
-.social-link {
+.cobalt-support-link {
display: flex;
flex-direction: row;
justify-content: flex-start;
diff --git a/src/front/updateBanners/valentines.webp b/src/front/updateBanners/valentines.webp
new file mode 100644
index 00000000..8e18cd1b
Binary files /dev/null and b/src/front/updateBanners/valentines.webp differ
diff --git a/src/localization/languages/en.json b/src/localization/languages/en.json
index 3e70c30b..384c1695 100644
--- a/src/localization/languages/en.json
+++ b/src/localization/languages/en.json
@@ -25,7 +25,7 @@
"ErrorBrokenLink": "{s} is supported, but something is wrong with your link. maybe you didn't copy it fully?",
"ErrorNoLink": "i can't guess what you want to download! please give me a link.",
"ErrorPageRenderFail": "something went wrong and page couldn't render. if it's a recurring or critical issue, please {ContactLink}. it'd be useful if you provided current commit hash ({s}) and error recreation steps. thank you in advance :D",
- "ErrorRateLimit": "you're making too many requests. calm down and try again in a bit.",
+ "ErrorRateLimit": "you're making too many requests. try again in a minute!",
"ErrorCouldntFetch": "couldn't get any info about your link. check if it's correct and try again.",
"ErrorLengthLimit": "current length limit is {s} minutes. video that you tried to download is longer than {s} minutes. pick something else!",
"ErrorBadFetch": "an error occurred when i tried to get info about your link. are you sure it works? check if it does, and try again.",
diff --git a/src/localization/languages/ru.json b/src/localization/languages/ru.json
index e09b016e..2a312065 100644
--- a/src/localization/languages/ru.json
+++ b/src/localization/languages/ru.json
@@ -7,7 +7,7 @@
"LinkInput": "вставь ссылку сюда",
"AboutSummary": "{appName} — твой друг при скачивании контента из соц. сетей. никакой рекламы или трекеров. вставляешь ссылку и получаешь файл. ничего лишнего.",
"EmbedBriefDescription": "сохраняй что хочешь, без мороки и вторжения в личное пространство",
- "MadeWithLove": "сделано с <3 wukko",
+ "MadeWithLove": "сделано wukko, с <3",
"AccessibilityInputArea": "зона вставки ссылки",
"AccessibilityOpenAbout": "открыть окно с инфой",
"AccessibilityDownloadButton": "кнопка скачивания",
@@ -23,9 +23,9 @@
"ErrorSomethingWentWrong": "что-то пошло совсем не так, и у меня не получилось ничего для тебя достать. ты можешь попробовать ещё раз, но если так и не получится, {ContactLink}.",
"ErrorUnsupported": "с твоей ссылкой что-то не так, или же этот сервис ещё не поддерживается. может быть, ты вставил не ту ссылку?",
"ErrorBrokenLink": "{s} поддерживается, но с твоей ссылкой что-то не так. может быть, ты её не полностью скопировал?",
- "ErrorNoLink": "я не гадалка и не могу угадывать, что ты хочешь скачать. попробуй в следующий раз вставить ссылку.",
+ "ErrorNoLink": "пока что я не умею угадывать, что ты хочешь скачать. дай мне, пожалуйста, ссылку :(",
"ErrorPageRenderFail": "что-то пошло не так и у меня не получилось срендерить страницу. если это повторится ещё раз, пожалуйста, {ContactLink}. также приложи хэш текущего коммита ({s}) с действиями для повторения этой ошибки. можно на русском языке. спасибо :)",
- "ErrorRateLimit": "ты делаешь слишком много запросов. успокойся и попробуй ещё раз через несколько минут.",
+ "ErrorRateLimit": "ты делаешь слишком много запросов. попробуй ещё раз через минуту!",
"ErrorCouldntFetch": "мне не удалось получить инфу о твоей ссылке. проверь её и попробуй ещё раз.",
"ErrorLengthLimit": "твоё видео длиннее чем {s} минут(ы). это превышает текущий лимит. скачай что-нибудь покороче, а не экранизацию \"войны и мира\".",
"ErrorBadFetch": "произошла ошибка при получении инфы о твоей ссылке. ты уверен, что она работает? проверь её, и попробуй ещё раз.",
diff --git a/src/modules/api.js b/src/modules/api.js
index 70f84016..9dc8a236 100644
--- a/src/modules/api.js
+++ b/src/modules/api.js
@@ -10,32 +10,41 @@ import match from "./processing/match.js";
export async function getJSON(originalURL, lang, obj) {
try {
let url = decodeURIComponent(originalURL);
- if (!url.includes('http://')) {
- let hostname = url.replace("https://", "").replace(' ', '').split('&')[0].split("/")[0].split("."),
- host = hostname[hostname.length - 2],
- patternMatch;
- if (host === "youtu") {
+ if (url.startsWith('http://')) {
+ return apiJSON(0, { t: errorUnsupported(lang) });
+ }
+ let hostname = url.replace("https://", "").replace(' ', '').split('&')[0].split("/")[0].split("."),
+ host = hostname[hostname.length - 2],
+ patternMatch;
+
+ // TO-DO: bring all tests into one unified module instead of placing them in several places
+ switch(host) {
+ case "youtu":
host = "youtube";
url = `https://youtube.com/watch?v=${url.replace("youtu.be/", "").replace("https://", "")}`;
- }
- if (host === "goo" && url.substring(0, 30) === "https://soundcloud.app.goo.gl/") {
- host = "soundcloud"
- url = `https://soundcloud.com/${url.replace("https://soundcloud.app.goo.gl/", "").split('/')[0]}`
- }
- if (host === "tumblr" && !url.includes("blog/view")) {
- if (url.slice(-1) == '/') url = url.slice(0, -1);
- url = url.replace(url.split('/')[5], '');
- }
- if (host && host.length < 20 && host in patterns && patterns[host]["enabled"]) {
- for (let i in patterns[host]["patterns"]) {
- patternMatch = new UrlPattern(patterns[host]["patterns"][i]).match(cleanURL(url, host).split(".com/")[1]);
- if (patternMatch) break;
+ break;
+ case "goo":
+ if (url.substring(0, 30) === "https://soundcloud.app.goo.gl/"){
+ host = "soundcloud"
+ url = `https://soundcloud.com/${url.replace("https://soundcloud.app.goo.gl/", "").split('/')[0]}`
}
- if (patternMatch) {
- return await match(host, patternMatch, url, lang, obj);
- } else return apiJSON(0, { t: errorUnsupported(lang) });
- } else return apiJSON(0, { t: errorUnsupported(lang) });
- } else return apiJSON(0, { t: errorUnsupported(lang) });
+ break;
+ case "tumblr":
+ if (!url.includes("blog/view")) {
+ if (url.slice(-1) === '/') url = url.slice(0, -1);
+ url = url.replace(url.split('/')[5], '');
+ }
+ break;
+ }
+ if (!(host && host.length < 20 && host in patterns && patterns[host]["enabled"])) return apiJSON(0, { t: errorUnsupported(lang) });
+
+ for (let i in patterns[host]["patterns"]) {
+ patternMatch = new UrlPattern(patterns[host]["patterns"][i]).match(cleanURL(url, host).split(".com/")[1]);
+ if (patternMatch) break;
+ }
+ if (!patternMatch) return apiJSON(0, { t: errorUnsupported(lang) });
+
+ return await match(host, patternMatch, url, lang, obj);
} catch (e) {
return apiJSON(0, { t: loc(lang, 'ErrorSomethingWentWrong') });
}
diff --git a/src/modules/build.js b/src/modules/build.js
index fdeebdab..0dd7279e 100644
--- a/src/modules/build.js
+++ b/src/modules/build.js
@@ -4,9 +4,9 @@ export async function buildFront() {
try {
await esbuild.build({
entryPoints: ['src/front/cobalt.js', 'src/front/cobalt.css'],
- outdir: `min/`,
+ outdir: 'min/',
minify: true,
- loader: { ".js": "js", ".css": "css" }
+ loader: { '.js': 'js', '.css': 'css' }
})
} catch (e) {
return;
diff --git a/src/modules/config.js b/src/modules/config.js
index 8f27e950..82a109ee 100644
--- a/src/modules/config.js
+++ b/src/modules/config.js
@@ -15,7 +15,6 @@ export const
repo = packageJson["bugs"]["url"].replace('/issues', ''),
authorInfo = config.authorInfo,
quality = config.quality,
- internetExplorerRedirect = config.internetExplorerRedirect,
donations = config.donations,
ffmpegArgs = config.ffmpegArgs,
supportedAudio = config.supportedAudio,
diff --git a/src/modules/pageRender/elements.js b/src/modules/pageRender/elements.js
index 909927ac..43a40363 100644
--- a/src/modules/pageRender/elements.js
+++ b/src/modules/pageRender/elements.js
@@ -2,7 +2,7 @@ import { celebrations } from "../config.js";
export function switcher(obj) {
let items = ``;
- if (obj.name == "download") {
+ if (obj.name === "download") {
items = obj.items;
} else {
for (let i = 0; i < obj.items.length; i++) {
@@ -129,7 +129,7 @@ export function backdropLink(link, text) {
return `${text}`
}
export function socialLink(emoji, name, handle, url) {
- return `
`
+ return ``
}
export function settingsCategory(obj) {
return `
diff --git a/src/modules/pageRender/onDemand.js b/src/modules/pageRender/onDemand.js
index 7ab1373a..6fed7c0b 100644
--- a/src/modules/pageRender/onDemand.js
+++ b/src/modules/pageRender/onDemand.js
@@ -6,7 +6,7 @@ export function changelogHistory() { // blockId 0
let historyLen = history.length
for (let i in history) {
- let separator = (i != 0 && i != historyLen) ? '
' : ''
+ let separator = (i !== 0 && i !== historyLen) ? '
' : ''
render += `${separator}${history[i]["banner"] ? `
` : ''}`
}
return render;
diff --git a/src/modules/pageRender/page.js b/src/modules/pageRender/page.js
index 87d4a0ad..0c90a756 100644
--- a/src/modules/pageRender/page.js
+++ b/src/modules/pageRender/page.js
@@ -186,7 +186,7 @@ export default function(obj) {
closeAria: t('AccessibilityClosePopup'),
header: {
aboveTitle: {
- text: `v.${version}-${obj.hash}`,
+ text: `v.${version}-${obj.hash} (${obj.branch})`,
url: `${repo}/commit/${obj.hash}`
},
title: `${emoji("⚙️", 30)} ${t('TitlePopupSettings')}`
diff --git a/src/modules/processing/match.js b/src/modules/processing/match.js
index ff150b83..3336bb4a 100644
--- a/src/modules/processing/match.js
+++ b/src/modules/processing/match.js
@@ -1,27 +1,28 @@
import { apiJSON } from "../sub/utils.js";
-import { errorUnsupported, genericError } from "../sub/errors.js";
+import { errorUnsupported, genericError, brokenLink } from "../sub/errors.js";
import loc from "../../localization/manager.js";
import { testers } from "./servicesPatternTesters.js";
-import bilibili from "../services/bilibili.js";
-import reddit from "../services/reddit.js";
-import twitter from "../services/twitter.js";
-import youtube from "../services/youtube.js";
-import vk from "../services/vk.js";
-import tiktok from "../services/tiktok.js";
-import tumblr from "../services/tumblr.js";
+import bilibili from "./services/bilibili.js";
+import reddit from "./services/reddit.js";
+import twitter from "./services/twitter.js";
+import youtube from "./services/youtube.js";
+import vk from "./services/vk.js";
+import tiktok from "./services/tiktok.js";
+import tumblr from "./services/tumblr.js";
import matchActionDecider from "./matchActionDecider.js";
-import vimeo from "../services/vimeo.js";
-import soundcloud from "../services/soundcloud.js";
+import vimeo from "./services/vimeo.js";
+import soundcloud from "./services/soundcloud.js";
export default async function (host, patternMatch, url, lang, obj) {
try {
- if (!testers[host]) return apiJSON(0, { t: errorUnsupported(lang) });
- if (!(testers[host](patternMatch))) throw Error();
+ let r, isAudioOnly = !!obj.isAudioOnly;
+
+ if (!testers[host]) return apiJSON(0, { t: errorUnsupported(lang) });
+ if (!(testers[host](patternMatch))) return apiJSON(0, { t: brokenLink(lang) });
- let r;
switch (host) {
case "twitter":
r = await twitter({
@@ -29,14 +30,14 @@ export default async function (host, patternMatch, url, lang, obj) {
spaceId: patternMatch["spaceId"] ? patternMatch["spaceId"] : false,
lang: lang
});
- if (r.isAudioOnly) obj.isAudioOnly = true
break;
case "vk":
r = await vk({
url: url,
userId: patternMatch["userId"],
videoId: patternMatch["videoId"],
- lang: lang, quality: obj.vQuality
+ lang: lang,
+ quality: obj.vQuality
});
break;
case "bilibili":
@@ -48,10 +49,11 @@ export default async function (host, patternMatch, url, lang, obj) {
case "youtube":
let fetchInfo = {
id: patternMatch["id"].slice(0, 11),
- lang: lang, quality: obj.vQuality,
+ lang: lang,
+ quality: obj.vQuality,
format: "webm"
};
- if (url.match('music.youtube.com') || obj.isAudioOnly == true) obj.vFormat = "audio";
+ if (url.match('music.youtube.com') || isAudioOnly === true) obj.vFormat = "audio";
switch (obj.vFormat) {
case "mp4":
fetchInfo["format"] = "mp4";
@@ -60,7 +62,7 @@ export default async function (host, patternMatch, url, lang, obj) {
fetchInfo["format"] = "webm";
fetchInfo["isAudioOnly"] = true;
fetchInfo["quality"] = "max";
- obj.isAudioOnly = true;
+ isAudioOnly = true;
break;
}
r = await youtube(fetchInfo);
@@ -69,7 +71,8 @@ export default async function (host, patternMatch, url, lang, obj) {
r = await reddit({
sub: patternMatch["sub"],
id: patternMatch["id"],
- title: patternMatch["title"], lang: lang,
+ title: patternMatch["title"],
+ lang: lang,
});
break;
case "douyin":
@@ -77,28 +80,33 @@ export default async function (host, patternMatch, url, lang, obj) {
r = await tiktok({
host: host,
postId: patternMatch["postId"],
- id: patternMatch["id"], lang: lang,
- noWatermark: obj.isNoTTWatermark, fullAudio: obj.isTTFullAudio,
- isAudioOnly: obj.isAudioOnly
+ id: patternMatch["id"],
+ lang: lang,
+ noWatermark: obj.isNoTTWatermark,
+ fullAudio: obj.isTTFullAudio,
+ isAudioOnly: isAudioOnly
});
- if (r.isAudioOnly) obj.isAudioOnly = true;
break;
case "tumblr":
r = await tumblr({
- id: patternMatch["id"], url: url, user: patternMatch["user"] ? patternMatch["user"] : false,
+ id: patternMatch["id"],
+ url: url,
+ user: patternMatch["user"] ? patternMatch["user"] : false,
lang: lang
});
break;
case "vimeo":
r = await vimeo({
- id: patternMatch["id"].slice(0, 11), quality: obj.vQuality,
+ id: patternMatch["id"].slice(0, 11),
+ quality: obj.vQuality,
lang: lang
});
break;
case "soundcloud":
- obj.isAudioOnly = true;
+ isAudioOnly = true;
r = await soundcloud({
- author: patternMatch["author"], song: patternMatch["song"], url: url,
+ author: patternMatch["author"],
+ song: patternMatch["song"], url: url,
shortLink: patternMatch["shortLink"] ? patternMatch["shortLink"] : false,
accessKey: patternMatch["accessKey"] ? patternMatch["accessKey"] : false,
format: obj.aFormat,
@@ -108,9 +116,13 @@ export default async function (host, patternMatch, url, lang, obj) {
default:
return apiJSON(0, { t: errorUnsupported(lang) });
}
- return !r.error ? matchActionDecider(r, host, obj.ip, obj.aFormat, obj.isAudioOnly, lang, obj.isAudioMuted) : apiJSON(0, {
- t: Array.isArray(r.error) ? loc(lang, r.error[0], r.error[1]) : loc(lang, r.error)
- });
+
+ if (r.isAudioOnly) isAudioOnly = true;
+ let isAudioMuted = isAudioOnly ? false : obj.isAudioMuted;
+
+ if (r.error) return apiJSON(0, { t: Array.isArray(r.error) ? loc(lang, r.error[0], r.error[1]) : loc(lang, r.error) });
+
+ return matchActionDecider(r, host, obj.ip, obj.aFormat, isAudioOnly, lang, isAudioMuted);
} catch (e) {
return apiJSON(0, { t: genericError(lang, host) })
}
diff --git a/src/modules/processing/matchActionDecider.js b/src/modules/processing/matchActionDecider.js
index a8070cde..fb0892a4 100644
--- a/src/modules/processing/matchActionDecider.js
+++ b/src/modules/processing/matchActionDecider.js
@@ -1,121 +1,137 @@
-import { audioIgnore, services, supportedAudio } from "../config.js"
-import { apiJSON } from "../sub/utils.js"
+import { audioIgnore, services, supportedAudio } from "../config.js";
+import { apiJSON } from "../sub/utils.js";
import loc from "../../localization/manager.js";
export default function(r, host, ip, audioFormat, isAudioOnly, lang, isAudioMuted) {
- if (!isAudioOnly && !r.picker && !isAudioMuted) {
- switch (host) {
- case "twitter":
- return apiJSON(1, { u: r.urls });
- case "vk":
- return apiJSON(2, {
- type: "bridge", u: r.urls, service: host, ip: ip,
- filename: r.filename,
- });
- case "bilibili":
- return apiJSON(2, {
- type: "render", u: r.urls, service: host, ip: ip,
- filename: r.filename,
- time: r.time
- });
- case "youtube":
- return apiJSON(2, {
- type: r.type, u: r.urls, service: host, ip: ip,
- filename: r.filename,
- time: r.time,
- });
- case "reddit":
- return apiJSON(r.typeId, {
- type: r.type, u: r.urls, service: host, ip: ip,
- filename: r.filename,
- });
- case "tiktok":
- return apiJSON(2, {
- type: "bridge", u: r.urls, service: host, ip: ip,
- filename: r.filename,
- });
- case "douyin":
- return apiJSON(2, {
- type: "bridge", u: r.urls, service: host, ip: ip,
- filename: r.filename,
- });
- case "tumblr":
- return apiJSON(1, { u: r.urls });
- case "vimeo":
- if (Array.isArray(r.urls)) {
- return apiJSON(2, {
- type: "render", u: r.urls, service: host, ip: ip,
- filename: r.filename
- });
- } else {
- return apiJSON(1, { u: r.urls });
- }
- }
- } else if (isAudioMuted && !isAudioOnly) {
- let isSplit = Array.isArray(r.urls);
- return apiJSON(2, {
- type: isSplit ? "bridge" : "mute",
- u: isSplit ? r.urls[0] : r.urls,
+ let action,
+ responseType = 2,
+ defaultParams = {
+ u: r.urls,
service: host,
ip: ip,
filename: r.filename,
- mute: true,
- });
- } else if (r.picker) {
- switch (host) {
- case "douyin":
- case "tiktok":
- let type = "render";
- if (audioFormat === "mp3" || audioFormat === "best") {
- audioFormat = "mp3"
- type = "bridge"
- }
- return apiJSON(5, {
- type: type,
- picker: r.picker,
- u: Array.isArray(r.urls) ? r.urls[1] : r.urls, service: host, ip: ip,
- filename: r.audioFilename, isAudioOnly: true, audioFormat: audioFormat, copy: audioFormat === "best" ? true : false,
- })
- case "twitter":
- return apiJSON(5, {
- picker: r.picker, service: host
- })
- }
- } else if (isAudioOnly) {
- if ((host === "reddit" && r.typeId === 1) || (host === "vimeo" && !r.filename) || audioIgnore.includes(host)) return apiJSON(0, { t: loc(lang, 'ErrorEmptyDownload') });
- let type = "render";
- let copy = false;
-
- if (!supportedAudio.includes(audioFormat)) audioFormat = "best";
- if ((host == "tiktok" || host == "douyin") && services.tiktok.audioFormats.includes(audioFormat)) {
- if (r.isMp3) {
- if (audioFormat === "mp3" || audioFormat === "best") {
- audioFormat = "mp3"
- type = "bridge"
- }
- } else if (audioFormat === "best") {
- audioFormat = "m4a"
- type = "bridge"
- }
- }
- if ((audioFormat === "best" && services[host]["bestAudio"]) || services[host]["bestAudio"] && (audioFormat === services[host]["bestAudio"])) {
- audioFormat = services[host]["bestAudio"]
- type = "bridge"
- } else if (audioFormat === "best") {
- audioFormat = "m4a"
- copy = true
- if (r.audioFilename.includes("twitterspaces")) {
- audioFormat = "mp3"
- copy = false
- }
- }
- return apiJSON(2, {
- type: type,
- u: Array.isArray(r.urls) ? r.urls[1] : r.urls, service: host, ip: ip,
- filename: r.audioFilename, isAudioOnly: true,
- audioFormat: audioFormat, copy: copy, fileMetadata: r.fileMetadata ? r.fileMetadata : false
- })
- } else {
- return apiJSON(0, { t: loc(lang, 'ErrorSomethingWentWrong') });
+ },
+ params = {}
+
+ if (!isAudioOnly && !r.picker && !isAudioMuted) action = "video";
+ if (isAudioOnly && !r.picker) action = "audio";
+ if (r.picker) action = "picker";
+ if (isAudioMuted) action = "muteVideo";
+
+ if (action === "picker" || action === "audio") {
+ defaultParams.filename = r.audioFilename;
+ defaultParams.isAudioOnly = true;
+ defaultParams.audioFormat = audioFormat;
}
+
+ switch (action) {
+ case "video":
+ switch (host) {
+ case "bilibili":
+ params = { type: "render", time: r.time };
+ break;
+ case "youtube":
+ params = { type: r.type, time: r.time };
+ break;
+ case "reddit":
+ responseType = r.typeId;
+ params = { type: r.type };
+ break;
+ case "vimeo":
+ if (Array.isArray(r.urls)) {
+ params = { type: "render" }
+ } else {
+ responseType = 1;
+ }
+ break;
+
+ case "vk":
+ case "douyin":
+ case "tiktok":
+ params = { type: "bridge" };
+ break;
+
+ case "tumblr":
+ case "twitter":
+ responseType = 1;
+ break;
+ }
+ break;
+
+ case "muteVideo":
+ params = {
+ type: Array.isArray(r.urls) ? "bridge" : "mute",
+ u: Array.isArray(r.urls) ? r.urls[0] : r.urls,
+ mute: true
+ }
+ break;
+
+ case "picker":
+ responseType = 5;
+ switch (host) {
+ case "twitter":
+ params = { picker: r.picker };
+ break;
+ case "douyin":
+ case "tiktok":
+ let pickerType = "render";
+ if (audioFormat === "mp3" || audioFormat === "best") {
+ audioFormat = "mp3";
+ pickerType = "bridge"
+ }
+ params = {
+ type: pickerType,
+ picker: r.picker,
+ u: Array.isArray(r.urls) ? r.urls[1] : r.urls,
+ copy: audioFormat === "best" ? true : false
+ }
+ }
+ break;
+
+ case "audio":
+ if ((host === "reddit" && r.typeId === 1) || (host === "vimeo" && !r.filename) || audioIgnore.includes(host)) return apiJSON(0, { t: loc(lang, 'ErrorEmptyDownload') });
+
+ let processType = "render";
+ let copy = false;
+
+ if (!supportedAudio.includes(audioFormat)) audioFormat = "best";
+
+ if ((host === "tiktok" || host === "douyin") && services.tiktok.audioFormats.includes(audioFormat)) {
+ if (r.isMp3) {
+ if (audioFormat === "mp3" || audioFormat === "best") {
+ audioFormat = "mp3";
+ processType = "bridge"
+ }
+ } else if (audioFormat === "best") {
+ audioFormat = "m4a";
+ processType = "bridge"
+ }
+ }
+
+ if ((audioFormat === "best" && services[host]["bestAudio"])
+ || services[host]["bestAudio"] && (audioFormat === services[host]["bestAudio"])) {
+ audioFormat = services[host]["bestAudio"];
+ processType = "bridge"
+ } else if (audioFormat === "best") {
+ audioFormat = "m4a";
+ copy = true;
+ if (r.audioFilename.includes("twitterspaces")) {
+ audioFormat = "mp3"
+ copy = false
+ }
+ }
+
+ params = {
+ type: processType,
+ u: Array.isArray(r.urls) ? r.urls[1] : r.urls,
+ audioFormat: audioFormat,
+ copy: copy,
+ fileMetadata: r.fileMetadata ? r.fileMetadata : false
+ }
+ break;
+ default:
+ return apiJSON(0, { t: loc(lang, 'ErrorEmptyDownload') });
+ }
+
+ return apiJSON(responseType, {...defaultParams, ...params})
}
diff --git a/src/modules/processing/services/bilibili.js b/src/modules/processing/services/bilibili.js
new file mode 100644
index 00000000..82964a3a
--- /dev/null
+++ b/src/modules/processing/services/bilibili.js
@@ -0,0 +1,28 @@
+import { genericUserAgent, maxVideoDuration } from "../../config.js";
+
+// TO-DO: quality picking, bilibili.tv support, and higher quality downloads (currently requires an account)
+export default async function(obj) {
+ let html = await fetch(`https://bilibili.com/video/${obj.id}`, {
+ headers: { "user-agent": genericUserAgent }
+ }).then((r) => { return r.text() }).catch(() => { return false });
+ if (!html) return { error: 'ErrorCouldntFetch' };
+ if (!(html.includes('')[0]);
+ if (streamData.data.timelength > maxVideoDuration) return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
+
+ let video = streamData["data"]["dash"]["video"].filter((v) => {
+ if (!v["baseUrl"].includes("https://upos-sz-mirrorcosov.bilivideo.com/")) return true;
+ }).sort((a, b) => Number(b.bandwidth) - Number(a.bandwidth));
+
+ let audio = streamData["data"]["dash"]["audio"].filter((a) => {
+ if (!a["baseUrl"].includes("https://upos-sz-mirrorcosov.bilivideo.com/")) return true;
+ }).sort((a, b) => Number(b.bandwidth) - Number(a.bandwidth));
+
+ return {
+ urls: [video[0]["baseUrl"], audio[0]["baseUrl"]],
+ time: streamData.data.timelength,
+ audioFilename: `bilibili_${obj.id}_audio`,
+ filename: `bilibili_${obj.id}_${video[0]["width"]}x${video[0]["height"]}.mp4`
+ };
+}
diff --git a/src/modules/processing/services/reddit.js b/src/modules/processing/services/reddit.js
new file mode 100644
index 00000000..816a8da6
--- /dev/null
+++ b/src/modules/processing/services/reddit.js
@@ -0,0 +1,28 @@
+import { maxVideoDuration } from "../../config.js";
+
+export default async function(obj) {
+ let data = await fetch(`https://www.reddit.com/r/${obj.sub}/comments/${obj.id}/${obj.name}.json`).then((r) => { return r.json() }).catch(() => { return false });
+ if (!data) return { error: 'ErrorCouldntFetch' };
+
+ data = data[0]["data"]["children"][0]["data"];
+
+ if (data.url.endsWith('.gif')) return { typeId: 1, urls: data.url };
+
+ if (!("reddit_video" in data["secure_media"])) return { error: 'ErrorEmptyDownload' };
+ if (data["secure_media"]["reddit_video"]["duration"] * 1000 > maxVideoDuration) return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
+
+ let video = data["secure_media"]["reddit_video"]["fallback_url"].split('?')[0],
+ audio = video.match('.mp4') ? `${video.split('_')[0]}_audio.mp4` : `${data["secure_media"]["reddit_video"]["fallback_url"].split('DASH')[0]}audio`;
+ await fetch(audio, { method: "HEAD" }).then((r) => {if (Number(r.status) !== 200) audio = ''}).catch(() => {audio = ''});
+
+ let id = data["secure_media"]["reddit_video"]["fallback_url"].split('/')[3];
+ if (!audio.length > 0) return { typeId: 1, urls: video };
+
+ return {
+ typeId: 2,
+ type: "render",
+ urls: [video, audio],
+ audioFilename: `reddit_${id}_audio`,
+ filename: `reddit_${id}.mp4`
+ };
+}
diff --git a/src/modules/processing/services/soundcloud.js b/src/modules/processing/services/soundcloud.js
new file mode 100644
index 00000000..aa3a695a
--- /dev/null
+++ b/src/modules/processing/services/soundcloud.js
@@ -0,0 +1,74 @@
+import { maxAudioDuration } from "../../config.js";
+
+let cachedID = {};
+
+async function findClientID() {
+ try {
+ let sc = await fetch('https://soundcloud.com/').then((r) => { return r.text() }).catch(() => { return false });
+ let scVersion = String(sc.match(/')[0])
+ if (!json["media"]["transcodings"]) return { error: 'ErrorEmptyDownload' };
+
+ let clientId = await findClientID();
+ if (!clientId) return { error: 'ErrorSoundCloudNoClientId' };
+
+ let fileUrlBase = json.media.transcodings[0]["url"].replace("/hls", "/progressive")
+ let fileUrl = `${fileUrlBase}${fileUrlBase.includes("?") ? "&" : "?"}client_id=${clientId}&track_authorization=${json.track_authorization}`;
+ if (fileUrl.substring(0, 54) !== "https://api-v2.soundcloud.com/media/soundcloud:tracks:") return { error: 'ErrorEmptyDownload' };
+
+ if (json.duration > maxAudioDuration) return { error: ['ErrorLengthAudioConvert', maxAudioDuration / 60000] };
+
+ let file = await fetch(fileUrl).then(async (r) => { return (await r.json()).url }).catch(() => { return false });
+ if (!file) return { error: 'ErrorCouldntFetch' };
+
+ return {
+ urls: file,
+ audioFilename: `soundcloud_${json.id}`,
+ fileMetadata: {
+ title: json.title,
+ artist: json.user.username,
+ }
+ }
+}
diff --git a/src/modules/processing/services/tiktok.js b/src/modules/processing/services/tiktok.js
new file mode 100644
index 00000000..dac6fe43
--- /dev/null
+++ b/src/modules/processing/services/tiktok.js
@@ -0,0 +1,112 @@
+import { genericUserAgent } from "../../config.js";
+
+let userAgent = genericUserAgent.split(' Chrome/1')[0],
+ config = {
+ tiktok: {
+ short: "https://vt.tiktok.com/",
+ api: "https://api2.musical.ly/aweme/v1/feed/?aweme_id={postId}&version_code=262&app_name=musical_ly&channel=App&device_id=null&os_version=14.4.2&device_platform=iphone&device_type=iPhone9®ion=US&carrier_region=US",
+ },
+ douyin: {
+ short: "https://v.douyin.com/",
+ api: "https://www.iesdouyin.com/aweme/v1/web/aweme/detail/?aweme_id={postId}",
+ }
+}
+
+function selector(j, h, id) {
+ if (!j) return false;
+ let t;
+ switch (h) {
+ case "tiktok":
+ t = j["aweme_list"].filter((v) => { if (v["aweme_id"] === id) return true })[0];
+ break;
+ case "douyin":
+ t = j['aweme_detail'];
+ break;
+ }
+ if (t.length < 3) return false;
+ return t;
+}
+
+export default async function(obj) {
+ let postId = obj.postId ? obj.postId : false;
+
+ if (!postId) {
+ let html = await fetch(`${config[obj.host]["short"]}${obj.id}`, {
+ redirect: "manual",
+ headers: { "user-agent": userAgent }
+ }).then((r) => { return r.text() }).catch(() => { return false });
+ if (!html) return { error: 'ErrorCouldntFetch' };
+
+ if (html.slice(0, 17) === '
{ return r.json() }).catch(() => { return false });
+
+ detail = selector(detail, obj.host, postId);
+ if (!detail) return { error: 'ErrorCouldntFetch' };
+
+ let video, videoFilename, audioFilename, isMp3, audio, images, filenameBase = `${obj.host}_${postId}`;
+ if (obj.host === "tiktok") {
+ images = detail["image_post_info"] ? detail["image_post_info"]["images"] : false
+ } else {
+ images = detail["images"] ? detail["images"] : false
+ }
+
+ if (!obj.isAudioOnly && !images) {
+ video = obj.host === "tiktok" ? detail["video"]["download_addr"]["url_list"][0] : detail["video"]["play_addr"]["url_list"][2].replace("/play/", "/playwm/");
+ videoFilename = `${filenameBase}_video.mp4`;
+ if (obj.noWatermark) {
+ video = obj.host === "tiktok" ? detail["video"]["play_addr"]["url_list"][0] : detail["video"]["play_addr"]["url_list"][0];
+ videoFilename = `${filenameBase}_video_nw.mp4` // nw - no watermark
+ }
+ } else {
+ let fallback = obj.host === "douyin" ? detail["video"]["play_addr"]["url_list"][0].replace("playwm", "play") : detail["video"]["play_addr"]["url_list"][0];
+ audio = fallback;
+ audioFilename = `${filenameBase}_audio_fv`; // fv - from video
+ if (obj.fullAudio || fallback.includes("music")) {
+ audio = detail["music"]["play_url"]["url_list"][0]
+ audioFilename = `${filenameBase}_audio`
+ }
+ if (audio.slice(-4) === ".mp3") isMp3 = true;
+ }
+
+ if (video) return {
+ urls: video,
+ filename: videoFilename
+ }
+ if (images && obj.isAudioOnly) return {
+ urls: audio,
+ audioFilename: audioFilename,
+ isAudioOnly: true,
+ isMp3: isMp3
+ }
+ if (images) {
+ let imageLinks = [];
+ for (let i in images) {
+ let sel = obj.host === "tiktok" ? images[i]["display_image"]["url_list"] : images[i]["url_list"];
+ sel = sel.filter((p) => { if (p.includes(".jpeg?")) return true; })
+ imageLinks.push({url: sel[0]})
+ }
+ return {
+ picker: imageLinks,
+ urls: audio,
+ audioFilename: audioFilename,
+ isAudioOnly: true,
+ isMp3: isMp3
+ }
+ }
+ if (audio) return {
+ urls: audio,
+ audioFilename: audioFilename,
+ isAudioOnly: true,
+ isMp3: isMp3
+ }
+}
diff --git a/src/modules/processing/services/tumblr.js b/src/modules/processing/services/tumblr.js
new file mode 100644
index 00000000..45291f07
--- /dev/null
+++ b/src/modules/processing/services/tumblr.js
@@ -0,0 +1,14 @@
+import { genericUserAgent } from "../../config.js";
+
+export default async function(obj) {
+ let html = await fetch(`https://${
+ obj.user ? obj.user : obj.url.split('.')[0].replace('https://', '')
+ }.tumblr.com/post/${obj.id}`, {
+ headers: { "user-agent": genericUserAgent }
+ }).then((r) => { return r.text() }).catch(() => { return false });
+
+ if (!html) return { error: 'ErrorCouldntFetch' };
+ if (!html.includes('property="og:video" content="https://va.media.tumblr.com/')) return { error: 'ErrorEmptyDownload' };
+
+ return { urls: `https://va.media.tumblr.com/${html.split('property="og:video" content="https://va.media.tumblr.com/')[1].split('"')[0]}`, audioFilename: `tumblr_${obj.id}_audio` }
+}
diff --git a/src/modules/processing/services/twitter.js b/src/modules/processing/services/twitter.js
new file mode 100644
index 00000000..77eb375e
--- /dev/null
+++ b/src/modules/processing/services/twitter.js
@@ -0,0 +1,101 @@
+import { genericUserAgent } from "../../config.js";
+
+function bestQuality(arr) {
+ return arr.filter((v) => { if (v["content_type"] === "video/mp4") return true }).sort((a, b) => Number(b.bitrate) - Number(a.bitrate))[0]["url"].split("?")[0]
+}
+const apiURL = "https://api.twitter.com/1.1"
+
+// TO-DO: move from 1.1 api to graphql
+export default async function(obj) {
+ let _headers = {
+ "user-agent": genericUserAgent,
+ "authorization": "Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA",
+ // ^ no explicit content, but with multi media support
+ "host": "api.twitter.com"
+ };
+ let req_act = await fetch(`${apiURL}/guest/activate.json`, {
+ method: "POST",
+ headers: _headers
+ }).then((r) => { return r.status === 200 ? r.json() : false }).catch(() => { return false });
+ if (!req_act) return { error: 'ErrorCouldntFetch' };
+
+ _headers["x-guest-token"] = req_act["guest_token"];
+ let showURL = `${apiURL}/statuses/show/${obj.id}.json?tweet_mode=extended&include_user_entities=0&trim_user=1&include_entities=0&cards_platform=Web-12&include_cards=1`;
+
+ if (!obj.spaceId) {
+ let req_status = await fetch(showURL, { headers: _headers }).then((r) => { return r.status === 200 ? r.json() : false }).catch((e) => { return false });
+ if (!req_status) {
+ _headers.authorization = "Bearer AAAAAAAAAAAAAAAAAAAAAPYXBAAAAAAACLXUNDekMxqa8h%2F40K4moUkGsoc%3DTYfbDKbT3jJPCEVnMYqilB28NHfOPqkca3qaAxGfsyKCs0wRbw";
+ // ^ explicit content, but no multi media support
+ delete _headers["x-guest-token"]
+
+ req_act = await fetch(`${apiURL}/guest/activate.json`, {
+ method: "POST",
+ headers: _headers
+ }).then((r) => { return r.status === 200 ? r.json() : false}).catch(() => { return false });
+ if (!req_act) return { error: 'ErrorCouldntFetch' };
+
+ _headers["x-guest-token"] = req_act["guest_token"];
+ req_status = await fetch(showURL, { headers: _headers }).then((r) => { return r.status === 200 ? r.json() : false }).catch(() => { return false });
+ }
+ if (!req_status) return { error: 'ErrorCouldntFetch' };
+ if (!req_status["extended_entities"] || !req_status["extended_entities"]["media"]) return { error: 'ErrorNoVideosInTweet' };
+
+ let single, multiple = [], media = req_status["extended_entities"]["media"];
+ media = media.filter((i) => { if (i["type"] === "video" || i["type"] === "animated_gif") return true })
+ if (media.length > 1) {
+ for (let i in media) { multiple.push({type: "video", thumb: media[i]["media_url_https"], url: bestQuality(media[i]["video_info"]["variants"])}) }
+ } else if (media.length === 1) {
+ single = bestQuality(media[0]["video_info"]["variants"])
+ } else {
+ return { error: 'ErrorNoVideosInTweet' }
+ }
+
+ if (single) {
+ return { urls: single, filename: `twitter_${obj.id}.mp4`, audioFilename: `twitter_${obj.id}_audio` }
+ } else if (multiple) {
+ return { picker: multiple }
+ } else {
+ return { error: 'ErrorNoVideosInTweet' }
+ }
+ } else {
+ _headers["host"] = "twitter.com"
+ _headers["content-type"] = "application/json"
+
+ let query = {
+ variables: {"id": obj.spaceId,"isMetatagsQuery":true,"withSuperFollowsUserFields":true,"withDownvotePerspective":false,"withReactionsMetadata":false,"withReactionsPerspective":false,"withSuperFollowsTweetFields":true,"withReplays":true},
+ features: {"spaces_2022_h2_clipping":true,"spaces_2022_h2_spaces_communities":true,"verified_phone_label_enabled":false,"tweetypie_unmention_optimization_enabled":true,"responsive_web_uc_gql_enabled":true,"vibe_api_enabled":true,"responsive_web_edit_tweet_api_enabled":true,"graphql_is_translatable_rweb_tweet_is_translatable_enabled":true,"standardized_nudges_misinfo":true,"tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled":false,"responsive_web_graphql_timeline_navigation_enabled":false,"interactive_text_enabled":true,"responsive_web_text_conversations_enabled":false,"responsive_web_enhance_cards_enabled":true}
+ }
+ query.variables = new URLSearchParams(JSON.stringify(query.variables)).toString().slice(0, -1);
+ query.features = new URLSearchParams(JSON.stringify(query.features)).toString().slice(0, -1);
+ query = `https://twitter.com/i/api/graphql/wJ5g4zf7v8qPHSQbaozYuw/AudioSpaceById?variables=${query.variables}&features=${query.features}`
+
+ let AudioSpaceById = await fetch(query, { headers: _headers }).then((r) => {return r.status === 200 ? r.json() : false}).catch((e) => { return false });
+ if (!AudioSpaceById) return { error: 'ErrorEmptyDownload' };
+
+ if (!AudioSpaceById.data.audioSpace.metadata) return { error: 'ErrorEmptyDownload' };
+ if (AudioSpaceById.data.audioSpace.metadata.is_space_available_for_replay !== true) return { error: 'TwitterSpaceWasntRecorded' };
+
+ let streamStatus = await fetch(
+ `https://twitter.com/i/api/1.1/live_video_stream/status/${AudioSpaceById.data.audioSpace.metadata.media_key}`, { headers: _headers }
+ ).then((r) =>{ return r.status === 200 ? r.json() : false }).catch(() => { return false });
+ if (!streamStatus) return { error: 'ErrorCouldntFetch' };
+
+ let participants = AudioSpaceById.data.audioSpace.participants.speakers;
+ let listOfParticipants = `Twitter Space speakers: `;
+ for (let i in participants) { listOfParticipants += `@${participants[i]["twitter_screen_name"]}, ` }
+ listOfParticipants = listOfParticipants.slice(0, -2);
+
+ return {
+ urls: streamStatus.source.noRedirectPlaybackUrl,
+ audioFilename: `twitterspaces_${obj.spaceId}`,
+ isAudioOnly: true,
+ fileMetadata: {
+ title: AudioSpaceById.data.audioSpace.metadata.title,
+ artist: `Twitter Space by @${AudioSpaceById.data.audioSpace.metadata.creator_results.result.legacy.screen_name}`,
+ comment: listOfParticipants,
+ // cover: AudioSpaceById.data.audioSpace.metadata.creator_results.result.legacy.profile_image_url_https.replace("_normal", "")
+ }
+ }
+ }
+}
diff --git a/src/modules/processing/services/vimeo.js b/src/modules/processing/services/vimeo.js
new file mode 100644
index 00000000..7f402597
--- /dev/null
+++ b/src/modules/processing/services/vimeo.js
@@ -0,0 +1,77 @@
+import { maxVideoDuration, quality, services } from "../../config.js";
+
+export default async function(obj) {
+ let api = await fetch(`https://player.vimeo.com/video/${obj.id}/config`).then((r) => { return r.json() }).catch(() => { return false });
+ if (!api) return { error: 'ErrorCouldntFetch' };
+
+ let downloadType = "dash";
+ if (JSON.stringify(api).includes('"progressive":[{')) downloadType = "progressive";
+
+ switch(downloadType) {
+ case "progressive":
+ let all = api["request"]["files"]["progressive"].sort((a, b) => Number(b.width) - Number(a.width));
+ let best = all[0];
+
+ try {
+ if (obj.quality !== "max") {
+ let pref = parseInt(quality[obj.quality], 10)
+ for (let i in all) {
+ let currQuality = parseInt(all[i]["quality"].replace('p', ''), 10)
+ if (currQuality === pref) {
+ best = all[i];
+ break
+ }
+ if (currQuality < pref) {
+ best = all[i-1];
+ break
+ }
+ }
+ }
+ } catch (e) {
+ best = all[0]
+ }
+
+ return { urls: best["url"], filename: `tumblr_${obj.id}.mp4` };
+ case "dash":
+ if (api.video.duration > maxVideoDuration / 1000) return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
+
+ let masterJSONURL = api["request"]["files"]["dash"]["cdns"]["akfire_interconnect_quic"]["url"];
+ let masterJSON = await fetch(masterJSONURL).then((r) => { return r.json() }).catch(() => { return false });
+
+ if (!masterJSON) return { error: 'ErrorCouldntFetch' };
+ if (!masterJSON.video) return { error: 'ErrorEmptyDownload' };
+
+ let type = "parcel";
+ if (masterJSON.base_url === "../") type = "chop";
+
+ let masterJSON_Video = masterJSON.video.sort((a, b) => Number(b.width) - Number(a.width));
+ let masterJSON_Audio = masterJSON.audio.sort((a, b) => Number(b.bitrate) - Number(a.bitrate)).filter((a)=> {if (a['mime_type'] === "audio/mp4") return true;});
+ let bestVideo = masterJSON_Video[0], bestAudio = masterJSON_Audio[0];
+
+ switch (type) {
+ case "parcel":
+ if (obj.quality !== "max") {
+ let pref = parseInt(quality[obj.quality], 10)
+ for (let i in masterJSON_Video) {
+ let currQuality = parseInt(services.vimeo.resolutionMatch[masterJSON_Video[i]["width"]], 10)
+ if (currQuality < pref) {
+ break;
+ } else if (String(currQuality) === String(pref)) {
+ bestVideo = masterJSON_Video[i]
+ }
+ }
+ }
+
+ let baseUrl = masterJSONURL.split("/sep/")[0];
+ let videoUrl = `${baseUrl}/parcel/video/${bestVideo.index_segment.split('?')[0]}`,
+ audioUrl = `${baseUrl}/parcel/audio/${bestAudio.index_segment.split('?')[0]}`;
+
+ return { urls: [videoUrl, audioUrl], audioFilename: `vimeo_${obj.id}_audio`, filename: `vimeo_${obj.id}_${bestVideo["width"]}x${bestVideo["height"]}.mp4` }
+ case "chop": // TO-DO: support for chop stream type
+ default:
+ return { error: 'ErrorEmptyDownload' }
+ }
+ default:
+ return { error: 'ErrorEmptyDownload' }
+ }
+}
diff --git a/src/modules/processing/services/vk.js b/src/modules/processing/services/vk.js
new file mode 100644
index 00000000..b7370af2
--- /dev/null
+++ b/src/modules/processing/services/vk.js
@@ -0,0 +1,45 @@
+import { xml2json } from "xml-js";
+import { genericUserAgent, maxVideoDuration, services } from "../../config.js";
+import selectQuality from "../../stream/selectQuality.js";
+
+export default async function(obj) {
+ let html;
+ html = await fetch(`https://vk.com/video-${obj.userId}_${obj.videoId}`, {
+ headers: { "user-agent": genericUserAgent }
+ }).then((r) => { return r.text() }).catch(() => { return false });
+ if (!html) return { error: 'ErrorCouldntFetch' };
+ if (!html.includes(`{"lang":`)) return { error: 'ErrorEmptyDownload' };
+
+ let js = JSON.parse('{"lang":' + html.split(`{"lang":`)[1].split(']);')[0]);
+
+ if (Number(js["mvData"]["is_active_live"]) !== 0) return { error: 'ErrorLiveVideo' };
+ if (js["mvData"]["duration"] > maxVideoDuration / 1000) return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
+
+ let mpd = JSON.parse(xml2json(js["player"]["params"][0]["manifest"], { compact: true, spaces: 4 }));
+ let repr = mpd["MPD"]["Period"]["AdaptationSet"]["Representation"];
+ if (!mpd["MPD"]["Period"]["AdaptationSet"]["Representation"]) repr = mpd["MPD"]["Period"]["AdaptationSet"][0]["Representation"];
+
+ let selectedQuality,
+ attr = repr[repr.length - 1]["_attributes"],
+ qualities = Object.keys(services.vk.quality_match);
+ for (let i in qualities) {
+ if (qualities[i] === attr["height"]) {
+ selectedQuality = `url${attr["height"]}`;
+ break
+ }
+ if (qualities[i] === attr["width"]) {
+ selectedQuality = `url${attr["width"]}`;
+ break
+ }
+ }
+
+ let maxQuality = js["player"]["params"][0][selectedQuality].split('type=')[1].slice(0, 1);
+ let userQuality = selectQuality('vk', obj.quality, Object.entries(services.vk.quality_match).reduce((r, [k, v]) => { r[v] = k; return r; })[maxQuality]);
+ let userRepr = repr[services.vk.representation_match[userQuality]]["_attributes"];
+ if (!(selectedQuality in js["player"]["params"][0])) return { error: 'ErrorEmptyDownload' };
+
+ return {
+ urls: js["player"]["params"][0][`url${userQuality}`],
+ filename: `vk_${obj.userId}_${obj.videoId}_${userRepr["width"]}x${userRepr['height']}.mp4`
+ }
+}
diff --git a/src/modules/processing/services/youtube.js b/src/modules/processing/services/youtube.js
new file mode 100644
index 00000000..d96af51f
--- /dev/null
+++ b/src/modules/processing/services/youtube.js
@@ -0,0 +1,88 @@
+import ytdl from "better-ytdl-core";
+import { maxVideoDuration, quality as mq } from "../../config.js";
+import selectQuality from "../../stream/selectQuality.js";
+
+export default async function(obj) {
+ let isAudioOnly = !!obj.isAudioOnly,
+ infoInitial = await ytdl.getInfo(obj.id);
+ if (!infoInitial) return { error: 'ErrorCantConnectToServiceAPI' };
+
+ let info = infoInitial.formats;
+ if (info[0]["isLive"]) return { error: 'ErrorLiveVideo' };
+
+ let videoMatch = [], fullVideoMatch = [], video = [],
+ audio = info.filter((a) => {
+ if (!a["isHLS"] && !a["isDashMPD"] && a["hasAudio"] && !a["hasVideo"] && a["container"] === obj.format) return true
+ }).sort((a, b) => Number(b.bitrate) - Number(a.bitrate));
+
+ if (audio.length === 0) return { error: 'ErrorBadFetch' };
+ if (audio[0]["approxDurationMs"] > maxVideoDuration) return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
+
+ if (!isAudioOnly) {
+ video = info.filter((a) => {
+ if (!a["isHLS"] && !a["isDashMPD"] && a["hasVideo"] && a["container"] === obj.format) {
+ if (obj.quality !== "max") {
+ if (a["hasAudio"] && String(mq[obj.quality]) === String(a["height"])) {
+ fullVideoMatch.push(a)
+ } else if (!a["hasAudio"] && String(mq[obj.quality]) === String(a["height"])) {
+ videoMatch.push(a)
+ }
+ }
+ return true
+ }
+ }).sort((a, b) => Number(b.bitrate) - Number(a.bitrate));
+
+ if (obj.quality !== "max") {
+ if (videoMatch.length === 0) {
+ let ss = selectQuality("youtube", obj.quality, video[0]["qualityLabel"].slice(0, 5).replace('p', '').trim());
+ videoMatch = video.filter((a) => {
+ if (a["qualityLabel"].slice(0, 5).replace('p', '').trim() === String(ss)) return true
+ })
+ } else if (fullVideoMatch.length > 0) {
+ videoMatch = [fullVideoMatch[0]]
+ }
+ } else videoMatch = [video[0]];
+ if (obj.quality === "los") videoMatch = [video[video.length - 1]];
+ }
+ if (video.length === 0) isAudioOnly = true;
+
+ if (isAudioOnly) {
+ let r = {
+ type: "render",
+ isAudioOnly: true,
+ urls: audio[0]["url"],
+ audioFilename: `youtube_${obj.id}_audio`,
+ fileMetadata: {
+ title: infoInitial.videoDetails.title,
+ artist: infoInitial.videoDetails.ownerChannelName.replace("- Topic", "").trim(),
+ }
+ }
+ if (infoInitial.videoDetails.description) {
+ let isAutoGenAudio = infoInitial.videoDetails.description.startsWith("Provided to YouTube by");
+ if (isAutoGenAudio) {
+ let descItems = infoInitial.videoDetails.description.split("\n\n")
+ r.fileMetadata.album = descItems[2]
+ r.fileMetadata.copyright = descItems[3]
+ if (descItems[4].startsWith("Released on:")) r.fileMetadata.date = descItems[4].replace("Released on: ", '').trim();
+ }
+ }
+ return r
+ }
+ let singleTest;
+ if (videoMatch.length > 0) {
+ singleTest = videoMatch[0]["hasVideo"] && videoMatch[0]["hasAudio"];
+ return {
+ type: singleTest ? "bridge" : "render",
+ urls: singleTest ? videoMatch[0]["url"] : [videoMatch[0]["url"], audio[0]["url"]],
+ time: videoMatch[0]["approxDurationMs"],
+ filename: `youtube_${obj.id}_${videoMatch[0]["width"]}x${videoMatch[0]["height"]}.${obj.format}`
+ }
+ }
+ singleTest = video[0]["hasVideo"] && video[0]["hasAudio"];
+ return {
+ type: singleTest ? "bridge" : "render",
+ urls: singleTest ? video[0]["url"] : [video[0]["url"], audio[0]["url"]],
+ time: video[0]["approxDurationMs"],
+ filename: `youtube_${obj.id}_${video[0]["width"]}x${video[0]["height"]}.${obj.format}`
+ }
+}
diff --git a/src/modules/processing/servicesConfig.json b/src/modules/processing/servicesConfig.json
index fa1fd968..98384e15 100644
--- a/src/modules/processing/servicesConfig.json
+++ b/src/modules/processing/servicesConfig.json
@@ -8,17 +8,18 @@
"enabled": true
},
"reddit": {
+ "alias": "reddit videos & gifs",
"patterns": ["r/:sub/comments/:id/:title"],
"enabled": true
},
"twitter": {
- "alias": "twitter posts & spaces",
+ "alias": "twitter posts & spaces & voice",
"patterns": [":user/status/:id", ":user/status/:id/video/:v", "i/spaces/:spaceId"],
"enabled": true
},
"vk": {
"alias": "vk video & clips",
- "patterns": ["video-:userId_:videoId", "clip-:userId_:videoId", "clips-:userId?z=clip-:userId_:videoId"],
+ "patterns": ["video-:userId_:videoId", "clip-:userId_:videoId", "clips-:duplicate?z=clip-:userId_:videoId"],
"quality_match": {
"2160": 7,
"1440": 6,
@@ -69,7 +70,7 @@
"enabled": true
},
"douyin": {
- "alias": "douyin videos & photos & audio",
+ "alias": "douyin videos & audio",
"patterns": ["video/:postId", ":id"],
"enabled": true
},
diff --git a/src/modules/processing/servicesPatternTesters.js b/src/modules/processing/servicesPatternTesters.js
index 56ca2b28..13695714 100644
--- a/src/modules/processing/servicesPatternTesters.js
+++ b/src/modules/processing/servicesPatternTesters.js
@@ -1,27 +1,28 @@
export const testers = {
- "twitter": (patternMatch) => (patternMatch["id"] && patternMatch["id"].length < 20) || (patternMatch["spaceId"] && patternMatch["spaceId"].length === 13),
+ "twitter": (patternMatch) => (patternMatch["id"] && patternMatch["id"].length < 20)
+ || (patternMatch["spaceId"] && patternMatch["spaceId"].length === 13),
- "vk": (patternMatch) => (patternMatch["userId"] && patternMatch["videoId"] &&
- patternMatch["userId"].length <= 10 && patternMatch["videoId"].length === 9),
+ "vk": (patternMatch) => (patternMatch["userId"] && patternMatch["videoId"]
+ && patternMatch["userId"].length <= 10 && patternMatch["videoId"].length === 9),
"bilibili": (patternMatch) => (patternMatch["id"] && patternMatch["id"].length >= 12),
"youtube": (patternMatch) => (patternMatch["id"] && patternMatch["id"].length >= 11),
- "reddit": (patternMatch) => (patternMatch["sub"] && patternMatch["id"] && patternMatch["title"] &&
- patternMatch["sub"].length <= 22 && patternMatch["id"].length <= 10 && patternMatch["title"].length <= 96),
+ "reddit": (patternMatch) => (patternMatch["sub"] && patternMatch["id"] && patternMatch["title"]
+ && patternMatch["sub"].length <= 22 && patternMatch["id"].length <= 10 && patternMatch["title"].length <= 96),
- "tiktok": (patternMatch) => ((patternMatch["user"] && patternMatch["postId"] && patternMatch["postId"].length <= 21) ||
- (patternMatch["id"] && patternMatch["id"].length <= 13)),
+ "tiktok": (patternMatch) => ((patternMatch["user"] && patternMatch["postId"] && patternMatch["postId"].length <= 21)
+ || (patternMatch["id"] && patternMatch["id"].length <= 13)),
- "douyin": (patternMatch) => ((patternMatch["postId"] && patternMatch["postId"].length <= 21) ||
- (patternMatch["id"] && patternMatch["id"].length <= 13)),
+ "douyin": (patternMatch) => ((patternMatch["postId"] && patternMatch["postId"].length <= 21)
+ || (patternMatch["id"] && patternMatch["id"].length <= 13)),
- "tumblr": (patternMatch) => ((patternMatch["id"] && patternMatch["id"].length < 21) ||
- (patternMatch["id"] && patternMatch["id"].length < 21 && patternMatch["user"] && patternMatch["user"].length <= 32)),
+ "tumblr": (patternMatch) => ((patternMatch["id"] && patternMatch["id"].length < 21)
+ || (patternMatch["id"] && patternMatch["id"].length < 21 && patternMatch["user"] && patternMatch["user"].length <= 32)),
"vimeo": (patternMatch) => ((patternMatch["id"] && patternMatch["id"].length <= 11)),
- "soundcloud": (patternMatch) => ((patternMatch["author"] && patternMatch["song"] && (patternMatch["author"].length + patternMatch["song"].length) <= 96) ||
- (patternMatch["shortLink"] && patternMatch["shortLink"].length <= 32))
-};
\ No newline at end of file
+ "soundcloud": (patternMatch) => ((patternMatch["author"] && patternMatch["song"]
+ && (patternMatch["author"].length + patternMatch["song"].length) <= 96) || (patternMatch["shortLink"] && patternMatch["shortLink"].length <= 32))
+}
diff --git a/src/modules/services/bilibili.js b/src/modules/services/bilibili.js
deleted file mode 100644
index 5d1902f7..00000000
--- a/src/modules/services/bilibili.js
+++ /dev/null
@@ -1,29 +0,0 @@
-import { genericUserAgent, maxVideoDuration } from "../config.js";
-
-export default async function(obj) {
- try {
- let html = await fetch(`https://bilibili.com/video/${obj.id}`, {
- headers: {"user-agent": genericUserAgent}
- }).then((r) => {return r.text()}).catch(() => {return false});
- if (!html) return { error: 'ErrorCouldntFetch' };
-
- if (html.includes('')[0]);
- if (streamData.data.timelength <= maxVideoDuration) {
- let video = streamData["data"]["dash"]["video"].filter((v) => {
- if (!v["baseUrl"].includes("https://upos-sz-mirrorcosov.bilivideo.com/")) return true;
- }).sort((a, b) => Number(b.bandwidth) - Number(a.bandwidth));
- let audio = streamData["data"]["dash"]["audio"].filter((a) => {
- if (!a["baseUrl"].includes("https://upos-sz-mirrorcosov.bilivideo.com/")) return true;
- }).sort((a, b) => Number(b.bandwidth) - Number(a.bandwidth));
- return { urls: [video[0]["baseUrl"], audio[0]["baseUrl"]], time: streamData.data.timelength, audioFilename: `bilibili_${obj.id}_audio`, filename: `bilibili_${obj.id}_${video[0]["width"]}x${video[0]["height"]}.mp4` };
- } else {
- return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
- }
- } else {
- return { error: 'ErrorEmptyDownload' };
- }
- } catch (e) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/reddit.js b/src/modules/services/reddit.js
deleted file mode 100644
index 2323a028..00000000
--- a/src/modules/services/reddit.js
+++ /dev/null
@@ -1,27 +0,0 @@
-import { maxVideoDuration } from "../config.js";
-
-export default async function(obj) {
- try {
- let data = await fetch(`https://www.reddit.com/r/${obj.sub}/comments/${obj.id}/${obj.name}.json`).then((r) => {return r.json()}).catch(() => {return false});
- if (!data) return { error: 'ErrorCouldntFetch' };
- data = data[0]["data"]["children"][0]["data"];
-
- if ("reddit_video" in data["secure_media"] && data["secure_media"]["reddit_video"]["duration"] * 1000 < maxVideoDuration) {
- let video = data["secure_media"]["reddit_video"]["fallback_url"].split('?')[0],
- audio = video.match('.mp4') ? `${video.split('_')[0]}_audio.mp4` : `${data["secure_media"]["reddit_video"]["fallback_url"].split('DASH')[0]}audio`;
-
- await fetch(audio, {method: "HEAD"}).then((r) => {if (r.status != 200) audio = ''}).catch(() => {audio = ''});
-
- let id = data["secure_media"]["reddit_video"]["fallback_url"].split('/')[3]
- if (audio.length > 0) {
- return { typeId: 2, type: "render", urls: [video, audio], audioFilename: `reddit_${id}_audio`, filename: `reddit_${id}.mp4` };
- } else {
- return { typeId: 1, urls: video };
- }
- } else {
- return { error: 'ErrorEmptyDownload' };
- }
- } catch (err) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/soundcloud.js b/src/modules/services/soundcloud.js
deleted file mode 100644
index af26a92c..00000000
--- a/src/modules/services/soundcloud.js
+++ /dev/null
@@ -1,79 +0,0 @@
-import { genericUserAgent, maxAudioDuration } from "../config.js";
-
-let cachedID = {}
-
-async function findClientID() {
- try {
- let sc = await fetch('https://soundcloud.com/').then((r) => {return r.text()}).catch(() => {return false});
- let sc_version = String(sc.match(/')[0])
- if (json["media"]["transcodings"]) {
- let clientId = await findClientID();
- if (clientId) {
- let fileUrlBase = json.media.transcodings[0]["url"].replace("/hls", "/progressive")
- let fileUrl = `${fileUrlBase}${fileUrlBase.includes("?") ? "&" : "?"}client_id=${clientId}&track_authorization=${json.track_authorization}`;
- if (fileUrl.substring(0, 54) === "https://api-v2.soundcloud.com/media/soundcloud:tracks:") {
- if (json.duration < maxAudioDuration) {
- let file = await fetch(fileUrl).then(async (r) => {return (await r.json()).url}).catch(() => {return false});
- if (!file) return { error: 'ErrorCouldntFetch' };
- return {
- urls: file,
- audioFilename: `soundcloud_${json.id}`,
- fileMetadata: {
- title: json.title,
- artist: json.user.username,
- }
- }
- } else return { error: ['ErrorLengthAudioConvert', maxAudioDuration / 60000] }
- }
- } else return { error: 'ErrorSoundCloudNoClientId' }
- } else return { error: 'ErrorEmptyDownload' }
- } else return { error: ['ErrorBrokenLink', 'soundcloud'] }
- } catch (e) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/tiktok.js b/src/modules/services/tiktok.js
deleted file mode 100644
index ec1b1ce9..00000000
--- a/src/modules/services/tiktok.js
+++ /dev/null
@@ -1,116 +0,0 @@
-import { genericUserAgent } from "../config.js";
-
-let userAgent = genericUserAgent.split(' Chrome/1')[0]
-let config = {
- tiktok: {
- short: "https://vt.tiktok.com/",
- api: "https://api2.musical.ly/aweme/v1/feed/?aweme_id={postId}&version_code=262&app_name=musical_ly&channel=App&device_id=null&os_version=14.4.2&device_platform=iphone&device_type=iPhone9®ion=US&carrier_region=US",
- },
- douyin: {
- short: "https://v.douyin.com/",
- api: "https://www.iesdouyin.com/web/api/v2/aweme/iteminfo/?item_ids={postId}",
- }
-}
-function selector(j, h, id) {
- if (j) {
- let t;
- switch (h) {
- case "tiktok":
- t = j["aweme_list"].filter((v) => { if (v["aweme_id"] == id) return true })
- break;
- case "douyin":
- t = j['item_list'].filter((v) => { if (v["aweme_id"] == id) return true })
- break;
- }
- if (t.length > 0) { return t[0] } else return false
- } else return false
-}
-
-export default async function(obj) {
- try {
- if (!obj.postId) {
- let html = await fetch(`${config[obj.host]["short"]}${obj.id}`, {
- redirect: "manual",
- headers: { "user-agent": userAgent }
- }).then((r) => {return r.text()}).catch(() => {return false});
- if (!html) return { error: 'ErrorCouldntFetch' };
-
- if (html.slice(0, 17) === ' {return r.json()}).catch(() => {return false});
-
- detail = selector(detail, obj.host, obj.postId);
-
- if (!detail) return { error: 'ErrorCouldntFetch' }
-
- let video, videoFilename, audioFilename, isMp3, audio, images,
- filenameBase = `${obj.host}_${obj.postId}`;
- if (obj.host == "tiktok") {
- images = detail["image_post_info"] ? detail["image_post_info"]["images"] : false
- } else {
- images = detail["images"] ? detail["images"] : false
- }
- if (!obj.isAudioOnly && !images) {
- video = obj.host === "tiktok" ? detail["video"]["play_addr"]["url_list"][0] : detail["video"]["play_addr"]["url_list"][0].replace("playwm", "play");
- videoFilename = `${filenameBase}_video_nw.mp4` // nw - no watermark
- if (!obj.noWatermark) {
- video = obj.host === "tiktok" ? detail["video"]["download_addr"]["url_list"][0] : detail['video']['play_addr']['url_list'][0]
- videoFilename = `${filenameBase}_video.mp4`
- }
- } else {
- let fallback = obj.host === "douyin" ? detail["video"]["play_addr"]["url_list"][0].replace("playwm", "play") : detail["video"]["play_addr"]["url_list"][0];
- if (obj.fullAudio || fallback.includes("music")) {
- audio = detail["music"]["play_url"]["url_list"][0]
- audioFilename = `${filenameBase}_audio`
- } else {
- audio = fallback
- audioFilename = `${filenameBase}_audio_fv` // fv - from video
- }
- if (audio.slice(-4) === ".mp3") isMp3 = true;
- }
- if (video) return {
- urls: video,
- filename: videoFilename
- }
- if (images && obj.isAudioOnly) {
- return {
- urls: audio,
- audioFilename: audioFilename,
- isAudioOnly: true,
- isMp3: isMp3,
- }
- }
- if (images) {
- let imageLinks = [];
- for (let i in images) {
- let sel = obj.host == "tiktok" ? images[i]["display_image"]["url_list"] : images[i]["url_list"];
- sel = sel.filter((p) => { if (p.includes(".jpeg?")) return true; })
- imageLinks.push({url: sel[0]})
- }
- return {
- picker: imageLinks,
- urls: audio,
- audioFilename: audioFilename,
- isAudioOnly: true,
- isMp3: isMp3,
- }
- }
- if (audio) return {
- urls: audio,
- audioFilename: audioFilename,
- isAudioOnly: true,
- isMp3: isMp3,
- }
- } catch (e) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/tumblr.js b/src/modules/services/tumblr.js
deleted file mode 100644
index e0bde0a1..00000000
--- a/src/modules/services/tumblr.js
+++ /dev/null
@@ -1,16 +0,0 @@
-import { genericUserAgent } from "../config.js";
-
-export default async function(obj) {
- try {
- let user = obj.user ? obj.user : obj.url.split('.')[0].replace('https://', '');
- let html = await fetch(`https://${user}.tumblr.com/post/${obj.id}`, {
- headers: {"user-agent": genericUserAgent}
- }).then((r) => {return r.text()}).catch(() => {return false});
- if (!html) return { error: 'ErrorCouldntFetch' };
- if (html.includes('property="og:video" content="https://va.media.tumblr.com/')) {
- return { urls: `https://va.media.tumblr.com/${html.split('property="og:video" content="https://va.media.tumblr.com/')[1].split('"')[0]}`, audioFilename: `tumblr_${obj.id}_audio` }
- } else return { error: 'ErrorEmptyDownload' }
- } catch (e) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/twitter.js b/src/modules/services/twitter.js
deleted file mode 100644
index c9766c2e..00000000
--- a/src/modules/services/twitter.js
+++ /dev/null
@@ -1,102 +0,0 @@
-import { genericUserAgent } from "../config.js";
-
-function bestQuality(arr) {
- return arr.filter((v) => { if (v["content_type"] === "video/mp4") return true; }).sort((a, b) => Number(b.bitrate) - Number(a.bitrate))[0]["url"].split("?")[0]
-}
-const apiURL = "https://api.twitter.com/1.1"
-
-export default async function(obj) {
- try {
- let _headers = {
- "user-agent": genericUserAgent,
- "authorization": "Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA",
- "host": "api.twitter.com"
- };
- let req_act = await fetch(`${apiURL}/guest/activate.json`, {
- method: "POST",
- headers: _headers
- }).then((r) => { return r.status == 200 ? r.json() : false;}).catch(() => {return false});
-
- if (!req_act) return { error: 'ErrorCouldntFetch' };
- _headers["x-guest-token"] = req_act["guest_token"];
- let showURL = `${apiURL}/statuses/show/${obj.id}.json?tweet_mode=extended&include_user_entities=0&trim_user=1&include_entities=0&cards_platform=Web-12&include_cards=1`
- if (!obj.spaceId) {
- let req_status = await fetch(showURL, { headers: _headers }).then((r) => { return r.status == 200 ? r.json() : false;}).catch((e) => { return false});
- if (!req_status) {
- _headers.authorization = "Bearer AAAAAAAAAAAAAAAAAAAAAPYXBAAAAAAACLXUNDekMxqa8h%2F40K4moUkGsoc%3DTYfbDKbT3jJPCEVnMYqilB28NHfOPqkca3qaAxGfsyKCs0wRbw";
- delete _headers["x-guest-token"]
-
- req_act = await fetch(`${apiURL}/guest/activate.json`, {
- method: "POST",
- headers: _headers
- }).then((r) => { return r.status == 200 ? r.json() : false;}).catch(() => {return false});
- if (!req_act) return { error: 'ErrorCouldntFetch' };
-
- _headers["x-guest-token"] = req_act["guest_token"];
- req_status = await fetch(showURL, { headers: _headers }).then((r) => { return r.status == 200 ? r.json() : false;}).catch(() => {return false});
- }
- if (!req_status) return { error: 'ErrorCouldntFetch' }
- if (req_status["extended_entities"] && req_status["extended_entities"]["media"]) {
- let single, multiple = [], media = req_status["extended_entities"]["media"];
- media = media.filter((i) => { if (i["type"] === "video" || i["type"] === "animated_gif") return true })
- if (media.length > 1) {
- for (let i in media) { multiple.push({type: "video", thumb: media[i]["media_url_https"], url: bestQuality(media[i]["video_info"]["variants"])}) }
- } else if (media.length > 0) {
- single = bestQuality(media[0]["video_info"]["variants"])
- } else {
- return { error: 'ErrorNoVideosInTweet' }
- }
- if (single) {
- return { urls: single, filename: `twitter_${obj.id}.mp4`, audioFilename: `twitter_${obj.id}_audio` }
- } else if (multiple) {
- return { picker: multiple }
- } else {
- return { error: 'ErrorNoVideosInTweet' }
- }
- } else {
- return { error: 'ErrorNoVideosInTweet' }
- }
- } else {
- _headers["host"] = "twitter.com"
- _headers["content-type"] = "application/json"
- let query = {
- variables: {"id": obj.spaceId,"isMetatagsQuery":true,"withSuperFollowsUserFields":true,"withDownvotePerspective":false,"withReactionsMetadata":false,"withReactionsPerspective":false,"withSuperFollowsTweetFields":true,"withReplays":true}, features: {"spaces_2022_h2_clipping":true,"spaces_2022_h2_spaces_communities":true,"verified_phone_label_enabled":false,"tweetypie_unmention_optimization_enabled":true,"responsive_web_uc_gql_enabled":true,"vibe_api_enabled":true,"responsive_web_edit_tweet_api_enabled":true,"graphql_is_translatable_rweb_tweet_is_translatable_enabled":true,"standardized_nudges_misinfo":true,"tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled":false,"responsive_web_graphql_timeline_navigation_enabled":false,"interactive_text_enabled":true,"responsive_web_text_conversations_enabled":false,"responsive_web_enhance_cards_enabled":true}
- }
-
- let AudioSpaceById = await fetch(`https://twitter.com/i/api/graphql/wJ5g4zf7v8qPHSQbaozYuw/AudioSpaceById?variables=${new URLSearchParams(JSON.stringify(query.variables)).toString().slice(0, -1)}&features=${new URLSearchParams(JSON.stringify(query.features)).toString().slice(0, -1)}`, { headers: _headers }).then((r) => {
- return r.status == 200 ? r.json() : false;
- }).catch((e) => {return false});
-
- if (AudioSpaceById) {
- if (AudioSpaceById.data.audioSpace.metadata.is_space_available_for_replay === true) {
- let streamStatus = await fetch(`https://twitter.com/i/api/1.1/live_video_stream/status/${AudioSpaceById.data.audioSpace.metadata.media_key}`, { headers: _headers }).then((r) => {return r.status == 200 ? r.json() : false;}).catch(() => {return false;});
- if (!streamStatus) return { error: 'ErrorCouldntFetch' };
-
- let participants = AudioSpaceById.data.audioSpace.participants.speakers
- let listOfParticipants = `Twitter Space speakers: `
- for (let i in participants) {
- listOfParticipants += `@${participants[i]["twitter_screen_name"]}, `
- }
- listOfParticipants = listOfParticipants.slice(0, -2);
- return {
- urls: streamStatus.source.noRedirectPlaybackUrl,
- audioFilename: `twitterspaces_${obj.spaceId}`,
- isAudioOnly: true,
- fileMetadata: {
- title: AudioSpaceById.data.audioSpace.metadata.title,
- artist: `Twitter Space by @${AudioSpaceById.data.audioSpace.metadata.creator_results.result.legacy.screen_name}`,
- comment: listOfParticipants,
- // cover: AudioSpaceById.data.audioSpace.metadata.creator_results.result.legacy.profile_image_url_https.replace("_normal", "")
- }
- }
- } else {
- return { error: 'TwitterSpaceWasntRecorded' };
- }
- } else {
- return { error: 'ErrorEmptyDownload' }
- }
- }
- } catch (err) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/vimeo.js b/src/modules/services/vimeo.js
deleted file mode 100644
index a5e88f3c..00000000
--- a/src/modules/services/vimeo.js
+++ /dev/null
@@ -1,80 +0,0 @@
-import { quality, services } from "../config.js";
-
-export default async function(obj) {
- try {
- let api = await fetch(`https://player.vimeo.com/video/${obj.id}/config`).then((r) => {return r.json()}).catch(() => {return false});
- if (!api) return { error: 'ErrorCouldntFetch' };
-
- let downloadType = "";
- if (JSON.stringify(api).includes('"progressive":[{')) {
- downloadType = "progressive";
- } else if (JSON.stringify(api).includes('"files":{"dash":{"')) downloadType = "dash";
-
- switch(downloadType) {
- case "progressive":
- let all = api["request"]["files"]["progressive"].sort((a, b) => Number(b.width) - Number(a.width));
- let best = all[0]
- try {
- if (obj.quality != "max") {
- let pref = parseInt(quality[obj.quality], 10)
- for (let i in all) {
- let currQuality = parseInt(all[i]["quality"].replace('p', ''), 10)
- if (currQuality < pref) {
- break;
- } else if (currQuality == pref) {
- best = all[i]
- }
- }
- }
- } catch (e) {
- best = all[0]
- }
- return { urls: best["url"], filename: `tumblr_${obj.id}.mp4` };
- case "dash":
- let masterJSONURL = api["request"]["files"]["dash"]["cdns"]["akfire_interconnect_quic"]["url"];
- let masterJSON = await fetch(masterJSONURL).then((r) => {return r.json()}).catch(() => {return false});
- if (!masterJSON) return { error: 'ErrorCouldntFetch' };
- if (masterJSON.video) {
- let type = "";
- if (masterJSON.base_url.includes("parcel")) {
- type = "parcel"
- } else if (masterJSON.base_url == "../") {
- type = "chop"
- }
- let masterJSON_Video = masterJSON.video.sort((a, b) => Number(b.width) - Number(a.width));
- let masterJSON_Audio = masterJSON.audio.sort((a, b) => Number(b.bitrate) - Number(a.bitrate)).filter((a)=> {if (a['mime_type'] === "audio/mp4") return true;});
-
- let bestVideo = masterJSON_Video[0]
- let bestAudio = masterJSON_Audio[0]
- switch (type) {
- case "parcel":
- if (obj.quality != "max") {
- let pref = parseInt(quality[obj.quality], 10)
- for (let i in masterJSON_Video) {
- let currQuality = parseInt(services.vimeo.resolutionMatch[masterJSON_Video[i]["width"]], 10)
- if (currQuality < pref) {
- break;
- } else if (currQuality == pref) {
- bestVideo = masterJSON_Video[i]
- }
- }
- }
- let baseUrl = masterJSONURL.split("/sep/")[0]
- let videoUrl = `${baseUrl}/parcel/video/${bestVideo.index_segment.split('?')[0]}`;
- let audioUrl = `${baseUrl}/parcel/audio/${bestAudio.index_segment.split('?')[0]}`;
-
- return { urls: [videoUrl, audioUrl], audioFilename: `vimeo_${obj.id}_audio`, filename: `vimeo_${obj.id}_${bestVideo["width"]}x${bestVideo["height"]}.mp4` }
- case "chop": // TO-DO: support chop type of streams
- default:
- return { error: 'ErrorEmptyDownload' }
- }
- } else {
- return { error: 'ErrorEmptyDownload' }
- }
- default:
- return { error: 'ErrorEmptyDownload' }
- }
- } catch (e) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/vk.js b/src/modules/services/vk.js
deleted file mode 100644
index ca2d826e..00000000
--- a/src/modules/services/vk.js
+++ /dev/null
@@ -1,58 +0,0 @@
-import { xml2json } from "xml-js";
-import { genericUserAgent, maxVideoDuration, services } from "../config.js";
-import selectQuality from "../stream/selectQuality.js";
-
-export default async function(obj) {
- try {
- let html;
- html = await fetch(`https://vk.com/video-${obj.userId}_${obj.videoId}`, {
- headers: {"user-agent": genericUserAgent}
- }).then((r) => {return r.text()}).catch(() => {return false});
- if (!html) return { error: 'ErrorCouldntFetch' };
- if (html.includes(`{"lang":`)) {
- let js = JSON.parse('{"lang":' + html.split(`{"lang":`)[1].split(']);')[0]);
- if (js["mvData"]["is_active_live"] == '0') {
- if (js["mvData"]["duration"] <= maxVideoDuration / 1000) {
- let mpd = JSON.parse(xml2json(js["player"]["params"][0]["manifest"], { compact: true, spaces: 4 }));
-
- let repr = mpd["MPD"]["Period"]["AdaptationSet"]["Representation"];
- if (!mpd["MPD"]["Period"]["AdaptationSet"]["Representation"]) {
- repr = mpd["MPD"]["Period"]["AdaptationSet"][0]["Representation"];
- }
- let attr = repr[repr.length - 1]["_attributes"];
- let selectedQuality;
- let qualities = Object.keys(services.vk.quality_match);
- for (let i in qualities) {
- if (qualities[i] == attr["height"]) {
- selectedQuality = `url${attr["height"]}`;
- break;
- }
- if (qualities[i] == attr["width"]) {
- selectedQuality = `url${attr["width"]}`;
- break;
- }
- }
- let maxQuality = js["player"]["params"][0][selectedQuality].split('type=')[1].slice(0, 1)
- let userQuality = selectQuality('vk', obj.quality, Object.entries(services.vk.quality_match).reduce((r, [k, v]) => { r[v] = k; return r; })[maxQuality]);
- let userRepr = repr[services.vk.representation_match[userQuality]]["_attributes"];
- if (selectedQuality in js["player"]["params"][0]) {
- return {
- urls: js["player"]["params"][0][`url${userQuality}`],
- filename: `vk_${obj.userId}_${obj.videoId}_${userRepr["width"]}x${userRepr['height']}.mp4`
- };
- } else {
- return { error: 'ErrorEmptyDownload' };
- }
- } else {
- return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
- }
- } else {
- return { error: 'ErrorLiveVideo' };
- }
- } else {
- return { error: 'ErrorEmptyDownload' };
- }
- } catch (err) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/services/youtube.js b/src/modules/services/youtube.js
deleted file mode 100644
index 6d247e4d..00000000
--- a/src/modules/services/youtube.js
+++ /dev/null
@@ -1,98 +0,0 @@
-import ytdl from "better-ytdl-core";
-import { maxVideoDuration, quality as mq } from "../config.js";
-import selectQuality from "../stream/selectQuality.js";
-
-export default async function(obj) {
- try {
- let infoInitial = await ytdl.getInfo(obj.id);
- if (infoInitial) {
- let info = infoInitial.formats;
- if (!info[0]["isLive"]) {
- let videoMatch = [], fullVideoMatch = [], video = [], audio = info.filter((a) => {
- if (!a["isHLS"] && !a["isDashMPD"] && a["hasAudio"] && !a["hasVideo"] && a["container"] == obj.format) return true;
- }).sort((a, b) => Number(b.bitrate) - Number(a.bitrate));
- if (!obj.isAudioOnly) {
- video = info.filter((a) => {
- if (!a["isHLS"] && !a["isDashMPD"] && a["hasVideo"] && a["container"] == obj.format) {
- if (obj.quality != "max") {
- if (a["hasAudio"] && mq[obj.quality] == a["height"]) {
- fullVideoMatch.push(a)
- } else if (!a["hasAudio"] && mq[obj.quality] == a["height"]) {
- videoMatch.push(a);
- }
- }
- return true
- }
- }).sort((a, b) => Number(b.bitrate) - Number(a.bitrate));
- if (obj.quality != "max") {
- if (videoMatch.length == 0) {
- let ss = selectQuality("youtube", obj.quality, video[0]["qualityLabel"].slice(0, 5).replace('p', '').trim())
- videoMatch = video.filter((a) => {
- if (a["qualityLabel"].slice(0, 5).replace('p', '').trim() == ss) return true;
- })
- } else if (fullVideoMatch.length > 0) {
- videoMatch = [fullVideoMatch[0]]
- }
- } else videoMatch = [video[0]];
- if (obj.quality == "los") videoMatch = [video[video.length - 1]];
- }
- let generalMeta = {
- title: infoInitial.videoDetails.title,
- artist: infoInitial.videoDetails.ownerChannelName.replace("- Topic", "").trim(),
- }
- if (audio[0]["approxDurationMs"] <= maxVideoDuration) {
- if (!obj.isAudioOnly && videoMatch.length > 0) {
- if (video.length > 0 && audio.length > 0) {
- if (videoMatch[0]["hasVideo"] && videoMatch[0]["hasAudio"]) {
- return {
- type: "bridge", urls: videoMatch[0]["url"], time: videoMatch[0]["approxDurationMs"],
- filename: `youtube_${obj.id}_${videoMatch[0]["width"]}x${videoMatch[0]["height"]}.${obj.format}`
- };
- } else {
- return {
- type: "render", urls: [videoMatch[0]["url"], audio[0]["url"]], time: videoMatch[0]["approxDurationMs"],
- filename: `youtube_${obj.id}_${videoMatch[0]["width"]}x${videoMatch[0]["height"]}.${obj.format}`
- };
- }
- } else {
- return { error: 'ErrorBadFetch' };
- }
- } else if (!obj.isAudioOnly) {
- return {
- type: "render", urls: [video[0]["url"], audio[0]["url"]], time: video[0]["approxDurationMs"],
- filename: `youtube_${obj.id}_${video[0]["width"]}x${video[0]["height"]}.${video[0]["container"]}`
- };
- } else if (audio.length > 0) {
- let r = {
- type: "render",
- isAudioOnly: true,
- urls: audio[0]["url"],
- audioFilename: `youtube_${obj.id}_audio`,
- fileMetadata: generalMeta
- };
- if (infoInitial.videoDetails.description) {
- let isAutoGenAudio = infoInitial.videoDetails.description.startsWith("Provided to YouTube by");
- if (isAutoGenAudio) {
- let descItems = infoInitial.videoDetails.description.split("\n\n")
- r.fileMetadata.album = descItems[2]
- r.fileMetadata.copyright = descItems[3]
- if (descItems[4].startsWith("Released on:")) r.fileMetadata.date = descItems[4].replace("Released on: ", '').trim();
- }
- }
- return r
- } else {
- return { error: 'ErrorBadFetch' };
- }
- } else {
- return { error: ['ErrorLengthLimit', maxVideoDuration / 60000] };
- }
- } else {
- return { error: 'ErrorLiveVideo' };
- }
- } else {
- return { error: 'ErrorCantConnectToServiceAPI' };
- }
- } catch (e) {
- return { error: 'ErrorBadFetch' };
- }
-}
diff --git a/src/modules/setup.js b/src/modules/setup.js
index 625d4beb..aa64370e 100644
--- a/src/modules/setup.js
+++ b/src/modules/setup.js
@@ -33,22 +33,15 @@ console.log(
)
rl.question(q, r1 => {
- if (r1) {
- ob['selfURL'] = `https://${r1}/`
- } else {
- ob['selfURL'] = `http://localhost`
- }
+ ob['selfURL'] = `http://localhost:9000/`
+ ob['port'] = 9000
+ if (r1) ob['selfURL'] = `https://${r1}/`
+
console.log(Bright("\nGreat! Now, what's the port it'll be running on? (9000)"))
+
rl.question(q, r2 => {
- if (!r1 && !r2) {
- ob['selfURL'] = `http://localhost:9000/`
- ob['port'] = 9000
- } else if (!r1 && r2) {
- ob['selfURL'] = `http://localhost:${r2}/`
- ob['port'] = r2
- } else {
- ob['port'] = r2
- }
+ if (r2) ob['port'] = r2
+ if (!r1 && r2) ob['selfURL'] = `http://localhost:${r2}/`
final()
});
})
diff --git a/src/modules/stream/manage.js b/src/modules/stream/manage.js
index 7da39d9c..5da23c3c 100644
--- a/src/modules/stream/manage.js
+++ b/src/modules/stream/manage.js
@@ -43,16 +43,15 @@ export function createStream(obj) {
export function verifyStream(ip, id, hmac, exp) {
try {
let streamInfo = streamCache.get(id);
- if (streamInfo) {
- let ghmac = sha256(`${id},${streamInfo.service},${ip},${exp}`, salt);
- if (hmac == ghmac && exp.toString() == streamInfo.exp && ghmac == streamInfo.hmac && ip == streamInfo.ip && exp > Math.floor(new Date().getTime())) {
- return streamInfo;
- } else {
- return { error: 'Unauthorized', status: 401 };
- }
- } else {
+ if (!streamInfo) {
return { error: 'this stream token does not exist', status: 400 };
}
+ let ghmac = sha256(`${id},${streamInfo.service},${ip},${exp}`, salt);
+ if (String(hmac) === ghmac && String(exp) === String(streamInfo.exp) && ghmac === String(streamInfo.hmac)
+ && String(ip) === streamInfo.ip && Number(exp) > Math.floor(new Date().getTime())) {
+ return streamInfo;
+ }
+ return { error: 'Unauthorized', status: 401 };
} catch (e) {
return { status: 500, body: { status: "error", text: "Internal Server Error" } };
}
diff --git a/src/modules/stream/selectQuality.js b/src/modules/stream/selectQuality.js
index d21a5f23..56244482 100644
--- a/src/modules/stream/selectQuality.js
+++ b/src/modules/stream/selectQuality.js
@@ -1,5 +1,6 @@
import { services, quality as mq } from "../config.js";
+// TO-DO: remake entirety of this module to be more of how quality picking is done in vimeo module
function closest(goal, array) {
return array.sort().reduce(function (prev, curr) {
return (Math.abs(curr - goal) < Math.abs(prev - goal) ? curr : prev);
@@ -7,17 +8,15 @@ function closest(goal, array) {
}
export default function(service, quality, maxQuality) {
- if (quality == "max") return maxQuality;
+ if (quality === "max") return maxQuality;
quality = parseInt(mq[quality], 10)
maxQuality = parseInt(maxQuality, 10)
- if (quality >= maxQuality || quality == maxQuality) return maxQuality;
+ if (quality >= maxQuality || quality === maxQuality) return maxQuality;
if (quality < maxQuality) {
- if (services[service]["quality"][quality]) {
- return quality
- } else {
+ if (!services[service]["quality"][quality]) {
let s = Object.keys(services[service]["quality_match"]).filter((q) => {
if (q <= quality) {
return true
@@ -25,5 +24,6 @@ export default function(service, quality, maxQuality) {
})
return closest(quality, s)
}
+ return quality
}
}
diff --git a/src/modules/stream/stream.js b/src/modules/stream/stream.js
index eb843086..7f9b42e6 100644
--- a/src/modules/stream/stream.js
+++ b/src/modules/stream/stream.js
@@ -5,24 +5,24 @@ import { streamAudioOnly, streamDefault, streamLiveRender, streamVideoOnly } fro
export default function(res, ip, id, hmac, exp) {
try {
let streamInfo = verifyStream(ip, id, hmac, exp);
- if (!streamInfo.error) {
- if (streamInfo.isAudioOnly && streamInfo.type !== "bridge") {
- streamAudioOnly(streamInfo, res);
- } else {
- switch (streamInfo.type) {
- case "render":
- streamLiveRender(streamInfo, res);
- break;
- case "mute":
- streamVideoOnly(streamInfo, res);
- break;
- default:
- streamDefault(streamInfo, res);
- break;
- }
- }
- } else {
+ if (streamInfo.error) {
res.status(streamInfo.status).json(apiJSON(0, { t: streamInfo.error }).body);
+ return;
+ }
+ if (streamInfo.isAudioOnly && streamInfo.type !== "bridge") {
+ streamAudioOnly(streamInfo, res);
+ return;
+ }
+ switch (streamInfo.type) {
+ case "render":
+ streamLiveRender(streamInfo, res);
+ break;
+ case "mute":
+ streamVideoOnly(streamInfo, res);
+ break;
+ default:
+ streamDefault(streamInfo, res);
+ break;
}
} catch (e) {
res.status(500).json({ status: "error", text: "Internal Server Error" });
diff --git a/src/modules/stream/types.js b/src/modules/stream/types.js
index ef6ac22c..6c7fd691 100644
--- a/src/modules/stream/types.js
+++ b/src/modules/stream/types.js
@@ -15,10 +15,10 @@ export function streamDefault(streamInfo, res) {
},
isStream: true
});
- stream.pipe(res).on('error', (err) => {
+ stream.pipe(res).on('error', () => {
res.end();
});
- stream.on('error', (err) => {
+ stream.on('error', () => {
res.end();
});
} catch (e) {
@@ -27,39 +27,41 @@ export function streamDefault(streamInfo, res) {
}
export function streamLiveRender(streamInfo, res) {
try {
- if (streamInfo.urls.length === 2) {
- let format = streamInfo.filename.split('.')[streamInfo.filename.split('.').length - 1], args = [
- '-loglevel', '-8',
- '-i', streamInfo.urls[0],
- '-i', streamInfo.urls[1],
- '-map', '0:v',
- '-map', '1:a',
- ];
- args = args.concat(ffmpegArgs[format])
- if (streamInfo.time) args.push('-t', msToTime(streamInfo.time));
- args.push('-f', format, 'pipe:3');
- const ffmpegProcess = spawn(ffmpeg, args, {
- windowsHide: true,
- stdio: [
- 'inherit', 'inherit', 'inherit',
- 'pipe'
- ],
- });
- res.setHeader('Connection', 'keep-alive');
- res.setHeader('Content-Disposition', `attachment; filename="${streamInfo.filename}"`);
- ffmpegProcess.stdio[3].pipe(res);
- ffmpegProcess.on('disconnect', () => ffmpegProcess.kill());
- ffmpegProcess.on('close', () => ffmpegProcess.kill());
- ffmpegProcess.on('exit', () => ffmpegProcess.kill());
- res.on('finish', () => ffmpegProcess.kill());
- res.on('close', () => ffmpegProcess.kill());
- ffmpegProcess.on('error', (err) => {
- ffmpegProcess.kill();
- res.end();
- });
- } else {
+ if (streamInfo.urls.length !== 2) {
res.end();
+ return;
}
+ let format = streamInfo.filename.split('.')[streamInfo.filename.split('.').length - 1], args = [
+ '-loglevel', '-8',
+ '-i', streamInfo.urls[0],
+ '-i', streamInfo.urls[1],
+ '-map', '0:v',
+ '-map', '1:a',
+ ];
+ args = args.concat(ffmpegArgs[format])
+ if (streamInfo.time) args.push('-t', msToTime(streamInfo.time));
+ args.push('-f', format, 'pipe:3');
+ const ffmpegProcess = spawn(ffmpeg, args, {
+ windowsHide: true,
+ stdio: [
+ 'inherit', 'inherit', 'inherit',
+ 'pipe'
+ ],
+ });
+ res.setHeader('Connection', 'keep-alive');
+ res.setHeader('Content-Disposition', `attachment; filename="${streamInfo.filename}"`);
+ ffmpegProcess.stdio[3].pipe(res);
+
+ ffmpegProcess.on('disconnect', () => ffmpegProcess.kill());
+ ffmpegProcess.on('close', () => ffmpegProcess.kill());
+ ffmpegProcess.on('exit', () => ffmpegProcess.kill());
+ res.on('finish', () => ffmpegProcess.kill());
+ res.on('close', () => ffmpegProcess.kill());
+ ffmpegProcess.on('error', () => {
+ ffmpegProcess.kill();
+ res.end();
+ });
+
} catch (e) {
res.end();
}
@@ -93,12 +95,13 @@ export function streamAudioOnly(streamInfo, res) {
res.setHeader('Connection', 'keep-alive');
res.setHeader('Content-Disposition', `attachment; filename="${streamInfo.filename}.${streamInfo.audioFormat}"`);
ffmpegProcess.stdio[3].pipe(res);
+
ffmpegProcess.on('disconnect', () => ffmpegProcess.kill());
ffmpegProcess.on('close', () => ffmpegProcess.kill());
ffmpegProcess.on('exit', () => ffmpegProcess.kill());
res.on('finish', () => ffmpegProcess.kill());
res.on('close', () => ffmpegProcess.kill());
- ffmpegProcess.on('error', (err) => {
+ ffmpegProcess.on('error', () => {
ffmpegProcess.kill();
res.end();
});
@@ -113,7 +116,7 @@ export function streamVideoOnly(streamInfo, res) {
'-i', streamInfo.urls,
'-c', 'copy', '-an'
]
- if (format == "mp4") args.push('-movflags', 'faststart+frag_keyframe+empty_moov')
+ if (format === "mp4") args.push('-movflags', 'faststart+frag_keyframe+empty_moov')
args.push('-f', format, 'pipe:3');
const ffmpegProcess = spawn(ffmpeg, args, {
windowsHide: true,
@@ -125,12 +128,13 @@ export function streamVideoOnly(streamInfo, res) {
res.setHeader('Connection', 'keep-alive');
res.setHeader('Content-Disposition', `attachment; filename="${streamInfo.filename.split('.')[0]}_mute.${format}"`);
ffmpegProcess.stdio[3].pipe(res);
+
ffmpegProcess.on('disconnect', () => ffmpegProcess.kill());
ffmpegProcess.on('close', () => ffmpegProcess.kill());
ffmpegProcess.on('exit', () => ffmpegProcess.kill());
res.on('finish', () => ffmpegProcess.kill());
res.on('close', () => ffmpegProcess.kill());
- ffmpegProcess.on('error', (err) => {
+ ffmpegProcess.on('error', () => {
ffmpegProcess.kill();
res.end();
});
diff --git a/src/modules/sub/currentCommit.js b/src/modules/sub/currentCommit.js
index 7dfb1f9b..f3c145f5 100644
--- a/src/modules/sub/currentCommit.js
+++ b/src/modules/sub/currentCommit.js
@@ -1,10 +1,23 @@
import { execSync } from "child_process";
+let commit, commitInfo, branch;
+
export function shortCommit() {
- return execSync('git rev-parse --short HEAD').toString().trim()
+ if (commit) return commit;
+ let c = execSync('git rev-parse --short HEAD').toString().trim();
+ commit = c;
+ return c
}
export function getCommitInfo() {
- let d = execSync(`git show -s --format='%s;;;%B'`).toString().trim().replace(/[\r\n]/gm, '\n').split(';;;')
- d[1] = d[1].replace(d[0], '').trim().toString().replace(/[\r\n]/gm, '
')
+ if (commitInfo) return commitInfo;
+ let d = execSync(`git show -s --format='%s;;;%B'`).toString().trim().replace(/[\r\n]/gm, '\n').split(';;;');
+ d[1] = d[1].replace(d[0], '').trim().toString().replace(/[\r\n]/gm, '
');
+ commitInfo = d;
return d
}
+export function getCurrentBranch() {
+ if (branch) return branch;
+ let b = execSync('git branch --show-current').toString().trim();
+ branch = b;
+ return b
+}
diff --git a/src/modules/sub/errors.js b/src/modules/sub/errors.js
index d4570ae3..c73ffe85 100644
--- a/src/modules/sub/errors.js
+++ b/src/modules/sub/errors.js
@@ -3,6 +3,9 @@ import loc from "../../localization/manager.js";
export function errorUnsupported(lang) {
return loc(lang, 'ErrorUnsupported');
}
-export function genericError(lang, host) {
+export function brokenLink(lang, host) {
return loc(lang, 'ErrorBrokenLink', host);
}
+export function genericError(lang, host) {
+ return loc(lang, 'ErrorBadFetch', host);
+}
diff --git a/src/modules/sub/utils.js b/src/modules/sub/utils.js
index 6c65f5fb..64ccb526 100644
--- a/src/modules/sub/utils.js
+++ b/src/modules/sub/utils.js
@@ -64,6 +64,7 @@ export function msToTime(d) {
export function cleanURL(url, host) {
let forbiddenChars = ['}', '{', '(', ')', '\\', '%', '>', '<', '^', '*', '!', '~', ';', ':', ',', '`', '[', ']', '#', '$', '"', "'", "@"]
switch(host) {
+ case "vk":
case "youtube":
url = url.split('&')[0];
break;
@@ -103,26 +104,26 @@ export function checkJSONPost(obj) {
}
try {
let objKeys = Object.keys(obj);
- if (objKeys.length < 8 && obj.url) {
- let defKeys = Object.keys(def);
- for (let i in objKeys) {
- if (String(objKeys[i]) !== "url" && defKeys.includes(objKeys[i])) {
- if (apiVar.booleanOnly.includes(objKeys[i])) {
- def[objKeys[i]] = obj[objKeys[i]] ? true : false;
- } else {
- if (apiVar.allowed[objKeys[i]] && apiVar.allowed[objKeys[i]].includes(obj[objKeys[i]])) def[objKeys[i]] = String(obj[objKeys[i]])
- }
+ if (!(objKeys.length <= 8 && obj.url)) return false;
+ let defKeys = Object.keys(def);
+
+ for (let i in objKeys) {
+ if (String(objKeys[i]) !== "url" && defKeys.includes(objKeys[i])) {
+ if (apiVar.booleanOnly.includes(objKeys[i])) {
+ def[objKeys[i]] = obj[objKeys[i]] ? true : false;
+ } else {
+ if (apiVar.allowed[objKeys[i]] && apiVar.allowed[objKeys[i]].includes(obj[objKeys[i]])) def[objKeys[i]] = String(obj[objKeys[i]])
}
}
- obj["url"] = decodeURIComponent(String(obj["url"]))
- let hostname = obj["url"].replace("https://", "").replace(' ', '').split('&')[0].split("/")[0].split("."),
- host = hostname[hostname.length - 2]
- def["url"] = encodeURIComponent(cleanURL(obj["url"], host))
- return def
- } else {
- return false
}
+
+ obj["url"] = decodeURIComponent(String(obj["url"]));
+ let hostname = obj["url"].replace("https://", "").replace(' ', '').split('&')[0].split("/")[0].split("."),
+ host = hostname[hostname.length - 2];
+ def["url"] = encodeURIComponent(cleanURL(obj["url"], host));
+
+ return def
} catch (e) {
- return false;
+ return false
}
}
diff --git a/src/test/test.js b/src/test/test.js
new file mode 100644
index 00000000..f03c8dbb
--- /dev/null
+++ b/src/test/test.js
@@ -0,0 +1,71 @@
+import "dotenv/config";
+
+import { getJSON } from "../modules/api.js";
+import { services } from "../modules/config.js";
+import loadJSON from "../modules/sub/loadJSON.js";
+import { checkJSONPost } from "../modules/sub/utils.js";
+
+let tests = loadJSON('./src/test/tests.json');
+
+let noTest = [];
+let failed = [];
+let success = 0;
+
+function addToFail(service, testName, url, status, response) {
+ failed.push({
+ service: service,
+ name: testName,
+ url: url,
+ status: status,
+ response: response
+ })
+}
+for (let i in services) {
+ if (tests[i]) {
+ console.log(`\nRunning tests for ${i}...\n`)
+ for (let k = 0; k < tests[i].length; k++) {
+ let test = tests[i][k];
+
+ console.log(`Running test ${k+1}: ${test.name}`);
+ console.log('params:');
+ let params = {...{url: test.url}, ...test.params};
+ console.log(params);
+
+ let chck = checkJSONPost(params);
+ if (chck) {
+ chck["ip"] = "d21ec524bc2ade41bef569c0361ac57728c69e2764b5cb3cb310fe36568ca53f"; // random sha256
+ let j = await getJSON(chck["url"], "en", chck);
+ console.log('\nReceived:');
+ console.log(j)
+ if (j.status === test.expected.code && j.body.status === test.expected.status) {
+ console.log("\n✅ Success.\n");
+ success++
+ } else {
+ console.log(`\n❌ Fail. Expected: ${test.expected.code} & ${test.expected.status}, received: ${j.status} & ${j.body.status}\n`);
+ addToFail(i, test.name, test.url, j.body.status, j)
+ }
+ } else {
+ console.log("\n❌ couldn't validate the request JSON.\n");
+ addToFail(i, test.name, test.url, "unknown", {})
+ }
+ }
+ console.log("\n\n")
+ } else {
+ console.warn(`No tests found for ${i}.`);
+ noTest.push(i)
+ }
+}
+
+console.log(`✅ ${success} tests succeeded.`);
+console.log(`❌ ${failed.length} tests failed.`);
+console.log(`❔ ${noTest.length} services weren't tested.`);
+
+if (failed.length > 0) {
+ console.log(`\nFailed tests:`);
+ console.log(failed)
+}
+
+if (noTest.length > 0) {
+ console.log(`\nMissing tests:`);
+ console.log(noTest)
+}
diff --git a/src/test/tests.json b/src/test/tests.json
new file mode 100644
index 00000000..e4935177
--- /dev/null
+++ b/src/test/tests.json
@@ -0,0 +1,742 @@
+{
+ "twitter": [{
+ "name": "regular video",
+ "url": "https://twitter.com/TwitterSpaces/status/1526955853743546372?s=20",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "embedded twitter video",
+ "url": "https://twitter.com/dustbin_nie/status/1624596567188717568?s=20",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "mixed media (image + gif)",
+ "url": "https://twitter.com/Twitter/status/1580661436132757506?s=20",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "picker: mixed media (3 gifs + image)",
+ "url": "https://twitter.com/emerald_pedrod/status/1582418163521581063?s=20",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "picker"
+ }
+ }, {
+ "name": "audio from embedded twitter video (mp3, isAudioOnly)",
+ "url": "https://twitter.com/dustbin_nie/status/1624596567188717568?s=20",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": true,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "audio from embedded twitter video (best, isAudioOnly)",
+ "url": "https://twitter.com/dustbin_nie/status/1624596567188717568?s=20",
+ "params": {
+ "aFormat": "best",
+ "isAudioOnly": true,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "audio from embedded twitter video (ogg, isAudioOnly, isAudioMuted)",
+ "url": "https://twitter.com/dustbin_nie/status/1624596567188717568?s=20",
+ "params": {
+ "aFormat": "best",
+ "isAudioOnly": true,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "muted embedded twitter video",
+ "url": "https://twitter.com/dustbin_nie/status/1624596567188717568?s=20",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "inexistent post",
+ "url": "https://twitter.com/test/status/9487653",
+ "params": {
+ "aFormat": "best",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }, {
+ "name": "post with no media content",
+ "url": "https://twitter.com/elonmusk/status/1604617643973124097?s=20",
+ "params": {
+ "aFormat": "best",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }, {
+ "name": "recorded space by nyc (best)",
+ "url": "https://twitter.com/i/spaces/1gqxvyLoYQkJB",
+ "params": {
+ "aFormat": "best",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "recorded space by nyc (mp3)",
+ "url": "https://twitter.com/i/spaces/1gqxvyLoYQkJB",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "recorded space by nyc (wav, isAudioMuted)",
+ "url": "https://twitter.com/i/spaces/1gqxvyLoYQkJB",
+ "params": {
+ "aFormat": "wav",
+ "isAudioOnly": false,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "recorded space by service95 & dualipa (mp3, isAudioMuted, isAudioOnly)",
+ "url": "https://twitter.com/i/spaces/1nAJErvvVXgxL",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": true,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "unavailable space",
+ "url": "https://twitter.com/i/spaces/1OwGWwjRjVVGQ?s=20",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }, {
+ "name": "inexistent space",
+ "url": "https://twitter.com/i/spaces/10Wkie2j29iiI",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }],
+ "soundcloud": [{
+ "name": "public song (best)",
+ "url": "https://soundcloud.com/l2share77/loona-butterfly?utm_source=clipboard&utm_medium=text&utm_campaign=social_sharing",
+ "params": {
+ "aFormat": "best",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "public song (mp3, isAudioMuted)",
+ "url": "https://soundcloud.com/l2share77/loona-butterfly?utm_source=clipboard&utm_medium=text&utm_campaign=social_sharing",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "private song",
+ "url": "https://soundcloud.com/4kayy/unhappy-new-year-prod4kay/s-9bKbvwLdRWG",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "private song (wav, isAudioMuted)",
+ "url": "https://soundcloud.com/4kayy/unhappy-new-year-prod4kay/s-9bKbvwLdRWG",
+ "params": {
+ "aFormat": "wav",
+ "isAudioOnly": false,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "private song (ogg, isAudioMuted, isAudioOnly)",
+ "url": "https://soundcloud.com/4kayy/unhappy-new-year-prod4kay/s-9bKbvwLdRWG",
+ "params": {
+ "aFormat": "ogg",
+ "isAudioOnly": true,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }],
+ "youtube": [{
+ "name": "4k video (mp4, hig)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "mp4",
+ "vQuality": "hig",
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "4k video (webm, mid)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "webm",
+ "vQuality": "mid",
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "4k video (mp4, max)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "mp4",
+ "vQuality": "max",
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "4k video (webm, max)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "webm",
+ "vQuality": "max",
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "4k video (webm, max, isAudioMuted)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "webm",
+ "vQuality": "max",
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "4k video (mp4, max, isAudioMuted)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "webm",
+ "vQuality": "max",
+ "aFormat": "mp3",
+ "isAudioOnly": true,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "4k video (mp4, max, isAudioMuted, isAudioOnly, mp3)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "webm",
+ "vQuality": "max",
+ "aFormat": "mp3",
+ "isAudioOnly": true,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "4k video (mp4, max, isAudioMuted, isAudioOnly, best)",
+ "url": "https://www.youtube.com/watch?v=vPwaXytZcgI",
+ "params": {
+ "vFormat": "webm",
+ "vQuality": "max",
+ "aFormat": "best",
+ "isAudioOnly": true,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "music (mp3, isAudioOnly, isAudioMuted)",
+ "url": "https://music.youtube.com/watch?v=5rGTsvZCEdk&feature=share",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": true,
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "music (mp3)",
+ "url": "https://music.youtube.com/watch?v=5rGTsvZCEdk&feature=share",
+ "params": {
+ "aFormat": "mp3",
+ "isAudioOnly": false,
+ "isAudioMuted": false
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "short, defaults",
+ "url": "https://www.youtube.com/shorts/r5FpeOJItbw",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "inexistent video",
+ "url": "https://youtube.com/watch?v=gnjuHYWGEW",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }],
+ "vk": [{
+ "name": "clip, defaults",
+ "url": "https://vk.com/clip-57274055_456239788",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "clip, low",
+ "url": "https://vk.com/clip-57274055_456239788",
+ "params": {
+ "vQuality": "low"
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "clip different link, max",
+ "url": "https://vk.com/clips-57274055?z=clip-57274055_456239788",
+ "params": {
+ "vQuality": "max"
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "video, defaults",
+ "url": "https://vk.com/video-57274055_456239399",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "inexistent video",
+ "url": "https://vk.com/video-53333333_456233333",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }],
+ "douyin": [{
+ "name": "short link video, with watermark",
+ "url": "https://v.douyin.com/2p4Aya7/",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "short link video (isNoTTWatermark)",
+ "url": "https://v.douyin.com/2p4Aya7/",
+ "params": {
+ "isNoTTWatermark": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "short link video (isAudioOnly)",
+ "url": "https://v.douyin.com/2p4Aya7/",
+ "params": {
+ "isAudioOnly": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "short link video (isAudioOnly, isTTFullAudio)",
+ "url": "https://v.douyin.com/2p4Aya7/",
+ "params": {
+ "isAudioOnly": true,
+ "isTTFullAudio": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "long link video (isNoTTWatermark)",
+ "url": "https://www.douyin.com/video/7120601033314716968",
+ "params": {
+ "isNoTTWatermark": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "images",
+ "url": "https://v.douyin.com/MdVwo31/",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "picker"
+ }
+ }, {
+ "name": "long link inexistent",
+ "url": "https://www.douyin.com/video/7120851458451417478",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }, {
+ "name": "short link inexistent",
+ "url": "https://v.douyin.com/2p4ewa7/",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }],
+ "tiktok": [{
+ "name": "short link (vt) video, with watermark",
+ "url": "https://vt.tiktok.com/ZS85U86aa/",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "short link (vt) video (isNoTTWatermark)",
+ "url": "https://vt.tiktok.com/ZS85U86aa/",
+ "params": {
+ "isNoTTWatermark": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "short link (vm) video (isAudioOnly)",
+ "url": "https://vm.tiktok.com/ZMYrYAf34/",
+ "params": {
+ "isAudioOnly": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "short link (vm) video (isAudioOnly, isTTFullAudio)",
+ "url": "https://vm.tiktok.com/ZMYrYAf34/",
+ "params": {
+ "isAudioOnly": true,
+ "isTTFullAudio": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "long link video (isNoTTWatermark)",
+ "url": "https://www.tiktok.com/@fatfatmillycat/video/7195741644585454894",
+ "params": {
+ "isNoTTWatermark": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "images",
+ "url": "https://vt.tiktok.com/ZS8JP89eB/",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "picker"
+ }
+ }, {
+ "name": "long link inexistent",
+ "url": "https://www.tiktok.com/@blablabla/video/7120851458451417478",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }, {
+ "name": "short link inexistent",
+ "url": "https://vt.tiktok.com/2p4ewa7/",
+ "params": {},
+ "expected": {
+ "code": 400,
+ "status": "error"
+ }
+ }],
+ "bilibili": [{
+ "name": "1080p video",
+ "url": "https://www.bilibili.com/video/BV18i4y1m7xV/",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "1080p video muted",
+ "url": "https://www.bilibili.com/video/BV18i4y1m7xV/",
+ "params": {
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "1080p vertical video",
+ "url": "https://www.bilibili.com/video/BV1uu411z7VV/",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "1080p vertical video muted",
+ "url": "https://www.bilibili.com/video/BV1uu411z7VV/",
+ "params": {
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }],
+ "tumblr": [{
+ "name": "at.tumblr link",
+ "url": "https://at.tumblr.com/music/704177038274281472/n7x7pr7x4w2b",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "user subdomain link",
+ "url": "https://garfield-69.tumblr.com/post/696499862852780032",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "web app link",
+ "url": "https://www.tumblr.com/rongzhi/707729381162958848/english-added-by-me?source=share",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }],
+ "vimeo": [{
+ "name": "4k progressive",
+ "url": "https://vimeo.com/288386543",
+ "params": {
+ "vQuality": "max"
+ },
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "720p progressive",
+ "url": "https://vimeo.com/288386543",
+ "params": {
+ "vQuality": "mid"
+ },
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "1080p dash parcel",
+ "url": "https://vimeo.com/774694040",
+ "params": {
+ "vQuality": "hig"
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "720p dash parcel",
+ "url": "https://vimeo.com/774694040",
+ "params": {
+ "vQuality": "mid"
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }],
+ "reddit": [{
+ "name": "video with audio",
+ "url": "https://www.reddit.com/r/catvideos/comments/b2rygq/my_new_kittens_1st_day_checking_out_his_new_home/?utm_source=share&utm_medium=web2x&context=3",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "video with audio (isAudioOnly)",
+ "url": "https://www.reddit.com/r/catvideos/comments/b2rygq/my_new_kittens_1st_day_checking_out_his_new_home/?utm_source=share&utm_medium=web2x&context=3",
+ "params": {
+ "isAudioOnly": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "video with audio (isAudioMuted)",
+ "url": "https://www.reddit.com/r/catvideos/comments/b2rygq/my_new_kittens_1st_day_checking_out_his_new_home/?utm_source=share&utm_medium=web2x&context=3",
+ "params": {
+ "isAudioMuted": true
+ },
+ "expected": {
+ "code": 200,
+ "status": "stream"
+ }
+ }, {
+ "name": "video without audio",
+ "url": "https://www.reddit.com/r/catvideos/comments/ftoeo7/luna_doesnt_want_to_be_bothered_while_shes_napping/?utm_source=share&utm_medium=web2x&context=3",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }, {
+ "name": "actual gif, not looping video",
+ "url": "https://www.reddit.com/r/whenthe/comments/109wqy1/god_really_did_some_trolling/?utm_source=share&utm_medium=web2x&context=3",
+ "params": {},
+ "expected": {
+ "code": 200,
+ "status": "redirect"
+ }
+ }]
+}
\ No newline at end of file