mirror of
https://github.com/ourongxing/newsnow.git
synced 2025-01-19 03:09:14 +08:00
86 lines
2.3 KiB
TypeScript
86 lines
2.3 KiB
TypeScript
import process from "node:process"
|
|
import type { NewsItem } from "@shared/types"
|
|
import type { Database } from "db0"
|
|
import type { CacheInfo, CacheRow } from "../types"
|
|
|
|
export class Cache {
|
|
private db
|
|
constructor(db: Database) {
|
|
this.db = db
|
|
}
|
|
|
|
async init() {
|
|
await this.db.prepare(`
|
|
CREATE TABLE IF NOT EXISTS cache (
|
|
id TEXT PRIMARY KEY,
|
|
updated INTEGER,
|
|
data TEXT
|
|
);
|
|
`).run()
|
|
logger.success(`init cache table`)
|
|
}
|
|
|
|
async set(key: string, value: NewsItem[]) {
|
|
const now = Date.now()
|
|
await this.db.prepare(
|
|
`INSERT OR REPLACE INTO cache (id, data, updated) VALUES (?, ?, ?)`,
|
|
).run(key, JSON.stringify(value), now)
|
|
logger.success(`set ${key} cache`)
|
|
}
|
|
|
|
async get(key: string): Promise<CacheInfo | undefined > {
|
|
const row = (await this.db.prepare(`SELECT id, data, updated FROM cache WHERE id = ?`).get(key)) as CacheRow | undefined
|
|
if (row) {
|
|
logger.success(`get ${key} cache`)
|
|
return {
|
|
id: row.id,
|
|
updated: row.updated,
|
|
items: JSON.parse(row.data),
|
|
}
|
|
}
|
|
}
|
|
|
|
async getEntire(keys: string[]): Promise<CacheInfo[]> {
|
|
const keysStr = keys.map(k => `id = '${k}'`).join(" or ")
|
|
const res = await this.db.prepare(`SELECT id, data, updated FROM cache WHERE ${keysStr}`).all() as any
|
|
const rows = (res.results ?? res) as CacheRow[]
|
|
|
|
/**
|
|
* https://developers.cloudflare.com/d1/build-with-d1/d1-client-api/#return-object
|
|
* cloudflare d1 .all() will return
|
|
* {
|
|
* success: boolean
|
|
* meta:
|
|
* results:
|
|
* }
|
|
*/
|
|
if (rows?.length) {
|
|
logger.success(`get entire (...) cache`)
|
|
return rows.map(row => ({
|
|
id: row.id,
|
|
updated: row.updated,
|
|
items: JSON.parse(row.data) as NewsItem[],
|
|
}))
|
|
} else {
|
|
return []
|
|
}
|
|
}
|
|
|
|
async delete(key: string) {
|
|
return await this.db.prepare(`DELETE FROM cache WHERE id = ?`).run(key)
|
|
}
|
|
}
|
|
|
|
export async function getCacheTable() {
|
|
try {
|
|
// 如果没有数据库,这里不会报错,只会在第一次访问的时候报错
|
|
const db = useDatabase()
|
|
if (process.env.ENABLE_CACHE === "false") return
|
|
const cacheTable = new Cache(db)
|
|
if (process.env.INIT_TABLE !== "false") await cacheTable.init()
|
|
return cacheTable
|
|
} catch {
|
|
// logger.error("failed to init database ", e)
|
|
}
|
|
}
|