Merge remote-tracking branch 'gitea/develop' into feat/page-templates

# Conflicts:
#	apps/server/src/integrations/throttle/throttle.module.ts
#	apps/server/src/integrations/throttle/throttler-names.ts
This commit is contained in:
claude_code
2026-06-20 20:18:42 +03:00
130 changed files with 9951 additions and 3095 deletions

View File

@@ -0,0 +1,85 @@
import { type Kysely, sql } from 'kysely';
export async function up(db: Kysely<any>): Promise<void> {
// Reusable, workspace-scoped agent roles (admin-owned). A role REPLACES the
// persona layer of the system prompt (instructions) and may optionally
// override the chat model. The non-removable SAFETY_FRAMEWORK is always still
// appended downstream — a role only shapes the persona, never the safety rules.
await db.schema
.createTable('ai_agent_roles')
.ifNotExists()
.addColumn('id', 'uuid', (col) =>
col.primaryKey().defaultTo(sql`gen_uuid_v7()`),
)
.addColumn('workspace_id', 'uuid', (col) =>
col.references('workspaces.id').onDelete('cascade').notNull(),
)
// Who created the role (audit). The role is shared and outlives its author,
// so SET NULL on user deletion (unlike ai_chats.creator_id which is NOT NULL).
.addColumn('creator_id', 'uuid', (col) =>
col.references('users.id').onDelete('set null'),
)
// Display name, e.g. 'Proofreader'.
.addColumn('name', 'varchar', (col) => col.notNull())
// Optional presentation emoji for the role badge.
.addColumn('emoji', 'varchar', (col) => col)
// Optional short description shown in the management UI.
.addColumn('description', 'text', (col) => col)
// The persona fragment injected into the system prompt (replaces the admin
// persona / DEFAULT_PROMPT). Required.
.addColumn('instructions', 'text', (col) => col.notNull())
// Optional model override: { chatModel } or { driver, chatModel }. NULL =>
// use the workspace default model. Driver creds come from the matching
// provider in ai_provider_credentials (no per-role creds).
.addColumn('model_config', 'jsonb', (col) => col)
.addColumn('enabled', 'boolean', (col) => col.notNull().defaultTo(true))
.addColumn('created_at', 'timestamptz', (col) =>
col.notNull().defaultTo(sql`now()`),
)
.addColumn('updated_at', 'timestamptz', (col) =>
col.notNull().defaultTo(sql`now()`),
)
// Soft delete (consistent with ai_chats): the role disappears from the
// picker but lookups can still resolve it for already-bound chats.
.addColumn('deleted_at', 'timestamptz', (col) => col)
.execute();
// Scoped lookups (listByWorkspace) hit workspace_id first.
await db.schema
.createIndex('idx_ai_agent_roles_workspace_id')
.ifNotExists()
.on('ai_agent_roles')
.column('workspace_id')
.execute();
// A role name is unique per workspace. Partial (WHERE deleted_at IS NULL) so a
// soft-deleted role does not block re-creating a role with the same name.
await db.schema
.createIndex('ai_agent_roles_workspace_id_name_unique')
.ifNotExists()
.on('ai_agent_roles')
.columns(['workspace_id', 'name'])
.unique()
.where(sql.ref('deleted_at'), 'is', null)
.execute();
// Bind a chat to a role. ON DELETE SET NULL: a hard-deleted role degrades the
// chat to the universal assistant instead of breaking it. The role is read
// from this column on every turn — the client only sends roleId on chat
// creation (first message).
await db.schema
.alterTable('ai_chats')
.addColumn('role_id', 'uuid', (col) =>
col.references('ai_agent_roles.id').onDelete('set null'),
)
.execute();
}
export async function down(db: Kysely<any>): Promise<void> {
await db.schema.alterTable('ai_chats').dropColumn('role_id').execute();
await db.schema
.dropIndex('ai_agent_roles_workspace_id_name_unique')
.ifExists()
.execute();
await db.schema.dropTable('ai_agent_roles').execute();
}