From cc6611df6087fdd0836982d15e7c224d7062f359 Mon Sep 17 00:00:00 2001 From: Xeu Date: Fri, 12 Jul 2024 00:16:03 +0800 Subject: [PATCH 1/4] fix: broken seo workflow (#215) just because wrong puppeteer version --- .github/workflows/deploy.yaml | 2 +- .github/workflows/seo.yaml | 2 +- bun.lockb | Bin 483796 -> 484204 bytes package.json | 2 +- scripts/render.ts | 2 +- 5 files changed, 4 insertions(+), 4 deletions(-) diff --git a/.github/workflows/deploy.yaml b/.github/workflows/deploy.yaml index 288c7619..d5a5987e 100644 --- a/.github/workflows/deploy.yaml +++ b/.github/workflows/deploy.yaml @@ -50,5 +50,5 @@ jobs: RSS_DESCRIPTION: ${{ vars.RSS_DESCRIPTION }} run: | cd Rin/ - bun install + bun install --frozen-lockfile bun scripts/migrator.ts diff --git a/.github/workflows/seo.yaml b/.github/workflows/seo.yaml index 7c8d2bf4..bbc847b1 100644 --- a/.github/workflows/seo.yaml +++ b/.github/workflows/seo.yaml @@ -45,5 +45,5 @@ jobs: S3_FORCE_PATH_STYLE: ${{ vars.S3_FORCE_PATH_STYLE }} run: | cd Rin/ - bun i + bun install --frozen-lockfile bun scripts/render.ts diff --git a/bun.lockb b/bun.lockb index 82dbbf8a5a6064cd5b952bc3849731fd7168bb82..dd88d71f9fb6678dd69eae3a4e6c58bbd383306c 100755 GIT binary patch delta 51684 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 YWwu&*xtVhgGb>lhSI3ZMv~kXV0Q?i%C;$Ke delta 52040 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 Rcks5)mKrUoc$96-e*q4nwY~rV diff --git a/package.json b/package.json index 1366b09b..c084a5dc 100644 --- a/package.json +++ b/package.json @@ -25,7 +25,7 @@ "autoprefixer": "^10.4.19", "i18next-parser": "^9.0.0", "postcss": "^8.4.38", - "puppeteer": "^22.10.0", + "puppeteer": "^22.13.0", "tailwindcss": "^3.4.3", "turbo": "^1.13.3", "vitest": "1.3.0" diff --git a/scripts/render.ts b/scripts/render.ts index 9fc84484..6244ce38 100644 --- a/scripts/render.ts +++ b/scripts/render.ts @@ -82,7 +82,7 @@ async function fetchPage(url: string) { const anchors = Array.from(document.querySelectorAll('a')); return anchors.map(anchor => anchor.href); }); - for (const link of links.filter(link => (link.startsWith(baseUrl) || (containsKey !== '' && link.includes(containsKey))))) { + for (const link of links.filter(link => (link.startsWith(baseUrl) || (containsKey != '' && link.includes(containsKey))))) { const linkWithoutHash = link.split('#')[0]; if (fetchedLinks.has(linkWithoutHash)) { continue; From ed174b77764975884a68a66150efe9e3aa510aa3 Mon Sep 17 00:00:00 2001 From: hzy <1448424184@qq.com> Date: Fri, 12 Jul 2024 12:33:03 +0800 Subject: [PATCH 2/4] formula overflow --- client/src/index.css | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/client/src/index.css b/client/src/index.css index 7081662f..91c5a973 100644 --- a/client/src/index.css +++ b/client/src/index.css @@ -91,6 +91,12 @@ ul { .octicon { @apply mr-1 w-3; } + +.katex { + overflow-x: auto; + overflow-y: hidden; +} + /* animation */ @keyframes anvil { @@ -127,6 +133,8 @@ ul { .table { @apply w-full border-collapse my-4; + overflow-x: auto; + display: block !important; } .table th, @@ -197,4 +205,4 @@ a.toc-link { .is-active-link::before { @apply bg-theme; -} \ No newline at end of file +} From aacb021493b9f0ae1f8430a37ca78535ee0fe6cc Mon Sep 17 00:00:00 2001 From: Xeu Date: Fri, 12 Jul 2024 22:07:45 +0800 Subject: [PATCH 3/4] fix: new database migration strategy (#219) * formula overflow * fix: new database migration strategy * ci: add more error message * fix: change typ to 'local' | 'remote' --------- Co-authored-by: hzy <1448424184@qq.com> --- package.json | 2 +- scripts/dev-migrator.ts | 29 +++++++++++++++++++--- scripts/fix-top-field.ts | 53 ++++++++++++++++++++++++++++++++++++++++ scripts/migrator.ts | 34 +++++++++++++++++++++++--- server/sql/0002.sql | 28 ++++----------------- server/src/db/schema.ts | 5 ++++ 6 files changed, 119 insertions(+), 32 deletions(-) create mode 100644 scripts/fix-top-field.ts diff --git a/package.json b/package.json index c084a5dc..72b1c9c3 100644 --- a/package.json +++ b/package.json @@ -11,7 +11,7 @@ "dev:client": "bun --filter './client' dev", "dev:server": "bun wrangler dev --port 11498", "check": "turbo check", - "cf-deploy": "bun server/migrator.ts", + "cf-deploy": "bun scripts/migrator.ts", "b": "turbo build", "t": "turbo t", "g": "turbo run g", diff --git a/scripts/dev-migrator.ts b/scripts/dev-migrator.ts index 129cd341..aadaa563 100644 --- a/scripts/dev-migrator.ts +++ b/scripts/dev-migrator.ts @@ -1,26 +1,35 @@ import * as fs from 'fs'; import * as path from 'path'; -import {execSync} from 'child_process'; +import { execSync } from 'child_process'; +import { fixTopField, getMigrationVersion, isInfoExist, updateMigrationVersion } from './fix-top-field'; const DB_NAME = "rin"; const SQL_DIR = path.join(__dirname, '..', 'server', 'sql'); // Change to the server/sql directory process.chdir(SQL_DIR); - +const typ = 'local'; +const migrationVersion = await getMigrationVersion(typ, DB_NAME); +const isInfoExistResult = await isInfoExist(typ, DB_NAME); // List all SQL files and sort them const sqlFiles = fs - .readdirSync(SQL_DIR, {withFileTypes: true}) + .readdirSync(SQL_DIR, { withFileTypes: true }) .filter(dirent => dirent.isFile() && dirent.name.endsWith('.sql')) .map(dirent => dirent.name) + .filter(file => { + const version = parseInt(file.split('-')[0]); + return version > migrationVersion; + }) .sort(); +console.log("migration_version:", migrationVersion, "Migration SQL List: ", sqlFiles) + // For each file in the sorted list for (const file of sqlFiles) { const filePath = path.join(SQL_DIR, file); // Run the migration try { - execSync(`bunx wrangler d1 execute ${DB_NAME} --local --file "${filePath}"`, {stdio: 'inherit'}); + execSync(`bunx wrangler d1 execute ${DB_NAME} --local --file "${filePath}"`, { stdio: 'inherit' }); console.log(`Executed ${file}`); } catch (error) { console.error(`Failed to execute ${file}: ${error}`); @@ -28,5 +37,17 @@ for (const file of sqlFiles) { } } +if (sqlFiles.length === 0) { + console.log("No migration needed.") +} else { + const lastVersion = parseInt(sqlFiles[sqlFiles.length - 1].split('-')[0]); + if (lastVersion > migrationVersion) { + // Update the migration version + await updateMigrationVersion(typ, DB_NAME, lastVersion); + } +} + +await fixTopField(typ, DB_NAME, isInfoExistResult); + // Back to the root directory (optional, as the script ends) process.chdir(__dirname); \ No newline at end of file diff --git a/scripts/fix-top-field.ts b/scripts/fix-top-field.ts new file mode 100644 index 00000000..a4162327 --- /dev/null +++ b/scripts/fix-top-field.ts @@ -0,0 +1,53 @@ +import { $ } from "bun" + +export async function fixTopField(typ: 'local' | 'remote', db: string, isInfoExistResult: boolean) { + if (!isInfoExistResult) { + console.log("Legacy database, check top field") + const result = await $`bunx wrangler d1 execute ${db} --${typ} --json --command "SELECT name FROM pragma_table_info('feeds') WHERE name='top'"`.quiet().json() + if (result[0].results.length === 0) { + console.log("Adding top field to feeds table") + await $`bunx wrangler d1 execute ${db} --${typ} --json --command "ALTER TABLE feeds ADD COLUMN top INTEGER DEFAULT 0"`.quiet() + } else { + console.log("Top field already exists in feeds table") + } + } else { + console.log("New database, skip top field check") + } +} + +export async function isInfoExist(typ: 'local' | 'remote', db: string) { + const result = await $`bunx wrangler d1 execute ${db} --${typ} --json --command "SELECT name FROM sqlite_master WHERE type='table' AND name='info'"`.quiet().json() + if (result[0].results.length === 0) { + console.log("info table not exists") + return false + } else { + console.log("info table already exists") + return true + } +} + +export async function getMigrationVersion(typ: 'local' | 'remote', db: string) { + const isInfoExistResult = await isInfoExist(typ, db) + if (!isInfoExistResult) { + console.log("Legacy database, migration_version not exists") + return -1 + } + const result = await $`bunx wrangler d1 execute ${db} --${typ} --json --command "SELECT value FROM info WHERE key='migration_version'"`.quiet().json() + if (result[0].results.length === 0) { + console.log("migration_version not exists") + return -1 + } else { + console.log("migration_version:", result[0].results[0].value) + return parseInt(result[0].results[0].value) + } +} + +export async function updateMigrationVersion(typ: 'local' | 'remote', db: string, version: number) { + const exists = await isInfoExist(typ, db) + if (!exists) { + console.log("info table not exists, skip update migration_version") + throw new Error("info table not exists") + } + await $`bunx wrangler d1 execute ${db} --${typ} --json --command "UPDATE info SET value='${version}' WHERE key='migration_version'"`.quiet() + console.log("Updated migration_version to", version) +} \ No newline at end of file diff --git a/scripts/migrator.ts b/scripts/migrator.ts index 7d928702..153d0c48 100644 --- a/scripts/migrator.ts +++ b/scripts/migrator.ts @@ -1,6 +1,7 @@ import { $ } from "bun" import { readdir } from "node:fs/promises" import stripIndent from 'strip-indent' +import { fixTopField, getMigrationVersion, isInfoExist, updateMigrationVersion } from "./fix-top-field" function env(name: string, defaultValue?: string, required = false) { const env = process.env @@ -12,7 +13,7 @@ function env(name: string, defaultValue?: string, required = false) { } // must be defined -const renv = (name: string, defaultValue?: string) => env(name, defaultValue, true) +const renv = (name: string, defaultValue?: string) => env(name, defaultValue, true)! const DB_NAME = renv("DB_NAME", 'rin') const WORKER_NAME = renv("WORKER_NAME", 'rin-server') @@ -102,21 +103,46 @@ if (existing) { } console.log(`----------------------------`) - console.log(`Migrating D1 "${DB_NAME}"`) +const typ = 'remote'; +const migrationVersion = await getMigrationVersion(typ, DB_NAME); +const isInfoExistResult = await isInfoExist(typ, DB_NAME); + try { const files = await readdir("./server/sql", { recursive: false }) - for (const file of files) { + const sqlFiles = files + .filter(name => name.endsWith('.sql')) + .filter(name => { + const version = parseInt(name.split('-')[0]); + return version > migrationVersion; + }) + .sort(); + console.log("migration_version:", migrationVersion, "Migration SQL List: ", sqlFiles) + for (const file of sqlFiles) { await $`bunx wrangler d1 execute ${DB_NAME} --remote --file ./server/sql/${file} -y` console.log(`Migrated ${file}`) } + if (sqlFiles.length === 0) { + console.log("No migration needed.") + } else { + const lastVersion = parseInt(sqlFiles[sqlFiles.length - 1].split('-')[0]); + if (lastVersion > migrationVersion) { + // Update the migration version + await updateMigrationVersion(typ, DB_NAME, lastVersion); + } + } } catch (e: any) { - console.error(e.stderr.toString()) + console.error(e.stdio?.toString()) + console.error(e.stdout?.toString()) + console.error(e.stderr?.toString()) process.exit(1) } console.log(`Migrated D1 "${DB_NAME}"`) console.log(`----------------------------`) +console.log(`Patch D1`) +await fixTopField(typ, DB_NAME, isInfoExistResult); +console.log(`----------------------------`) console.log(`Put secrets`) async function putSecret(name: string, value?: string) { diff --git a/server/sql/0002.sql b/server/sql/0002.sql index ddcc88da..c132b6bb 100644 --- a/server/sql/0002.sql +++ b/server/sql/0002.sql @@ -1,26 +1,8 @@ -PRAGMA foreign_keys=off; ---> statement-breakpoint -CREATE TABLE IF NOT EXISTS `feeds_new` ( - `id` integer PRIMARY KEY NOT NULL, - `alias` text, - `title` text, - `content` text NOT NULL, - `summary` text DEFAULT '' NOT NULL, - `listed` integer DEFAULT 1 NOT NULL, - `draft` integer DEFAULT 1 NOT NULL, - `uid` integer NOT NULL, - `top` integer DEFAULT 0 NOT NULL, - `created_at` integer DEFAULT (unixepoch()) NOT NULL, - `updated_at` integer DEFAULT (unixepoch()) NOT NULL, - FOREIGN KEY (`uid`) REFERENCES `users`(`id`) ON UPDATE no action ON DELETE no action +CREATE TABLE IF NOT EXISTS `info` ( + `key` text NOT NULL, + `value` text NOT NULL ); --> statement-breakpoint -INSERT INTO `feeds_new` (`id`, `alias`, `title`, `content`, `summary`, `listed`, `draft`, `uid`, `created_at`, `updated_at`) -SELECT `id`, `alias`, `title`, `content`, `summary`, `listed`, `draft`, `uid`, `created_at`, `updated_at` -FROM `feeds`; ---> statement-breakpoint -DROP TABLE `feeds`; ---> statement-breakpoint -ALTER TABLE `feeds_new` RENAME TO `feeds`; +CREATE UNIQUE INDEX `info_key_unique` ON `info` (`key`); --> statement-breakpoint -PRAGMA foreign_keys=on; \ No newline at end of file +INSERT INTO `info` (`key`, `value`) VALUES ('migration_version', '2'); \ No newline at end of file diff --git a/server/src/db/schema.ts b/server/src/db/schema.ts index 24fe34d8..76a7f768 100644 --- a/server/src/db/schema.ts +++ b/server/src/db/schema.ts @@ -25,6 +25,11 @@ export const visits = sqliteTable("visits", { createdAt: created_at, }); +export const info = sqliteTable("info", { + key: text("key").notNull().unique(), + value: text("value").notNull(), +}); + export const friends = sqliteTable("friends", { id: integer("id").primaryKey(), name: text("name").notNull(), From f6151e897b7dfb64f9155fb4b2792a831ff45c8a Mon Sep 17 00:00:00 2001 From: PBK Bin Date: Fri, 26 Jul 2024 16:39:57 +0800 Subject: [PATCH 4/4] fix: processing S3_ACCESS_HOST configuration without protocol header obtained in rss acquisition #228 (#229) --- server/src/services/rss.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/server/src/services/rss.ts b/server/src/services/rss.ts index 935bea55..be5745a9 100644 --- a/server/src/services/rss.ts +++ b/server/src/services/rss.ts @@ -23,7 +23,8 @@ export function RSSService() { const folder = env.S3_CACHE_FOLDER || 'cache/'; return new Elysia({ aot: false }) .get('/sub/:name', async ({ set, params: { name } }) => { - if (!accessHost) { + const host = `${(accessHost.startsWith("http://") || accessHost.startsWith("https://") ? '' :'https://')}${accessHost}`; + if (!host) { set.status = 500; return 'S3_ACCESS_HOST is not defined' } @@ -33,7 +34,7 @@ export function RSSService() { if (['rss.xml', 'atom.xml', 'rss.json'].includes(name)) { const key = path.join(folder, name); try { - const url = `${accessHost}/${key}`; + const url = `${host}/${key}`; console.log(`Fetching ${url}`); const response = await fetch(new Request(url)) const contentType = name === 'rss.xml' ? 'application/rss+xml; charset=UTF-8' : name === 'atom.xml' ? 'application/atom+xml; charset=UTF-8' : 'application/feed+json; charset=UTF-8';