feat(builder): upload thumbnails via storage plugin (#137)

Signed-off-by: Innei <tukon479@gmail.com>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
Co-authored-by: Copilot <198982749+Copilot@users.noreply.github.com>
This commit is contained in:
Innei
2025-10-31 21:49:31 +08:00
committed by GitHub
parent ba68be802e
commit e27b45ec49
116 changed files with 3541 additions and 1761 deletions

View File

@@ -0,0 +1,9 @@
CREATE TABLE "reactions" (
"id" text PRIMARY KEY NOT NULL,
"tenant_id" text NOT NULL,
"created_at" timestamp DEFAULT now() NOT NULL,
"ref_key" text NOT NULL,
"reaction" text NOT NULL
);
--> statement-breakpoint
ALTER TABLE "reactions" ADD CONSTRAINT "reactions_tenant_id_tenant_id_fk" FOREIGN KEY ("tenant_id") REFERENCES "public"."tenant"("id") ON DELETE cascade ON UPDATE no action;

View File

@@ -0,0 +1,834 @@
{
"id": "73c602e7-efc2-46a3-ab83-cecf071c2035",
"prevId": "40bebd58-4e8e-4961-aca6-df9bf5f091e3",
"version": "7",
"dialect": "postgresql",
"tables": {
"public.auth_account": {
"name": "auth_account",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"account_id": {
"name": "account_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"provider_id": {
"name": "provider_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"user_id": {
"name": "user_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"access_token": {
"name": "access_token",
"type": "text",
"primaryKey": false,
"notNull": false
},
"refresh_token": {
"name": "refresh_token",
"type": "text",
"primaryKey": false,
"notNull": false
},
"id_token": {
"name": "id_token",
"type": "text",
"primaryKey": false,
"notNull": false
},
"access_token_expires_at": {
"name": "access_token_expires_at",
"type": "timestamp",
"primaryKey": false,
"notNull": false
},
"refresh_token_expires_at": {
"name": "refresh_token_expires_at",
"type": "timestamp",
"primaryKey": false,
"notNull": false
},
"scope": {
"name": "scope",
"type": "text",
"primaryKey": false,
"notNull": false
},
"password": {
"name": "password",
"type": "text",
"primaryKey": false,
"notNull": false
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
}
},
"indexes": {},
"foreignKeys": {
"auth_account_user_id_auth_user_id_fk": {
"name": "auth_account_user_id_auth_user_id_fk",
"tableFrom": "auth_account",
"tableTo": "auth_user",
"columnsFrom": ["user_id"],
"columnsTo": ["id"],
"onDelete": "cascade",
"onUpdate": "no action"
}
},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.auth_session": {
"name": "auth_session",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"expires_at": {
"name": "expires_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true
},
"token": {
"name": "token",
"type": "text",
"primaryKey": false,
"notNull": true
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"ip_address": {
"name": "ip_address",
"type": "text",
"primaryKey": false,
"notNull": false
},
"user_agent": {
"name": "user_agent",
"type": "text",
"primaryKey": false,
"notNull": false
},
"tenant_id": {
"name": "tenant_id",
"type": "text",
"primaryKey": false,
"notNull": false
},
"user_id": {
"name": "user_id",
"type": "text",
"primaryKey": false,
"notNull": true
}
},
"indexes": {},
"foreignKeys": {
"auth_session_tenant_id_tenant_id_fk": {
"name": "auth_session_tenant_id_tenant_id_fk",
"tableFrom": "auth_session",
"tableTo": "tenant",
"columnsFrom": ["tenant_id"],
"columnsTo": ["id"],
"onDelete": "set null",
"onUpdate": "no action"
},
"auth_session_user_id_auth_user_id_fk": {
"name": "auth_session_user_id_auth_user_id_fk",
"tableFrom": "auth_session",
"tableTo": "auth_user",
"columnsFrom": ["user_id"],
"columnsTo": ["id"],
"onDelete": "cascade",
"onUpdate": "no action"
}
},
"compositePrimaryKeys": {},
"uniqueConstraints": {
"auth_session_token_unique": {
"name": "auth_session_token_unique",
"nullsNotDistinct": false,
"columns": ["token"]
}
},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.auth_user": {
"name": "auth_user",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"name": {
"name": "name",
"type": "text",
"primaryKey": false,
"notNull": true
},
"email": {
"name": "email",
"type": "text",
"primaryKey": false,
"notNull": true
},
"email_verified": {
"name": "email_verified",
"type": "boolean",
"primaryKey": false,
"notNull": true,
"default": false
},
"image": {
"name": "image",
"type": "text",
"primaryKey": false,
"notNull": false
},
"role": {
"name": "role",
"type": "user_role",
"typeSchema": "public",
"primaryKey": false,
"notNull": true,
"default": "'user'"
},
"tenant_id": {
"name": "tenant_id",
"type": "text",
"primaryKey": false,
"notNull": false
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"two_factor_enabled": {
"name": "two_factor_enabled",
"type": "boolean",
"primaryKey": false,
"notNull": true,
"default": false
},
"username": {
"name": "username",
"type": "text",
"primaryKey": false,
"notNull": false
},
"display_username": {
"name": "display_username",
"type": "text",
"primaryKey": false,
"notNull": false
},
"banned": {
"name": "banned",
"type": "boolean",
"primaryKey": false,
"notNull": true,
"default": false
},
"ban_reason": {
"name": "ban_reason",
"type": "text",
"primaryKey": false,
"notNull": false
},
"ban_expires_at": {
"name": "ban_expires_at",
"type": "timestamp",
"primaryKey": false,
"notNull": false
}
},
"indexes": {},
"foreignKeys": {
"auth_user_tenant_id_tenant_id_fk": {
"name": "auth_user_tenant_id_tenant_id_fk",
"tableFrom": "auth_user",
"tableTo": "tenant",
"columnsFrom": ["tenant_id"],
"columnsTo": ["id"],
"onDelete": "set null",
"onUpdate": "no action"
}
},
"compositePrimaryKeys": {},
"uniqueConstraints": {
"auth_user_email_unique": {
"name": "auth_user_email_unique",
"nullsNotDistinct": false,
"columns": ["email"]
}
},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.photo_asset": {
"name": "photo_asset",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"tenant_id": {
"name": "tenant_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"photo_id": {
"name": "photo_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"storage_key": {
"name": "storage_key",
"type": "text",
"primaryKey": false,
"notNull": true
},
"storage_provider": {
"name": "storage_provider",
"type": "text",
"primaryKey": false,
"notNull": true
},
"size": {
"name": "size",
"type": "bigint",
"primaryKey": false,
"notNull": false
},
"etag": {
"name": "etag",
"type": "text",
"primaryKey": false,
"notNull": false
},
"last_modified": {
"name": "last_modified",
"type": "timestamp",
"primaryKey": false,
"notNull": false
},
"metadata_hash": {
"name": "metadata_hash",
"type": "text",
"primaryKey": false,
"notNull": false
},
"manifest_version": {
"name": "manifest_version",
"type": "text",
"primaryKey": false,
"notNull": true,
"default": "'v7'"
},
"manifest": {
"name": "manifest",
"type": "jsonb",
"primaryKey": false,
"notNull": true
},
"sync_status": {
"name": "sync_status",
"type": "photo_sync_status",
"typeSchema": "public",
"primaryKey": false,
"notNull": true,
"default": "'pending'"
},
"conflict_reason": {
"name": "conflict_reason",
"type": "text",
"primaryKey": false,
"notNull": false
},
"conflict_payload": {
"name": "conflict_payload",
"type": "jsonb",
"primaryKey": false,
"notNull": false,
"default": "'null'::jsonb"
},
"synced_at": {
"name": "synced_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
}
},
"indexes": {},
"foreignKeys": {
"photo_asset_tenant_id_tenant_id_fk": {
"name": "photo_asset_tenant_id_tenant_id_fk",
"tableFrom": "photo_asset",
"tableTo": "tenant",
"columnsFrom": ["tenant_id"],
"columnsTo": ["id"],
"onDelete": "cascade",
"onUpdate": "no action"
}
},
"compositePrimaryKeys": {},
"uniqueConstraints": {
"uq_photo_asset_tenant_storage_key": {
"name": "uq_photo_asset_tenant_storage_key",
"nullsNotDistinct": false,
"columns": ["tenant_id", "storage_key"]
},
"uq_photo_asset_tenant_photo_id": {
"name": "uq_photo_asset_tenant_photo_id",
"nullsNotDistinct": false,
"columns": ["tenant_id", "photo_id"]
}
},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.reactions": {
"name": "reactions",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"tenant_id": {
"name": "tenant_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"ref_key": {
"name": "ref_key",
"type": "text",
"primaryKey": false,
"notNull": true
},
"reaction": {
"name": "reaction",
"type": "text",
"primaryKey": false,
"notNull": true
}
},
"indexes": {},
"foreignKeys": {
"reactions_tenant_id_tenant_id_fk": {
"name": "reactions_tenant_id_tenant_id_fk",
"tableFrom": "reactions",
"tableTo": "tenant",
"columnsFrom": ["tenant_id"],
"columnsTo": ["id"],
"onDelete": "cascade",
"onUpdate": "no action"
}
},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.settings": {
"name": "settings",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"tenant_id": {
"name": "tenant_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"key": {
"name": "key",
"type": "text",
"primaryKey": false,
"notNull": true
},
"value": {
"name": "value",
"type": "text",
"primaryKey": false,
"notNull": true
},
"is_sensitive": {
"name": "is_sensitive",
"type": "boolean",
"primaryKey": false,
"notNull": true,
"default": false
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
}
},
"indexes": {},
"foreignKeys": {
"settings_tenant_id_tenant_id_fk": {
"name": "settings_tenant_id_tenant_id_fk",
"tableFrom": "settings",
"tableTo": "tenant",
"columnsFrom": ["tenant_id"],
"columnsTo": ["id"],
"onDelete": "cascade",
"onUpdate": "no action"
}
},
"compositePrimaryKeys": {},
"uniqueConstraints": {
"uq_settings_tenant_key": {
"name": "uq_settings_tenant_key",
"nullsNotDistinct": false,
"columns": ["tenant_id", "key"]
}
},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.system_setting": {
"name": "system_setting",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"key": {
"name": "key",
"type": "text",
"primaryKey": false,
"notNull": true
},
"value": {
"name": "value",
"type": "jsonb",
"primaryKey": false,
"notNull": false,
"default": "'null'::jsonb"
},
"is_sensitive": {
"name": "is_sensitive",
"type": "boolean",
"primaryKey": false,
"notNull": true,
"default": false
},
"description": {
"name": "description",
"type": "text",
"primaryKey": false,
"notNull": false
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {
"uq_system_setting_key": {
"name": "uq_system_setting_key",
"nullsNotDistinct": false,
"columns": ["key"]
}
},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.tenant_domain": {
"name": "tenant_domain",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"tenant_id": {
"name": "tenant_id",
"type": "text",
"primaryKey": false,
"notNull": true
},
"domain": {
"name": "domain",
"type": "text",
"primaryKey": false,
"notNull": true
},
"is_primary": {
"name": "is_primary",
"type": "boolean",
"primaryKey": false,
"notNull": true,
"default": false
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
}
},
"indexes": {},
"foreignKeys": {
"tenant_domain_tenant_id_tenant_id_fk": {
"name": "tenant_domain_tenant_id_tenant_id_fk",
"tableFrom": "tenant_domain",
"tableTo": "tenant",
"columnsFrom": ["tenant_id"],
"columnsTo": ["id"],
"onDelete": "cascade",
"onUpdate": "no action"
}
},
"compositePrimaryKeys": {},
"uniqueConstraints": {
"uq_tenant_domain_domain": {
"name": "uq_tenant_domain_domain",
"nullsNotDistinct": false,
"columns": ["domain"]
}
},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
},
"public.tenant": {
"name": "tenant",
"schema": "",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true
},
"slug": {
"name": "slug",
"type": "text",
"primaryKey": false,
"notNull": true
},
"name": {
"name": "name",
"type": "text",
"primaryKey": false,
"notNull": true
},
"status": {
"name": "status",
"type": "tenant_status",
"typeSchema": "public",
"primaryKey": false,
"notNull": true,
"default": "'inactive'"
},
"primary_domain": {
"name": "primary_domain",
"type": "text",
"primaryKey": false,
"notNull": false
},
"is_primary": {
"name": "is_primary",
"type": "boolean",
"primaryKey": false,
"notNull": true,
"default": false
},
"created_at": {
"name": "created_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
},
"updated_at": {
"name": "updated_at",
"type": "timestamp",
"primaryKey": false,
"notNull": true,
"default": "now()"
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {
"uq_tenant_slug": {
"name": "uq_tenant_slug",
"nullsNotDistinct": false,
"columns": ["slug"]
}
},
"policies": {},
"checkConstraints": {},
"isRLSEnabled": false
}
},
"enums": {
"public.photo_sync_status": {
"name": "photo_sync_status",
"schema": "public",
"values": ["pending", "synced", "conflict"]
},
"public.tenant_status": {
"name": "tenant_status",
"schema": "public",
"values": ["active", "inactive", "suspended"]
},
"public.user_role": {
"name": "user_role",
"schema": "public",
"values": ["user", "admin", "superadmin"]
}
},
"schemas": {},
"sequences": {},
"roles": {},
"policies": {},
"views": {},
"_meta": {
"columns": {},
"schemas": {},
"tables": {}
}
}

