- 全表新增 deleted_at 列,统一软删除替代硬删除+archived_at - models.model_id 重命名为 external_id,消除语义混淆 - conversations.model_id 改为可空(模型为建议而非绑定) - messages 新增 updated_at,移除 CASCADE 改为 DAO 层级联 - 移除 DB 层 UNIQUE 约束,改为应用层检查(配合软删除) - 新增 helpers.ts(baseColumns + 构造层防御)、ESLint 规则、契约测试 - 迁移 0004 补全 CHECK 约束(providers.type/materials.status/messages.role) - DAO 层全面重写:级联软删除、应用层唯一、provider 删除保护 - 路由/前端/测试全量适配 externalId 重命名及类型变更
171 lines
5.0 KiB
JavaScript
171 lines
5.0 KiB
JavaScript
import js from "@eslint/js";
|
||
import importPlugin from "eslint-plugin-import";
|
||
import perfectionist from "eslint-plugin-perfectionist";
|
||
import eslintPluginPrettierRecommended from "eslint-plugin-prettier/recommended";
|
||
import reactHooks from "eslint-plugin-react-hooks";
|
||
import reactRefresh from "eslint-plugin-react-refresh";
|
||
import tseslint from "typescript-eslint";
|
||
|
||
import { enforceCatchType } from "./eslint-rules/enforce-catch-type.js";
|
||
import { noEmptyFunction } from "./eslint-rules/no-empty-function.js";
|
||
|
||
const noDirectConsoleMessage =
|
||
"后端运行时代码禁止直接使用 console.*;请通过注入的 Logger 实例输出日志,配置加载失败前使用 createConsoleFallback()。";
|
||
|
||
const noDirectConsoleFrontendMessage =
|
||
"前端代码禁止直接使用 console.*;请使用 useLogger() hook(组件内)或 createConsoleLogger()(非组件纯函数)。";
|
||
|
||
export default tseslint.config(
|
||
{
|
||
ignores: [
|
||
"node_modules/**",
|
||
"dist/**",
|
||
".build/**",
|
||
"*.bun-build",
|
||
"openspec/**",
|
||
".opencode/**",
|
||
".claude/**",
|
||
".codex/**",
|
||
".agents/**",
|
||
".worktrees/**",
|
||
"bin/**",
|
||
"bun.lock",
|
||
"data/**",
|
||
"eslint-rules/**",
|
||
],
|
||
},
|
||
js.configs.recommended,
|
||
...tseslint.configs.recommendedTypeChecked,
|
||
...tseslint.configs.stylisticTypeChecked,
|
||
importPlugin.flatConfigs.recommended,
|
||
importPlugin.flatConfigs.typescript,
|
||
perfectionist.configs["recommended-natural"],
|
||
{
|
||
languageOptions: {
|
||
parserOptions: {
|
||
projectService: true,
|
||
tsconfigRootDir: import.meta.dirname,
|
||
},
|
||
},
|
||
settings: {
|
||
"import/resolver": { node: true, typescript: true },
|
||
},
|
||
},
|
||
{
|
||
rules: {
|
||
"@typescript-eslint/array-type": ["error", { default: "array-simple" }],
|
||
"@typescript-eslint/consistent-type-assertions": ["error", { assertionStyle: "as" }],
|
||
"@typescript-eslint/consistent-type-imports": ["error", { prefer: "type-imports" }],
|
||
"@typescript-eslint/no-empty-function": "off",
|
||
"@typescript-eslint/no-unused-vars": ["error", { argsIgnorePattern: "^_" }],
|
||
"@typescript-eslint/only-throw-error": "error",
|
||
"@typescript-eslint/prefer-nullish-coalescing": "error",
|
||
"@typescript-eslint/prefer-optional-chain": "error",
|
||
"import/no-unresolved": ["error", { ignore: ["^bun:"] }],
|
||
"no-restricted-syntax": [
|
||
"error",
|
||
{
|
||
message:
|
||
"禁止 throw 字面量。项目约定只允许 throw new Error(...) 或 throw new AppError(msg, statusCode)。Re-throw 已捕获的 Error 实例时使用 throw e。",
|
||
selector: "ThrowStatement > Literal",
|
||
},
|
||
],
|
||
"no-undef": "off",
|
||
},
|
||
},
|
||
{
|
||
files: ["src/**/*.{ts,tsx}"],
|
||
plugins: {
|
||
local: {
|
||
rules: {
|
||
"enforce-catch-type": enforceCatchType,
|
||
"no-empty-function": noEmptyFunction,
|
||
},
|
||
},
|
||
},
|
||
rules: {
|
||
"local/enforce-catch-type": "warn",
|
||
"local/no-empty-function": "error",
|
||
},
|
||
},
|
||
{
|
||
files: ["eslint.config.js"],
|
||
rules: {
|
||
"import/no-named-as-default": "off",
|
||
"import/no-named-as-default-member": "off",
|
||
},
|
||
},
|
||
{
|
||
files: ["src/server/db/**/*.ts"],
|
||
ignores: ["src/server/db/helpers.ts"],
|
||
rules: {
|
||
"no-restricted-imports": [
|
||
"error",
|
||
{
|
||
paths: [
|
||
{
|
||
importNames: ["sqliteTable"],
|
||
message:
|
||
"请从 ./helpers.ts 导入 sqliteTable,并在列定义中展开 baseColumns。参见 src/server/db/helpers.ts。",
|
||
name: "drizzle-orm/sqlite-core",
|
||
},
|
||
],
|
||
},
|
||
],
|
||
},
|
||
},
|
||
{
|
||
files: ["src/server/**/*.ts"],
|
||
ignores: ["src/server/logger.ts"],
|
||
rules: {
|
||
"no-restricted-syntax": [
|
||
"error",
|
||
{
|
||
message: noDirectConsoleMessage,
|
||
selector: "MemberExpression[object.name='console']",
|
||
},
|
||
],
|
||
},
|
||
},
|
||
{
|
||
files: ["src/web/**/*.{ts,tsx}"],
|
||
ignores: ["src/web/**/logger.ts"],
|
||
plugins: {
|
||
"react-hooks": reactHooks,
|
||
"react-refresh": reactRefresh,
|
||
},
|
||
rules: {
|
||
...reactHooks.configs.recommended.rules,
|
||
"no-restricted-imports": [
|
||
"error",
|
||
{
|
||
patterns: [
|
||
{
|
||
group: [
|
||
"../server/*",
|
||
"../server/**",
|
||
"../**/server/*",
|
||
"../**/server/**",
|
||
"../../server/*",
|
||
"../../server/**",
|
||
"src/server/*",
|
||
"src/server/**",
|
||
],
|
||
message: "前端不得导入 src/server 后端运行时实现;请改用 src/shared 类型或 HTTP API。",
|
||
},
|
||
],
|
||
},
|
||
],
|
||
"no-restricted-syntax": [
|
||
"error",
|
||
{
|
||
message: noDirectConsoleFrontendMessage,
|
||
selector: "MemberExpression[object.name='console']",
|
||
},
|
||
],
|
||
"react-refresh/only-export-components": ["warn", { allowConstantExport: true }],
|
||
},
|
||
},
|
||
eslintPluginPrettierRecommended,
|
||
);
|