newsnow/server/api/s/index.ts

91 lines
2.9 KiB
TypeScript
Raw Normal View History

2024-10-08 20:49:19 +08:00
import type { SourceID, SourceResponse } from "@shared/types"
2024-10-25 03:06:03 +08:00
import { getters } from "#/getters"
import { getCacheTable } from "#/database/cache"
import type { CacheInfo } from "#/types"
2024-09-29 20:57:24 +08:00
2024-10-05 17:20:49 +08:00
export default defineEventHandler(async (event): Promise<SourceResponse> => {
2024-10-03 17:24:29 +08:00
try {
const query = getQuery(event)
const latest = query.latest !== undefined && query.latest !== "false"
2024-11-03 00:59:44 +08:00
let id = query.id as SourceID
2024-10-25 03:06:03 +08:00
const isValid = (id: SourceID) => !id || !sources[id] || !getters[id]
2024-10-08 20:49:19 +08:00
if (isValid(id)) {
2024-10-21 17:00:51 +08:00
const redirectID = sources?.[id]?.redirect
2024-10-08 20:49:19 +08:00
if (redirectID) id = redirectID
if (isValid(id)) throw new Error("Invalid source id")
}
2024-10-03 13:16:14 +08:00
const cacheTable = await getCacheTable()
// Date.now() in Cloudflare Worker will not update throughout the entire runtime.
2024-10-05 17:20:49 +08:00
const now = Date.now()
let cache: CacheInfo | undefined
2024-10-23 03:11:20 +08:00
if (cacheTable) {
cache = await cacheTable.get(id)
2024-10-03 17:24:29 +08:00
if (cache) {
// if (cache) {
2024-10-08 22:41:38 +08:00
// interval 刷新间隔,对于缓存失效也要执行的。本质上表示本来内容更新就很慢,这个间隔内可能内容压根不会更新。
2024-10-14 21:50:27 +08:00
// 默认 10 分钟,是低于 TTL 的,但部分 Source 的更新间隔会超过 TTL甚至有的一天更新一次。
if (now - cache.updated < sources[id].interval) {
2024-10-08 22:41:38 +08:00
return {
status: "success",
id,
2024-10-15 12:05:03 +08:00
updatedTime: now,
items: cache.items,
2024-10-08 22:41:38 +08:00
}
}
2024-10-14 00:02:58 +08:00
2024-10-08 22:41:38 +08:00
// 而 TTL 缓存失效时间,在时间范围内,就算内容更新了也要用这个缓存。
// 复用缓存是不会更新时间的。
2024-10-14 21:50:27 +08:00
if (now - cache.updated < TTL) {
// 有 latest
// 没有 latest但服务器禁止登录
// 没有 latest
// 有 latest服务器可以登录但没有登录
if (!latest || (!event.context.disabledLogin && !event.context.user)) {
2024-10-19 15:28:02 +08:00
return {
status: "cache",
id,
2024-10-19 15:28:02 +08:00
updatedTime: cache.updated,
items: cache.items,
2024-10-14 21:50:27 +08:00
}
2024-10-03 17:24:29 +08:00
}
}
}
}
try {
const newData = (await getters[id]()).slice(0, 30)
if (cacheTable && newData.length) {
if (event.context.waitUntil) event.context.waitUntil(cacheTable.set(id, newData))
else await cacheTable.set(id, newData)
}
logger.success(`fetch ${id} latest`)
return {
status: "success",
id,
updatedTime: now,
items: newData,
}
} catch (e) {
if (cache!) {
return {
status: "cache",
id,
updatedTime: cache.updated,
items: cache.items,
}
} else {
throw e
}
2024-10-03 17:24:29 +08:00
}
} catch (e: any) {
2024-10-05 23:55:30 +08:00
logger.error(e)
2024-10-15 12:05:03 +08:00
throw createError({
statusCode: 500,
message: e instanceof Error ? e.message : "Internal Server Error",
})
2024-10-03 13:16:14 +08:00
}
2024-09-29 20:57:24 +08:00
})