Compare commits

...
Sign in to create a new pull request.

17 commits

Author SHA1 Message Date
4a8f0f8cd2
feat(proxy): ignore certificate errors for proxy
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 1m18s
2025-01-11 18:13:24 -03:00
5560d9f986
feat(metrics): better metrics and support for failed and successful requests
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 1m26s
2025-01-11 14:58:37 -03:00
12c965ceaf
add resolution limit to save bandwidth
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 1m8s
add environment variable for the resolution limit
2024-12-18 19:38:54 -03:00
309015454b
Merge remote-tracking branch 'upstream/master'
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 1m27s
2024-12-26 03:44:43 -03:00
d53d10b774
add health endpoint for healthchecks (#27) 2024-12-25 11:55:41 +00:00
Emilien Devos
f89f41380a Revert "workaround issue with TV not returning any shortDescription"
This reverts commit 6ca59654ba.
2024-12-25 10:13:37 +01:00
Emilien Devos
fd76a51933 true to yes and false to no 2024-12-24 23:19:29 +01:00
Emilien Devos
f318c94bd8 include alr=false in order to disable application redirect for now
fixes #30
2024-12-24 20:57:13 +01:00
72f3df37e6
Merge remote-tracking branch 'upstream/master'
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 1m3s
2024-12-24 15:20:15 -03:00
Emilien Devos
6ca59654ba workaround issue with TV not returning any shortDescription 2024-12-24 15:45:50 +01:00
Émilien (perso)
7c0e26f7f8
add section about documentation 2024-12-24 11:57:59 +00:00
fa9a3ffb3e
support for video_cache on disk
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 1m2s
2024-12-21 01:20:25 -03:00
9f579c806a
add support for prometheus metrics
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 54s
2024-12-17 21:56:44 -03:00
f73ed00b6d
add support for external token generator key
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 53s
2024-12-17 03:28:49 -03:00
465878355b
add support for external token generator
All checks were successful
Build and Push Docker Image / build-and-push (push) Successful in 55s
2024-12-17 01:16:59 -03:00
Emilien
8084936d5a add message about secret key size 2024-12-16 11:51:41 +01:00
Émilien (perso)
70cf366639
Update to deno 2.1.4 2024-12-16 11:46:18 +01:00
13 changed files with 246 additions and 37 deletions

View file

@ -1,4 +1,4 @@
FROM i.sanxian.tech/denoland/deno:debian-2.1.3 AS builder
FROM denoland/deno:debian-2.1.4 AS builder
ARG TINI_VERSION=0.19.0

View file

@ -6,8 +6,12 @@ Companion for Invidious which handle all the video stream retrieval from YouTube
- [deno](https://docs.deno.com/runtime/)
## Run Locally
## Documentation
- Installation guide: https://docs.invidious.io/companion-installation/
- Extra documentation for Invidious companion: https://github.com/iv-org/invidious-companion/wiki
## Run Locally (development)
```
SERVER_SECRET_KEY=CHANGEME deno task dev
```
```

View file

@ -1,9 +1,11 @@
[server]
port = 8282
host = "127.0.0.1"
secret_key = "myBeautifulKey"
# secret key needs to be 16 characters long or more
secret_key = "CHANGE_ME"
base_url = "http://localhost:8282"
verify_requests = false
# max_dash_resolution = 1080
[cache]
enabled = true

View file

@ -1,12 +1,13 @@
{
"tasks": {
"dev": "deno run --allow-import=github.com:443,jsr.io:443,raw.githubusercontent.com:443,esm.sh:443,deno.land:443 --allow-net --allow-env --allow-sys=hostname --allow-read --allow-write=/var/tmp/youtubei.js --watch src/main.ts",
"compile": "deno compile --include ./src/lib/helpers/youtubePlayerReq.ts --include ./src/lib/helpers/getFetchClient.ts --output invidious_companion --allow-import=github.com:443,jsr.io:443,raw.githubusercontent.com:443,esm.sh:443,deno.land:443 --allow-net --allow-env --allow-read --allow-sys=hostname --allow-write=/var/tmp/youtubei.js src/main.ts"
"compile": "deno compile --include ./src/lib/helpers/youtubePlayerReq.ts --include ./src/lib/helpers/getFetchClient.ts --output invidious_companion --allow-import=github.com:443,jsr.io:443,raw.githubusercontent.com:443,esm.sh:443,deno.land:443 --allow-net --allow-env --allow-read --allow-sys=hostname --allow-write=/var/tmp/youtubei.js --unsafely-ignore-certificate-errors src/main.ts"
},
"imports": {
"hono": "jsr:@hono/hono@^4.6.5",
"hono/logger": "jsr:@hono/hono@^4.6.5/logger",
"hono/bearer-auth": "jsr:@hono/hono@^4.6.5/bearer-auth",
"prom-client": "npm:prom-client@^15.1.3",
"youtubei.js": "https://raw.githubusercontent.com/LuanRT/YouTube.js/refs/tags/v12.2.0-deno/deno.ts",
"youtubei.js/Utils": "https://raw.githubusercontent.com/LuanRT/YouTube.js/refs/tags/v12.2.0-deno/deno/src/utils/Utils.ts",
"youtubei.js/NavigationEndpoint": "https://raw.githubusercontent.com/LuanRT/YouTube.js/refs/tags/v12.2.0-deno/deno/src/parser/classes/NavigationEndpoint.ts",

View file

@ -2,6 +2,7 @@ import { ApiResponse, Innertube, YT } from "youtubei.js";
import { generateRandomString } from "youtubei.js/Utils";
import { compress, decompress } from "https://deno.land/x/brotli@0.1.7/mod.ts";
import { Store } from "@willsoto/node-konfig-core";
import { failedRequests, successfulRequests } from "../../routes/index.ts";
let youtubePlayerReqLocation = "youtubePlayerReq";
if (Deno.env.get("YT_PLAYER_REQ_LOCATION")) {
if (Deno.env.has("DENO_COMPILED")) {
@ -15,7 +16,15 @@ if (Deno.env.get("YT_PLAYER_REQ_LOCATION")) {
}
const { youtubePlayerReq } = await import(youtubePlayerReqLocation);
const kv = await Deno.openKv();
const videoCachePath = Deno.env.get("VIDEO_CACHE_PATH") as string || "/var/tmp/youtubei.js/video_cache"
export let kv : Deno.Kv
if ((Deno.env.get("VIDEO_CACHE_ON_DISK")?.toLowerCase() ?? false) == "true") {
console.log("[INFO] Storing video cache on disk")
kv = await Deno.openKv(videoCachePath);
} else {
kv = await Deno.openKv();
}
export const youtubePlayerParsing = async (
innertubeClient: Innertube,
@ -72,6 +81,11 @@ export const youtubePlayerParsing = async (
delete videoData.streamingData.formats[index]
.signatureCipher;
}
if (videoData.streamingData.formats[index].url.includes("alr=yes")) {
videoData.streamingData.formats[index].url.replace("alr=yes", "alr=no");
} else {
videoData.streamingData.formats[index].url += "&alr=no";
}
}
for (
const [index, adaptive_format] of streamingData
@ -91,6 +105,11 @@ export const youtubePlayerParsing = async (
delete videoData.streamingData.adaptiveFormats[index]
.signatureCipher;
}
if (videoData.streamingData.adaptiveFormats[index].url.includes("alr=yes")) {
videoData.streamingData.adaptiveFormats[index].url.replace("alr=yes", "alr=no");
} else {
videoData.streamingData.adaptiveFormats[index].url += "&alr=no";
}
}
}
}
@ -119,22 +138,27 @@ export const youtubePlayerParsing = async (
},
}))(videoData);
if (
cacheEnabled == true && videoData.playabilityStatus?.status == "OK"
) {
(async () => {
await kv.set(
["video_cache", videoId],
compress(
new TextEncoder().encode(
JSON.stringify(videoOnlyNecessaryInfo),
if (videoData.playabilityStatus?.status == "OK") {
successfulRequests.inc()
if (
cacheEnabled == true
) {
(async () => {
await kv.set(
["video_cache", videoId],
compress(
new TextEncoder().encode(
JSON.stringify(videoOnlyNecessaryInfo),
),
),
),
{
expireIn: 1000 * 60 * 60,
},
);
})();
{
expireIn: 1000 * 60 * 60,
},
);
})();
}
} else {
failedRequests.inc()
}
return videoOnlyNecessaryInfo;

View file

@ -3,6 +3,7 @@ import type { BgConfig } from "bgutils";
import { JSDOM } from "jsdom";
import { Innertube, UniversalCache } from "youtubei.js";
import { Store } from "@willsoto/node-konfig-core";
import { poTokenFail, externalTokenGeneratorFail } from "../../routes/index.ts";
let getFetchClientLocation = "getFetchClient";
if (Deno.env.get("GET_FETCH_CLIENT_LOCATION")) {
if (Deno.env.has("DENO_COMPILED")) {
@ -22,8 +23,58 @@ export const poTokenGenerate = async (
konfigStore: Store<Record<string, unknown>>,
innertubeClientCache: UniversalCache,
): Promise<Innertube> => {
const externalTokenGenerator = konfigStore.get(
"server.external_token_generator",
) as string;
const externalTokenGeneratorKey = konfigStore.get(
"server.external_token_generator_key",
) as string;
const requestKey = "O43z0dpjhgX20SCx4KAo";
if (externalTokenGenerator != "" && externalTokenGenerator != undefined) {
console.log(
"poTokenGenerate: using external token generator at " +
externalTokenGenerator,
);
try {
let response: Response;
if (
externalTokenGeneratorKey != "" &&
externalTokenGeneratorKey != undefined
) {
response = await fetch(
`${externalTokenGenerator}/generate`, {
headers: {
'Authorization': `Bearer ${externalTokenGeneratorKey}`
}
}
);
if (response.status == 401) {
throw new Error(`Key '${externalTokenGeneratorKey}' is invalid!`)
}
} else {
response = await fetch(
`${externalTokenGenerator}/generate`,
);
}
const data = await response.json();
return (await Innertube.create({
po_token: data.potoken,
visitor_data: data.visitorData,
fetch: getFetchClient(konfigStore),
cache: innertubeClientCache,
generate_session_locally: true,
}));
} catch (e) {
console.error(
"poTokenGenerate: error fetch token from the external token generator: " +
e,
);
console.log("poTokenGenerate: Using built-in token generator");
externalTokenGeneratorFail.inc()
}
}
if (innertubeClient.session.po_token) {
innertubeClient = await Innertube.create({ retrieve_player: false });
}
@ -51,6 +102,7 @@ export const poTokenGenerate = async (
const bgChallenge = await BG.Challenge.create(bgConfig);
if (!bgChallenge) {
poTokenFail.inc()
throw new Error("Could not get challenge");
}
@ -59,7 +111,10 @@ export const poTokenGenerate = async (
if (interpreterJavascript) {
new Function(interpreterJavascript)();
} else throw new Error("Could not load VM");
} else {
poTokenFail.inc()
throw new Error("Could not load VM");
}
const poTokenResult = await BG.PoToken.generate({
program: bgChallenge.program,

View file

@ -67,17 +67,12 @@ if (!innertubeClientOauthEnabled) {
}
setInterval(
async () => {
const currentDateTime = new Date();
console.log("regenrate token called: " + currentDateTime)
if (innertubeClientJobPoTokenEnabled) {
innertubeClient = await poTokenGenerate(
innertubeClient,
konfigStore,
innertubeClientCache,
);
const currentDateTimexd = new Date();
console.log("regenerate token finished: " + currentDateTimexd)
console.log("po_token: " + innertubeClient.session.po_token)
} else {
innertubeClient = await Innertube.create({
cache: innertubeClientCache,
@ -85,7 +80,9 @@ if (!innertubeClientOauthEnabled) {
});
}
},
konfigStore.get("jobs.youtube_session.frequency_seconds") as number * 1000 || 50000);
konfigStore.get("jobs.youtube_session.frequency_seconds") as number *
1000 || 50000,
);
} else if (innertubeClientOauthEnabled) {
// Fired when waiting for the user to authorize the sign in attempt.
innertubeClient.session.on("auth-pending", (data) => {
@ -117,11 +114,11 @@ app.use("*", async (c, next) => {
});
routes(app, konfigStore);
const https = Deno.env.get("HTTPS");
const port = konfigStore.get("server.port") as number;
const host = konfigStore.get("server.host") as string;
if (https == "TRUE" || https == "true") {
const cert = Deno.readTextFileSync("/data/cert.pem");
const key = Deno.readTextFileSync("/data/key.key");

12
src/routes/health.ts Normal file
View file

@ -0,0 +1,12 @@
import { Hono } from "hono";
const health = new Hono();
health.get("/", () => {
return new Response("OK", {
status: 200,
headers: { "Content-Type": "text/plain" },
});
});
export default health;

View file

@ -2,11 +2,73 @@ import { Hono } from "hono";
import { logger } from "hono/logger";
import { Store } from "@willsoto/node-konfig-core";
import { bearerAuth } from "hono/bearer-auth";
import { Registry, Counter, Gauge } from "prom-client"
import youtubeApiPlayer from "./youtube_api_routes/player.ts";
import invidiousRouteLatestVersion from "./invidious_routes/latestVersion.ts";
import invidiousRouteDashManifest from "./invidious_routes/dashManifest.ts";
import videoPlaybackProxy from "./videoPlaybackProxy.ts";
import metrics from "./metrics.ts";
const METRICS_PREFIX = "invidious_companion_";
export const register = new Registry();
export const externalTokenGeneratorFail = new Counter({
name: `${METRICS_PREFIX}externaltokengenerator_fail`,
help: 'TODO',
registers: [register]
});
export const cachedEntries = new Gauge({
name: `${METRICS_PREFIX}cached_entries`,
help: 'TODO',
registers: [register]
});
export const subreasonProtectCommunity = new Counter({
name: `${METRICS_PREFIX}subreason_protect_community`,
help: 'TODO',
registers: [register]
});
export const poTokenFail = new Counter({
name: `${METRICS_PREFIX}potoken_fail`,
help: 'TODO',
registers: [register]
});
export const reasonBot = new Counter({
name: `${METRICS_PREFIX}reason_bot`,
help: 'TODO',
registers: [register]
});
export const videoUnavailable = new Counter({
name: `${METRICS_PREFIX}video_unavailable`,
help: 'TODO',
registers: [register]
});
export const videoRestricted = new Counter({
name: `${METRICS_PREFIX}video_restricted`,
help: 'TODO',
registers: [register]
});
export const failedRequests = new Counter({
name: `${METRICS_PREFIX}failed_requests`,
help: 'TODO',
registers: [register]
});
export const successfulRequests = new Counter({
name: `${METRICS_PREFIX}successful_requests`,
help: 'TODO',
registers: [register]
});
import health from "./health.ts";
export const routes = (app: Hono, konfigStore: Store<Record<string, unknown>>) => {
app.use("*", logger());
@ -22,4 +84,6 @@ export const routes = (app: Hono, konfigStore: Store<Record<string, unknown>>) =
app.route("/latest_version", invidiousRouteLatestVersion);
app.route("/api/manifest/dash/id", invidiousRouteDashManifest);
app.route("/videoplayback", videoPlaybackProxy);
app.route("/metrics", metrics)
app.route("/healthz", health);
};

View file

@ -24,6 +24,8 @@ dashManifest.get("/:videoId", async (c) => {
Record<string, unknown>
>;
const maxDashResolution = Deno.env.get("SERVER_MAX_DASH_RESOLUTION") || konfigStore.get("server.max_dash_resolution") as number;
if (konfigStore.get("server.verify_requests") && check == undefined) {
throw new HTTPException(400, {
res: new Response("No check ID."),
@ -67,7 +69,12 @@ dashManifest.get("/:videoId", async (c) => {
).includes("av01")
) {
if (i.mime_type.includes("av01")) {
return true;
// @ts-ignore 'i.height' is possibly 'undefined'.
if (i.height > maxDashResolution) {
return false;
} else {
return true;
}
} else {
return false;
}

19
src/routes/metrics.ts Normal file
View file

@ -0,0 +1,19 @@
import { Hono } from "hono";
import { register, cachedEntries } from "./index.ts";
import { kv } from "../lib/helpers/youtubePlayerHandling.ts"
const metrics = new Hono();
metrics.get("/", async () => {
let i = 0;
const entries = kv.list({ prefix: ["video_cache"] })
for await (const _ of entries) {
i += 1
}
cachedEntries.set(i)
return new Response(await register.metrics(), {
headers: { "Content-Type": "text/plain" },
});
});
export default metrics;

View file

@ -52,10 +52,6 @@ videoPlaybackProxy.get("/", async (c) => {
// deno-lint-ignore prefer-const
let queryParams = new URLSearchParams(urlReq.search);
queryParams.delete("host");
// alr parameter is only for WEB/HTML5 clients
if (client.includes("WEB")) {
queryParams.append("alr", "yes");
}
if (rangeHeader) {
queryParams.append(
"range",

View file

@ -3,9 +3,33 @@ import { youtubePlayerParsing } from "../../lib/helpers/youtubePlayerHandling.ts
import { HonoVariables } from "../../lib/types/HonoVariables.ts";
import { Innertube } from "youtubei.js";
import { Store } from "@willsoto/node-konfig-core";
import { reasonBot, subreasonProtectCommunity, videoRestricted, videoUnavailable } from "../index.ts";
const player = new Hono<{ Variables: HonoVariables }>();
const errors = [
{
// @ts-ignore: Property 'playabilityStatus' does not exist on type 'object'
check: (yt: object) => yt.playabilityStatus?.reason?.includes("Sign in to confirm youre not a bot"),
action: () => reasonBot.inc(),
},
{
// @ts-ignore: Property 'playabilityStatus' does not exist on type 'object'
check: (yt: object) => yt.playabilityStatus?.errorScreen?.playerErrorMessageRenderer?.subreason?.runs?.[0]?.text?.includes("This helps protect our community"),
action: () => subreasonProtectCommunity.inc(),
},
{
// @ts-ignore: Property 'playabilityStatus' does not exist on type 'object'
check: (yt: object) => yt.playabilityStatus?.errorScreen?.playerErrorMessageRenderer?.subreason?.runs?.[0]?.text?.includes("Video unavailable"),
action: () => videoUnavailable.inc(),
},
{
// @ts-ignore: Property 'playabilityStatus' does not exist on type 'object'
check: (yt: object) => yt.playabilityStatus?.reason?.includes("This video is restricted"),
action: () => videoRestricted.inc(),
},
];
player.post("/player", async (c) => {
const jsonReq = await c.req.json();
const innertubeClient = await c.get("innertubeClient") as Innertube;
@ -14,9 +38,13 @@ player.post("/player", async (c) => {
Record<string, unknown>
>;
if (jsonReq.videoId) {
return c.json(
await youtubePlayerParsing(innertubeClient, jsonReq.videoId, konfigStore)
);
const yt = await youtubePlayerParsing(innertubeClient, jsonReq.videoId, konfigStore)
errors.forEach((error) => {
if (error.check(yt)) {
error.action()
}
})
return c.json(yt);
}
});