mirror of
https://github.com/imputnet/cobalt.git
synced 2025-07-18 11:18:28 +00:00
Merge branch 'current' into support/videoclip
This commit is contained in:
commit
a9ef23755b
1
.github/FUNDING.yml
vendored
1
.github/FUNDING.yml
vendored
@ -1 +0,0 @@
|
|||||||
custom: https://boosty.to/wukko/donate
|
|
2
.github/test.sh
vendored
2
.github/test.sh
vendored
@ -18,7 +18,7 @@ test_api() {
|
|||||||
-X POST \
|
-X POST \
|
||||||
-H "Accept: application/json" \
|
-H "Accept: application/json" \
|
||||||
-H "Content-Type: application/json" \
|
-H "Content-Type: application/json" \
|
||||||
-d '{"url":"https://www.youtube.com/watch?v=jNQXAC9IVRw"}')
|
-d '{"url":"https://vine.co/v/huwVJIEJW50", "isAudioOnly": true}')
|
||||||
|
|
||||||
echo "$API_RESPONSE"
|
echo "$API_RESPONSE"
|
||||||
STATUS=$(echo "$API_RESPONSE" | jq -r .status)
|
STATUS=$(echo "$API_RESPONSE" | jq -r .status)
|
||||||
|
24
.github/workflows/test.yml
vendored
24
.github/workflows/test.yml
vendored
@ -36,3 +36,27 @@ jobs:
|
|||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
- name: Run test script
|
- name: Run test script
|
||||||
run: .github/test.sh api
|
run: .github/test.sh api
|
||||||
|
|
||||||
|
check-services:
|
||||||
|
name: test service functionality
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
outputs:
|
||||||
|
services: ${{ steps.checkServices.outputs.service_list }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- id: checkServices
|
||||||
|
run: npm ci && echo "service_list=$(node src/util/test-ci get-services)" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
test-services:
|
||||||
|
needs: check-services
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
service: ${{ fromJson(needs.check-services.outputs.services) }}
|
||||||
|
name: "test service: ${{ matrix.service }}"
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- run: npm ci && node src/util/test-ci run-tests-for ${{ matrix.service }}
|
@ -26,7 +26,7 @@ the scope is not strictly defined, you can write whatever you find most fitting
|
|||||||
|
|
||||||
if you think a change deserves further explanation, we encourage you to write a short explanation in the commit message ([example](https://github.com/imputnet/cobalt/commit/d2e5b6542f71f3809ba94d56c26f382b5cb62762)), which will save both you and us time having to enquire about the change, and you explaining the reason behind it.
|
if you think a change deserves further explanation, we encourage you to write a short explanation in the commit message ([example](https://github.com/imputnet/cobalt/commit/d2e5b6542f71f3809ba94d56c26f382b5cb62762)), which will save both you and us time having to enquire about the change, and you explaining the reason behind it.
|
||||||
|
|
||||||
if your contribution has uninformative commit messages, you may be asked to interactively rebase your branch and amend each commit to include a meaningful message.
|
if your contribution has uninformative commit titles, you may be asked to interactively rebase your branch and amend each commit to include a meaningful title.
|
||||||
|
|
||||||
### clean commit history
|
### clean commit history
|
||||||
if your branch is out of date and/or has some merge conflicts with the `current` branch, you should **rebase** it instead of merging. this prevents meaningless merge commits from being included in your branch, which would then end up in the cobalt git history.
|
if your branch is out of date and/or has some merge conflicts with the `current` branch, you should **rebase** it instead of merging. this prevents meaningless merge commits from being included in your branch, which would then end up in the cobalt git history.
|
||||||
|
@ -19,11 +19,13 @@ this list is not final and keeps expanding over time. if support for a service y
|
|||||||
| bilibili.com & bilibili.tv | ✅ | ✅ | ✅ | ➖ | ➖ |
|
| bilibili.com & bilibili.tv | ✅ | ✅ | ✅ | ➖ | ➖ |
|
||||||
| dailymotion | ✅ | ✅ | ✅ | ✅ | ✅ |
|
| dailymotion | ✅ | ✅ | ✅ | ✅ | ✅ |
|
||||||
| instagram posts & reels | ✅ | ✅ | ✅ | ➖ | ➖ |
|
| instagram posts & reels | ✅ | ✅ | ✅ | ➖ | ➖ |
|
||||||
|
| facebook videos | ✅ | ❌ | ❌ | ➖ | ➖ |
|
||||||
| loom | ✅ | ❌ | ✅ | ✅ | ➖ |
|
| loom | ✅ | ❌ | ✅ | ✅ | ➖ |
|
||||||
| ok video | ✅ | ❌ | ✅ | ✅ | ✅ |
|
| ok video | ✅ | ❌ | ✅ | ✅ | ✅ |
|
||||||
| pinterest | ✅ | ✅ | ✅ | ➖ | ➖ |
|
| pinterest | ✅ | ✅ | ✅ | ➖ | ➖ |
|
||||||
| reddit | ✅ | ✅ | ✅ | ❌ | ❌ |
|
| reddit | ✅ | ✅ | ✅ | ❌ | ❌ |
|
||||||
| rutube | ✅ | ✅ | ✅ | ✅ | ✅ |
|
| rutube | ✅ | ✅ | ✅ | ✅ | ✅ |
|
||||||
|
| snapchat stories & spotlights | ✅ | ✅ | ✅ | ➖ | ➖ |
|
||||||
| soundcloud | ➖ | ✅ | ➖ | ✅ | ✅ |
|
| soundcloud | ➖ | ✅ | ➖ | ✅ | ✅ |
|
||||||
| streamable | ✅ | ✅ | ✅ | ➖ | ➖ |
|
| streamable | ✅ | ✅ | ✅ | ➖ | ➖ |
|
||||||
| tiktok | ✅ | ✅ | ✅ | ❌ | ❌ |
|
| tiktok | ✅ | ✅ | ✅ | ❌ | ❌ |
|
||||||
@ -45,8 +47,10 @@ this list is not final and keeps expanding over time. if support for a service y
|
|||||||
| service | notes or features |
|
| service | notes or features |
|
||||||
| :-------- | :----- |
|
| :-------- | :----- |
|
||||||
| instagram | supports reels, photos, and videos. lets you pick what to save from multi-media posts. |
|
| instagram | supports reels, photos, and videos. lets you pick what to save from multi-media posts. |
|
||||||
|
| facebook | supports public accessible videos content only. |
|
||||||
| pinterest | supports photos, gifs, videos and stories. |
|
| pinterest | supports photos, gifs, videos and stories. |
|
||||||
| reddit | supports gifs and videos. |
|
| reddit | supports gifs and videos. |
|
||||||
|
| snapchat | supports spotlights and stories. lets you pick what to save from stories. |
|
||||||
| rutube | supports yappy & private links. |
|
| rutube | supports yappy & private links. |
|
||||||
| soundcloud | supports private links. |
|
| soundcloud | supports private links. |
|
||||||
| tiktok | supports videos with or without watermark, images from slideshow without watermark, and full (original) audios. |
|
| tiktok | supports videos with or without watermark, images from slideshow without watermark, and full (original) audios. |
|
||||||
|
@ -49,9 +49,9 @@ item type: `object`
|
|||||||
|
|
||||||
| key | type | variables | description |
|
| key | type | variables | description |
|
||||||
|:--------|:---------|:--------------------------------------------------------|:---------------------------------------|
|
|:--------|:---------|:--------------------------------------------------------|:---------------------------------------|
|
||||||
| `type` | `string` | `video` | used only if `pickerType`is `various`. |
|
| `type` | `string` | `video / photo / gif` | used only if `pickerType` is `various` |
|
||||||
| `url` | `string` | direct link to a file or a link to cobalt's live render | |
|
| `url` | `string` | direct link to a file or a link to cobalt's live render | |
|
||||||
| `thumb` | `string` | item thumbnail that's displayed in the picker | used only for `video` type. |
|
| `thumb` | `string` | item thumbnail that's displayed in the picker | used for `video` and `gif` types |
|
||||||
|
|
||||||
## GET: `/api/stream`
|
## GET: `/api/stream`
|
||||||
cobalt's live render (or stream) endpoint. usually, you will receive a url to this endpoint
|
cobalt's live render (or stream) endpoint. usually, you will receive a url to this endpoint
|
||||||
|
@ -56,8 +56,9 @@ sudo service nscd start
|
|||||||
| `API_LISTEN_ADDRESS` | `0.0.0.0` | `127.0.0.1` | changes address from which api server is accessible. **if you are using docker, you usually don't need to configure this.** |
|
| `API_LISTEN_ADDRESS` | `0.0.0.0` | `127.0.0.1` | changes address from which api server is accessible. **if you are using docker, you usually don't need to configure this.** |
|
||||||
| `API_URL` | ➖ | `https://api.cobalt.tools/` | changes url from which api server is accessible. <br> ***REQUIRED TO RUN THE API***. |
|
| `API_URL` | ➖ | `https://api.cobalt.tools/` | changes url from which api server is accessible. <br> ***REQUIRED TO RUN THE API***. |
|
||||||
| `API_NAME` | `unknown` | `ams-1` | api server name that is shown in `/api/serverInfo`. |
|
| `API_NAME` | `unknown` | `ams-1` | api server name that is shown in `/api/serverInfo`. |
|
||||||
|
| `API_EXTERNAL_PROXY` | ➖ | `http://user:password@127.0.0.1:8080`| url of the proxy that will be passed to [`ProxyAgent`](https://undici.nodejs.org/#/docs/api/ProxyAgent) and used for all external requests. HTTP(S) only. |
|
||||||
| `CORS_WILDCARD` | `1` | `0` | toggles cross-origin resource sharing. <br> `0`: disabled. `1`: enabled. |
|
| `CORS_WILDCARD` | `1` | `0` | toggles cross-origin resource sharing. <br> `0`: disabled. `1`: enabled. |
|
||||||
| `CORS_URL` | not used | `https://cobalt.tools/` | cross-origin resource sharing url. api will be available only from this url if `CORS_WILDCARD` is set to `0`. |
|
| `CORS_URL` | not used | `https://cobalt.tools` | cross-origin resource sharing url. api will be available only from this url if `CORS_WILDCARD` is set to `0`. |
|
||||||
| `COOKIE_PATH` | not used | `/cookies.json` | path for cookie file relative to main folder. |
|
| `COOKIE_PATH` | not used | `/cookies.json` | path for cookie file relative to main folder. |
|
||||||
| `PROCESSING_PRIORITY` | not used | `10` | changes `nice` value* for ffmpeg subprocess. available only on unix systems. |
|
| `PROCESSING_PRIORITY` | not used | `10` | changes `nice` value* for ffmpeg subprocess. available only on unix systems. |
|
||||||
| `FREEBIND_CIDR` | ➖ | `2001:db8::/32` | IPv6 prefix used for randomly assigning addresses to cobalt requests. only supported on linux systems. see below for more info. |
|
| `FREEBIND_CIDR` | ➖ | `2001:db8::/32` | IPv6 prefix used for randomly assigning addresses to cobalt requests. only supported on linux systems. see below for more info. |
|
||||||
|
27
package-lock.json
generated
27
package-lock.json
generated
@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "cobalt",
|
"name": "cobalt",
|
||||||
"version": "7.14.5",
|
"version": "7.15",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "cobalt",
|
"name": "cobalt",
|
||||||
"version": "7.14.5",
|
"version": "7.15",
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"content-disposition-header": "0.6.0",
|
"content-disposition-header": "0.6.0",
|
||||||
@ -24,7 +24,7 @@
|
|||||||
"set-cookie-parser": "2.6.0",
|
"set-cookie-parser": "2.6.0",
|
||||||
"undici": "^5.19.1",
|
"undici": "^5.19.1",
|
||||||
"url-pattern": "1.0.3",
|
"url-pattern": "1.0.3",
|
||||||
"youtubei.js": "^9.3.0"
|
"youtubei.js": "^10.2.0"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18"
|
"node": ">=18"
|
||||||
@ -73,9 +73,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/acorn": {
|
"node_modules/acorn": {
|
||||||
"version": "8.11.3",
|
"version": "8.12.1",
|
||||||
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.3.tgz",
|
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.1.tgz",
|
||||||
"integrity": "sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==",
|
"integrity": "sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==",
|
||||||
"bin": {
|
"bin": {
|
||||||
"acorn": "bin/acorn"
|
"acorn": "bin/acorn"
|
||||||
},
|
},
|
||||||
@ -683,9 +683,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/jintr": {
|
"node_modules/jintr": {
|
||||||
"version": "1.1.0",
|
"version": "2.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/jintr/-/jintr-1.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/jintr/-/jintr-2.0.0.tgz",
|
||||||
"integrity": "sha512-Tu9wk3BpN2v+kb8yT6YBtue+/nbjeLFv4vvVC4PJ7oCidHKbifWhvORrAbQfxVIQZG+67am/mDagpiGSVtvrZg==",
|
"integrity": "sha512-RiVlevxttZ4eHEYB2dXKXDXluzHfRuw0DJQGsYuKCc5IvZj5/GbOakeqVX+Bar/G9kTty9xDJREcxukurkmYLA==",
|
||||||
"funding": [
|
"funding": [
|
||||||
"https://github.com/sponsors/LuanRT"
|
"https://github.com/sponsors/LuanRT"
|
||||||
],
|
],
|
||||||
@ -1123,14 +1123,15 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/youtubei.js": {
|
"node_modules/youtubei.js": {
|
||||||
"version": "9.4.0",
|
"version": "10.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/youtubei.js/-/youtubei.js-9.4.0.tgz",
|
"resolved": "https://registry.npmjs.org/youtubei.js/-/youtubei.js-10.2.0.tgz",
|
||||||
"integrity": "sha512-8plCOZD2WabqWSEgZU3RjzigIIeR7sF028EERJENYrC9xO/6awpLMZfeoE1gNrNEbKcA+bzbMvonqlvBdxGdKg==",
|
"integrity": "sha512-JLKW9AHQ1qrTwBbre1aDkH8UJFmNcc4+kOSaVou5jSY7AzfFPFJK0yvX6afnLst0UVC9wfXHrLiNx93sutVErA==",
|
||||||
"funding": [
|
"funding": [
|
||||||
"https://github.com/sponsors/LuanRT"
|
"https://github.com/sponsors/LuanRT"
|
||||||
],
|
],
|
||||||
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"jintr": "^1.1.0",
|
"jintr": "^2.0.0",
|
||||||
"tslib": "^2.5.0",
|
"tslib": "^2.5.0",
|
||||||
"undici": "^5.19.1"
|
"undici": "^5.19.1"
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "cobalt",
|
"name": "cobalt",
|
||||||
"description": "save what you love",
|
"description": "save what you love",
|
||||||
"version": "7.14.5",
|
"version": "7.15",
|
||||||
"author": "imput",
|
"author": "imput",
|
||||||
"exports": "./src/cobalt.js",
|
"exports": "./src/cobalt.js",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
@ -40,7 +40,7 @@
|
|||||||
"set-cookie-parser": "2.6.0",
|
"set-cookie-parser": "2.6.0",
|
||||||
"undici": "^5.19.1",
|
"undici": "^5.19.1",
|
||||||
"url-pattern": "1.0.3",
|
"url-pattern": "1.0.3",
|
||||||
"youtubei.js": "^9.3.0"
|
"youtubei.js": "^10.2.0"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"freebind": "^0.2.2"
|
"freebind": "^0.2.2"
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import cors from "cors";
|
import cors from "cors";
|
||||||
import rateLimit from "express-rate-limit";
|
import rateLimit from "express-rate-limit";
|
||||||
|
import { setGlobalDispatcher, ProxyAgent } from "undici";
|
||||||
|
|
||||||
import { env, version } from "../modules/config.js";
|
import { env, version } from "../modules/config.js";
|
||||||
|
|
||||||
@ -26,7 +27,7 @@ const corsConfig = env.corsWildcard ? {} : {
|
|||||||
export function runAPI(express, app, gitCommit, gitBranch, __dirname) {
|
export function runAPI(express, app, gitCommit, gitBranch, __dirname) {
|
||||||
const startTime = new Date();
|
const startTime = new Date();
|
||||||
const startTimestamp = startTime.getTime();
|
const startTimestamp = startTime.getTime();
|
||||||
|
|
||||||
const serverInfo = {
|
const serverInfo = {
|
||||||
version: version,
|
version: version,
|
||||||
commit: gitCommit,
|
commit: gitCommit,
|
||||||
@ -81,38 +82,23 @@ export function runAPI(express, app, gitCommit, gitBranch, __dirname) {
|
|||||||
app.use((req, res, next) => {
|
app.use((req, res, next) => {
|
||||||
try {
|
try {
|
||||||
decodeURIComponent(req.path)
|
decodeURIComponent(req.path)
|
||||||
} catch {
|
} catch {
|
||||||
return res.redirect('/')
|
return res.redirect('/')
|
||||||
}
|
}
|
||||||
next();
|
next();
|
||||||
})
|
})
|
||||||
|
|
||||||
app.use('/api/json', express.json({
|
app.use('/api/json', express.json({ limit: 1024 }));
|
||||||
verify: (req, res, buf) => {
|
app.use('/api/json', (err, _, res, next) => {
|
||||||
if (String(req.header('Accept')) === "application/json") {
|
if (err) {
|
||||||
if (buf.length > 720) throw new Error();
|
|
||||||
JSON.parse(buf);
|
|
||||||
} else {
|
|
||||||
throw new Error();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
|
|
||||||
// handle express.json errors properly (https://github.com/expressjs/express/issues/4065)
|
|
||||||
app.use('/api/json', (err, req, res, next) => {
|
|
||||||
let errorText = "invalid json body";
|
|
||||||
const acceptHeader = String(req.header('Accept')) !== "application/json";
|
|
||||||
|
|
||||||
if (err || acceptHeader) {
|
|
||||||
if (acceptHeader) errorText = "invalid accept header";
|
|
||||||
return res.status(400).json({
|
return res.status(400).json({
|
||||||
status: "error",
|
status: "error",
|
||||||
text: errorText
|
text: "invalid json body"
|
||||||
});
|
});
|
||||||
} else {
|
|
||||||
next();
|
|
||||||
}
|
}
|
||||||
})
|
|
||||||
|
next();
|
||||||
|
});
|
||||||
|
|
||||||
app.post('/api/json', async (req, res) => {
|
app.post('/api/json', async (req, res) => {
|
||||||
const request = req.body;
|
const request = req.body;
|
||||||
@ -123,6 +109,10 @@ export function runAPI(express, app, gitCommit, gitBranch, __dirname) {
|
|||||||
res.status(status).json(body);
|
res.status(status).json(body);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!acceptRegex.test(req.header('Accept'))) {
|
||||||
|
return fail('ErrorInvalidAcceptHeader');
|
||||||
|
}
|
||||||
|
|
||||||
if (!acceptRegex.test(req.header('Content-Type'))) {
|
if (!acceptRegex.test(req.header('Content-Type'))) {
|
||||||
return fail('ErrorInvalidContentType');
|
return fail('ErrorInvalidContentType');
|
||||||
}
|
}
|
||||||
@ -219,6 +209,14 @@ export function runAPI(express, app, gitCommit, gitBranch, __dirname) {
|
|||||||
randomizeCiphers();
|
randomizeCiphers();
|
||||||
setInterval(randomizeCiphers, 1000 * 60 * 30); // shuffle ciphers every 30 minutes
|
setInterval(randomizeCiphers, 1000 * 60 * 30); // shuffle ciphers every 30 minutes
|
||||||
|
|
||||||
|
if (env.externalProxy) {
|
||||||
|
if (env.freebindCIDR) {
|
||||||
|
throw new Error('Freebind is not available when external proxy is enabled')
|
||||||
|
}
|
||||||
|
|
||||||
|
setGlobalDispatcher(new ProxyAgent(env.externalProxy))
|
||||||
|
}
|
||||||
|
|
||||||
app.listen(env.apiPort, env.listenAddress, () => {
|
app.listen(env.apiPort, env.listenAddress, () => {
|
||||||
console.log(`\n` +
|
console.log(`\n` +
|
||||||
`${Cyan("cobalt")} API ${Bright(`v.${version}-${gitCommit} (${gitBranch})`)}\n` +
|
`${Cyan("cobalt")} API ${Bright(`v.${version}-${gitCommit} (${gitBranch})`)}\n` +
|
||||||
|
@ -159,6 +159,7 @@
|
|||||||
"UpdateOneMillion": "1 million users and blazing speed",
|
"UpdateOneMillion": "1 million users and blazing speed",
|
||||||
"ErrorYTAgeRestrict": "this youtube video is age-restricted, so i can't see it. try another one!",
|
"ErrorYTAgeRestrict": "this youtube video is age-restricted, so i can't see it. try another one!",
|
||||||
"ErrorYTLogin": "couldn't get this youtube video because it requires an account to view.\n\nthis limitation is done by google to seemingly stop scraping, affecting all 3rd party tools and even their own clients.\n\ntry again, but if issue persists, {ContactLink}.",
|
"ErrorYTLogin": "couldn't get this youtube video because it requires an account to view.\n\nthis limitation is done by google to seemingly stop scraping, affecting all 3rd party tools and even their own clients.\n\ntry again, but if issue persists, {ContactLink}.",
|
||||||
"ErrorYTRateLimit": "i got rate limited by youtube. try again in a few seconds, but if issue persists, {ContactLink}."
|
"ErrorYTRateLimit": "i got rate limited by youtube. try again in a few seconds, but if issue persists, {ContactLink}.",
|
||||||
|
"ErrorInvalidAcceptHeader": "invalid accept header"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -48,7 +48,9 @@ const
|
|||||||
|
|
||||||
processingPriority: process.platform !== 'win32'
|
processingPriority: process.platform !== 'win32'
|
||||||
&& process.env.PROCESSING_PRIORITY
|
&& process.env.PROCESSING_PRIORITY
|
||||||
&& parseInt(process.env.PROCESSING_PRIORITY)
|
&& parseInt(process.env.PROCESSING_PRIORITY),
|
||||||
|
|
||||||
|
externalProxy: process.env.API_EXTERNAL_PROXY,
|
||||||
}
|
}
|
||||||
|
|
||||||
export const
|
export const
|
||||||
|
@ -24,8 +24,10 @@ import streamable from "./services/streamable.js";
|
|||||||
import twitch from "./services/twitch.js";
|
import twitch from "./services/twitch.js";
|
||||||
import rutube from "./services/rutube.js";
|
import rutube from "./services/rutube.js";
|
||||||
import dailymotion from "./services/dailymotion.js";
|
import dailymotion from "./services/dailymotion.js";
|
||||||
|
import snapchat from "./services/snapchat.js";
|
||||||
import loom from "./services/loom.js";
|
import loom from "./services/loom.js";
|
||||||
import videoclip from "./services/videoclip.js";
|
import videoclip from "./services/videoclip.js";
|
||||||
|
import facebook from "./services/facebook.js";
|
||||||
|
|
||||||
let freebind;
|
let freebind;
|
||||||
|
|
||||||
@ -194,11 +196,22 @@ export default async function(host, patternMatch, lang, obj) {
|
|||||||
id: patternMatch.id
|
id: patternMatch.id
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
case "snapchat":
|
||||||
|
r = await snapchat({
|
||||||
|
hostname: url.hostname,
|
||||||
|
...patternMatch
|
||||||
|
});
|
||||||
|
break;
|
||||||
case "loom":
|
case "loom":
|
||||||
r = await loom({
|
r = await loom({
|
||||||
id: patternMatch.id
|
id: patternMatch.id
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
case "facebook":
|
||||||
|
r = await facebook({
|
||||||
|
...patternMatch
|
||||||
|
});
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
return createResponse("error", {
|
return createResponse("error", {
|
||||||
t: loc(lang, 'ErrorUnsupported')
|
t: loc(lang, 'ErrorUnsupported')
|
||||||
|
@ -73,6 +73,7 @@ export default function(r, host, userFormat, isAudioOnly, lang, isAudioMuted, di
|
|||||||
switch (host) {
|
switch (host) {
|
||||||
case "instagram":
|
case "instagram":
|
||||||
case "twitter":
|
case "twitter":
|
||||||
|
case "snapchat":
|
||||||
params = { picker: r.picker };
|
params = { picker: r.picker };
|
||||||
break;
|
break;
|
||||||
case "tiktok":
|
case "tiktok":
|
||||||
@ -137,11 +138,13 @@ export default function(r, host, userFormat, isAudioOnly, lang, isAudioMuted, di
|
|||||||
params = { type: "bridge" };
|
params = { type: "bridge" };
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case "facebook":
|
||||||
case "vine":
|
case "vine":
|
||||||
case "instagram":
|
case "instagram":
|
||||||
case "tumblr":
|
case "tumblr":
|
||||||
case "pinterest":
|
case "pinterest":
|
||||||
case "streamable":
|
case "streamable":
|
||||||
|
case "snapchat":
|
||||||
case "loom":
|
case "loom":
|
||||||
responseType = "redirect";
|
responseType = "redirect";
|
||||||
break;
|
break;
|
||||||
|
56
src/modules/processing/services/facebook.js
Normal file
56
src/modules/processing/services/facebook.js
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
import { genericUserAgent } from "../../config.js";
|
||||||
|
|
||||||
|
const headers = {
|
||||||
|
'User-Agent': genericUserAgent,
|
||||||
|
'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8',
|
||||||
|
'Accept-Language': 'en-US,en;q=0.5',
|
||||||
|
'Sec-Fetch-Mode': 'navigate',
|
||||||
|
'Sec-Fetch-Site': 'none',
|
||||||
|
}
|
||||||
|
|
||||||
|
const resolveUrl = (url) => {
|
||||||
|
return fetch(url, { headers })
|
||||||
|
.then(r => {
|
||||||
|
if (r.headers.get('location')) {
|
||||||
|
return decodeURIComponent(r.headers.get('location'));
|
||||||
|
}
|
||||||
|
if (r.headers.get('link')) {
|
||||||
|
const linkMatch = r.headers.get('link').match(/<(.*?)\/>/);
|
||||||
|
return decodeURIComponent(linkMatch[1]);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
})
|
||||||
|
.catch(() => false);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function({ id, shareType, shortLink }) {
|
||||||
|
let url = `https://web.facebook.com/i/videos/${id}`;
|
||||||
|
|
||||||
|
if (shareType) url = `https://web.facebook.com/share/${shareType}/${id}`;
|
||||||
|
if (shortLink) url = await resolveUrl(`https://fb.watch/${shortLink}`);
|
||||||
|
|
||||||
|
const html = await fetch(url, { headers })
|
||||||
|
.then(r => r.text())
|
||||||
|
.catch(() => false);
|
||||||
|
|
||||||
|
if (!html) return { error: 'ErrorCouldntFetch' };
|
||||||
|
|
||||||
|
const urls = [];
|
||||||
|
const hd = html.match('"browser_native_hd_url":(".*?")');
|
||||||
|
const sd = html.match('"browser_native_sd_url":(".*?")');
|
||||||
|
|
||||||
|
if (hd?.[1]) urls.push(JSON.parse(hd[1]));
|
||||||
|
if (sd?.[1]) urls.push(JSON.parse(sd[1]));
|
||||||
|
|
||||||
|
if (!urls.length) {
|
||||||
|
return { error: 'ErrorEmptyDownload' };
|
||||||
|
}
|
||||||
|
|
||||||
|
const baseFilename = `facebook_${id || shortLink}`;
|
||||||
|
|
||||||
|
return {
|
||||||
|
urls: urls[0],
|
||||||
|
filename: `${baseFilename}.mp4`,
|
||||||
|
audioFilename: `${baseFilename}_audio`,
|
||||||
|
};
|
||||||
|
}
|
@ -45,7 +45,7 @@ export default async function(o) {
|
|||||||
|
|
||||||
let fileMetadata = {
|
let fileMetadata = {
|
||||||
title: cleanString(videoData.movie.title.trim()),
|
title: cleanString(videoData.movie.title.trim()),
|
||||||
author: cleanString(videoData.author.name.trim()),
|
author: cleanString((videoData.author?.name || videoData.compilationTitle).trim()),
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bestVideo) return {
|
if (bestVideo) return {
|
||||||
|
@ -10,6 +10,8 @@ async function requestJSON(url) {
|
|||||||
} catch {}
|
} catch {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const delta = (a, b) => Math.abs(a - b);
|
||||||
|
|
||||||
export default async function(obj) {
|
export default async function(obj) {
|
||||||
if (obj.yappyId) {
|
if (obj.yappyId) {
|
||||||
const yappy = await requestJSON(
|
const yappy = await requestJSON(
|
||||||
@ -25,7 +27,7 @@ export default async function(obj) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const quality = obj.quality === "max" ? "9000" : obj.quality;
|
const quality = Number(obj.quality) || 9000;
|
||||||
|
|
||||||
const requestURL = new URL(`https://rutube.ru/api/play/options/${obj.id}/?no_404=true&referer&pver=v2`);
|
const requestURL = new URL(`https://rutube.ru/api/play/options/${obj.id}/?no_404=true&referer&pver=v2`);
|
||||||
if (obj.key) requestURL.searchParams.set('p', obj.key);
|
if (obj.key) requestURL.searchParams.set('p', obj.key);
|
||||||
@ -45,12 +47,16 @@ export default async function(obj) {
|
|||||||
|
|
||||||
if (!m3u8) return { error: 'ErrorCouldntFetch' };
|
if (!m3u8) return { error: 'ErrorCouldntFetch' };
|
||||||
|
|
||||||
m3u8 = HLS.parse(m3u8).variants.sort((a, b) => Number(b.bandwidth) - Number(a.bandwidth));
|
m3u8 = HLS.parse(m3u8).variants;
|
||||||
|
|
||||||
let bestQuality = m3u8[0];
|
const matchingQuality = m3u8.reduce((prev, next) => {
|
||||||
if (Number(quality) < bestQuality.resolution.height) {
|
const diff = {
|
||||||
bestQuality = m3u8.find((i) => (Number(quality) === i.resolution.height));
|
prev: delta(quality, prev.resolution.height),
|
||||||
}
|
next: delta(quality, next.resolution.height)
|
||||||
|
};
|
||||||
|
|
||||||
|
return diff.prev < diff.next ? prev : next;
|
||||||
|
});
|
||||||
|
|
||||||
const fileMetadata = {
|
const fileMetadata = {
|
||||||
title: cleanString(play.title.trim()),
|
title: cleanString(play.title.trim()),
|
||||||
@ -58,15 +64,15 @@ export default async function(obj) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
urls: bestQuality.uri,
|
urls: matchingQuality.uri,
|
||||||
isM3U8: true,
|
isM3U8: true,
|
||||||
filenameAttributes: {
|
filenameAttributes: {
|
||||||
service: "rutube",
|
service: "rutube",
|
||||||
id: obj.id,
|
id: obj.id,
|
||||||
title: fileMetadata.title,
|
title: fileMetadata.title,
|
||||||
author: fileMetadata.artist,
|
author: fileMetadata.artist,
|
||||||
resolution: `${bestQuality.resolution.width}x${bestQuality.resolution.height}`,
|
resolution: `${matchingQuality.resolution.width}x${matchingQuality.resolution.height}`,
|
||||||
qualityLabel: `${bestQuality.resolution.height}p`,
|
qualityLabel: `${matchingQuality.resolution.height}p`,
|
||||||
extension: "mp4"
|
extension: "mp4"
|
||||||
},
|
},
|
||||||
fileMetadata: fileMetadata
|
fileMetadata: fileMetadata
|
||||||
|
96
src/modules/processing/services/snapchat.js
Normal file
96
src/modules/processing/services/snapchat.js
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
import { genericUserAgent } from "../../config.js";
|
||||||
|
import { getRedirectingURL } from "../../sub/utils.js";
|
||||||
|
import { extract, normalizeURL } from "../url.js";
|
||||||
|
|
||||||
|
const SPOTLIGHT_VIDEO_REGEX = /<link data-react-helmet="true" rel="preload" href="(https:\/\/cf-st\.sc-cdn\.net\/d\/[\w.?=]+&uc=\d+)" as="video"\/>/;
|
||||||
|
const NEXT_DATA_REGEX = /<script id="__NEXT_DATA__" type="application\/json">({.+})<\/script><\/body><\/html>$/;
|
||||||
|
|
||||||
|
async function getSpotlight(id) {
|
||||||
|
const html = await fetch(`https://www.snapchat.com/spotlight/${id}`, {
|
||||||
|
headers: { 'User-Agent': genericUserAgent }
|
||||||
|
}).then((r) => r.text()).catch(() => null);
|
||||||
|
if (!html) {
|
||||||
|
return { error: 'ErrorCouldntFetch' };
|
||||||
|
}
|
||||||
|
|
||||||
|
const videoURL = html.match(SPOTLIGHT_VIDEO_REGEX)?.[1];
|
||||||
|
if (videoURL) {
|
||||||
|
return {
|
||||||
|
urls: videoURL,
|
||||||
|
filename: `snapchat_${id}.mp4`,
|
||||||
|
audioFilename: `snapchat_${id}_audio`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getStory(username, storyId) {
|
||||||
|
const html = await fetch(`https://www.snapchat.com/add/${username}${storyId ? `/${storyId}` : ''}`, {
|
||||||
|
headers: { 'User-Agent': genericUserAgent }
|
||||||
|
}).then((r) => r.text()).catch(() => null);
|
||||||
|
if (!html) {
|
||||||
|
return { error: 'ErrorCouldntFetch' };
|
||||||
|
}
|
||||||
|
|
||||||
|
const nextDataString = html.match(NEXT_DATA_REGEX)?.[1];
|
||||||
|
if (nextDataString) {
|
||||||
|
const data = JSON.parse(nextDataString);
|
||||||
|
const storyIdParam = data.query.profileParams[1];
|
||||||
|
|
||||||
|
if (storyIdParam && data.props.pageProps.story) {
|
||||||
|
const story = data.props.pageProps.story.snapList.find((snap) => snap.snapId.value === storyIdParam);
|
||||||
|
if (story) {
|
||||||
|
if (story.snapMediaType === 0) {
|
||||||
|
return {
|
||||||
|
urls: story.snapUrls.mediaUrl,
|
||||||
|
isPhoto: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
urls: story.snapUrls.mediaUrl,
|
||||||
|
filename: `snapchat_${storyId}.mp4`,
|
||||||
|
audioFilename: `snapchat_${storyId}_audio`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const defaultStory = data.props.pageProps.curatedHighlights[0];
|
||||||
|
if (defaultStory) {
|
||||||
|
return {
|
||||||
|
picker: defaultStory.snapList.map((snap) => ({
|
||||||
|
type: snap.snapMediaType === 0 ? 'photo' : 'video',
|
||||||
|
url: snap.snapUrls.mediaUrl,
|
||||||
|
thumb: snap.snapUrls.mediaPreviewUrl.value
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function (obj) {
|
||||||
|
let params = obj;
|
||||||
|
if (obj.hostname === 't.snapchat.com' && obj.shortLink) {
|
||||||
|
const link = await getRedirectingURL(`https://t.snapchat.com/${obj.shortLink}`);
|
||||||
|
|
||||||
|
if (!link?.startsWith('https://www.snapchat.com/')) {
|
||||||
|
return { error: 'ErrorCouldntFetch' };
|
||||||
|
}
|
||||||
|
|
||||||
|
const extractResult = extract(normalizeURL(link));
|
||||||
|
if (extractResult?.host !== 'snapchat') {
|
||||||
|
return { error: 'ErrorCouldntFetch' };
|
||||||
|
}
|
||||||
|
|
||||||
|
params = extractResult.patternMatch;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (params.spotlightId) {
|
||||||
|
const result = await getSpotlight(params.spotlightId);
|
||||||
|
if (result) return result;
|
||||||
|
} else if (params.username) {
|
||||||
|
const result = await getStory(params.username, params.storyId);
|
||||||
|
if (result) return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
return { error: 'ErrorCouldntFetch' };
|
||||||
|
}
|
@ -147,7 +147,6 @@ export default async function({ id, index, toGif, dispatcher }) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let media = (repostedTweet?.media || baseTweet?.extended_entities?.media);
|
let media = (repostedTweet?.media || baseTweet?.extended_entities?.media);
|
||||||
media = media?.filter(m => m.video_info?.variants?.length);
|
|
||||||
|
|
||||||
// check if there's a video at given index (/video/<index>)
|
// check if there's a video at given index (/video/<index>)
|
||||||
if (index >= 0 && index < media?.length) {
|
if (index >= 0 && index < media?.length) {
|
||||||
@ -159,18 +158,38 @@ export default async function({ id, index, toGif, dispatcher }) {
|
|||||||
case 0:
|
case 0:
|
||||||
return { error: 'ErrorNoVideosInTweet' };
|
return { error: 'ErrorNoVideosInTweet' };
|
||||||
case 1:
|
case 1:
|
||||||
|
if (media[0].type === "photo") {
|
||||||
|
return {
|
||||||
|
type: "normal",
|
||||||
|
isPhoto: true,
|
||||||
|
urls: `${media[0].media_url_https}?name=4096x4096`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
type: needsFixing(media[0]) ? "remux" : "normal",
|
type: needsFixing(media[0]) ? "remux" : "normal",
|
||||||
urls: bestQuality(media[0].video_info.variants),
|
urls: bestQuality(media[0].video_info.variants),
|
||||||
filename: `twitter_${id}.mp4`,
|
filename: `twitter_${id}.mp4`,
|
||||||
audioFilename: `twitter_${id}_audio`,
|
audioFilename: `twitter_${id}_audio`,
|
||||||
isGif: media[0].type === "animated_gif"
|
isGif: media[0].type === "animated_gif"
|
||||||
};
|
}
|
||||||
default:
|
default:
|
||||||
const picker = media.map((content, i) => {
|
const picker = media.map((content, i) => {
|
||||||
|
if (content.type === "photo") {
|
||||||
|
let url = `${content.media_url_https}?name=4096x4096`;
|
||||||
|
return {
|
||||||
|
type: "photo",
|
||||||
|
url,
|
||||||
|
thumb: url,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let url = bestQuality(content.video_info.variants);
|
let url = bestQuality(content.video_info.variants);
|
||||||
const shouldRenderGif = content.type === 'animated_gif' && toGif;
|
const shouldRenderGif = content.type === 'animated_gif' && toGif;
|
||||||
|
|
||||||
|
let type = "video";
|
||||||
|
if (shouldRenderGif) type = "gif";
|
||||||
|
|
||||||
if (needsFixing(content) || shouldRenderGif) {
|
if (needsFixing(content) || shouldRenderGif) {
|
||||||
url = createStream({
|
url = createStream({
|
||||||
service: 'twitter',
|
service: 'twitter',
|
||||||
@ -181,9 +200,9 @@ export default async function({ id, index, toGif, dispatcher }) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
type: 'video',
|
type,
|
||||||
url,
|
url,
|
||||||
thumb: content.media_url_https,
|
thumb: content.media_url_https
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
return { picker };
|
return { picker };
|
||||||
|
@ -1,25 +1,27 @@
|
|||||||
import { Innertube, Session } from 'youtubei.js';
|
import { fetch } from "undici";
|
||||||
import { env } from '../../config.js';
|
|
||||||
import { cleanString } from '../../sub/utils.js';
|
import { Innertube, Session } from "youtubei.js";
|
||||||
import { fetch } from 'undici'
|
|
||||||
import { getCookie, updateCookieValues } from '../cookie/manager.js'
|
import { env } from "../../config.js";
|
||||||
|
import { cleanString } from "../../sub/utils.js";
|
||||||
|
import { getCookie, updateCookieValues } from "../cookie/manager.js";
|
||||||
|
|
||||||
const ytBase = Innertube.create().catch(e => e);
|
const ytBase = Innertube.create().catch(e => e);
|
||||||
|
|
||||||
const codecMatch = {
|
const codecMatch = {
|
||||||
h264: {
|
h264: {
|
||||||
codec: "avc1",
|
videoCodec: "avc1",
|
||||||
aCodec: "mp4a",
|
audioCodec: "mp4a",
|
||||||
container: "mp4"
|
container: "mp4"
|
||||||
},
|
},
|
||||||
av1: {
|
av1: {
|
||||||
codec: "av01",
|
videoCodec: "av01",
|
||||||
aCodec: "mp4a",
|
audioCodec: "mp4a",
|
||||||
container: "mp4"
|
container: "mp4"
|
||||||
},
|
},
|
||||||
vp9: {
|
vp9: {
|
||||||
codec: "vp9",
|
videoCodec: "vp9",
|
||||||
aCodec: "opus",
|
audioCodec: "opus",
|
||||||
container: "webm"
|
container: "webm"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -28,20 +30,21 @@ const transformSessionData = (cookie) => {
|
|||||||
if (!cookie)
|
if (!cookie)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
const values = cookie.values();
|
const values = { ...cookie.values() };
|
||||||
const REQUIRED_VALUES = [
|
const REQUIRED_VALUES = [ 'access_token', 'refresh_token' ];
|
||||||
'access_token', 'refresh_token',
|
|
||||||
'client_id', 'client_secret',
|
|
||||||
'expires'
|
|
||||||
];
|
|
||||||
|
|
||||||
if (REQUIRED_VALUES.some(x => typeof values[x] !== 'string')) {
|
if (REQUIRED_VALUES.some(x => typeof values[x] !== 'string')) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
return {
|
|
||||||
...values,
|
if (values.expires) {
|
||||||
expires: new Date(values.expires),
|
values.expiry_date = values.expires;
|
||||||
};
|
delete values.expires;
|
||||||
|
} else if (!values.expiry_date) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return values;
|
||||||
}
|
}
|
||||||
|
|
||||||
const cloneInnertube = async (customFetch) => {
|
const cloneInnertube = async (customFetch) => {
|
||||||
@ -70,14 +73,19 @@ const cloneInnertube = async (customFetch) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (session.logged_in) {
|
if (session.logged_in) {
|
||||||
await session.oauth.refreshIfRequired();
|
if (session.oauth.shouldRefreshToken()) {
|
||||||
const oldExpiry = new Date(cookie.values().expires);
|
await session.oauth.refreshAccessToken();
|
||||||
const newExpiry = session.oauth.credentials.expires;
|
}
|
||||||
|
|
||||||
|
const cookieValues = cookie.values();
|
||||||
|
const oldExpiry = new Date(cookieValues.expiry_date);
|
||||||
|
const newExpiry = new Date(session.oauth.oauth2_tokens.expiry_date);
|
||||||
|
|
||||||
if (oldExpiry.getTime() !== newExpiry.getTime()) {
|
if (oldExpiry.getTime() !== newExpiry.getTime()) {
|
||||||
updateCookieValues(cookie, {
|
updateCookieValues(cookie, {
|
||||||
...session.oauth.credentials,
|
...session.oauth.client_id,
|
||||||
expires: session.oauth.credentials.expires.toISOString()
|
...session.oauth.oauth2_tokens,
|
||||||
|
expiry_date: newExpiry.toISOString()
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -88,11 +96,16 @@ const cloneInnertube = async (customFetch) => {
|
|||||||
|
|
||||||
export default async function(o) {
|
export default async function(o) {
|
||||||
const yt = await cloneInnertube(
|
const yt = await cloneInnertube(
|
||||||
(input, init) => fetch(input, { ...init, dispatcher: o.dispatcher })
|
(input, init) => fetch(input, {
|
||||||
|
...init,
|
||||||
|
dispatcher: o.dispatcher
|
||||||
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
let info, isDubbed, format = o.format || "h264";
|
const quality = o.quality === "max" ? "9000" : o.quality;
|
||||||
let quality = o.quality === "max" ? "9000" : o.quality; // 9000(p) - max quality
|
|
||||||
|
let info, isDubbed,
|
||||||
|
format = o.format || "h264";
|
||||||
|
|
||||||
function qual(i) {
|
function qual(i) {
|
||||||
if (!i.quality_label) {
|
if (!i.quality_label) {
|
||||||
@ -115,6 +128,7 @@ export default async function(o) {
|
|||||||
if (!info) return { error: 'ErrorCantConnectToServiceAPI' };
|
if (!info) return { error: 'ErrorCantConnectToServiceAPI' };
|
||||||
|
|
||||||
const playability = info.playability_status;
|
const playability = info.playability_status;
|
||||||
|
const basicInfo = info.basic_info;
|
||||||
|
|
||||||
if (playability.status === 'LOGIN_REQUIRED') {
|
if (playability.status === 'LOGIN_REQUIRED') {
|
||||||
if (playability.reason.endsWith('bot')) {
|
if (playability.reason.endsWith('bot')) {
|
||||||
@ -129,11 +143,11 @@ export default async function(o) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (playability.status !== 'OK') return { error: 'ErrorYTUnavailable' };
|
if (playability.status !== 'OK') return { error: 'ErrorYTUnavailable' };
|
||||||
if (info.basic_info.is_live) return { error: 'ErrorLiveVideo' };
|
if (basicInfo.is_live) return { error: 'ErrorLiveVideo' };
|
||||||
|
|
||||||
// return a critical error if returned video is "Video Not Available"
|
// return a critical error if returned video is "Video Not Available"
|
||||||
// or a similar stub by youtube
|
// or a similar stub by youtube
|
||||||
if (info.basic_info.id !== o.id) {
|
if (basicInfo.id !== o.id) {
|
||||||
return {
|
return {
|
||||||
error: 'ErrorCantConnectToServiceAPI',
|
error: 'ErrorCantConnectToServiceAPI',
|
||||||
critical: true
|
critical: true
|
||||||
@ -142,11 +156,16 @@ export default async function(o) {
|
|||||||
|
|
||||||
let bestQuality, hasAudio;
|
let bestQuality, hasAudio;
|
||||||
|
|
||||||
const filterByCodec = (formats) => formats.filter(e =>
|
const filterByCodec = (formats) =>
|
||||||
e.mime_type.includes(codecMatch[format].codec) || e.mime_type.includes(codecMatch[format].aCodec)
|
formats
|
||||||
).sort((a, b) => Number(b.bitrate) - Number(a.bitrate));
|
.filter(e =>
|
||||||
|
e.mime_type.includes(codecMatch[format].videoCodec)
|
||||||
|
|| e.mime_type.includes(codecMatch[format].audioCodec)
|
||||||
|
)
|
||||||
|
.sort((a, b) => Number(b.bitrate) - Number(a.bitrate));
|
||||||
|
|
||||||
let adaptive_formats = filterByCodec(info.streaming_data.adaptive_formats);
|
let adaptive_formats = filterByCodec(info.streaming_data.adaptive_formats);
|
||||||
|
|
||||||
if (adaptive_formats.length === 0 && format === "vp9") {
|
if (adaptive_formats.length === 0 && format === "vp9") {
|
||||||
format = "h264"
|
format = "h264"
|
||||||
adaptive_formats = filterByCodec(info.streaming_data.adaptive_formats)
|
adaptive_formats = filterByCodec(info.streaming_data.adaptive_formats)
|
||||||
@ -156,27 +175,43 @@ export default async function(o) {
|
|||||||
hasAudio = adaptive_formats.find(i => i.has_audio && i.content_length);
|
hasAudio = adaptive_formats.find(i => i.has_audio && i.content_length);
|
||||||
|
|
||||||
if (bestQuality) bestQuality = qual(bestQuality);
|
if (bestQuality) bestQuality = qual(bestQuality);
|
||||||
if (!bestQuality && !o.isAudioOnly || !hasAudio) return { error: 'ErrorYTTryOtherCodec' };
|
|
||||||
if (info.basic_info.duration > env.durationLimit) return { error: ['ErrorLengthLimit', env.durationLimit / 60] };
|
|
||||||
|
|
||||||
let checkBestAudio = (i) => (i.has_audio && !i.has_video),
|
if ((!bestQuality && !o.isAudioOnly) || !hasAudio)
|
||||||
audio = adaptive_formats.find(i => checkBestAudio(i) && !i.is_dubbed);
|
return { error: 'ErrorYTTryOtherCodec' };
|
||||||
|
|
||||||
|
if (basicInfo.duration > env.durationLimit)
|
||||||
|
return { error: ['ErrorLengthLimit', env.durationLimit / 60] };
|
||||||
|
|
||||||
|
const checkBestAudio = (i) => (i.has_audio && !i.has_video);
|
||||||
|
|
||||||
|
let audio = adaptive_formats.find(i =>
|
||||||
|
checkBestAudio(i) && i.is_original
|
||||||
|
);
|
||||||
|
|
||||||
if (o.dubLang) {
|
if (o.dubLang) {
|
||||||
let dubbedAudio = adaptive_formats.find(i =>
|
let dubbedAudio = adaptive_formats.find(i =>
|
||||||
checkBestAudio(i) && i.language === o.dubLang && i.audio_track && !i.audio_track.audio_is_default
|
checkBestAudio(i)
|
||||||
);
|
&& i.language === o.dubLang
|
||||||
|
&& i.audio_track
|
||||||
|
)
|
||||||
|
|
||||||
if (dubbedAudio) {
|
if (dubbedAudio) {
|
||||||
audio = dubbedAudio;
|
audio = dubbedAudio;
|
||||||
isDubbed = true
|
isDubbed = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
let fileMetadata = {
|
|
||||||
title: cleanString(info.basic_info.title.trim()),
|
if (!audio) {
|
||||||
artist: cleanString(info.basic_info.author.replace("- Topic", "").trim()),
|
audio = adaptive_formats.find(i => checkBestAudio(i));
|
||||||
}
|
}
|
||||||
if (info.basic_info.short_description && info.basic_info.short_description.startsWith("Provided to YouTube by")) {
|
|
||||||
let descItems = info.basic_info.short_description.split("\n\n");
|
let fileMetadata = {
|
||||||
|
title: cleanString(basicInfo.title.trim()),
|
||||||
|
artist: cleanString(basicInfo.author.replace("- Topic", "").trim()),
|
||||||
|
}
|
||||||
|
|
||||||
|
if (basicInfo?.short_description?.startsWith("Provided to YouTube by")) {
|
||||||
|
let descItems = basicInfo.short_description.split("\n\n");
|
||||||
fileMetadata.album = descItems[2];
|
fileMetadata.album = descItems[2];
|
||||||
fileMetadata.copyright = descItems[3];
|
fileMetadata.copyright = descItems[3];
|
||||||
if (descItems[4].startsWith("Released on:")) {
|
if (descItems[4].startsWith("Released on:")) {
|
||||||
@ -192,19 +227,23 @@ export default async function(o) {
|
|||||||
youtubeDubName: isDubbed ? o.dubLang : false
|
youtubeDubName: isDubbed ? o.dubLang : false
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hasAudio && o.isAudioOnly) return {
|
if (audio && o.isAudioOnly) return {
|
||||||
type: "render",
|
type: "render",
|
||||||
isAudioOnly: true,
|
isAudioOnly: true,
|
||||||
urls: audio.decipher(yt.session.player),
|
urls: audio.decipher(yt.session.player),
|
||||||
filenameAttributes: filenameAttributes,
|
filenameAttributes: filenameAttributes,
|
||||||
fileMetadata: fileMetadata,
|
fileMetadata: fileMetadata,
|
||||||
bestAudio: format === "h264" ? 'm4a' : 'opus'
|
bestAudio: format === "h264" ? "m4a" : "opus"
|
||||||
}
|
}
|
||||||
|
|
||||||
const matchingQuality = Number(quality) > Number(bestQuality) ? bestQuality : quality,
|
const matchingQuality = Number(quality) > Number(bestQuality) ? bestQuality : quality,
|
||||||
checkSingle = i => qual(i) === matchingQuality && i.mime_type.includes(codecMatch[format].codec),
|
checkSingle = i =>
|
||||||
checkRender = i => qual(i) === matchingQuality && i.has_video && !i.has_audio;
|
qual(i) === matchingQuality && i.mime_type.includes(codecMatch[format].videoCodec),
|
||||||
|
checkRender = i =>
|
||||||
|
qual(i) === matchingQuality && i.has_video && !i.has_audio;
|
||||||
|
|
||||||
let match, type, urls;
|
let match, type, urls;
|
||||||
|
|
||||||
if (!o.isAudioOnly && !o.isAudioMuted && format === 'h264') {
|
if (!o.isAudioOnly && !o.isAudioMuted && format === 'h264') {
|
||||||
match = info.streaming_data.formats.find(checkSingle);
|
match = info.streaming_data.formats.find(checkSingle);
|
||||||
type = "bridge";
|
type = "bridge";
|
||||||
@ -212,10 +251,14 @@ export default async function(o) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const video = adaptive_formats.find(checkRender);
|
const video = adaptive_formats.find(checkRender);
|
||||||
if (!match && video) {
|
|
||||||
|
if (!match && video && audio) {
|
||||||
match = video;
|
match = video;
|
||||||
type = "render";
|
type = "render";
|
||||||
urls = [video.decipher(yt.session.player), audio.decipher(yt.session.player)];
|
urls = [
|
||||||
|
video.decipher(yt.session.player),
|
||||||
|
audio.decipher(yt.session.player)
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
@ -226,7 +269,7 @@ export default async function(o) {
|
|||||||
return {
|
return {
|
||||||
type,
|
type,
|
||||||
urls,
|
urls,
|
||||||
filenameAttributes,
|
filenameAttributes,
|
||||||
fileMetadata
|
fileMetadata
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -114,6 +114,12 @@
|
|||||||
"patterns": ["video/:id"],
|
"patterns": ["video/:id"],
|
||||||
"enabled": true
|
"enabled": true
|
||||||
},
|
},
|
||||||
|
"snapchat": {
|
||||||
|
"alias": "snapchat stories & spotlights",
|
||||||
|
"subdomains": ["t", "story"],
|
||||||
|
"patterns": [":shortLink", "spotlight/:spotlightId", "add/:username/:storyId", "u/:username/:storyId", "add/:username", "u/:username"],
|
||||||
|
"enabled": true
|
||||||
|
},
|
||||||
"loom": {
|
"loom": {
|
||||||
"alias": "loom videos",
|
"alias": "loom videos",
|
||||||
"patterns": ["share/:id"],
|
"patterns": ["share/:id"],
|
||||||
@ -124,6 +130,19 @@
|
|||||||
"tld": "bg",
|
"tld": "bg",
|
||||||
"patterns": ["watch/:id"],
|
"patterns": ["watch/:id"],
|
||||||
"enabled": true
|
"enabled": true
|
||||||
|
},
|
||||||
|
"facebook": {
|
||||||
|
"alias": "facebook videos",
|
||||||
|
"altDomains": ["fb.watch"],
|
||||||
|
"subdomains": ["web"],
|
||||||
|
"patterns": [
|
||||||
|
"_shortLink/:shortLink",
|
||||||
|
":username/videos/:caption/:id",
|
||||||
|
":username/videos/:id",
|
||||||
|
"reel/:id",
|
||||||
|
"share/:shareType/:id"
|
||||||
|
],
|
||||||
|
"enabled": true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -29,9 +29,14 @@ export const testers = {
|
|||||||
patternMatch.id?.length === 32 || patternMatch.yappyId?.length === 32,
|
patternMatch.id?.length === 32 || patternMatch.yappyId?.length === 32,
|
||||||
|
|
||||||
"soundcloud": (patternMatch) =>
|
"soundcloud": (patternMatch) =>
|
||||||
(patternMatch.author?.length <= 255 && patternMatch.song?.length <= 255)
|
(patternMatch.author?.length <= 255 && patternMatch.song?.length <= 255)
|
||||||
|| patternMatch.shortLink?.length <= 32,
|
|| patternMatch.shortLink?.length <= 32,
|
||||||
|
|
||||||
|
"snapchat": (patternMatch) =>
|
||||||
|
(patternMatch.username?.length <= 32 && (!patternMatch.storyId || patternMatch.storyId?.length <= 255))
|
||||||
|
|| patternMatch.spotlightId?.length <= 255
|
||||||
|
|| patternMatch.shortLink?.length <= 16,
|
||||||
|
|
||||||
"streamable": (patternMatch) =>
|
"streamable": (patternMatch) =>
|
||||||
patternMatch.id?.length === 6,
|
patternMatch.id?.length === 6,
|
||||||
|
|
||||||
@ -60,4 +65,11 @@ export const testers = {
|
|||||||
|
|
||||||
"youtube": (patternMatch) =>
|
"youtube": (patternMatch) =>
|
||||||
patternMatch.id?.length <= 11,
|
patternMatch.id?.length <= 11,
|
||||||
|
|
||||||
|
"facebook": (patternMatch) =>
|
||||||
|
patternMatch.shortLink?.length <= 11
|
||||||
|
|| patternMatch.username?.length <= 30
|
||||||
|
|| patternMatch.caption?.length <= 255
|
||||||
|
|| patternMatch.id?.length <= 20 && !patternMatch.shareType
|
||||||
|
|| patternMatch.id?.length <= 20 && patternMatch.shareType?.length === 1,
|
||||||
}
|
}
|
||||||
|
@ -64,7 +64,17 @@ function aliasURL(url) {
|
|||||||
if (url.hostname === 'dai.ly' && parts.length === 2) {
|
if (url.hostname === 'dai.ly' && parts.length === 2) {
|
||||||
url = new URL(`https://dailymotion.com/video/${parts[1]}`)
|
url = new URL(`https://dailymotion.com/video/${parts[1]}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case "facebook":
|
||||||
|
case "fb":
|
||||||
|
if (url.searchParams.get('v')) {
|
||||||
|
url = new URL(`https://web.facebook.com/user/videos/${url.searchParams.get('v')}`)
|
||||||
|
}
|
||||||
|
if (url.hostname === 'fb.watch') {
|
||||||
|
url = new URL(`https://web.facebook.com/_shortLink/${parts[1]}`)
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case "ddinstagram":
|
case "ddinstagram":
|
||||||
if (services.instagram.altDomains.includes(host.domain) && [null, 'd', 'g'].includes(host.subdomain)) {
|
if (services.instagram.altDomains.includes(host.domain) && [null, 'd', 'g'].includes(host.subdomain)) {
|
||||||
url.hostname = 'instagram.com';
|
url.hostname = 'instagram.com';
|
||||||
|
@ -25,7 +25,7 @@ function killProcess(p) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function getCommand(args) {
|
function getCommand(args) {
|
||||||
if (!isNaN(env.processingPriority)) {
|
if (typeof env.processingPriority === 'number' && !isNaN(env.processingPriority)) {
|
||||||
return ['nice', ['-n', env.processingPriority.toString(), ffmpeg, ...args]]
|
return ['nice', ['-n', env.processingPriority.toString(), ffmpeg, ...args]]
|
||||||
}
|
}
|
||||||
return [ffmpeg, args]
|
return [ffmpeg, args]
|
||||||
|
@ -45,6 +45,13 @@ export function cleanHTML(html) {
|
|||||||
return clean
|
return clean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getRedirectingURL(url) {
|
||||||
|
return fetch(url, { redirect: 'manual' }).then((r) => {
|
||||||
|
if ([301, 302, 303].includes(r.status) && r.headers.has('location'))
|
||||||
|
return r.headers.get('location');
|
||||||
|
}).catch(() => null);
|
||||||
|
}
|
||||||
|
|
||||||
export function merge(a, b) {
|
export function merge(a, b) {
|
||||||
for (const k of Object.keys(b)) {
|
for (const k of Object.keys(b)) {
|
||||||
if (Array.isArray(b[k])) {
|
if (Array.isArray(b[k])) {
|
||||||
|
42
src/modules/test.js
Normal file
42
src/modules/test.js
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
import { normalizeRequest } from "../modules/processing/request.js";
|
||||||
|
import match from "./processing/match.js";
|
||||||
|
import { extract } from "./processing/url.js";
|
||||||
|
|
||||||
|
export async function runTest(url, params, expect) {
|
||||||
|
const normalized = normalizeRequest({ url, ...params });
|
||||||
|
if (!normalized) {
|
||||||
|
throw "invalid request";
|
||||||
|
}
|
||||||
|
|
||||||
|
const parsed = extract(normalized.url);
|
||||||
|
if (parsed === null) {
|
||||||
|
throw `invalid url: ${normalized.url}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = await match(
|
||||||
|
parsed.host, parsed.patternMatch, "en", normalized
|
||||||
|
);
|
||||||
|
|
||||||
|
let error = [];
|
||||||
|
if (expect.status !== result.body.status) {
|
||||||
|
const detail = `${expect.status} (expected) != ${result.body.status} (actual)`;
|
||||||
|
error.push(`status mismatch: ${detail}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (expect.code !== result.status) {
|
||||||
|
const detail = `${expect.code} (expected) != ${result.status} (actual)`;
|
||||||
|
error.push(`status code mismatch: ${detail}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (error.length) {
|
||||||
|
if (result.body.text) {
|
||||||
|
error.push(`error message: ${result.body.text}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw error.join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.body.status === 'stream') {
|
||||||
|
// TODO: stream testing
|
||||||
|
}
|
||||||
|
}
|
@ -20,9 +20,9 @@ tube.session.once(
|
|||||||
);
|
);
|
||||||
|
|
||||||
tube.session.once('auth-error', (err) => bail('An error occurred:', err));
|
tube.session.once('auth-error', (err) => bail('An error occurred:', err));
|
||||||
tube.session.once('auth', ({ status, credentials, ...rest }) => {
|
tube.session.once('auth', ({ credentials }) => {
|
||||||
if (status !== 'SUCCESS') {
|
if (!credentials.access_token) {
|
||||||
bail('something went wrong', rest);
|
bail('something went wrong');
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(
|
console.log(
|
||||||
|
81
src/util/test-ci.js
Normal file
81
src/util/test-ci.js
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
import { env } from "../modules/config.js";
|
||||||
|
import { runTest } from "../modules/test.js";
|
||||||
|
import { loadLoc } from "../localization/manager.js";
|
||||||
|
import { loadJSON } from "../modules/sub/loadFromFs.js";
|
||||||
|
import { Red, Bright } from "../modules/sub/consoleText.js";
|
||||||
|
|
||||||
|
const tests = loadJSON('./src/util/tests.json');
|
||||||
|
const services = loadJSON('./src/modules/processing/servicesConfig.json');
|
||||||
|
|
||||||
|
// services that are known to frequently fail due to external
|
||||||
|
// factors (e.g. rate limiting)
|
||||||
|
const finnicky = new Set(['bilibili', 'instagram', 'youtube'])
|
||||||
|
|
||||||
|
const action = process.argv[2];
|
||||||
|
switch (action) {
|
||||||
|
case "get-services":
|
||||||
|
const fromConfig = Object.keys(services.config);
|
||||||
|
|
||||||
|
const missingTests = fromConfig.filter(
|
||||||
|
service => !tests[service] || tests[service].length === 0
|
||||||
|
);
|
||||||
|
|
||||||
|
if (missingTests.length) {
|
||||||
|
console.error('services have no tests:', missingTests);
|
||||||
|
console.log('[]');
|
||||||
|
process.exitCode = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(JSON.stringify(fromConfig));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "run-tests-for":
|
||||||
|
const service = process.argv[3];
|
||||||
|
let failed = false;
|
||||||
|
|
||||||
|
if (!tests[service]) {
|
||||||
|
console.error('no such service:', service);
|
||||||
|
}
|
||||||
|
|
||||||
|
await loadLoc();
|
||||||
|
env.streamLifespan = 10000;
|
||||||
|
env.apiURL = 'http://x';
|
||||||
|
|
||||||
|
for (const test of tests[service]) {
|
||||||
|
const { name, url, params, expected } = test;
|
||||||
|
const canFail = test.canFail || finnicky.has(service);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await runTest(url, params, expected);
|
||||||
|
console.log(`${service}/${name}: ok`);
|
||||||
|
|
||||||
|
} catch(e) {
|
||||||
|
failed = !canFail;
|
||||||
|
|
||||||
|
let failText = canFail ? `${Red('FAIL')} (ignored)` : Bright(Red('FAIL'));
|
||||||
|
if (canFail && process.env.GITHUB_ACTION) {
|
||||||
|
console.log(`::warning title=${service}/${name.replace(/,/g, ';')}::failed and was ignored`);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.error(`${service}/${name}: ${failText}`);
|
||||||
|
const errorString = e.toString().split('\n');
|
||||||
|
let c = '┃';
|
||||||
|
errorString.forEach((line, index) => {
|
||||||
|
line = line.replace('!=', Red('!='));
|
||||||
|
|
||||||
|
if (index === errorString.length - 1) {
|
||||||
|
c = '┗';
|
||||||
|
}
|
||||||
|
|
||||||
|
console.error(` ${c}`, line);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
process.exitCode = Number(failed);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
console.error('invalid action:', action);
|
||||||
|
process.exitCode = 1;
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"twitter": [{
|
"twitter": [{
|
||||||
"name": "regular video",
|
"name": "regular video",
|
||||||
"url": "https://twitter.com/TwitterSpaces/status/1526955853743546372?s=20",
|
"url": "https://twitter.com/X/status/1697304622749086011",
|
||||||
"params": {
|
"params": {
|
||||||
"aFormat": "mp3",
|
"aFormat": "mp3",
|
||||||
"isAudioOnly": false,
|
"isAudioOnly": false,
|
||||||
@ -11,10 +11,9 @@
|
|||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "redirect"
|
"status": "redirect"
|
||||||
}
|
}
|
||||||
},
|
}, {
|
||||||
{
|
|
||||||
"name": "video with mobile web mediaviewer",
|
"name": "video with mobile web mediaviewer",
|
||||||
"url": "https://twitter.com/XSpaces/status/1526955853743546372/mediaViewer?currentTweet=1526955853743546372¤tTweetUser=XSpaces¤tTweet=1526955853743546372¤tTweetUser=XSpaces",
|
"url": "https://twitter.com/X/status/1697304622749086011/mediaViewer?currentTweet=1697304622749086011¤tTweetUser=X¤tTweet=1697304622749086011¤tTweetUser=X",
|
||||||
"params": {},
|
"params": {},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
@ -34,7 +33,7 @@
|
|||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "mixed media (image + gif)",
|
"name": "mixed media (image + gif)",
|
||||||
"url": "https://twitter.com/Twitter/status/1580661436132757506?s=20",
|
"url": "https://twitter.com/sky_mj26/status/1807756010712428565",
|
||||||
"params": {
|
"params": {
|
||||||
"aFormat": "mp3",
|
"aFormat": "mp3",
|
||||||
"isAudioOnly": false,
|
"isAudioOnly": false,
|
||||||
@ -42,7 +41,7 @@
|
|||||||
},
|
},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "redirect"
|
"status": "picker"
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "picker: mixed media (3 videos)",
|
"name": "picker: mixed media (3 videos)",
|
||||||
@ -113,9 +112,10 @@
|
|||||||
"status": "redirect"
|
"status": "redirect"
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "age-restricted video",
|
"name": "age restricted video",
|
||||||
"url": "https://twitter.com/FckyeahCharli/status/1650987582749065220",
|
"url": "https://twitter.com/FckyeahCharli/status/1650987582749065220",
|
||||||
"params": {},
|
"params": {},
|
||||||
|
"canFail": true,
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "redirect"
|
"status": "redirect"
|
||||||
@ -136,6 +136,22 @@
|
|||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "redirect"
|
"status": "redirect"
|
||||||
}
|
}
|
||||||
|
}, {
|
||||||
|
"name": "post with 1 image",
|
||||||
|
"url": "https://x.com/PopCrave/status/1815960083475423235",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "post with 4 images",
|
||||||
|
"url": "https://x.com/PopCrave/status/1816260887147114696",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "picker"
|
||||||
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "retweeted video, isAudioOnly",
|
"name": "retweeted video, isAudioOnly",
|
||||||
"url": "https://twitter.com/hugekiwinuts/status/1618671150829309953?s=46&t=gItGzgwGQQJJaJrO6qc1Pg",
|
"url": "https://twitter.com/hugekiwinuts/status/1618671150829309953?s=46&t=gItGzgwGQQJJaJrO6qc1Pg",
|
||||||
@ -650,23 +666,23 @@
|
|||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "1080p dash parcel",
|
"name": "1080p dash parcel",
|
||||||
"url": "https://vimeo.com/774694040",
|
"url": "https://vimeo.com/967252742",
|
||||||
"params": {
|
"params": {
|
||||||
"vQuality": "1440"
|
"vQuality": "1440"
|
||||||
},
|
},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "stream"
|
"status": "redirect"
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "720p dash parcel",
|
"name": "720p dash parcel",
|
||||||
"url": "https://vimeo.com/774694040",
|
"url": "https://vimeo.com/967252742",
|
||||||
"params": {
|
"params": {
|
||||||
"vQuality": "360"
|
"vQuality": "360"
|
||||||
},
|
},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "stream"
|
"status": "redirect"
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "private video",
|
"name": "private video",
|
||||||
@ -676,6 +692,14 @@
|
|||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "redirect"
|
"status": "redirect"
|
||||||
}
|
}
|
||||||
|
}, {
|
||||||
|
"name": "mature video",
|
||||||
|
"url": "https://vimeo.com/973212054",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
}],
|
}],
|
||||||
"reddit": [{
|
"reddit": [{
|
||||||
"name": "video with audio",
|
"name": "video with audio",
|
||||||
@ -947,7 +971,7 @@
|
|||||||
}],
|
}],
|
||||||
"streamable": [{
|
"streamable": [{
|
||||||
"name": "regular video",
|
"name": "regular video",
|
||||||
"url": "https://streamable.com/03r3c2",
|
"url": "https://streamable.com/p9cln4",
|
||||||
"params": {},
|
"params": {},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
@ -963,7 +987,7 @@
|
|||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "regular video (isAudioOnly)",
|
"name": "regular video (isAudioOnly)",
|
||||||
"url": "https://streamable.com/03r3c2",
|
"url": "https://streamable.com/p9cln4",
|
||||||
"params": {
|
"params": {
|
||||||
"isAudioOnly": true
|
"isAudioOnly": true
|
||||||
},
|
},
|
||||||
@ -973,7 +997,7 @@
|
|||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "regular video (isAudioMuted)",
|
"name": "regular video (isAudioMuted)",
|
||||||
"url": "https://streamable.com/03r3c2",
|
"url": "https://streamable.com/p9cln4",
|
||||||
"params": {
|
"params": {
|
||||||
"isAudioMuted": true
|
"isAudioMuted": true
|
||||||
},
|
},
|
||||||
@ -1042,8 +1066,8 @@
|
|||||||
"url": "https://rutube.ru/video/b521653b4f71ece57b8ff54e57ca9b82/",
|
"url": "https://rutube.ru/video/b521653b4f71ece57b8ff54e57ca9b82/",
|
||||||
"params": {},
|
"params": {},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 400,
|
||||||
"status": "stream"
|
"status": "error"
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "vertical video",
|
"name": "vertical video",
|
||||||
@ -1055,7 +1079,8 @@
|
|||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
"name": "yappy",
|
"name": "yappy",
|
||||||
"url": "https://rutube.ru/yappy/f1771e86294748eea8ecb2ac88e55740/",
|
"url": "https://rutube.ru/yappy/c8c32bf7aee04412837656ea26c2b25b/",
|
||||||
|
"canFail": true,
|
||||||
"params": {},
|
"params": {},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
@ -1132,6 +1157,31 @@
|
|||||||
"status": "stream"
|
"status": "stream"
|
||||||
}
|
}
|
||||||
}],
|
}],
|
||||||
|
"snapchat": [{
|
||||||
|
"name": "spotlight",
|
||||||
|
"url": "https://www.snapchat.com/spotlight/W7_EDlXWTBiXAEEniNoMPwAAYdWxucG9pZmNqAY46j0a5AY46j0YbAAAAAQ",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "shortlinked spotlight",
|
||||||
|
"url": "https://t.snapchat.com/4ZsiBLDi",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "story",
|
||||||
|
"url": "https://www.snapchat.com/add/bazerkmakane",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "picker"
|
||||||
|
}
|
||||||
|
}],
|
||||||
"loom": [{
|
"loom": [{
|
||||||
"name": "1080p video",
|
"name": "1080p video",
|
||||||
"url": "https://www.loom.com/share/313bf71d20ca47b2a35b6634cefdb761",
|
"url": "https://www.loom.com/share/313bf71d20ca47b2a35b6634cefdb761",
|
||||||
@ -1156,9 +1206,68 @@
|
|||||||
"params": {
|
"params": {
|
||||||
"isAudioOnly": true
|
"isAudioOnly": true
|
||||||
},
|
},
|
||||||
|
"expected": {
|
||||||
|
"code": 400,
|
||||||
|
"status": "error"
|
||||||
|
}
|
||||||
|
}],
|
||||||
|
"facebook": [{
|
||||||
|
"name": "direct video with username and id",
|
||||||
|
"url": "https://web.facebook.com/100048111287134/videos/1157798148685638/",
|
||||||
|
"params": {},
|
||||||
"expected": {
|
"expected": {
|
||||||
"code": 200,
|
"code": 200,
|
||||||
"status": "stream"
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "direct video with id as query param",
|
||||||
|
"url": "https://web.facebook.com/watch/?v=883839773514682&ref=sharing",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "direct video with caption",
|
||||||
|
"url": "https://web.facebook.com/wood57/videos/𝐒𝐞𝐛𝐚𝐬𝐤𝐨𝐦-𝐟𝐮𝐥𝐥/883839773514682",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "shortlink video",
|
||||||
|
"url": "https://fb.watch/r1K6XHMfGT/",
|
||||||
|
"canFail": true,
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "reel video",
|
||||||
|
"url": "https://web.facebook.com/reel/730293269054758",
|
||||||
|
"canFail": true,
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "shared video link",
|
||||||
|
"url": "https://www.facebook.com/share/v/NEf87jbPTvFE8LsL/",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
"name": "shared video link v2",
|
||||||
|
"url": "https://web.facebook.com/share/r/JFZfPVgLkiJQmWrr/",
|
||||||
|
"params": {},
|
||||||
|
"expected": {
|
||||||
|
"code": 200,
|
||||||
|
"status": "redirect"
|
||||||
}
|
}
|
||||||
}]
|
}]
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user