View File

@@ -22,6 +22,13 @@
"when": 1761586496779,
"tag": "0002_funny_captain_cross",
"breakpoints": true
},
{
"idx": 3,
"version": "7",
"when": 1761917402844,
"tag": "0003_lovely_wendell_rand",
"breakpoints": true
}
]
}

View File

@@ -22,6 +22,6 @@
"zod": "^4.1.11"
},
"devDependencies": {
"drizzle-kit": "0.31.5"
"drizzle-kit": "0.31.6"
}
}

View File

@@ -1,5 +1,5 @@
import type { PhotoManifestItem } from '@afilmory/builder'
import { bigint, boolean, jsonb, pgEnum, pgTable, text, timestamp, unique } from 'drizzle-orm/pg-core'
import { bigint, boolean, index, jsonb, pgEnum, pgTable, text, timestamp, unique } from 'drizzle-orm/pg-core'
import { generateId } from './snowflake'
@@ -158,6 +158,23 @@ export const systemSettings = pgTable(
(t) => [unique('uq_system_setting_key').on(t.key)],
)
export const reactions = pgTable(
'reactions',
{
id: snowflakeId,
tenantId: text('tenant_id')
.notNull()
.references(() => tenants.id, { onDelete: 'cascade' }),
createdAt: timestamp('created_at', { mode: 'string' }).defaultNow().notNull(),
refKey: text('ref_key').notNull(),
reaction: text('reaction').notNull(),
},
(t) => [
index('idx_reactions_tenant_ref_key').on(t.tenantId, t.refKey),
unique('uq_reactions_tenant_ref_key').on(t.tenantId, t.refKey),
],
)
export const photoAssets = pgTable(
'photo_asset',
{
@@ -195,6 +212,7 @@ export const dbSchema = {
authAccounts,
settings,
systemSettings,
reactions,
photoAssets,
}

View File

@@ -19,17 +19,17 @@
"ioredis": ">=5.8.0"
},
"dependencies": {
"hono": "4.10.2",
"hono": "4.10.4",
"picocolors": "1.1.1",
"reflect-metadata": "0.2.2",
"tsyringe": "4.10.0",
"zod": "^4.1.11"
},
"devDependencies": {
"@types/node": "^24.9.1",
"@vitest/coverage-v8": "4.0.3",
"@types/node": "^24.9.2",
"@vitest/coverage-v8": "4.0.6",
"ioredis": "5.8.2",
"unplugin-swc": "1.5.8",
"vitest": "4.0.3"
"vitest": "4.0.6"
}
}

View File

@@ -22,9 +22,9 @@
"tsyringe": "^4.10.0"
},
"devDependencies": {
"@types/node": "^24.9.1",
"@vitest/coverage-v8": "4.0.3",
"@types/node": "^24.9.2",
"@vitest/coverage-v8": "4.0.6",
"ioredis": "^5.8.2",
"vitest": "4.0.3"
"vitest": "4.0.6"
}
}

View File

@@ -19,6 +19,6 @@
"zod": "^4.1.11"
},
"devDependencies": {
"@types/node": "^24.9.1"
"@types/node": "^24.9.2"
}
}