生成视频

This commit is contained in:
小帅 2026-03-23 15:13:38 +08:00
parent 8b8dbc224c
commit 558bc93a88
7 changed files with 225 additions and 273 deletions

View File

@ -1,4 +1,4 @@
// @routes-hash 3d0673061005074e704638efccc539d1 // @routes-hash 35b38e94f3f229944c2a0c85a8439b87
import { Express } from "express"; import { Express } from "express";
import route1 from "./routes/agents/clearMemory"; import route1 from "./routes/agents/clearMemory";
@ -40,34 +40,37 @@ import route36 from "./routes/production/editStoryboard/saveStoryboardFlow";
import route37 from "./routes/production/editStoryboard/updateStoryboardFlow"; import route37 from "./routes/production/editStoryboard/updateStoryboardFlow";
import route38 from "./routes/production/getProductionData"; import route38 from "./routes/production/getProductionData";
import route39 from "./routes/production/getStoryboardData"; import route39 from "./routes/production/getStoryboardData";
import route40 from "./routes/production/workbench/generateVideo"; import route40 from "./routes/production/workbench/confirmSelection";
import route41 from "./routes/production/workbench/getVideoModelDetail"; import route41 from "./routes/production/workbench/delVideo";
import route42 from "./routes/project/addProject"; import route42 from "./routes/production/workbench/generateVideo";
import route43 from "./routes/project/delProject"; import route43 from "./routes/production/workbench/getVideoModelDetail";
import route44 from "./routes/project/editProject"; import route44 from "./routes/production/workbench/videoPolling";
import route45 from "./routes/project/getProject"; import route45 from "./routes/project/addProject";
import route46 from "./routes/script/addScript"; import route46 from "./routes/project/delProject";
import route47 from "./routes/script/delScript"; import route47 from "./routes/project/editProject";
import route48 from "./routes/script/getScrptApi"; import route48 from "./routes/project/getProject";
import route49 from "./routes/script/updateScript"; import route49 from "./routes/script/addScript";
import route50 from "./routes/setting/agentDeploy/deployAgentModel"; import route50 from "./routes/script/delScript";
import route51 from "./routes/setting/agentDeploy/getAgentDeploy"; import route51 from "./routes/script/getScrptApi";
import route52 from "./routes/setting/agentDeploy/updateKey"; import route52 from "./routes/script/updateScript";
import route53 from "./routes/setting/dbConfig/clearData"; import route53 from "./routes/setting/agentDeploy/deployAgentModel";
import route54 from "./routes/setting/getTextModel"; import route54 from "./routes/setting/agentDeploy/getAgentDeploy";
import route55 from "./routes/setting/loginConfig/getUser"; import route55 from "./routes/setting/agentDeploy/updateKey";
import route56 from "./routes/setting/loginConfig/updateUserPwd"; import route56 from "./routes/setting/dbConfig/clearData";
import route57 from "./routes/setting/memoryConfig/getMemory"; import route57 from "./routes/setting/getTextModel";
import route58 from "./routes/setting/memoryConfig/sureMemory"; import route58 from "./routes/setting/loginConfig/getUser";
import route59 from "./routes/setting/vendorConfig/addVendor"; import route59 from "./routes/setting/loginConfig/updateUserPwd";
import route60 from "./routes/setting/vendorConfig/deleteVendor"; import route60 from "./routes/setting/memoryConfig/getMemory";
import route61 from "./routes/setting/vendorConfig/getVendorList"; import route61 from "./routes/setting/memoryConfig/sureMemory";
import route62 from "./routes/setting/vendorConfig/modelTest"; import route62 from "./routes/setting/vendorConfig/addVendor";
import route63 from "./routes/setting/vendorConfig/updateVendor"; import route63 from "./routes/setting/vendorConfig/deleteVendor";
import route64 from "./routes/task/getMyTaskApi"; import route64 from "./routes/setting/vendorConfig/getVendorList";
import route65 from "./routes/task/getTaskCategories"; import route65 from "./routes/setting/vendorConfig/modelTest";
import route66 from "./routes/task/taskDetails"; import route66 from "./routes/setting/vendorConfig/updateVendor";
import route67 from "./routes/test/test"; import route67 from "./routes/task/getMyTaskApi";
import route68 from "./routes/task/getTaskCategories";
import route69 from "./routes/task/taskDetails";
import route70 from "./routes/test/test";
export default async (app: Express) => { export default async (app: Express) => {
app.use("/api/agents/clearMemory", route1); app.use("/api/agents/clearMemory", route1);
@ -109,32 +112,35 @@ export default async (app: Express) => {
app.use("/api/production/editStoryboard/updateStoryboardFlow", route37); app.use("/api/production/editStoryboard/updateStoryboardFlow", route37);
app.use("/api/production/getProductionData", route38); app.use("/api/production/getProductionData", route38);
app.use("/api/production/getStoryboardData", route39); app.use("/api/production/getStoryboardData", route39);
app.use("/api/production/workbench/generateVideo", route40); app.use("/api/production/workbench/confirmSelection", route40);
app.use("/api/production/workbench/getVideoModelDetail", route41); app.use("/api/production/workbench/delVideo", route41);
app.use("/api/project/addProject", route42); app.use("/api/production/workbench/generateVideo", route42);
app.use("/api/project/delProject", route43); app.use("/api/production/workbench/getVideoModelDetail", route43);
app.use("/api/project/editProject", route44); app.use("/api/production/workbench/videoPolling", route44);
app.use("/api/project/getProject", route45); app.use("/api/project/addProject", route45);
app.use("/api/script/addScript", route46); app.use("/api/project/delProject", route46);
app.use("/api/script/delScript", route47); app.use("/api/project/editProject", route47);
app.use("/api/script/getScrptApi", route48); app.use("/api/project/getProject", route48);
app.use("/api/script/updateScript", route49); app.use("/api/script/addScript", route49);
app.use("/api/setting/agentDeploy/deployAgentModel", route50); app.use("/api/script/delScript", route50);
app.use("/api/setting/agentDeploy/getAgentDeploy", route51); app.use("/api/script/getScrptApi", route51);
app.use("/api/setting/agentDeploy/updateKey", route52); app.use("/api/script/updateScript", route52);
app.use("/api/setting/dbConfig/clearData", route53); app.use("/api/setting/agentDeploy/deployAgentModel", route53);
app.use("/api/setting/getTextModel", route54); app.use("/api/setting/agentDeploy/getAgentDeploy", route54);
app.use("/api/setting/loginConfig/getUser", route55); app.use("/api/setting/agentDeploy/updateKey", route55);
app.use("/api/setting/loginConfig/updateUserPwd", route56); app.use("/api/setting/dbConfig/clearData", route56);
app.use("/api/setting/memoryConfig/getMemory", route57); app.use("/api/setting/getTextModel", route57);
app.use("/api/setting/memoryConfig/sureMemory", route58); app.use("/api/setting/loginConfig/getUser", route58);
app.use("/api/setting/vendorConfig/addVendor", route59); app.use("/api/setting/loginConfig/updateUserPwd", route59);
app.use("/api/setting/vendorConfig/deleteVendor", route60); app.use("/api/setting/memoryConfig/getMemory", route60);
app.use("/api/setting/vendorConfig/getVendorList", route61); app.use("/api/setting/memoryConfig/sureMemory", route61);
app.use("/api/setting/vendorConfig/modelTest", route62); app.use("/api/setting/vendorConfig/addVendor", route62);
app.use("/api/setting/vendorConfig/updateVendor", route63); app.use("/api/setting/vendorConfig/deleteVendor", route63);
app.use("/api/task/getMyTaskApi", route64); app.use("/api/setting/vendorConfig/getVendorList", route64);
app.use("/api/task/getTaskCategories", route65); app.use("/api/setting/vendorConfig/modelTest", route65);
app.use("/api/task/taskDetails", route66); app.use("/api/setting/vendorConfig/updateVendor", route66);
app.use("/api/test/test", route67); app.use("/api/task/getMyTaskApi", route67);
app.use("/api/task/getTaskCategories", route68);
app.use("/api/task/taskDetails", route69);
app.use("/api/test/test", route70);
} }

View File

@ -12,89 +12,75 @@ export default router.post(
async (req, res) => { async (req, res) => {
const { scriptId } = req.body; const { scriptId } = req.body;
// 1. 查出该剧本下所有分镜 //查询分镜数据
const storyboards = await u.db("o_storyboard").where("o_storyboard.scriptId", scriptId).select("*").orderBy("o_storyboard.createTime", "asc"); const storyboards = await u.db("o_storyboard").where("o_storyboard.scriptId", scriptId).select("*").orderBy("o_storyboard.createTime", "asc");
if (storyboards.length === 0) { const storyboardsList = await Promise.all(
return res.status(200).send(success([])); storyboards.map(async (item) => {
} return {
...item,
const storyboardIds = storyboards.map((s) => s.id as number); filePath: item.filePath ? await u.oss.getFileUrl(item.filePath) : null,
};
// 2. 批量查出所有相关视频 }),
const videos = await u
.db("o_video")
.whereIn("o_video.storyboardId", storyboardIds)
.select("o_video.id", "o_video.storyboardId", "o_video.filePath", "o_video.state", "o_video.errorReason")
.orderBy("o_video.time", "desc");
// 3. 批量查出所有相关配置
const configs = await u
.db("o_videoConfig")
.whereIn("o_videoConfig.storyboardId", storyboardIds)
.select(
"o_videoConfig.id",
"o_videoConfig.storyboardId",
"o_videoConfig.videoId",
"o_videoConfig.prompt",
"o_videoConfig.model",
"o_videoConfig.mode",
"o_videoConfig.resolution",
"o_videoConfig.duration",
"o_videoConfig.audio",
"o_videoConfig.data",
); );
// 4. 按 storyboardId 建立 Map 方便聚合 const storyboardIds = storyboardsList.map((s) => s.id as number);
const videoMap = new Map<number, typeof videos>();
for (const video of videos) {
const sid = video.storyboardId as number;
if (!videoMap.has(sid)) videoMap.set(sid, []);
videoMap.get(sid)!.push(video);
}
const configMap = new Map(configs.map((c) => [c.storyboardId as number, c]));
// 5. 组装结果:分镜平铺 + config 对象 + videos 数组 //查询分镜配置
const data = await Promise.all( const storyboardConfigs = await u.db("o_videoConfig").whereIn("storyboardId", storyboardIds).select("*");
storyboards.map(async (storyboard) => { const storyboardConfigsList = await Promise.all(
const sid = storyboard.id as number; storyboardConfigs.map(async (item) => {
const config = configMap.get(sid) ?? null; if (item.data) {
let configDataWithFilePath: any[] = []; const parsedData = JSON.parse(item.data);
if (config?.data) { const dataWithFilePath = await Promise.all(
const parsedData: { id: number; type: string }[] = JSON.parse(config.data); parsedData.map(async (d: { type: string; id: number }) => {
configDataWithFilePath = await Promise.all( if (d.type === "assets") {
parsedData.map(async (item) => {
if (item.type === "storyboard") {
const row = await u.db("o_storyboard").where("id", item.id).select("filePath").first();
return row?.filePath ? await u.oss.getFileUrl(row.filePath) : null;
}
if (item.type === "assets") {
const row = await u const row = await u
.db("o_assets") .db("o_assets")
.where("o_assets.id", item.id) .where("o_assets.id", item.id)
.leftJoin("o_image", "o_assets.imageId", "o_image.id") .leftJoin("o_image", "o_assets.imageId", "o_image.id")
.select("o_image.filePath") .select("o_image.filePath")
.first(); .first();
return row?.filePath ? await u.oss.getFileUrl(row.filePath) : null; if (row?.filePath) {
return await u.oss.getFileUrl(row.filePath);
} }
return null; } else {
const row = await u.db("o_storyboard").where("id", item.id).select("filePath").first();
if (row?.filePath) {
return await u.oss.getFileUrl(row.filePath);
}
}
return d;
}), }),
); );
}
return { return {
...storyboard, ...item,
filePath: storyboard.filePath && (await u.oss.getFileUrl(storyboard.filePath!)), data: dataWithFilePath,
config: config ? { ...config, data: configDataWithFilePath } : null, };
videos: await Promise.all( }
(videoMap.get(sid) ?? []).map(async (video) => ({ }),
...video, );
filePath: video.filePath ? await u.oss.getFileUrl(video.filePath) : null, //查询视频数据
})), const videos = await u.db("o_video").whereIn("storyboardId", storyboardIds).select("*");
),
const videosList = await Promise.all(
videos.map(async (item) => {
return {
...item,
filePath: item.filePath ? await u.oss.getFileUrl(item.filePath) : null,
}; };
}), }),
); );
//组装数据
const data = storyboardsList.map((storyboard) => {
const config = storyboardConfigsList.find((item) => item?.storyboardId === storyboard.id) || null;
return {
...storyboard,
config,
videos: videosList.filter((video) => video.storyboardId === storyboard.id),
};
});
return res.status(200).send(success(data)); return res.status(200).send(success(data));
}, },
); );

View File

@ -0,0 +1,19 @@
import express from "express";
import u from "@/utils";
import { z } from "zod";
import { success } from "@/lib/responseFormat";
import { validateFields } from "@/middleware/middleware";
const router = express.Router();
export default router.post(
"/",
validateFields({
storyboardId: z.number(),
videoId: z.number(),
}),
async (req, res) => {
const { storyboardId, videoId } = req.body;
await u.db("o_videoConfig").where("storyboardId", storyboardId).update({ videoId, updateTime: Date.now() });
res.status(200).send(success({ message: "选择确认成功" }));
},
);

View File

@ -0,0 +1,19 @@
import express from "express";
import u from "@/utils";
import { z } from "zod";
import { success } from "@/lib/responseFormat";
import { validateFields } from "@/middleware/middleware";
const router = express.Router();
export default router.post(
"/",
validateFields({
videoId: z.number(),
}),
async (req, res) => {
const { videoId } = req.body;
await u.db("o_video").where("id", videoId).delete();
await u.db("o_videoConfig").where("videoId", videoId).update({ videoId: null, updateTime: Date.now() });
res.status(200).send(success({ message: "视频删除成功" }));
},
);

View File

@ -31,13 +31,14 @@ export default router.post(
const { scriptId, projectId, storyboardId, prompt, data, model, duration, resolution, audio, mode } = req.body; const { scriptId, projectId, storyboardId, prompt, data, model, duration, resolution, audio, mode } = req.body;
const videoPath = `/${projectId}/video/${uuidv4()}.mp4`; //视频保存路径 const videoPath = `/${projectId}/video/${uuidv4()}.mp4`; //视频保存路径
//新增 //新增
const [videoId] = await u.db("o_video").insert({ const videoData = {
filePath: videoPath, filePath: videoPath,
time: Date.now(), time: Date.now(),
state: "生成中", state: "生成中",
scriptId, scriptId,
storyboardId, storyboardId,
}); };
const [videoId] = await u.db("o_video").insert(videoData);
//查询分镜是否已有配置 //查询分镜是否已有配置
const config = await u.db("o_videoConfig").where({ storyboardId }).first(); const config = await u.db("o_videoConfig").where({ storyboardId }).first();
//保存配置 //保存配置
@ -85,7 +86,8 @@ export default router.post(
return await u.oss.getImageBase64(item); return await u.oss.getImageBase64(item);
}), }),
); );
//开始生成 res.status(200).send(success(videoId));
(async () => {
try { try {
const relatedObjects = { const relatedObjects = {
id: storyboardId, id: storyboardId,
@ -116,7 +118,6 @@ export default router.post(
}); });
await aiVideo.save(videoPath); await aiVideo.save(videoPath);
await u.db("o_video").where("id", videoId).update({ state: "生成成功" }); await u.db("o_video").where("id", videoId).update({ state: "生成成功" });
res.status(200).send(success({ videoId, message: "视频生成成功" }));
} catch (error: any) { } catch (error: any) {
await u await u
.db("o_video") .db("o_video")
@ -125,7 +126,7 @@ export default router.post(
state: "生成失败", state: "生成失败",
errorReason: error instanceof Error ? error.message : "未知错误", errorReason: error instanceof Error ? error.message : "未知错误",
}); });
res.status(500).send({ error: "视频生成失败" });
} }
})();
}, },
); );

