sa2kit 2.0.0 → 2.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -1
- package/dist/CollisionBalls-BpHufX3H.d.mts +41 -0
- package/dist/CollisionBalls-BpHufX3H.d.ts +41 -0
- package/dist/ConfigService-BxK06xP6.d.mts +262 -0
- package/dist/ConfigService-BxK06xP6.d.ts +262 -0
- package/dist/UniversalFileService-BpvbZitV.d.mts +139 -0
- package/dist/UniversalFileService-GsP6D3Rc.d.ts +139 -0
- package/dist/audioDetection/index.d.mts +449 -0
- package/dist/audioDetection/index.d.ts +449 -0
- package/dist/audioDetection/index.js +1244 -0
- package/dist/audioDetection/index.js.map +1 -0
- package/dist/audioDetection/index.mjs +1227 -0
- package/dist/audioDetection/index.mjs.map +1 -0
- package/dist/auth/legacy/core/index.d.mts +42 -0
- package/dist/auth/legacy/core/index.d.ts +42 -0
- package/dist/auth/legacy/core/index.js +242 -0
- package/dist/auth/legacy/core/index.js.map +1 -0
- package/dist/auth/legacy/core/index.mjs +226 -0
- package/dist/auth/legacy/core/index.mjs.map +1 -0
- package/dist/auth/legacy/db/index.d.mts +5 -0
- package/dist/auth/legacy/db/index.d.ts +5 -0
- package/dist/auth/legacy/db/index.js +261 -0
- package/dist/auth/legacy/db/index.js.map +1 -0
- package/dist/auth/legacy/db/index.mjs +250 -0
- package/dist/auth/legacy/db/index.mjs.map +1 -0
- package/dist/auth/legacy/index.d.mts +5 -0
- package/dist/auth/legacy/index.d.ts +5 -0
- package/dist/auth/legacy/index.js +1107 -0
- package/dist/auth/legacy/index.js.map +1 -0
- package/dist/auth/legacy/index.mjs +1086 -0
- package/dist/auth/legacy/index.mjs.map +1 -0
- package/dist/auth/legacy/logic/index.d.mts +9 -0
- package/dist/auth/legacy/logic/index.d.ts +9 -0
- package/dist/auth/legacy/logic/index.js +194 -0
- package/dist/auth/legacy/logic/index.js.map +1 -0
- package/dist/auth/legacy/logic/index.mjs +187 -0
- package/dist/auth/legacy/logic/index.mjs.map +1 -0
- package/dist/auth/legacy/miniapp/index.d.mts +5 -0
- package/dist/auth/legacy/miniapp/index.d.ts +5 -0
- package/dist/auth/legacy/miniapp/index.js +506 -0
- package/dist/auth/legacy/miniapp/index.js.map +1 -0
- package/dist/auth/legacy/miniapp/index.mjs +487 -0
- package/dist/auth/legacy/miniapp/index.mjs.map +1 -0
- package/dist/auth/legacy/routes/index.d.mts +53 -0
- package/dist/auth/legacy/routes/index.d.ts +53 -0
- package/dist/auth/legacy/routes/index.js +278 -0
- package/dist/auth/legacy/routes/index.js.map +1 -0
- package/dist/auth/legacy/routes/index.mjs +271 -0
- package/dist/auth/legacy/routes/index.mjs.map +1 -0
- package/dist/auth/legacy/schema/index.d.mts +401 -0
- package/dist/auth/legacy/schema/index.d.ts +401 -0
- package/dist/auth/legacy/schema/index.js +50 -0
- package/dist/auth/legacy/schema/index.js.map +1 -0
- package/dist/auth/legacy/schema/index.mjs +44 -0
- package/dist/auth/legacy/schema/index.mjs.map +1 -0
- package/dist/auth/legacy/server/index.d.mts +13 -0
- package/dist/auth/legacy/server/index.d.ts +13 -0
- package/dist/auth/legacy/server/index.js +21 -0
- package/dist/auth/legacy/server/index.js.map +1 -0
- package/dist/auth/legacy/server/index.mjs +19 -0
- package/dist/auth/legacy/server/index.mjs.map +1 -0
- package/dist/auth/legacy/services/index.d.mts +40 -0
- package/dist/auth/legacy/services/index.d.ts +40 -0
- package/dist/auth/legacy/services/index.js +258 -0
- package/dist/auth/legacy/services/index.js.map +1 -0
- package/dist/auth/legacy/services/index.mjs +252 -0
- package/dist/auth/legacy/services/index.mjs.map +1 -0
- package/dist/auth/legacy/ui/miniapp/index.d.mts +10 -0
- package/dist/auth/legacy/ui/miniapp/index.d.ts +10 -0
- package/dist/auth/legacy/ui/miniapp/index.js +298 -0
- package/dist/auth/legacy/ui/miniapp/index.js.map +1 -0
- package/dist/auth/legacy/ui/miniapp/index.mjs +290 -0
- package/dist/auth/legacy/ui/miniapp/index.mjs.map +1 -0
- package/dist/auth/legacy/ui/web/index.d.mts +22 -0
- package/dist/auth/legacy/ui/web/index.d.ts +22 -0
- package/dist/auth/legacy/ui/web/index.js +899 -0
- package/dist/auth/legacy/ui/web/index.js.map +1 -0
- package/dist/auth/legacy/ui/web/index.mjs +889 -0
- package/dist/auth/legacy/ui/web/index.mjs.map +1 -0
- package/dist/auth/legacy/web/index.d.mts +5 -0
- package/dist/auth/legacy/web/index.d.ts +5 -0
- package/dist/auth/legacy/web/index.js +1107 -0
- package/dist/auth/legacy/web/index.js.map +1 -0
- package/dist/auth/legacy/web/index.mjs +1086 -0
- package/dist/auth/legacy/web/index.mjs.map +1 -0
- package/dist/auth/rn/index.d.mts +64 -0
- package/dist/auth/rn/index.d.ts +64 -0
- package/dist/auth/rn/index.js +765 -0
- package/dist/auth/rn/index.js.map +1 -0
- package/dist/auth/rn/index.mjs +754 -0
- package/dist/auth/rn/index.mjs.map +1 -0
- package/dist/base-api-client-ACKKt13v.d.mts +277 -0
- package/dist/base-api-client-ACKKt13v.d.ts +277 -0
- package/dist/boothVaultService-Cn4WPhjg.d.mts +83 -0
- package/dist/boothVaultService-Cn4WPhjg.d.ts +83 -0
- package/dist/business/index.d.mts +6 -0
- package/dist/business/index.d.ts +6 -0
- package/dist/business/index.js +1682 -0
- package/dist/business/index.js.map +1 -0
- package/dist/business/index.mjs +1675 -0
- package/dist/business/index.mjs.map +1 -0
- package/dist/calendar/index.d.mts +1325 -0
- package/dist/calendar/index.d.ts +1325 -0
- package/dist/calendar/index.js +5964 -0
- package/dist/calendar/index.js.map +1 -0
- package/dist/calendar/index.mjs +5878 -0
- package/dist/calendar/index.mjs.map +1 -0
- package/dist/components/index.d.mts +405 -0
- package/dist/components/index.d.ts +405 -0
- package/dist/components/index.js +2516 -0
- package/dist/components/index.js.map +1 -0
- package/dist/components/index.mjs +2396 -0
- package/dist/components/index.mjs.map +1 -0
- package/dist/drizzle-schema-BNhqj2AZ.d.mts +1114 -0
- package/dist/drizzle-schema-BNhqj2AZ.d.ts +1114 -0
- package/dist/festivalCard/index.d.mts +75 -0
- package/dist/festivalCard/index.d.ts +75 -0
- package/dist/festivalCard/index.js +1492 -0
- package/dist/festivalCard/index.js.map +1 -0
- package/dist/festivalCard/index.mjs +1475 -0
- package/dist/festivalCard/index.mjs.map +1 -0
- package/dist/festivalCard/server/index.d.mts +120 -0
- package/dist/festivalCard/server/index.d.ts +120 -0
- package/dist/festivalCard/server/index.js +272 -0
- package/dist/festivalCard/server/index.js.map +1 -0
- package/dist/festivalCard/server/index.mjs +265 -0
- package/dist/festivalCard/server/index.mjs.map +1 -0
- package/dist/festivalCardService-CZomuQ4E.d.mts +80 -0
- package/dist/festivalCardService-CZomuQ4E.d.ts +80 -0
- package/dist/index-1Ag7IBXN.d.ts +144 -0
- package/dist/index-DNKZ7-R_.d.mts +184 -0
- package/dist/index-DNKZ7-R_.d.ts +184 -0
- package/dist/index-DSel44Ke.d.mts +93 -0
- package/dist/index-DSel44Ke.d.ts +93 -0
- package/dist/index-DdeZSeTJ.d.mts +144 -0
- package/dist/index-DrPcMJPc.d.mts +250 -0
- package/dist/index-DrPcMJPc.d.ts +250 -0
- package/dist/index.d.mts +5333 -0
- package/dist/index.d.ts +5333 -0
- package/dist/index.js +18809 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +18533 -0
- package/dist/index.mjs.map +1 -0
- package/dist/mikuContest/ui/web/index.d.mts +2 -0
- package/dist/mikuContest/ui/web/index.d.ts +2 -0
- package/dist/mikuContest/ui/web/index.js +353 -0
- package/dist/mikuContest/ui/web/index.js.map +1 -0
- package/dist/mikuContest/ui/web/index.mjs +343 -0
- package/dist/mikuContest/ui/web/index.mjs.map +1 -0
- package/dist/mikuFireworks3D/index.d.mts +268 -0
- package/dist/mikuFireworks3D/index.d.ts +268 -0
- package/dist/mikuFireworks3D/index.js +1267 -0
- package/dist/mikuFireworks3D/index.js.map +1 -0
- package/dist/mikuFireworks3D/index.mjs +1228 -0
- package/dist/mikuFireworks3D/index.mjs.map +1 -0
- package/dist/mikuFusionGame/index.d.mts +117 -0
- package/dist/mikuFusionGame/index.d.ts +117 -0
- package/dist/mikuFusionGame/index.js +1208 -0
- package/dist/mikuFusionGame/index.js.map +1 -0
- package/dist/mikuFusionGame/index.mjs +1195 -0
- package/dist/mikuFusionGame/index.mjs.map +1 -0
- package/dist/mmd/admin/index.d.mts +487 -0
- package/dist/mmd/admin/index.d.ts +487 -0
- package/dist/mmd/admin/index.js +1058 -0
- package/dist/mmd/admin/index.js.map +1 -0
- package/dist/mmd/admin/index.mjs +1027 -0
- package/dist/mmd/admin/index.mjs.map +1 -0
- package/dist/mmd/index.d.mts +2467 -0
- package/dist/mmd/index.d.ts +2467 -0
- package/dist/mmd/index.js +10119 -0
- package/dist/mmd/index.js.map +1 -0
- package/dist/mmd/index.mjs +10028 -0
- package/dist/mmd/index.mjs.map +1 -0
- package/dist/mmd/server/index.d.mts +139 -0
- package/dist/mmd/server/index.d.ts +139 -0
- package/dist/mmd/server/index.js +424 -0
- package/dist/mmd/server/index.js.map +1 -0
- package/dist/mmd/server/index.mjs +404 -0
- package/dist/mmd/server/index.mjs.map +1 -0
- package/dist/music/index.d.mts +74 -0
- package/dist/music/index.d.ts +74 -0
- package/dist/music/index.js +830 -0
- package/dist/music/index.js.map +1 -0
- package/dist/music/index.mjs +809 -0
- package/dist/music/index.mjs.map +1 -0
- package/dist/music/server/index.d.mts +1 -0
- package/dist/music/server/index.d.ts +1 -0
- package/dist/music/server/index.js +194 -0
- package/dist/music/server/index.js.map +1 -0
- package/dist/music/server/index.mjs +182 -0
- package/dist/music/server/index.mjs.map +1 -0
- package/dist/navigation/index.d.mts +93 -0
- package/dist/navigation/index.d.ts +93 -0
- package/dist/navigation/index.js +453 -0
- package/dist/navigation/index.js.map +1 -0
- package/dist/navigation/index.mjs +443 -0
- package/dist/navigation/index.mjs.map +1 -0
- package/dist/portfolio/index.d.mts +66 -0
- package/dist/portfolio/index.d.ts +66 -0
- package/dist/portfolio/index.js +736 -0
- package/dist/portfolio/index.js.map +1 -0
- package/dist/portfolio/index.mjs +724 -0
- package/dist/portfolio/index.mjs.map +1 -0
- package/dist/qqbot/server/index.d.mts +216 -0
- package/dist/qqbot/server/index.d.ts +216 -0
- package/dist/qqbot/server/index.js +394 -0
- package/dist/qqbot/server/index.js.map +1 -0
- package/dist/qqbot/server/index.mjs +385 -0
- package/dist/qqbot/server/index.mjs.map +1 -0
- package/dist/qqbot/ui/web/index.d.mts +10 -0
- package/dist/qqbot/ui/web/index.d.ts +10 -0
- package/dist/qqbot/ui/web/index.js +105 -0
- package/dist/qqbot/ui/web/index.js.map +1 -0
- package/dist/qqbot/ui/web/index.mjs +99 -0
- package/dist/qqbot/ui/web/index.mjs.map +1 -0
- package/dist/screenReceiver/index.d.mts +86 -0
- package/dist/screenReceiver/index.d.ts +86 -0
- package/dist/screenReceiver/index.js +281 -0
- package/dist/screenReceiver/index.js.map +1 -0
- package/dist/screenReceiver/index.mjs +273 -0
- package/dist/screenReceiver/index.mjs.map +1 -0
- package/dist/testYourself/admin/index.d.mts +58 -0
- package/dist/testYourself/admin/index.d.ts +58 -0
- package/dist/testYourself/admin/index.js +1009 -0
- package/dist/testYourself/admin/index.js.map +1 -0
- package/dist/testYourself/admin/index.mjs +1002 -0
- package/dist/testYourself/admin/index.mjs.map +1 -0
- package/dist/testYourself/index.d.mts +53 -0
- package/dist/testYourself/index.d.ts +53 -0
- package/dist/testYourself/index.js +2551 -0
- package/dist/testYourself/index.js.map +1 -0
- package/dist/testYourself/index.mjs +2531 -0
- package/dist/testYourself/index.mjs.map +1 -0
- package/dist/testYourself/server/index.d.mts +1029 -0
- package/dist/testYourself/server/index.d.ts +1029 -0
- package/dist/testYourself/server/index.js +825 -0
- package/dist/testYourself/server/index.js.map +1 -0
- package/dist/testYourself/server/index.mjs +816 -0
- package/dist/testYourself/server/index.mjs.map +1 -0
- package/dist/types-BTiaMsBz.d.mts +292 -0
- package/dist/types-DyG3ZV9V.d.mts +270 -0
- package/dist/types-DyG3ZV9V.d.ts +270 -0
- package/dist/types-ERmJyjx8.d.ts +292 -0
- package/dist/types-HorDyIRv.d.mts +303 -0
- package/dist/types-HorDyIRv.d.ts +303 -0
- package/dist/vocaloidBooth/index.d.mts +64 -0
- package/dist/vocaloidBooth/index.d.ts +64 -0
- package/dist/vocaloidBooth/index.js +376 -0
- package/dist/vocaloidBooth/index.js.map +1 -0
- package/dist/vocaloidBooth/index.mjs +362 -0
- package/dist/vocaloidBooth/index.mjs.map +1 -0
- package/dist/vocaloidBooth/server/index.d.mts +111 -0
- package/dist/vocaloidBooth/server/index.d.ts +111 -0
- package/dist/vocaloidBooth/server/index.js +247 -0
- package/dist/vocaloidBooth/server/index.js.map +1 -0
- package/dist/vocaloidBooth/server/index.mjs +237 -0
- package/dist/vocaloidBooth/server/index.mjs.map +1 -0
- package/dist/vocaloidBooth/web/index.d.mts +3 -0
- package/dist/vocaloidBooth/web/index.d.ts +3 -0
- package/dist/vocaloidBooth/web/index.js +376 -0
- package/dist/vocaloidBooth/web/index.js.map +1 -0
- package/dist/vocaloidBooth/web/index.mjs +362 -0
- package/dist/vocaloidBooth/web/index.mjs.map +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,424 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var drizzleOrm = require('drizzle-orm');
|
|
4
|
+
var pgCore = require('drizzle-orm/pg-core');
|
|
5
|
+
var AdmZip = require('adm-zip');
|
|
6
|
+
var crypto = require('crypto');
|
|
7
|
+
var promises = require('fs/promises');
|
|
8
|
+
var path = require('path');
|
|
9
|
+
|
|
10
|
+
function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
|
|
11
|
+
|
|
12
|
+
var AdmZip__default = /*#__PURE__*/_interopDefault(AdmZip);
|
|
13
|
+
var path__default = /*#__PURE__*/_interopDefault(path);
|
|
14
|
+
|
|
15
|
+
// src/mmd/server/drizzle-schema.ts
|
|
16
|
+
var mmdPlaylists = pgCore.pgTable(
|
|
17
|
+
"mmd_playlists",
|
|
18
|
+
{
|
|
19
|
+
/** 主键ID */
|
|
20
|
+
id: pgCore.uuid("id").primaryKey().defaultRandom(),
|
|
21
|
+
/** 播放列表名称 */
|
|
22
|
+
name: pgCore.varchar("name", { length: 255 }).notNull(),
|
|
23
|
+
/** 播放列表描述 */
|
|
24
|
+
description: pgCore.text("description"),
|
|
25
|
+
/** 是否启用列表循环 */
|
|
26
|
+
loop: pgCore.boolean("loop").notNull().default(false),
|
|
27
|
+
/** 预加载策略: none, next, all */
|
|
28
|
+
preloadStrategy: pgCore.varchar("preload_strategy", { length: 20 }).notNull().default("none"),
|
|
29
|
+
/** 是否自动播放 */
|
|
30
|
+
autoPlay: pgCore.boolean("auto_play").notNull().default(false),
|
|
31
|
+
/** 播放列表缩略图文件ID (关联 file_metadata.id) */
|
|
32
|
+
thumbnailFileId: pgCore.uuid("thumbnail_file_id"),
|
|
33
|
+
/** 播放列表状态: draft, published, archived */
|
|
34
|
+
status: pgCore.varchar("status", { length: 20 }).notNull().default("draft"),
|
|
35
|
+
/** 显示顺序 */
|
|
36
|
+
sortOrder: pgCore.integer("sort_order").notNull().default(0),
|
|
37
|
+
/** 额外配置(JSON格式,存储舞台配置等) */
|
|
38
|
+
config: pgCore.json("config"),
|
|
39
|
+
/** 创建者ID */
|
|
40
|
+
createdBy: pgCore.varchar("created_by", { length: 255 }).notNull(),
|
|
41
|
+
/** 创建时间 */
|
|
42
|
+
createdAt: pgCore.timestamp("created_at").defaultNow().notNull(),
|
|
43
|
+
/** 更新时间 */
|
|
44
|
+
updatedAt: pgCore.timestamp("updated_at").defaultNow().notNull(),
|
|
45
|
+
/** 删除时间(软删除) */
|
|
46
|
+
deletedAt: pgCore.timestamp("deleted_at")
|
|
47
|
+
},
|
|
48
|
+
(table) => ({
|
|
49
|
+
/** 按状态查询的索引 */
|
|
50
|
+
statusIndex: pgCore.index("mmd_playlists_status_idx").on(table.status),
|
|
51
|
+
/** 按创建者查询的索引 */
|
|
52
|
+
createdByIndex: pgCore.index("mmd_playlists_created_by_idx").on(table.createdBy),
|
|
53
|
+
/** 按删除状态查询的索引 */
|
|
54
|
+
deletedAtIndex: pgCore.index("mmd_playlists_deleted_at_idx").on(table.deletedAt),
|
|
55
|
+
/** 按排序查询的索引 */
|
|
56
|
+
sortOrderIndex: pgCore.index("mmd_playlists_sort_order_idx").on(table.sortOrder)
|
|
57
|
+
})
|
|
58
|
+
);
|
|
59
|
+
var mmdPlaylistNodes = pgCore.pgTable(
|
|
60
|
+
"mmd_playlist_nodes",
|
|
61
|
+
{
|
|
62
|
+
/** 主键ID */
|
|
63
|
+
id: pgCore.uuid("id").primaryKey().defaultRandom(),
|
|
64
|
+
/** 所属播放列表ID */
|
|
65
|
+
playlistId: pgCore.uuid("playlist_id").references(() => mmdPlaylists.id, { onDelete: "cascade" }).notNull(),
|
|
66
|
+
/** 节点名称 */
|
|
67
|
+
name: pgCore.varchar("name", { length: 255 }).notNull(),
|
|
68
|
+
/** 节点描述 */
|
|
69
|
+
description: pgCore.text("description"),
|
|
70
|
+
/** 是否启用节点循环 */
|
|
71
|
+
loop: pgCore.boolean("loop").notNull().default(false),
|
|
72
|
+
/** 预计时长(秒) */
|
|
73
|
+
duration: pgCore.integer("duration"),
|
|
74
|
+
/** 节点缩略图文件ID */
|
|
75
|
+
thumbnailFileId: pgCore.uuid("thumbnail_file_id"),
|
|
76
|
+
/** 显示顺序 */
|
|
77
|
+
sortOrder: pgCore.integer("sort_order").notNull().default(0),
|
|
78
|
+
/** 模型文件ID (关联 file_metadata.id) */
|
|
79
|
+
modelFileId: pgCore.uuid("model_file_id").notNull(),
|
|
80
|
+
/** 动作文件ID */
|
|
81
|
+
motionFileId: pgCore.uuid("motion_file_id"),
|
|
82
|
+
/** 相机动画文件ID */
|
|
83
|
+
cameraFileId: pgCore.uuid("camera_file_id"),
|
|
84
|
+
/** 音频文件ID */
|
|
85
|
+
audioFileId: pgCore.uuid("audio_file_id"),
|
|
86
|
+
/** 舞台模型文件ID */
|
|
87
|
+
stageModelFileId: pgCore.uuid("stage_model_file_id"),
|
|
88
|
+
/** 附加动作文件ID列表(JSON数组) */
|
|
89
|
+
additionalMotionFileIds: pgCore.json("additional_motion_file_ids").$type(),
|
|
90
|
+
/** 额外配置(JSON格式) */
|
|
91
|
+
config: pgCore.json("config"),
|
|
92
|
+
/** 创建时间 */
|
|
93
|
+
createdAt: pgCore.timestamp("created_at").defaultNow().notNull(),
|
|
94
|
+
/** 更新时间 */
|
|
95
|
+
updatedAt: pgCore.timestamp("updated_at").defaultNow().notNull()
|
|
96
|
+
},
|
|
97
|
+
(table) => ({
|
|
98
|
+
/** 按播放列表查询的索引 */
|
|
99
|
+
playlistIndex: pgCore.index("mmd_playlist_nodes_playlist_idx").on(table.playlistId),
|
|
100
|
+
/** 按排序查询的索引 */
|
|
101
|
+
sortOrderIndex: pgCore.index("mmd_playlist_nodes_sort_order_idx").on(table.sortOrder),
|
|
102
|
+
/** 按模型文件查询的索引 */
|
|
103
|
+
modelFileIndex: pgCore.index("mmd_playlist_nodes_model_file_idx").on(table.modelFileId),
|
|
104
|
+
/** 组合索引:播放列表+排序 */
|
|
105
|
+
playlistSortIndex: pgCore.index("mmd_playlist_nodes_playlist_sort_idx").on(
|
|
106
|
+
table.playlistId,
|
|
107
|
+
table.sortOrder
|
|
108
|
+
)
|
|
109
|
+
})
|
|
110
|
+
);
|
|
111
|
+
var mmdResourceOptions = pgCore.pgTable(
|
|
112
|
+
"mmd_resource_options",
|
|
113
|
+
{
|
|
114
|
+
/** 主键ID */
|
|
115
|
+
id: pgCore.uuid("id").primaryKey().defaultRandom(),
|
|
116
|
+
/** 资源名称 */
|
|
117
|
+
name: pgCore.varchar("name", { length: 255 }).notNull(),
|
|
118
|
+
/** 资源描述 */
|
|
119
|
+
description: pgCore.text("description"),
|
|
120
|
+
/** 资源类型: model, motion, camera, audio, stage */
|
|
121
|
+
resourceType: pgCore.varchar("resource_type", { length: 20 }).notNull(),
|
|
122
|
+
/** 文件ID (关联 file_metadata.id) */
|
|
123
|
+
fileId: pgCore.uuid("file_id").notNull(),
|
|
124
|
+
/** 缩略图文件ID */
|
|
125
|
+
thumbnailFileId: pgCore.uuid("thumbnail_file_id"),
|
|
126
|
+
/** 资源标签(JSON数组,用于分类和筛选) */
|
|
127
|
+
tags: pgCore.json("tags").$type(),
|
|
128
|
+
/** 显示顺序 */
|
|
129
|
+
sortOrder: pgCore.integer("sort_order").notNull().default(0),
|
|
130
|
+
/** 是否启用 */
|
|
131
|
+
isActive: pgCore.boolean("is_active").notNull().default(true),
|
|
132
|
+
/** 创建者ID */
|
|
133
|
+
createdBy: pgCore.varchar("created_by", { length: 255 }).notNull(),
|
|
134
|
+
/** 创建时间 */
|
|
135
|
+
createdAt: pgCore.timestamp("created_at").defaultNow().notNull(),
|
|
136
|
+
/** 更新时间 */
|
|
137
|
+
updatedAt: pgCore.timestamp("updated_at").defaultNow().notNull()
|
|
138
|
+
},
|
|
139
|
+
(table) => ({
|
|
140
|
+
/** 按资源类型查询的索引 */
|
|
141
|
+
resourceTypeIndex: pgCore.index("mmd_resource_options_resource_type_idx").on(table.resourceType),
|
|
142
|
+
/** 按文件ID查询的索引 */
|
|
143
|
+
fileIdIndex: pgCore.index("mmd_resource_options_file_id_idx").on(table.fileId),
|
|
144
|
+
/** 按活跃状态查询的索引 */
|
|
145
|
+
isActiveIndex: pgCore.index("mmd_resource_options_is_active_idx").on(table.isActive),
|
|
146
|
+
/** 按创建者查询的索引 */
|
|
147
|
+
createdByIndex: pgCore.index("mmd_resource_options_created_by_idx").on(table.createdBy),
|
|
148
|
+
/** 组合索引:资源类型+活跃状态+排序 */
|
|
149
|
+
typeActiveSortIndex: pgCore.index("mmd_resource_options_type_active_sort_idx").on(
|
|
150
|
+
table.resourceType,
|
|
151
|
+
table.isActive,
|
|
152
|
+
table.sortOrder
|
|
153
|
+
)
|
|
154
|
+
})
|
|
155
|
+
);
|
|
156
|
+
var mmdPresetItems = pgCore.pgTable(
|
|
157
|
+
"mmd_preset_items",
|
|
158
|
+
{
|
|
159
|
+
/** 主键ID */
|
|
160
|
+
id: pgCore.uuid("id").primaryKey().defaultRandom(),
|
|
161
|
+
/** 预设名称 */
|
|
162
|
+
name: pgCore.varchar("name", { length: 255 }).notNull(),
|
|
163
|
+
/** 预设描述 */
|
|
164
|
+
description: pgCore.text("description"),
|
|
165
|
+
/** 缩略图文件ID */
|
|
166
|
+
thumbnailFileId: pgCore.uuid("thumbnail_file_id"),
|
|
167
|
+
/** 模型文件ID */
|
|
168
|
+
modelFileId: pgCore.uuid("model_file_id").notNull(),
|
|
169
|
+
/** 动作文件ID */
|
|
170
|
+
motionFileId: pgCore.uuid("motion_file_id"),
|
|
171
|
+
/** 相机动画文件ID */
|
|
172
|
+
cameraFileId: pgCore.uuid("camera_file_id"),
|
|
173
|
+
/** 音频文件ID */
|
|
174
|
+
audioFileId: pgCore.uuid("audio_file_id"),
|
|
175
|
+
/** 舞台模型文件ID */
|
|
176
|
+
stageModelFileId: pgCore.uuid("stage_model_file_id"),
|
|
177
|
+
/** 附加动作文件ID列表(JSON数组) */
|
|
178
|
+
additionalMotionFileIds: pgCore.json("additional_motion_file_ids").$type(),
|
|
179
|
+
/** 显示顺序 */
|
|
180
|
+
sortOrder: pgCore.integer("sort_order").notNull().default(0),
|
|
181
|
+
/** 是否启用 */
|
|
182
|
+
isActive: pgCore.boolean("is_active").notNull().default(true),
|
|
183
|
+
/** 预设标签(JSON数组) */
|
|
184
|
+
tags: pgCore.json("tags").$type(),
|
|
185
|
+
/** 创建者ID */
|
|
186
|
+
createdBy: pgCore.varchar("created_by", { length: 255 }).notNull(),
|
|
187
|
+
/** 创建时间 */
|
|
188
|
+
createdAt: pgCore.timestamp("created_at").defaultNow().notNull(),
|
|
189
|
+
/** 更新时间 */
|
|
190
|
+
updatedAt: pgCore.timestamp("updated_at").defaultNow().notNull()
|
|
191
|
+
},
|
|
192
|
+
(table) => ({
|
|
193
|
+
/** 按活跃状态查询的索引 */
|
|
194
|
+
isActiveIndex: pgCore.index("mmd_preset_items_is_active_idx").on(table.isActive),
|
|
195
|
+
/** 按排序查询的索引 */
|
|
196
|
+
sortOrderIndex: pgCore.index("mmd_preset_items_sort_order_idx").on(table.sortOrder),
|
|
197
|
+
/** 按创建者查询的索引 */
|
|
198
|
+
createdByIndex: pgCore.index("mmd_preset_items_created_by_idx").on(table.createdBy),
|
|
199
|
+
/** 按模型文件查询的索引 */
|
|
200
|
+
modelFileIndex: pgCore.index("mmd_preset_items_model_file_idx").on(table.modelFileId)
|
|
201
|
+
})
|
|
202
|
+
);
|
|
203
|
+
var mmdPlaylistsRelations = drizzleOrm.relations(mmdPlaylists, ({ many }) => ({
|
|
204
|
+
nodes: many(mmdPlaylistNodes)
|
|
205
|
+
}));
|
|
206
|
+
var mmdPlaylistNodesRelations = drizzleOrm.relations(mmdPlaylistNodes, ({ one }) => ({
|
|
207
|
+
playlist: one(mmdPlaylists, {
|
|
208
|
+
fields: [mmdPlaylistNodes.playlistId],
|
|
209
|
+
references: [mmdPlaylists.id]
|
|
210
|
+
})
|
|
211
|
+
}));
|
|
212
|
+
var MMD_MODEL_ARCHIVE_MIME_TYPES = [
|
|
213
|
+
"application/zip",
|
|
214
|
+
"application/x-zip-compressed",
|
|
215
|
+
"multipart/x-zip"
|
|
216
|
+
];
|
|
217
|
+
async function processMmdModelArchive(buffer, options) {
|
|
218
|
+
const folderName = options.folderName ?? crypto.randomUUID();
|
|
219
|
+
const targetDir = path__default.default.join(options.storageRoot, folderName);
|
|
220
|
+
await promises.mkdir(targetDir, { recursive: true });
|
|
221
|
+
const zip = new AdmZip__default.default(buffer);
|
|
222
|
+
const entries = zip.getEntries();
|
|
223
|
+
if (!entries.length) {
|
|
224
|
+
await promises.rm(targetDir, { recursive: true, force: true });
|
|
225
|
+
throw new Error("\u538B\u7F29\u5305\u4E3A\u7A7A\uFF0C\u8BF7\u68C0\u67E5\u4E0A\u4F20\u6587\u4EF6\u5185\u5BB9");
|
|
226
|
+
}
|
|
227
|
+
let modelRelativePath = null;
|
|
228
|
+
let filesExtracted = 0;
|
|
229
|
+
for (const entry of entries) {
|
|
230
|
+
if (!entry.entryName || entry.entryName.startsWith("__MACOSX") || entry.entryName.endsWith(".DS_Store")) {
|
|
231
|
+
continue;
|
|
232
|
+
}
|
|
233
|
+
const safeRelativePath = sanitizeEntryPath(entry.entryName);
|
|
234
|
+
if (!safeRelativePath) {
|
|
235
|
+
continue;
|
|
236
|
+
}
|
|
237
|
+
const destinationPath = path__default.default.join(targetDir, safeRelativePath);
|
|
238
|
+
if (entry.isDirectory) {
|
|
239
|
+
await promises.mkdir(destinationPath, { recursive: true });
|
|
240
|
+
continue;
|
|
241
|
+
}
|
|
242
|
+
await promises.mkdir(path__default.default.dirname(destinationPath), { recursive: true });
|
|
243
|
+
await promises.writeFile(destinationPath, entry.getData());
|
|
244
|
+
filesExtracted += 1;
|
|
245
|
+
const entryExt = path__default.default.extname(safeRelativePath).toLowerCase();
|
|
246
|
+
if (!modelRelativePath && (entryExt === ".pmx" || entryExt === ".pmd")) {
|
|
247
|
+
modelRelativePath = safeRelativePath.split(path__default.default.sep).join("/");
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
if (!modelRelativePath) {
|
|
251
|
+
await promises.rm(targetDir, { recursive: true, force: true });
|
|
252
|
+
throw new Error("\u538B\u7F29\u5305\u4E2D\u672A\u627E\u5230 PMX/PMD \u6A21\u578B\u6587\u4EF6\uFF0C\u8BF7\u786E\u8BA4\u76EE\u5F55\u7ED3\u6784\u662F\u5426\u6B63\u786E");
|
|
253
|
+
}
|
|
254
|
+
const format = path__default.default.extname(modelRelativePath).slice(1).toLowerCase();
|
|
255
|
+
const publicRoot = options.publicRoot ?? `/uploads/mmd/models`;
|
|
256
|
+
const modelUrl = joinPublicPath(publicRoot, folderName, modelRelativePath);
|
|
257
|
+
return {
|
|
258
|
+
directory: targetDir,
|
|
259
|
+
relativeDirectory: folderName,
|
|
260
|
+
modelRelativePath,
|
|
261
|
+
modelUrl,
|
|
262
|
+
format,
|
|
263
|
+
filesExtracted
|
|
264
|
+
};
|
|
265
|
+
}
|
|
266
|
+
function sanitizeEntryPath(entryName) {
|
|
267
|
+
const normalized = path__default.default.normalize(entryName).replace(/^(\.\.(\/|\\|$))+/, "");
|
|
268
|
+
if (!normalized || normalized === "." || normalized.startsWith("..") || path__default.default.isAbsolute(normalized)) {
|
|
269
|
+
return null;
|
|
270
|
+
}
|
|
271
|
+
return normalized;
|
|
272
|
+
}
|
|
273
|
+
function joinPublicPath(...segments) {
|
|
274
|
+
return segments.map((segment) => segment.replace(/\/+/g, "/").replace(/^\//, "").replace(/\/$/, "")).filter(Boolean).join("/").replace(/\/{2,}/g, "/").replace(/^/, "/");
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
// src/mmd/server/playlistBuilder.ts
|
|
278
|
+
var defaultNormalizer = (value) => {
|
|
279
|
+
if (!value) return "";
|
|
280
|
+
if (value.startsWith("http://") || value.startsWith("https://")) {
|
|
281
|
+
return value;
|
|
282
|
+
}
|
|
283
|
+
return value.startsWith("/") ? value : "/" + value;
|
|
284
|
+
};
|
|
285
|
+
function buildMmdPlaylistFromSources(options) {
|
|
286
|
+
if (!options.models.length) {
|
|
287
|
+
throw new Error("\u6784\u5EFA MMD \u64AD\u653E\u5217\u8868\u5931\u8D25\uFF1Amodels \u4E3A\u7A7A");
|
|
288
|
+
}
|
|
289
|
+
const limit = Math.max(1, Math.min(options.limit ?? options.models.length, options.models.length));
|
|
290
|
+
const normalizeUrl = options.normalizeUrl ?? defaultNormalizer;
|
|
291
|
+
const motions = options.motions ?? [];
|
|
292
|
+
const hasMotions = motions.length > 0;
|
|
293
|
+
const duration = options.nodeDuration ?? 30;
|
|
294
|
+
const nodes = options.models.slice(0, limit).map((model, index2) => {
|
|
295
|
+
const motion = hasMotions ? motions[index2 % motions.length] : void 0;
|
|
296
|
+
return {
|
|
297
|
+
id: String(model.id ?? index2),
|
|
298
|
+
name: model.name,
|
|
299
|
+
loop: options.loop ?? true,
|
|
300
|
+
duration,
|
|
301
|
+
thumbnail: model.thumbnailPath ? normalizeUrl(model.thumbnailPath) : void 0,
|
|
302
|
+
resources: {
|
|
303
|
+
modelPath: normalizeUrl(model.filePath),
|
|
304
|
+
motionPath: motion ? normalizeUrl(motion.filePath) : void 0,
|
|
305
|
+
cameraPath: void 0,
|
|
306
|
+
audioPath: void 0,
|
|
307
|
+
stageModelPath: void 0,
|
|
308
|
+
additionalMotions: void 0
|
|
309
|
+
}
|
|
310
|
+
};
|
|
311
|
+
});
|
|
312
|
+
return {
|
|
313
|
+
id: options.playlistId,
|
|
314
|
+
name: options.playlistName ?? "MMD \u64AD\u653E\u5217\u8868 - " + options.playlistId,
|
|
315
|
+
nodes,
|
|
316
|
+
loop: options.loop ?? true,
|
|
317
|
+
preload: options.preload ?? "next",
|
|
318
|
+
autoPlay: options.autoPlay ?? true
|
|
319
|
+
};
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
// src/mmd/server/mmdUpload.ts
|
|
323
|
+
var MMD_SUPPORTED_TYPES = {
|
|
324
|
+
model: [...MMD_MODEL_ARCHIVE_MIME_TYPES],
|
|
325
|
+
animation: ["application/octet-stream", "animation/vmd"],
|
|
326
|
+
audio: ["audio/wav", "audio/mp3", "audio/mpeg", "audio/ogg"]
|
|
327
|
+
};
|
|
328
|
+
var MMD_FILE_EXTENSIONS = {
|
|
329
|
+
model: [".zip"],
|
|
330
|
+
animation: [".vmd"],
|
|
331
|
+
audio: [".wav", ".mp3", ".ogg"]
|
|
332
|
+
};
|
|
333
|
+
async function uploadMmdResource(fileService, options) {
|
|
334
|
+
const { file, resourceType, name, description, userId } = options;
|
|
335
|
+
const ext = "." + file.name.split(".").pop()?.toLowerCase();
|
|
336
|
+
const allowedExtensions = MMD_FILE_EXTENSIONS[resourceType];
|
|
337
|
+
if (!allowedExtensions.includes(ext)) {
|
|
338
|
+
throw new Error(
|
|
339
|
+
"\u4E0D\u652F\u6301\u7684\u6587\u4EF6\u6269\u5C55\u540D: " + ext + "\u3002\u652F\u6301\u7684\u683C\u5F0F: " + allowedExtensions.join(", ")
|
|
340
|
+
);
|
|
341
|
+
}
|
|
342
|
+
const moduleId = "mmd-" + resourceType + "s";
|
|
343
|
+
const metadata = await fileService.uploadFile({
|
|
344
|
+
file,
|
|
345
|
+
moduleId,
|
|
346
|
+
businessId: "default",
|
|
347
|
+
permission: "public",
|
|
348
|
+
metadata: {
|
|
349
|
+
uploadedBy: userId,
|
|
350
|
+
uploadedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
351
|
+
originalFileName: file.name,
|
|
352
|
+
resourceType,
|
|
353
|
+
name,
|
|
354
|
+
description: description || ""
|
|
355
|
+
},
|
|
356
|
+
needsProcessing: false
|
|
357
|
+
});
|
|
358
|
+
const format = ext.slice(1).toUpperCase();
|
|
359
|
+
const fileUrl = metadata.cdnUrl || metadata.storagePath;
|
|
360
|
+
return {
|
|
361
|
+
id: metadata.id,
|
|
362
|
+
name,
|
|
363
|
+
url: fileUrl,
|
|
364
|
+
filePath: metadata.storagePath,
|
|
365
|
+
fileSize: metadata.size,
|
|
366
|
+
type: resourceType,
|
|
367
|
+
format,
|
|
368
|
+
uploadTime: metadata.uploadTime,
|
|
369
|
+
metadata
|
|
370
|
+
};
|
|
371
|
+
}
|
|
372
|
+
async function uploadMmdModel(fileService, options) {
|
|
373
|
+
const { file, storageRoot, publicRoot } = options;
|
|
374
|
+
const baseResult = await uploadMmdResource(fileService, options);
|
|
375
|
+
try {
|
|
376
|
+
const fileBuffer = Buffer.from(await file.arrayBuffer());
|
|
377
|
+
const archiveResult = await processMmdModelArchive(fileBuffer, {
|
|
378
|
+
storageRoot,
|
|
379
|
+
publicRoot,
|
|
380
|
+
folderName: baseResult.id
|
|
381
|
+
});
|
|
382
|
+
return {
|
|
383
|
+
...baseResult,
|
|
384
|
+
url: archiveResult.modelUrl,
|
|
385
|
+
filePath: archiveResult.modelUrl,
|
|
386
|
+
format: archiveResult.format.toUpperCase(),
|
|
387
|
+
extractedPath: archiveResult.modelUrl
|
|
388
|
+
};
|
|
389
|
+
} catch (extractError) {
|
|
390
|
+
console.error("\u6A21\u578B\u538B\u7F29\u5305\u5904\u7406\u5931\u8D25:", extractError);
|
|
391
|
+
throw new Error(
|
|
392
|
+
extractError instanceof Error ? extractError.message : "\u6A21\u578B\u538B\u7F29\u5305\u5904\u7406\u5931\u8D25"
|
|
393
|
+
);
|
|
394
|
+
}
|
|
395
|
+
}
|
|
396
|
+
async function batchUploadMmdResources(fileService, uploads) {
|
|
397
|
+
const results = [];
|
|
398
|
+
for (const uploadOptions of uploads) {
|
|
399
|
+
try {
|
|
400
|
+
const result = await uploadMmdResource(fileService, uploadOptions);
|
|
401
|
+
results.push(result);
|
|
402
|
+
} catch (error) {
|
|
403
|
+
console.error("\u4E0A\u4F20\u5931\u8D25: " + uploadOptions.file.name, error);
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
return results;
|
|
407
|
+
}
|
|
408
|
+
|
|
409
|
+
exports.MMD_FILE_EXTENSIONS = MMD_FILE_EXTENSIONS;
|
|
410
|
+
exports.MMD_MODEL_ARCHIVE_MIME_TYPES = MMD_MODEL_ARCHIVE_MIME_TYPES;
|
|
411
|
+
exports.MMD_SUPPORTED_TYPES = MMD_SUPPORTED_TYPES;
|
|
412
|
+
exports.batchUploadMmdResources = batchUploadMmdResources;
|
|
413
|
+
exports.buildMmdPlaylistFromSources = buildMmdPlaylistFromSources;
|
|
414
|
+
exports.mmdPlaylistNodes = mmdPlaylistNodes;
|
|
415
|
+
exports.mmdPlaylistNodesRelations = mmdPlaylistNodesRelations;
|
|
416
|
+
exports.mmdPlaylists = mmdPlaylists;
|
|
417
|
+
exports.mmdPlaylistsRelations = mmdPlaylistsRelations;
|
|
418
|
+
exports.mmdPresetItems = mmdPresetItems;
|
|
419
|
+
exports.mmdResourceOptions = mmdResourceOptions;
|
|
420
|
+
exports.processMmdModelArchive = processMmdModelArchive;
|
|
421
|
+
exports.uploadMmdModel = uploadMmdModel;
|
|
422
|
+
exports.uploadMmdResource = uploadMmdResource;
|
|
423
|
+
//# sourceMappingURL=index.js.map
|
|
424
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../src/mmd/server/drizzle-schema.ts","../../../src/mmd/server/modelArchive.ts","../../../src/mmd/server/playlistBuilder.ts","../../../src/mmd/server/mmdUpload.ts"],"names":["pgTable","uuid","varchar","text","boolean","integer","json","timestamp","index","relations","randomUUID","path","mkdir","AdmZip","rm","writeFile"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * MMD 后台管理数据库表结构定义 (PostgreSQL)\n * \n * 功能:\n * - 管理MMD播放列表配置\n * - 管理MMD播放节点\n * - 文件关联管理(与 universalFile 集成)\n * - 支持多种资源类型的映射\n * \n * 设计原则:\n * - 与 universalFile 的 file_metadata 表无缝集成\n * - 支持文件ID到OSS URL的映射\n * - 保持数据结构灵活性,便于扩展\n * \n * @package sa2kit/mmd/server\n */\n\nimport { relations } from 'drizzle-orm';\nimport {\n serial,\n text,\n timestamp,\n pgTable,\n json,\n integer,\n boolean,\n varchar,\n index,\n uuid,\n} from 'drizzle-orm/pg-core';\n\n/**\n * MMD播放列表表 (mmd_playlists)\n * \n * 管理MMD播放列表的基础信息\n */\nexport const mmdPlaylists = pgTable(\n 'mmd_playlists',\n {\n /** 主键ID */\n id: uuid('id').primaryKey().defaultRandom(),\n\n /** 播放列表名称 */\n name: varchar('name', { length: 255 }).notNull(),\n\n /** 播放列表描述 */\n description: text('description'),\n\n /** 是否启用列表循环 */\n loop: boolean('loop').notNull().default(false),\n\n /** 预加载策略: none, next, all */\n preloadStrategy: varchar('preload_strategy', { length: 20 }).notNull().default('none'),\n\n /** 是否自动播放 */\n autoPlay: boolean('auto_play').notNull().default(false),\n\n /** 播放列表缩略图文件ID (关联 file_metadata.id) */\n thumbnailFileId: uuid('thumbnail_file_id'),\n\n /** 播放列表状态: draft, published, archived */\n status: varchar('status', { length: 20 }).notNull().default('draft'),\n\n /** 显示顺序 */\n sortOrder: integer('sort_order').notNull().default(0),\n\n /** 额外配置(JSON格式,存储舞台配置等) */\n config: json('config'),\n\n /** 创建者ID */\n createdBy: varchar('created_by', { length: 255 }).notNull(),\n\n /** 创建时间 */\n createdAt: timestamp('created_at').defaultNow().notNull(),\n\n /** 更新时间 */\n updatedAt: timestamp('updated_at').defaultNow().notNull(),\n\n /** 删除时间(软删除) */\n deletedAt: timestamp('deleted_at'),\n },\n (table) => ({\n /** 按状态查询的索引 */\n statusIndex: index('mmd_playlists_status_idx').on(table.status),\n\n /** 按创建者查询的索引 */\n createdByIndex: index('mmd_playlists_created_by_idx').on(table.createdBy),\n\n /** 按删除状态查询的索引 */\n deletedAtIndex: index('mmd_playlists_deleted_at_idx').on(table.deletedAt),\n\n /** 按排序查询的索引 */\n sortOrderIndex: index('mmd_playlists_sort_order_idx').on(table.sortOrder),\n })\n);\n\n/**\n * MMD播放节点表 (mmd_playlist_nodes)\n * \n * 管理播放列表中的每个播放节点\n */\nexport const mmdPlaylistNodes = pgTable(\n 'mmd_playlist_nodes',\n {\n /** 主键ID */\n id: uuid('id').primaryKey().defaultRandom(),\n\n /** 所属播放列表ID */\n playlistId: uuid('playlist_id')\n .references(() => mmdPlaylists.id, { onDelete: 'cascade' })\n .notNull(),\n\n /** 节点名称 */\n name: varchar('name', { length: 255 }).notNull(),\n\n /** 节点描述 */\n description: text('description'),\n\n /** 是否启用节点循环 */\n loop: boolean('loop').notNull().default(false),\n\n /** 预计时长(秒) */\n duration: integer('duration'),\n\n /** 节点缩略图文件ID */\n thumbnailFileId: uuid('thumbnail_file_id'),\n\n /** 显示顺序 */\n sortOrder: integer('sort_order').notNull().default(0),\n\n /** 模型文件ID (关联 file_metadata.id) */\n modelFileId: uuid('model_file_id').notNull(),\n\n /** 动作文件ID */\n motionFileId: uuid('motion_file_id'),\n\n /** 相机动画文件ID */\n cameraFileId: uuid('camera_file_id'),\n\n /** 音频文件ID */\n audioFileId: uuid('audio_file_id'),\n\n /** 舞台模型文件ID */\n stageModelFileId: uuid('stage_model_file_id'),\n\n /** 附加动作文件ID列表(JSON数组) */\n additionalMotionFileIds: json('additional_motion_file_ids').$type<string[]>(),\n\n /** 额外配置(JSON格式) */\n config: json('config'),\n\n /** 创建时间 */\n createdAt: timestamp('created_at').defaultNow().notNull(),\n\n /** 更新时间 */\n updatedAt: timestamp('updated_at').defaultNow().notNull(),\n },\n (table) => ({\n /** 按播放列表查询的索引 */\n playlistIndex: index('mmd_playlist_nodes_playlist_idx').on(table.playlistId),\n\n /** 按排序查询的索引 */\n sortOrderIndex: index('mmd_playlist_nodes_sort_order_idx').on(table.sortOrder),\n\n /** 按模型文件查询的索引 */\n modelFileIndex: index('mmd_playlist_nodes_model_file_idx').on(table.modelFileId),\n\n /** 组合索引:播放列表+排序 */\n playlistSortIndex: index('mmd_playlist_nodes_playlist_sort_idx').on(\n table.playlistId,\n table.sortOrder\n ),\n })\n);\n\n/**\n * MMD资源选项表 (mmd_resource_options)\n * \n * 管理可选的MMD资源(用于自由组合模式)\n */\nexport const mmdResourceOptions = pgTable(\n 'mmd_resource_options',\n {\n /** 主键ID */\n id: uuid('id').primaryKey().defaultRandom(),\n\n /** 资源名称 */\n name: varchar('name', { length: 255 }).notNull(),\n\n /** 资源描述 */\n description: text('description'),\n\n /** 资源类型: model, motion, camera, audio, stage */\n resourceType: varchar('resource_type', { length: 20 }).notNull(),\n\n /** 文件ID (关联 file_metadata.id) */\n fileId: uuid('file_id').notNull(),\n\n /** 缩略图文件ID */\n thumbnailFileId: uuid('thumbnail_file_id'),\n\n /** 资源标签(JSON数组,用于分类和筛选) */\n tags: json('tags').$type<string[]>(),\n\n /** 显示顺序 */\n sortOrder: integer('sort_order').notNull().default(0),\n\n /** 是否启用 */\n isActive: boolean('is_active').notNull().default(true),\n\n /** 创建者ID */\n createdBy: varchar('created_by', { length: 255 }).notNull(),\n\n /** 创建时间 */\n createdAt: timestamp('created_at').defaultNow().notNull(),\n\n /** 更新时间 */\n updatedAt: timestamp('updated_at').defaultNow().notNull(),\n },\n (table) => ({\n /** 按资源类型查询的索引 */\n resourceTypeIndex: index('mmd_resource_options_resource_type_idx').on(table.resourceType),\n\n /** 按文件ID查询的索引 */\n fileIdIndex: index('mmd_resource_options_file_id_idx').on(table.fileId),\n\n /** 按活跃状态查询的索引 */\n isActiveIndex: index('mmd_resource_options_is_active_idx').on(table.isActive),\n\n /** 按创建者查询的索引 */\n createdByIndex: index('mmd_resource_options_created_by_idx').on(table.createdBy),\n\n /** 组合索引:资源类型+活跃状态+排序 */\n typeActiveSortIndex: index('mmd_resource_options_type_active_sort_idx').on(\n table.resourceType,\n table.isActive,\n table.sortOrder\n ),\n })\n);\n\n/**\n * MMD预设列表项表 (mmd_preset_items)\n * \n * 管理预设的MMD资源组合(用于列表模式)\n */\nexport const mmdPresetItems = pgTable(\n 'mmd_preset_items',\n {\n /** 主键ID */\n id: uuid('id').primaryKey().defaultRandom(),\n\n /** 预设名称 */\n name: varchar('name', { length: 255 }).notNull(),\n\n /** 预设描述 */\n description: text('description'),\n\n /** 缩略图文件ID */\n thumbnailFileId: uuid('thumbnail_file_id'),\n\n /** 模型文件ID */\n modelFileId: uuid('model_file_id').notNull(),\n\n /** 动作文件ID */\n motionFileId: uuid('motion_file_id'),\n\n /** 相机动画文件ID */\n cameraFileId: uuid('camera_file_id'),\n\n /** 音频文件ID */\n audioFileId: uuid('audio_file_id'),\n\n /** 舞台模型文件ID */\n stageModelFileId: uuid('stage_model_file_id'),\n\n /** 附加动作文件ID列表(JSON数组) */\n additionalMotionFileIds: json('additional_motion_file_ids').$type<string[]>(),\n\n /** 显示顺序 */\n sortOrder: integer('sort_order').notNull().default(0),\n\n /** 是否启用 */\n isActive: boolean('is_active').notNull().default(true),\n\n /** 预设标签(JSON数组) */\n tags: json('tags').$type<string[]>(),\n\n /** 创建者ID */\n createdBy: varchar('created_by', { length: 255 }).notNull(),\n\n /** 创建时间 */\n createdAt: timestamp('created_at').defaultNow().notNull(),\n\n /** 更新时间 */\n updatedAt: timestamp('updated_at').defaultNow().notNull(),\n },\n (table) => ({\n /** 按活跃状态查询的索引 */\n isActiveIndex: index('mmd_preset_items_is_active_idx').on(table.isActive),\n\n /** 按排序查询的索引 */\n sortOrderIndex: index('mmd_preset_items_sort_order_idx').on(table.sortOrder),\n\n /** 按创建者查询的索引 */\n createdByIndex: index('mmd_preset_items_created_by_idx').on(table.createdBy),\n\n /** 按模型文件查询的索引 */\n modelFileIndex: index('mmd_preset_items_model_file_idx').on(table.modelFileId),\n })\n);\n\n// ========== 关系定义 ==========\n\nexport const mmdPlaylistsRelations = relations(mmdPlaylists, ({ many }) => ({\n nodes: many(mmdPlaylistNodes),\n}));\n\nexport const mmdPlaylistNodesRelations = relations(mmdPlaylistNodes, ({ one }) => ({\n playlist: one(mmdPlaylists, {\n fields: [mmdPlaylistNodes.playlistId],\n references: [mmdPlaylists.id],\n }),\n}));\n\n// ========== 导出类型 ==========\n\nexport type MmdPlaylist = typeof mmdPlaylists.$inferSelect;\nexport type NewMmdPlaylist = typeof mmdPlaylists.$inferInsert;\n\nexport type MmdPlaylistNode = typeof mmdPlaylistNodes.$inferSelect;\nexport type NewMmdPlaylistNode = typeof mmdPlaylistNodes.$inferInsert;\n\nexport type MmdResourceOption = typeof mmdResourceOptions.$inferSelect;\nexport type NewMmdResourceOption = typeof mmdResourceOptions.$inferInsert;\n\nexport type MmdPresetItem = typeof mmdPresetItems.$inferSelect;\nexport type NewMmdPresetItem = typeof mmdPresetItems.$inferInsert;\n\n","import AdmZip from 'adm-zip';\nimport { randomUUID } from 'crypto';\nimport { mkdir, rm, writeFile } from 'fs/promises';\nimport path from 'path';\n\nexport type SupportedModelFormat = 'pmx' | 'pmd';\n\nexport interface ProcessMmdModelArchiveOptions {\n /**\n * 绝对路径,指向模型解压根目录,例如 /app/uploads/mmd/models\n */\n storageRoot: string;\n /**\n * 对外暴露的公共路径前缀,例如 /uploads/mmd/models\n * 默认与 storageRoot 中的叶子目录一致\n */\n publicRoot?: string;\n /**\n * 自定义文件夹名称,默认使用随机 UUID\n */\n folderName?: string;\n}\n\nexport interface ProcessMmdModelArchiveResult {\n /** 解压出来的目录(绝对路径) */\n directory: string;\n /** 与 storageRoot 相对的目录名 */\n relativeDirectory: string;\n /** 模型文件相对目录(包含子目录) */\n modelRelativePath: string;\n /** 可供前端使用的模型 URL */\n modelUrl: string;\n /** 模型格式 */\n format: SupportedModelFormat;\n /** 解压得到的文件数量 */\n filesExtracted: number;\n}\n\nexport const MMD_MODEL_ARCHIVE_MIME_TYPES = [\n 'application/zip',\n 'application/x-zip-compressed',\n 'multipart/x-zip',\n] as const;\n\n/**\n * 解析上传的 MMD 模型压缩包,保留目录结构并返回模型路径\n */\nexport async function processMmdModelArchive(\n buffer: Buffer,\n options: ProcessMmdModelArchiveOptions,\n): Promise<ProcessMmdModelArchiveResult> {\n const folderName = options.folderName ?? randomUUID();\n const targetDir = path.join(options.storageRoot, folderName);\n await mkdir(targetDir, { recursive: true });\n\n const zip = new AdmZip(buffer);\n const entries = zip.getEntries();\n if (!entries.length) {\n await rm(targetDir, { recursive: true, force: true });\n throw new Error('压缩包为空,请检查上传文件内容');\n }\n\n let modelRelativePath: string | null = null;\n let filesExtracted = 0;\n\n for (const entry of entries) {\n if (!entry.entryName || entry.entryName.startsWith('__MACOSX') || entry.entryName.endsWith('.DS_Store')) {\n continue;\n }\n\n const safeRelativePath = sanitizeEntryPath(entry.entryName);\n if (!safeRelativePath) {\n continue;\n }\n\n const destinationPath = path.join(targetDir, safeRelativePath);\n\n if (entry.isDirectory) {\n await mkdir(destinationPath, { recursive: true });\n continue;\n }\n\n await mkdir(path.dirname(destinationPath), { recursive: true });\n await writeFile(destinationPath, entry.getData());\n filesExtracted += 1;\n\n const entryExt = path.extname(safeRelativePath).toLowerCase();\n if (!modelRelativePath && (entryExt === '.pmx' || entryExt === '.pmd')) {\n modelRelativePath = safeRelativePath.split(path.sep).join('/');\n }\n }\n\n if (!modelRelativePath) {\n await rm(targetDir, { recursive: true, force: true });\n throw new Error('压缩包中未找到 PMX/PMD 模型文件,请确认目录结构是否正确');\n }\n\n const format = path.extname(modelRelativePath).slice(1).toLowerCase() as SupportedModelFormat;\n const publicRoot = options.publicRoot ?? `/uploads/mmd/models`;\n const modelUrl = joinPublicPath(publicRoot, folderName, modelRelativePath);\n\n return {\n directory: targetDir,\n relativeDirectory: folderName,\n modelRelativePath,\n modelUrl,\n format,\n filesExtracted,\n };\n}\n\nfunction sanitizeEntryPath(entryName: string): string | null {\n const normalized = path.normalize(entryName).replace(/^(\\.\\.(\\/|\\\\|$))+/, '');\n if (!normalized || normalized === '.' || normalized.startsWith('..') || path.isAbsolute(normalized)) {\n return null;\n }\n return normalized;\n}\n\nfunction joinPublicPath(...segments: string[]) {\n return segments\n .map((segment) => segment.replace(/\\/+/g, '/').replace(/^\\//, '').replace(/\\/$/, ''))\n .filter(Boolean)\n .join('/')\n .replace(/\\/{2,}/g, '/')\n .replace(/^/, '/');\n}\n\n","import type { MMDPlaylistConfig, MMDPlaylistNode } from '../types';\n\nexport interface PlaylistModelSource {\n id: string | number;\n name: string;\n filePath: string;\n thumbnailPath?: string | null;\n}\n\nexport interface PlaylistMotionSource {\n id: string | number;\n name?: string;\n filePath: string;\n}\n\nexport interface BuildMmdPlaylistOptions {\n playlistId: string;\n playlistName?: string;\n models: PlaylistModelSource[];\n motions?: PlaylistMotionSource[];\n limit?: number;\n loop?: boolean;\n preload?: 'none' | 'next' | 'all';\n autoPlay?: boolean;\n nodeDuration?: number;\n normalizeUrl?: (pathOrUrl: string) => string;\n}\n\nconst defaultNormalizer = (value: string) => {\n if (!value) return '';\n if (value.startsWith('http://') || value.startsWith('https://')) {\n return value;\n }\n return value.startsWith('/') ? value : '/' + (value);\n};\n\n/**\n * 根据数据库中的模型/动作记录快速构建 MMDPlaylistConfig\n */\nexport function buildMmdPlaylistFromSources(options: BuildMmdPlaylistOptions): MMDPlaylistConfig {\n if (!options.models.length) {\n throw new Error('构建 MMD 播放列表失败:models 为空');\n }\n\n const limit = Math.max(1, Math.min(options.limit ?? options.models.length, options.models.length));\n const normalizeUrl = options.normalizeUrl ?? defaultNormalizer;\n const motions = options.motions ?? [];\n const hasMotions = motions.length > 0;\n const duration = options.nodeDuration ?? 30;\n\n const nodes: MMDPlaylistNode[] = options.models.slice(0, limit).map((model, index) => {\n const motion = hasMotions ? motions[index % motions.length] : undefined;\n return {\n id: String(model.id ?? index),\n name: model.name,\n loop: options.loop ?? true,\n duration,\n thumbnail: model.thumbnailPath ? normalizeUrl(model.thumbnailPath) : undefined,\n resources: {\n modelPath: normalizeUrl(model.filePath),\n motionPath: motion ? normalizeUrl(motion.filePath) : undefined,\n cameraPath: undefined,\n audioPath: undefined,\n stageModelPath: undefined,\n additionalMotions: undefined,\n },\n };\n });\n\n return {\n id: options.playlistId,\n name: options.playlistName ?? 'MMD 播放列表 - ' + (options.playlistId),\n nodes,\n loop: options.loop ?? true,\n preload: options.preload ?? 'next',\n autoPlay: options.autoPlay ?? true,\n };\n}\n\n","/**\n * MMD 资源上传辅助函数\n * \n * 整合 UniversalFileService 用于 MMD 资源上传\n */\n\nimport type { UniversalFileService } from '../../universalFile/server/UniversalFileService';\nimport type { FileMetadata } from '../../universalFile/types';\nimport { processMmdModelArchive, MMD_MODEL_ARCHIVE_MIME_TYPES } from './modelArchive';\n\nexport const MMD_SUPPORTED_TYPES = {\n model: [...MMD_MODEL_ARCHIVE_MIME_TYPES],\n animation: ['application/octet-stream', 'animation/vmd'],\n audio: ['audio/wav', 'audio/mp3', 'audio/mpeg', 'audio/ogg'],\n};\n\nexport const MMD_FILE_EXTENSIONS = {\n model: ['.zip'],\n animation: ['.vmd'],\n audio: ['.wav', '.mp3', '.ogg'],\n};\n\nexport interface MmdUploadOptions {\n file: File;\n resourceType: 'model' | 'animation' | 'audio';\n name: string;\n description?: string;\n userId: string;\n}\n\nexport interface MmdUploadResult {\n id: string;\n name: string;\n url: string;\n filePath: string;\n fileSize: number;\n type: string;\n format: string;\n uploadTime: Date;\n metadata?: FileMetadata;\n}\n\n/**\n * 使用 UniversalFileService 上传 MMD 资源\n * \n * @param fileService - UniversalFileService 实例\n * @param options - 上传选项\n * @returns 上传结果\n */\nexport async function uploadMmdResource(\n fileService: UniversalFileService,\n options: MmdUploadOptions\n): Promise<MmdUploadResult> {\n const { file, resourceType, name, description, userId } = options;\n\n // 验证文件扩展名\n const ext = '.' + (file.name.split('.').pop()?.toLowerCase());\n const allowedExtensions = MMD_FILE_EXTENSIONS[resourceType];\n\n if (!allowedExtensions.includes(ext)) {\n throw new Error(\n '不支持的文件扩展名: ' + (ext) + '。支持的格式: ' + (allowedExtensions.join(', '))\n );\n }\n\n // 确定模块ID\n const moduleId = 'mmd-' + (resourceType) + 's';\n\n // 上传文件\n const metadata = await fileService.uploadFile({\n file,\n moduleId,\n businessId: 'default',\n permission: 'public',\n metadata: {\n uploadedBy: userId,\n uploadedAt: new Date().toISOString(),\n originalFileName: file.name,\n resourceType,\n name,\n description: description || '',\n },\n needsProcessing: false,\n });\n\n // 构建返回结果\n const format = ext.slice(1).toUpperCase();\n const fileUrl = metadata.cdnUrl || metadata.storagePath;\n\n return {\n id: metadata.id,\n name,\n url: fileUrl,\n filePath: metadata.storagePath,\n fileSize: metadata.size,\n type: resourceType,\n format,\n uploadTime: metadata.uploadTime,\n metadata,\n };\n}\n\n/**\n * 上传 MMD 模型 (ZIP 压缩包)\n * \n * @param fileService - UniversalFileService 实例\n * @param options - 上传选项\n * @returns 上传结果,包含解压后的模型路径\n */\nexport async function uploadMmdModel(\n fileService: UniversalFileService,\n options: MmdUploadOptions & { storageRoot: string; publicRoot: string }\n): Promise<MmdUploadResult & { extractedPath?: string }> {\n const { file, storageRoot, publicRoot } = options;\n\n // 首先上传 ZIP 文件\n const baseResult = await uploadMmdResource(fileService, options);\n\n // 处理 ZIP 压缩包\n try {\n const fileBuffer = Buffer.from(await file.arrayBuffer());\n const archiveResult = await processMmdModelArchive(fileBuffer, {\n storageRoot,\n publicRoot,\n folderName: baseResult.id,\n });\n\n return {\n ...baseResult,\n url: archiveResult.modelUrl,\n filePath: archiveResult.modelUrl,\n format: archiveResult.format.toUpperCase(),\n extractedPath: archiveResult.modelUrl,\n };\n } catch (extractError) {\n console.error('模型压缩包处理失败:', extractError);\n throw new Error(\n extractError instanceof Error ? extractError.message : '模型压缩包处理失败'\n );\n }\n}\n\n/**\n * 批量上传 MMD 资源\n * \n * @param fileService - UniversalFileService 实例\n * @param uploads - 上传选项数组\n * @returns 上传结果数组\n */\nexport async function batchUploadMmdResources(\n fileService: UniversalFileService,\n uploads: MmdUploadOptions[]\n): Promise<MmdUploadResult[]> {\n const results: MmdUploadResult[] = [];\n\n for (const uploadOptions of uploads) {\n try {\n const result = await uploadMmdResource(fileService, uploadOptions);\n results.push(result);\n } catch (error) {\n console.error('上传失败: ' + (uploadOptions.file.name), error);\n // 继续处理其他文件\n }\n }\n\n return results;\n}\n\n"]}
|