feat: videoInfo API in backend

This commit is contained in:
alikia2x (寒寒) 2025-04-07 21:29:53 +08:00
parent b4205049cb
commit 71ed0bd66b
Signed by: alikia2x
GPG Key ID: 56209E0CCD8420C6
10 changed files with 100 additions and 36 deletions

View File

@ -20,7 +20,7 @@ export const dbCredMiddleware = createMiddleware(async (c, next) => {
c.set("dbCred", connection);
await next();
connection.release();
})
});
declare module "hono" {
interface ContextVariableMap {

View File

@ -4,11 +4,12 @@
"@rabbit-company/argon2id": "jsr:@rabbit-company/argon2id@^2.1.0",
"hono": "jsr:@hono/hono@^4.7.5",
"zod": "npm:zod",
"yup": "npm:yup"
"yup": "npm:yup",
"@crawler/net/videoInfo": "../crawler/net/getVideoInfo.ts"
},
"tasks": {
"dev": "deno serve --env-file=.env --allow-env --allow-net --watch main.ts",
"start": "deno serve --env-file=.env --allow-env --allow-net --host 127.0.0.1 main.ts"
"dev": "deno serve --env-file=.env --allow-env --allow-net --allow-read --allow-write --allow-run --watch main.ts",
"start": "deno serve --env-file=.env --allow-env --allow-net --allow-read --allow-write --allow-run --host 127.0.0.1 main.ts"
},
"compilerOptions": {
"jsx": "precompile",

View File

@ -3,16 +3,19 @@ import { dbCredMiddleware, dbMiddleware } from "./database.ts";
import { rootHandler } from "./root.ts";
import { getSnapshotsHanlder } from "./snapshots.ts";
import { registerHandler } from "./register.ts";
import { videoInfoHandler } from "./videoInfo.ts";
export const app = new Hono();
app.use('/video/*', dbMiddleware);
app.use('/user', dbCredMiddleware);
app.use("/video/*", dbMiddleware);
app.use("/user", dbCredMiddleware);
app.get("/", ...rootHandler);
app.get('/video/:id/snapshots', ...getSnapshotsHanlder);
app.post('/user', ...registerHandler);
app.get("/video/:id/snapshots", ...getSnapshotsHanlder);
app.post("/user", ...registerHandler);
app.get("/video/:id/info", ...videoInfoHandler);
const fetch = app.fetch;
@ -20,4 +23,4 @@ export default {
fetch,
} satisfies Deno.ServeDefaultExport;
export const VERSION = "0.3.0";
export const VERSION = "0.4.0";

View File

@ -19,7 +19,7 @@ export const userExists = async (username: string, client: Client) => {
`;
const result = await client.queryObject(query, [username]);
return result.rows.length > 0;
}
};
export const registerHandler = createHandlers(async (c: ContextType) => {
const client = c.get("dbCred");

View File

@ -9,23 +9,21 @@ export const rootHandler = createHandlers((c) => {
singer = getSingerForBirthday();
for (const s of singer) {
delete s.birthday;
s.message = `${s.name}生日快乐~`
s.message = `${s.name}生日快乐~`;
}
}
else if (shouldShowSpecialSinger) {
} else if (shouldShowSpecialSinger) {
singer = pickSpecialSinger();
}
else {
} else {
singer = pickSinger();
}
return c.json({
"project": {
"name": "中V档案馆",
"motto": "一起唱吧,心中的歌!"
"motto": "一起唱吧,心中的歌!",
},
"status": 200,
"version": VERSION,
"time": Date.now(),
"singer": singer
})
})
"singer": singer,
});
});

View File

@ -12,7 +12,7 @@ const SnapshotQueryParamsSchema = object({
reverse: boolean().optional(),
});
const idSchema = mixed().test(
export const idSchema = mixed().test(
"is-valid-id",
'id must be a string starting with "av" followed by digits, or "BV" followed by 10 alphanumeric characters, or a positive integer',
async (value) => {
@ -46,8 +46,7 @@ export const getSnapshotsHanlder = createHandlers(async (c: ContextType) => {
let videoId: string | number = idParam as string;
if (videoId.startsWith("av")) {
videoId = parseInt(videoId.slice(2));
}
else if (await number().isValid(videoId)) {
} else if (await number().isValid(videoId)) {
videoId = parseInt(videoId);
}
const queryParams = await SnapshotQueryParamsSchema.validate(c.req.query());

View File

@ -1,4 +1,4 @@
import { createFactory } from 'hono/factory'
import { createFactory } from "hono/factory";
const factory = createFactory();

View File

@ -0,0 +1,39 @@
import type { Context } from "hono";
import { createHandlers } from "./utils.ts";
import type { BlankEnv, BlankInput } from "hono/types";
import { number, ValidationError } from "yup";
import { getVideoInfo, getVideoInfoByBV } from "@crawler/net/videoInfo";
import { idSchema } from "./snapshots.ts";
import type { VideoInfoData } from "../crawler/net/bilibili.d.ts";
type ContextType = Context<BlankEnv, "/video/:id/info", BlankInput>;
export const videoInfoHandler = createHandlers(async (c: ContextType) => {
try {
const id = await idSchema.validate(c.req.param("id"));
let videoId: string | number = id as string;
if (videoId.startsWith("av")) {
videoId = parseInt(videoId.slice(2));
} else if (await number().isValid(videoId)) {
videoId = parseInt(videoId);
}
let result: VideoInfoData | number;
if (typeof videoId === "number") {
result = await getVideoInfo(videoId, "getVideoInfo");
} else {
result = await getVideoInfoByBV(videoId, "getVideoInfo");
}
if (typeof result === "number") {
return c.json({ message: "Error fetching video info", code: result }, 500);
}
return c.json(result);
} catch (e) {
if (e instanceof ValidationError) {
return c.json({ message: "Invalid query parameters", errors: e.errors }, 400);
} else {
return c.json({ message: "Unhandled error", error: e }, 500);
}
}
});

View File

@ -25,3 +25,27 @@ export async function getVideoInfo(aid: number, task: string): Promise<VideoInfo
}
return data.data;
}
/*
* Fetch video metadata from bilibili API by BVID
* @param {string} bvid - The video's BVID
* @param {string} task - The task name used in scheduler. It can be one of the following:
* - snapshotVideo
* - getVideoInfo
* - snapshotMilestoneVideo
* @returns {Promise<VideoInfoData | number>} VideoInfoData or the error code returned by bilibili API
* @throws {NetSchedulerError} - The error will be thrown in following cases:
* - No proxy is available currently: with error code `NO_PROXY_AVAILABLE`
* - The native `fetch` function threw an error: with error code `FETCH_ERROR`
* - The alicloud-fc threw an error: with error code `ALICLOUD_FC_ERROR`
*/
export async function getVideoInfoByBV(bvid: string, task: string): Promise<VideoInfoData | number> {
const url = `https://api.bilibili.com/x/web-interface/view?bvid=${bvid}`;
const data = await netScheduler.request<VideoInfoResponse>(url, task);
const errMessage = `Error fetching metadata for ${bvid}:`;
if (data.code !== 0) {
logger.error(errMessage + data.code + "-" + data.message, "net", "fn:getVideoInfoByBV");
return data.code;
}
return data.data;
}