View File

@ -0,0 +1,19 @@
import express from "express";
import u from "@/utils";
import { z } from "zod";
import { success } from "@/lib/responseFormat";
import { validateFields } from "@/middleware/middleware";
const router = express.Router();
export default router.post(
"/",
validateFields({
scriptId: z.number(),
specifyIds: z.array(z.number()),
}),
async (req, res) => {
const { scriptId, specifyIds } = req.body;
const data = await u.db("o_video").where("scriptId", scriptId).whereIn("id", specifyIds).select("*");
res.status(200).send(success(data));
},
);

View File

@ -1,30 +1,6 @@
// @db-hash 307e7d70184bd3663540410a66e5c54d // @db-hash 481efc498b16db66100914dff7c6aab5
//该文件由脚本自动生成,请勿手动修改 //该文件由脚本自动生成,请勿手动修改
export interface _o_storyboard_old_20260321 {
'createTime'?: number | null;
'detail'?: string | null;
'filePath'?: string | null;
'frameType'?: string | null;
'id'?: number;
'name'?: string | null;
'prompt'?: string | null;
'seconds'?: string | null;
}
export interface _o_storyboard_old_20260321_1 {
'associateAssetsIds'?: string | null;
'camera'?: string | null;
'createTime'?: number | null;
'detail'?: string | null;
'filePath'?: string | null;
'frameType'?: string | null;
'id'?: number;
'name'?: string | null;
'prompt'?: string | null;
'seconds'?: string | null;
'sound'?: string | null;
'TEXT'?: any | null;
}
export interface memories { export interface memories {
'content': string; 'content': string;
'createTime': number; 'createTime': number;
@ -59,6 +35,7 @@ export interface o_assets {
'projectId'?: number | null; 'projectId'?: number | null;
'prompt'?: string | null; 'prompt'?: string | null;
'remark'?: string | null; 'remark'?: string | null;
'scriptId'?: number | null;
'sonId'?: number | null; 'sonId'?: number | null;
'startTime'?: number | null; 'startTime'?: number | null;
'state'?: string | null; 'state'?: string | null;
@ -68,13 +45,6 @@ export interface o_assets2Storyboard {
'assetId'?: number; 'assetId'?: number;
'storyboardId': number; 'storyboardId': number;
} }
export interface o_chatHistory {
'data'?: string | null;
'id'?: number;
'novel'?: string | null;
'projectId'?: number | null;
'type'?: string | null;
}
export interface o_event { export interface o_event {
'createTime'?: number | null; 'createTime'?: number | null;
'detail'?: string | null; 'detail'?: string | null;
@ -95,33 +65,10 @@ export interface o_image {
'assetsId'?: number | null; 'assetsId'?: number | null;
'filePath'?: string | null; 'filePath'?: string | null;
'id'?: number; 'id'?: number;
'projectId'?: number | null; 'model'?: string | null;
'scriptId'?: number | null; 'resolution'?: string | null;
'state'?: string | null; 'state'?: string | null;
'type'?: string | null; 'type'?: string | null;
'videoId'?: number | null;
}
export interface o_model {
'apiKey'?: string | null;
'baseUrl'?: string | null;
'createTime'?: number | null;
'id'?: number;
'index'?: number | null;
'manufacturer'?: string | null;
'model'?: string | null;
'modelType'?: string | null;
'type'?: string | null;
}
export interface o_myTasks {
'describe'?: string | null;
'id'?: number;
'model'?: string | null;
'projectId'?: number | null;
'reason'?: string | null;
'relatedObjects'?: string | null;
'startTime'?: number | null;
'state'?: string | null;
'taskClass'?: string | null;
} }
export interface o_novel { export interface o_novel {
'chapter'?: string | null; 'chapter'?: string | null;
@ -154,15 +101,6 @@ export interface o_project {
'userId'?: number | null; 'userId'?: number | null;
'videoRatio'?: string | null; 'videoRatio'?: string | null;
} }
export interface o_prompts {
'code'?: string | null;
'customValue'?: string | null;
'defaultValue'?: string | null;
'id'?: number;
'name'?: string | null;
'parentCode'?: string | null;
'type'?: string | null;
}
export interface o_script { export interface o_script {
'content'?: string | null; 'content'?: string | null;
'createTime'?: number | null; 'createTime'?: number | null;
@ -170,47 +108,30 @@ export interface o_script {
'name'?: string | null; 'name'?: string | null;
'projectId'?: number | null; 'projectId'?: number | null;
} }
export interface o_scriptAssets {
'assetsId'?: number | null;
'id'?: number;
'scriptId'?: number | null;
}
export interface o_scriptOutline {
'id'?: number;
'outlineId'?: number | null;
'scriptId'?: number | null;
}
export interface o_setting { export interface o_setting {
'key'?: string | null; 'key'?: string | null;
'value'?: string | null; 'value'?: string | null;
} }
export interface o_skills {
'id'?: number;
'name'?: string | null;
'startTime'?: number | null;
}
export interface o_storyboard { export interface o_storyboard {
'associateAssetsIds'?: string | null; 'associateAssetsIds'?: string | null;
'camera'?: string | null; 'camera'?: string | null;
'createTime'?: number | null; 'createTime'?: number | null;
'detail'?: string | null; 'duration'?: string | null;
'filePath'?: string | null; 'filePath'?: string | null;
'frameType'?: string | null; 'frameType'?: string | null;
'id'?: number; 'id'?: number;
'mode'?: string | null;
'model'?: string | null;
'name'?: string | null; 'name'?: string | null;
'prompt'?: string | null; 'prompt'?: string | null;
'seconds'?: string | null; 'resolution'?: string | null;
'scriptId'?: number | null;
'sound'?: string | null; 'sound'?: string | null;
} }
export interface o_storyboardFlow { export interface o_storyboardFlow {
'flowData': string; 'flowData': string;
'id'?: number; 'id'?: number;
'storyboardId': number; 'stroryboardId': number;
}
export interface o_storyboardScript {
'id'?: number;
'scriptId'?: number | null;
'storyboardId'?: number | null;
} }
export interface o_tasks { export interface o_tasks {
'describe'?: string | null; 'describe'?: string | null;
@ -240,66 +161,47 @@ export interface o_vendorConfig {
'version'?: string | null; 'version'?: string | null;
} }
export interface o_video { export interface o_video {
'configId'?: number | null;
'errorReason'?: string | null; 'errorReason'?: string | null;
'filePath'?: string | null; 'filePath'?: string | null;
'firstFrame'?: string | null;
'id'?: number; 'id'?: number;
'model'?: string | null;
'prompt'?: string | null;
'resolution'?: string | null;
'scriptId'?: number | null; 'scriptId'?: number | null;
'state'?: number | null; 'state'?: string | null;
'storyboardImgs'?: string | null; 'storyboardId'?: number | null;
'time'?: number | null; 'time'?: number | null;
} }
export interface o_videoConfig { export interface o_videoConfig {
'aiConfigId'?: number | null; 'audio'?: number | null;
'audioEnabled'?: number | null;
'createTime'?: number | null; 'createTime'?: number | null;
'data'?: string | null;
'duration'?: number | null; 'duration'?: number | null;
'endFrame'?: string | null;
'id'?: number; 'id'?: number;
'images'?: string | null;
'manufacturer'?: string | null;
'mode'?: string | null; 'mode'?: string | null;
'projectId'?: number | null; 'model'?: string | null;
'prompt'?: string | null; 'prompt'?: string | null;
'resolution'?: string | null; 'resolution'?: string | null;
'scriptId'?: number | null; 'storyboardId'?: number | null;
'selectedResultId'?: number | null;
'startFrame'?: string | null;
'updateTime'?: number | null; 'updateTime'?: number | null;
'videoId'?: number | null;
} }
export interface DB { export interface DB {
"_o_storyboard_old_20260321": _o_storyboard_old_20260321;
"_o_storyboard_old_20260321_1": _o_storyboard_old_20260321_1;
"memories": memories; "memories": memories;
"o_agentDeploy": o_agentDeploy; "o_agentDeploy": o_agentDeploy;
"o_artStyle": o_artStyle; "o_artStyle": o_artStyle;
"o_assets": o_assets; "o_assets": o_assets;
"o_assets2Storyboard": o_assets2Storyboard; "o_assets2Storyboard": o_assets2Storyboard;
"o_chatHistory": o_chatHistory;
"o_event": o_event; "o_event": o_event;
"o_eventChapter": o_eventChapter; "o_eventChapter": o_eventChapter;
"o_flowData": o_flowData; "o_flowData": o_flowData;
"o_image": o_image; "o_image": o_image;
"o_model": o_model;
"o_myTasks": o_myTasks;
"o_novel": o_novel; "o_novel": o_novel;
"o_outline": o_outline; "o_outline": o_outline;
"o_outlineNovel": o_outlineNovel; "o_outlineNovel": o_outlineNovel;
"o_project": o_project; "o_project": o_project;
"o_prompts": o_prompts;
"o_script": o_script; "o_script": o_script;
"o_scriptAssets": o_scriptAssets;
"o_scriptOutline": o_scriptOutline;
"o_setting": o_setting; "o_setting": o_setting;
"o_skills": o_skills;
"o_storyboard": o_storyboard; "o_storyboard": o_storyboard;
"o_storyboardFlow": o_storyboardFlow; "o_storyboardFlow": o_storyboardFlow;
"o_storyboardScript": o_storyboardScript;
"o_tasks": o_tasks; "o_tasks": o_tasks;
"o_user": o_user; "o_user": o_user;
"o_vendorConfig": o_vendorConfig; "o_vendorConfig": o_vendorConfig;