fmode-ng 0.0.39 → 0.0.41

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.
Files changed (100) hide show
  1. package/esm2022/fmode-ng.mjs +5 -10
  2. package/esm2022/lib/aigc/agent/agent.prompt.mjs +122 -10
  3. package/esm2022/lib/aigc/agent/index.mjs +2 -10
  4. package/esm2022/lib/aigc/avatar/avatar.module.mjs +45 -10
  5. package/esm2022/lib/aigc/avatar/comp-avatar-particle/avatar.role.mjs +2 -10
  6. package/esm2022/lib/aigc/avatar/comp-avatar-particle/comp-avatar-particle.component.mjs +315 -10
  7. package/esm2022/lib/aigc/avatar/comp-avatar-particle/index.mjs +3 -10
  8. package/esm2022/lib/aigc/avatar/comp-avatar-particle/role-points.class.mjs +57 -10
  9. package/esm2022/lib/aigc/avatar/comp-avatar-role-image/comp-avatar-role-image.component.mjs +97 -10
  10. package/esm2022/lib/aigc/avatar/comp-avatar-role-video/comp-avatar-role-video.component.mjs +104 -10
  11. package/esm2022/lib/aigc/avatar/comp-avatar-talk/comp-avatar-talk.component.mjs +111 -10
  12. package/esm2022/lib/aigc/avatar/index.mjs +8 -10
  13. package/esm2022/lib/aigc/avatar/interface-avatar-role.mjs +2 -10
  14. package/esm2022/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.mjs +166 -8
  15. package/esm2022/lib/aigc/chat/chat-header-area/comp-header-area.component.mjs +36 -10
  16. package/esm2022/lib/aigc/chat/chat-header-area/index.mjs +2 -10
  17. package/esm2022/lib/aigc/chat/chat-list/chat-list.component.mjs +141 -8
  18. package/esm2022/lib/aigc/chat/chat-list/index.mjs +2 -10
  19. package/esm2022/lib/aigc/chat/chat-message-area/comp-message-area.component.mjs +40 -10
  20. package/esm2022/lib/aigc/chat/chat-message-area/index.mjs +2 -10
  21. package/esm2022/lib/aigc/chat/chat-message-card/comp-message-card.component.mjs +92 -10
  22. package/esm2022/lib/aigc/chat/chat-message-card/index.mjs +2 -10
  23. package/esm2022/lib/aigc/chat/chat-modal-input/index.mjs +2 -10
  24. package/esm2022/lib/aigc/chat/chat-modal-input/modal-audio-message/modal-audio-message.component.mjs +207 -8
  25. package/esm2022/lib/aigc/chat/chat-modal-input/modal-input.component.mjs +236 -10
  26. package/esm2022/lib/aigc/chat/chat-panel/chat-panel.component.mjs +137 -10
  27. package/esm2022/lib/aigc/chat/comp-role-prompt/comp-role-prompt.component.mjs +69 -10
  28. package/esm2022/lib/aigc/chat/comp-role-prompt/index.mjs +2 -10
  29. package/esm2022/lib/aigc/chat/index.mjs +8 -10
  30. package/esm2022/lib/aigc/comp-markdown-preview/clipboard.service.mjs +82 -10
  31. package/esm2022/lib/aigc/comp-markdown-preview/markdown-parse.mjs +269 -8
  32. package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.component.mjs +51 -10
  33. package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.module.mjs +24 -10
  34. package/esm2022/lib/aigc/comp-markdown-preview/plugins/md-mathjax/index.mjs +94 -10
  35. package/esm2022/lib/aigc/index.mjs +13 -10
  36. package/esm2022/lib/aigc/service-fmai/fmai.service.mjs +21 -10
  37. package/esm2022/lib/aigc/service-fmai/service-chat/chat-class.mjs +736 -8
  38. package/esm2022/lib/aigc/service-fmai/service-chat/chat.service.mjs +181 -8
  39. package/esm2022/lib/aigc/service-fmai/service-chat/index.mjs +7 -10
  40. package/esm2022/lib/aigc/service-fmai/service-chat/mask-list.mjs +194 -9
  41. package/esm2022/lib/aigc/service-fmai/service-chat/pipes/chat-content.pipe.mjs +27 -10
  42. package/esm2022/lib/aigc/service-fmai/service-chat/pipes/hidexml.pipe.mjs +27 -10
  43. package/esm2022/lib/aigc/service-fmai/service-chat/utilnow.pipe.mjs +68 -10
  44. package/esm2022/lib/aigc/service-fmai/service-imagine/imagine.service.mjs +229 -8
  45. package/esm2022/lib/aigc/service-fmai/service-imagine/index.mjs +2 -10
  46. package/esm2022/lib/aigc/voice/audio.player.mjs +52 -10
  47. package/esm2022/lib/aigc/voice/class-asr.mjs +79 -8
  48. package/esm2022/lib/aigc/voice/fmode-voice.service.mjs +501 -8
  49. package/esm2022/lib/aigc/voice/index.mjs +3 -10
  50. package/esm2022/lib/aigc/voice/lib/pcm2wav.mjs +38 -10
  51. package/esm2022/lib/aigc/voice/lib/resample.mjs +34 -10
  52. package/esm2022/lib/aigc/voice/tts/fmode-tts-class.mjs +233 -8
  53. package/esm2022/lib/aigc/voice/tts/index.mjs +2 -10
  54. package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.component.mjs +190 -10
  55. package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.module.mjs +33 -10
  56. package/esm2022/lib/map/index.mjs +4 -10
  57. package/esm2022/lib/map/map.module.mjs +61 -10
  58. package/esm2022/lib/map/page-loca-scatter/page-loca-scatter.component.mjs +110 -10
  59. package/esm2022/lib/map/page-map.start/page-map.start.component.mjs +98 -8
  60. package/esm2022/lib/map/page-plan-route/page-plan-route.component.mjs +100 -8
  61. package/esm2022/lib/nova-cloud/index.mjs +2 -10
  62. package/esm2022/lib/nova-cloud/nova-cloud.service.mjs +148 -10
  63. package/esm2022/lib/platform/cross.service.mjs +63 -10
  64. package/esm2022/lib/platform/index.mjs +2 -10
  65. package/esm2022/lib/social/index.mjs +2 -10
  66. package/esm2022/lib/social/wechat/wechat-jssdk.service.mjs +236 -8
  67. package/esm2022/lib/storage/comp-hwobs-manager/hwobs-manager.component.mjs +59 -10
  68. package/esm2022/lib/storage/index.mjs +5 -10
  69. package/esm2022/lib/storage/service-hwobs/hwobs.service.mjs +130 -8
  70. package/esm2022/lib/storage/service-upload/index.mjs +2 -10
  71. package/esm2022/lib/storage/service-upload/nova-upload.service.mjs +462 -8
  72. package/esm2022/lib/storage/service-upload/util-file-md5.mjs +28 -10
  73. package/esm2022/lib/storage/storage.module.mjs +41 -10
  74. package/esm2022/lib/user/account/account.service.mjs +221 -10
  75. package/esm2022/lib/user/captcha/captcha.component.mjs +135 -10
  76. package/esm2022/lib/user/comp-user-avatar/comp-user-avatar.component.mjs +62 -10
  77. package/esm2022/lib/user/index.mjs +17 -10
  78. package/esm2022/lib/user/login/auth.guard.mjs +28 -10
  79. package/esm2022/lib/user/login/auth.service.mjs +373 -8
  80. package/esm2022/lib/user/login/login.component.mjs +913 -10
  81. package/esm2022/lib/user/modal-user-login/modal-user-login.component.mjs +273 -10
  82. package/esm2022/lib/user/profile/auth-profile.guard.mjs +27 -10
  83. package/esm2022/lib/user/profile/auth-profile.service.mjs +122 -10
  84. package/esm2022/lib/user/profile/profile-bind/profile-bind.component.mjs +115 -10
  85. package/esm2022/lib/user/profile/profile.module.mjs +57 -10
  86. package/esm2022/lib/user/staff/index.mjs +4 -10
  87. package/esm2022/lib/user/staff/staff.guard.mjs +26 -10
  88. package/esm2022/lib/user/staff/staff.module.mjs +18 -10
  89. package/esm2022/lib/user/staff/staff.service.mjs +85 -10
  90. package/esm2022/lib/user/user-name.pipe.mjs +29 -10
  91. package/esm2022/lib/user/user.module.mjs +106 -10
  92. package/esm2022/lib/video/fm-video/fm-video.component.mjs +67 -10
  93. package/esm2022/lib/video/index.mjs +2 -10
  94. package/esm2022/public-api.mjs +13 -10
  95. package/fesm2022/fmode-ng.mjs +8895 -7
  96. package/fesm2022/fmode-ng.mjs.map +1 -1
  97. package/lib/user/login/auth.service.d.ts +18 -3
  98. package/lib/user/modal-user-login/modal-user-login.component.d.ts +4 -2
  99. package/package.json +1 -1
  100. package/LICENSE.md +0 -8
@@ -1,10 +1,738 @@
1
-
1
+ import { bufferTime, concatMap, Observable, delay, finalize } from "rxjs";
2
+ // var bufferTime:any, concatMap:any, Observable:any, Observer:any,delay:any, finalize:any,Observer:any
3
+ import Parse from "parse";
4
+ import { AgentPrompt } from "../../agent";
5
+ import { FmodeTTS } from "../../voice/tts";
6
+ import { PromptTemplate } from "@langchain/core/prompts";
7
+ // var Parse:any = {}
8
+ // const API_BASE:string = "http://127.0.0.1:7337/api/apig/aigc/gpt"
9
+ const API_BASE = "https://server.fmode.cn/api/apig/aigc/gpt";
10
+ // const API_BASE:string = "https://test.fmode.cn/api/apig/aigc/gpt"
11
+ const agentPrompt = new AgentPrompt();
12
+ const PromptTplTalkSSMLOutputCode = "talk-ssml-output-tpl";
13
+ const PromptTplTalkTextSSMLCode = "talk-text-ssml-tpl";
14
+ export function getMessageContentText(content) {
15
+ let text = "";
16
+ if (typeof content == "string")
17
+ text = content;
18
+ if (typeof content == "object")
19
+ text = content?.find(item => item?.text)?.text || "";
20
+ return text;
21
+ }
22
+ export function getMessageImageUrl(content) {
23
+ if (typeof content == "object")
24
+ return content?.find(item => item?.image_url)?.image_url?.url || "";
25
+ return null;
26
+ }
27
+ /**
28
+ * FmodeChat 聊天对话类
29
+ * @public
30
+ */
31
+ export class FmodeChat {
32
+ showAvatar() {
33
+ this.avatarConfig = this.role?.get("avatarConfig");
34
+ if (this.avatarConfig) {
35
+ this.isAvatarShow = true;
36
+ if (this.avatarConfig?.image) {
37
+ this.avatarConfig.image.waiting = this.avatarConfig.image.waiting || this.role?.get("thumb") || this.role?.get("avatar");
38
+ this.avatarMode = "image";
39
+ }
40
+ if (this.avatarConfig?.video) {
41
+ this.avatarConfig.video.waiting = this.avatarConfig.video.waiting;
42
+ this.avatarMode = "video";
43
+ }
44
+ }
45
+ }
46
+ constructor(sessionId, role, chatSession, chatServ, navCtrl, ncloud, uploadServ) {
47
+ this.ChatSession = Parse.Object.extend("ChatSession");
48
+ this.messageList = [{ role: "system", content: "系统提示:AI仅供参考" }];
49
+ this.latestAIResponse = ``;
50
+ this.userInput = ``;
51
+ this.userImage = ``;
52
+ this.isDirect = false;
53
+ /**
54
+ * 虚拟形象展示状态
55
+ */
56
+ this.isAvatarShow = false;
57
+ this.avatarMode = "";
58
+ /**
59
+ * 预置提示词弹窗是否展示
60
+ */
61
+ this.isPromptModalOpen = false;
62
+ this.isPromptMessageAreaShow = true;
63
+ this.promptList = [];
64
+ this.leftButtons = [
65
+ // 提示 当角色配置预设提示词时 显示
66
+ { title: "灵感", icon: "color-wand-outline", onClick: () => {
67
+ this.isPromptModalOpen = true;
68
+ }, show: () => {
69
+ return this?.promptList?.length;
70
+ } },
71
+ { title: "角色", icon: "people-outline", onClick: () => {
72
+ this.navCtrl?.navigateRoot("/chat/pro/mask");
73
+ }, show: () => { return true; } },
74
+ { title: "呼叫", icon: "call-outline", onClick: () => {
75
+ this.chatServ?.callRole(this.role);
76
+ }, show: () => {
77
+ return this?.role?.get('voiceConfig');
78
+ } },
79
+ ];
80
+ /**
81
+ * 是否开启语音消息模式(单次)
82
+ */
83
+ this.isVoiceInputMode = false;
84
+ this.isTexting = false;
85
+ this.isTalkMode = false;
86
+ this.SSMLRoleVoice = "zh-CN-XiaoxiaoNeural";
87
+ /**
88
+ * 会话Avatar控制
89
+ */
90
+ this.playAnimation = (animName) => {
91
+ console.log(animName);
92
+ return;
93
+ };
94
+ this.welcome = async () => {
95
+ let msglist = this.messageList?.filter(item => item?.role == "assistant");
96
+ if (msglist?.length)
97
+ return; // 已有对话不开场问候
98
+ let user = Parse.User.current();
99
+ let person = await this.loadSelf("Person", "userVerify");
100
+ let profile = await this.loadSelf("Profile", "user");
101
+ let name = user?.get("realname") || profile?.get("name") || person?.get("name") || user?.get("nickname") || user?.get("name");
102
+ // 问候语/首个话题
103
+ let tpl = this.role.get("voiceConfig")?.welcome?.prompt;
104
+ if (!tpl)
105
+ return; // 无模板则返回
106
+ let welcomeContent = await PromptTemplate.fromTemplate(tpl, {
107
+ templateFormat: "mustache"
108
+ }).format({
109
+ name: name,
110
+ timeOfDay: this.getTimeOfDay()
111
+ });
112
+ // let callName = name?`${name},`:"";
113
+ // let callTime = `${this.getTimeOfDay()}好`;
114
+ // let welcomeContent = `${callName}${callTime},期待聆听您的人生故事,想和我聊些什么呢?`;
115
+ // 生成ChatVoice并播放
116
+ let voice = await this.getVoiceByContentText(welcomeContent);
117
+ let message = {
118
+ role: "assistant",
119
+ voice: voice,
120
+ content: welcomeContent,
121
+ complete: true
122
+ };
123
+ this.voiceMap[voice?.id];
124
+ this.playChatVoice(this.voiceMap[voice?.id]);
125
+ this.messageList.push(message);
126
+ };
127
+ this.self = {};
128
+ this.voiceMap = {};
129
+ this.chatServ = chatServ;
130
+ this.role = role;
131
+ this.sessionId = sessionId;
132
+ this.navCtrl = navCtrl;
133
+ this.ncloud = ncloud;
134
+ this.uploadServ = uploadServ;
135
+ if (chatSession?.id) {
136
+ this.chatSession = chatSession;
137
+ this.messageList = this.chatSession.get("messageList");
138
+ this.sessionId = chatSession?.id;
139
+ }
140
+ if (this.role?.id) {
141
+ this.voiceConfig = this.role?.get("voiceConfig");
142
+ if (this.voiceConfig?.autoTalk) {
143
+ this.isTalkMode = true;
144
+ this.isDirect = true;
145
+ }
146
+ }
147
+ }
148
+ getTimeOfDay() {
149
+ const now = new Date();
150
+ const hours = now.getHours();
151
+ if (hours >= 5 && hours < 12) {
152
+ return "早上";
153
+ }
154
+ else if (hours >= 12 && hours < 14) {
155
+ return "中午";
156
+ }
157
+ else if (hours >= 14 && hours < 18) {
158
+ return "下午";
159
+ }
160
+ else {
161
+ return "晚上";
162
+ }
163
+ }
164
+ async loadSelf(className, userKey) {
165
+ if (this.self[className])
166
+ return this.self[className];
167
+ let user = Parse.User.current();
168
+ let query = new Parse.Query(className);
169
+ query.equalTo(userKey, user?.id);
170
+ this.self[className] = await query.first();
171
+ }
2
172
  /**
3
- * @copyright © 未来飞马 © 未来全栈 www.fmode.cn
4
- * 版权所有 © 未来飞马 © 江西脑控科技有限公司 Copyright © Fmode Technology Co., Ltd.
5
- * 保留所有权利 All Rights Reserved.
6
- * /home/ryan/workspace/nova/nova-admin/dist/fmode-ng/esm2022/lib/aigc/service-fmai/service-chat/chat-class.mjs
173
+ * 对话模型提示词
7
174
  */
8
- import{bufferTime,concatMap,Observable,delay,finalize}from"rxjs";import Parse from"parse";import{AgentPrompt}from"../../agent";import{FmodeTTS}from"../../voice/tts";import{PromptTemplate}from"@langchain/core/prompts";const API_BASE="https://server.fmode.cn/api/apig/aigc/gpt",agentPrompt=new AgentPrompt,PromptTplTalkSSMLOutputCode="talk-ssml-output-tpl",PromptTplTalkTextSSMLCode="talk-text-ssml-tpl";export function getMessageContentText(t){let e="";return"string"==typeof t&&(e=t),"object"==typeof t&&(e=t?.find((t=>t?.text))?.text||""),e}export function getMessageImageUrl(t){return"object"==typeof t?t?.find((t=>t?.image_url))?.image_url?.url||"":null}export class FmodeChat{showAvatar(){this.avatarConfig=this.role?.get("avatarConfig"),this.avatarConfig&&(this.isAvatarShow=!0,this.avatarConfig?.image&&(this.avatarConfig.image.waiting=this.avatarConfig.image.waiting||this.role?.get("thumb")||this.role?.get("avatar"),this.avatarMode="image"),this.avatarConfig?.video&&(this.avatarConfig.video.waiting=this.avatarConfig.video.waiting,this.avatarMode="video"))}constructor(t,e,s,i,o,n,a){this.ChatSession=Parse.Object.extend("ChatSession"),this.messageList=[{role:"system",content:"系统提示:AI仅供参考"}],this.latestAIResponse="",this.userInput="",this.userImage="",this.isDirect=!1,this.isAvatarShow=!1,this.avatarMode="",this.isPromptModalOpen=!1,this.isPromptMessageAreaShow=!0,this.promptList=[],this.leftButtons=[{title:"灵感",icon:"color-wand-outline",onClick:()=>{this.isPromptModalOpen=!0},show:()=>this?.promptList?.length},{title:"角色",icon:"people-outline",onClick:()=>{this.navCtrl?.navigateRoot("/chat/pro/mask")},show:()=>!0},{title:"呼叫",icon:"call-outline",onClick:()=>{this.chatServ?.callRole(this.role)},show:()=>this?.role?.get("voiceConfig")}],this.isVoiceInputMode=!1,this.isTexting=!1,this.isTalkMode=!1,this.SSMLRoleVoice="zh-CN-XiaoxiaoNeural",this.playAnimation=t=>{console.log(t)},this.welcome=async()=>{let t=this.messageList?.filter((t=>"assistant"==t?.role));if(t?.length)return;let e=Parse.User.current(),s=await this.loadSelf("Person","userVerify"),i=await this.loadSelf("Profile","user"),o=e?.get("realname")||i?.get("name")||s?.get("name")||e?.get("nickname")||e?.get("name"),n=this.role.get("voiceConfig")?.welcome?.prompt;if(!n)return;let a=await PromptTemplate.fromTemplate(n,{templateFormat:"mustache"}).format({name:o,timeOfDay:this.getTimeOfDay()}),r=await this.getVoiceByContentText(a),l={role:"assistant",voice:r,content:a,complete:!0};this.voiceMap[r?.id],this.playChatVoice(this.voiceMap[r?.id]),this.messageList.push(l)},this.self={},this.voiceMap={},this.chatServ=i,this.role=e,this.sessionId=t,this.navCtrl=o,this.ncloud=n,this.uploadServ=a,s?.id&&(this.chatSession=s,this.messageList=this.chatSession.get("messageList"),this.sessionId=s?.id),this.role?.id&&(this.voiceConfig=this.role?.get("voiceConfig"),this.voiceConfig?.autoTalk&&(this.isTalkMode=!0,this.isDirect=!0))}getTimeOfDay(){const t=(new Date).getHours();return t>=5&&t<12?"早上":t>=12&&t<14?"中午":t>=14&&t<18?"下午":"晚上"}async loadSelf(t,e){if(this.self[t])return this.self[t];let s=Parse.User.current(),i=new Parse.Query(t);i.equalTo(e,s?.id),this.self[t]=await i.first()}async loadTalkSystemPrompt(t){if(!this.isTalkMode)return;if(!t)return;"男"==t?.get("gender")?this.SSMLRoleVoice="zh-CN-YunyeNeural":this.SSMLRoleVoice="zh-CN-XiaoxiaoNeural",this.SSMLRoleVoice=t?.get("voiceConfig")?.voice||this.SSMLRoleVoice;let e=await agentPrompt.getFormatTpl("talk-ssml-output-tpl",{SSMLRoleVoice:this.SSMLRoleVoice}),s=t.get("prompt")||"请你扮演飞码AI的人工智能专家。";s+=e;let i={role:"user",content:s,hidden:!0},o=this.messageList?.map((t=>t?.content)).join();if(o.indexOf(s)>-1)return;let n=this.messageList?.findIndex((t=>"system"==t?.role)),a=n+1;this.messageList.splice(a,0,i)}loadRolePrompt(){let t=this.role?.get("prompt"),e={role:"user",content:t,hidden:!0};if(!t)return;let s=this.messageList?.map((t=>t?.content)).join();if(s.indexOf(t)>-1)return;let i=this.messageList?.findIndex((t=>"system"==t?.role)),o=i+1;this.messageList.splice(o,0,e)}async sendMessage(t="FmodeAiTest测试问题",e,s,i,o){if(this.isPromptMessageAreaShow=!1,this.loadRolePrompt(),e){let s={role:"user",content:[{type:"image_url",image_url:{url:e}},{type:"text",text:t}],complete:!0,createdAt:new Date};o&&(s.voice={id:o?.id}),this.messageList.push({role:"user",content:[{type:"image_url",image_url:{url:e}},{type:"text",text:t}],complete:!0,createdAt:new Date})}else{let e={role:"user",content:t,complete:!0,createdAt:new Date};o&&(e.voice={id:o?.id,duration:o?.duration}),this.messageList.push(e)}let n=new FmodeChatCompletion(this.fixMessageList(this.messageList),{model:this.chatServ?.currentModel?.get("code")||"fmode-4.5-128k"});this.userInput="",this.userImage="";let a=this.isDirect||!1;this.isTalkMode&&(a=!0);let r=n.sendCompletion({isDirect:a,onComplete:s||null}).pipe(finalize((async()=>{if(this.isTalkMode){let t=this.messageList[n.indexOfList]?.content,e=await this.getVoiceByContentText(t,i);i?.onSSMLComplete&&i?.onSSMLComplete(e),this.messageList[n.indexOfList].voice=e,this.playChatVoice(this.voiceMap[e?.id])}this.messageList[n.indexOfList].complete=!0}))).subscribe((t=>{this.messageList[n.indexOfList]=t,this.latestAIResponse=this.getContentText(t?.content);let e=this.chatSession?.get("messageList")?.length;this.messageList?.length>e&&this.saveChatSession(),t?.complete&&(this.saveChatSession(),r.unsubscribe())}))}getVoiceByContentText(t,e,s=!1){let i=this.getContentText(t),o=new(Parse.Object.extend("ChatVoice")),n="";return this.SSMLRoleVoice=this.role?.get("voiceConfig")?.voice||this.SSMLRoleVoice,new Promise((async(t,e)=>{let resolveChatVoice=async()=>{o.set("content",i),o.set("ssml",n),o.set("role","assistant");let e=localStorage.getItem("company");e&&o.set("company",{__type:"Pointer",className:"Company",objectId:e}),Parse.User.current()?.id&&o.set("user",Parse.User.current().toPointer()),this.chatSession?.id&&o.set("session",this.chatSession?.toPointer()),o=await o.save(),this.voiceMap[o?.id]=o,t({id:o?.id})};if(0==s&&(n=`<speak xmlns="http://www.w3.org/2001/10/synthesis" xmlns:mstts="http://www.w3.org/2001/mstts" xmlns:emo="http://www.w3.org/2009/10/emotionml" version="1.0" xml:lang="zh-CN"><voice name="${this.SSMLRoleVoice}">${i}</voice></speak>`,resolveChatVoice()),1==s){let t=await agentPrompt.getFormatTpl("talk-text-ssml-tpl",{content:i,SSMLRoleVoice:this.SSMLRoleVoice});new FmodeChatCompletion(this.fixMessageList([{role:"user",content:t}]),{model:this.chatServ?.currentModel?.get("code")||"fmode-4.5-128k"}).sendCompletion({isDirect:!0}).subscribe((async t=>{t?.complete&&(n=this.getContentText(t?.content),resolveChatVoice())}))}}))}getContentText(t){return"string"==typeof t?t:t?.[0]?.text||""}async initTTS(){let t=await this.ncloud.apig("voice/tts/token",{company:localStorage.getItem("company")});if(console.log(t),t?.token){let e=new FmodeTTS(t,this.uploadServ);this.tts=e}}async playChatVoice(t,e){if(await this.initTTS(),this.tts)try{this.playAnimation("talking"),await this.tts.speakAsync(t?.get("ssml"),t,{onLoaded:t=>{e?.onLoaded&&e?.onLoaded(t)},onStop:()=>{e?.onStop&&e?.onStop(),this.playAnimation("waiting")}})}catch(t){console.error(t)}}async saveChatSession(){if("new"==this.sessionId&&(this.chatSession=new this.ChatSession),this.chatSession.set("title",this.genTitle()),this.chatSession.set("role",this.role?.toPointer()),this.chatSession.set("messageList",this.messageList),this.chatSession.set("user",Parse.User.current()?.toPointer()),this.chatSession=await this.chatSession.save(),this.sessionId=this.chatSession?.id,this.sessionId){let t=`${window.location.origin}/chat/pro/chat/${this.sessionId}`;window.location?.pathname?.indexOf("chat/session")>-1&&(t=`${window.location.origin}/chat/session/chat/${this.sessionId}`),t=this.getInviteUrl(t),window.history.replaceState(null,null,t+window.location.search);let e={sid:this.chatSession?.id,rid:this.role?.id,name:this.role?.get("name"),message:this.chatSession?.get("messageList")?.[this.chatSession?.get("messageList")?.length-1]?.content?.slice(0,20),latest:this.chatSession?.createdAt};this.chatServ?.chatList?.length||(this.chatServ.chatList=[]);let s=this.chatServ?.chatList?.find((t=>t?.sid==e?.sid));s>-1?this.chatServ.chatList[s]=e:this.chatServ?.chatList.unshift(e)}}getInviteUrl(t){let e="?";e=t?.indexOf("?")>-1?"&":"?";let s=Parse.User?.current()?.id;if(-1==t?.indexOf("invite="+s)){if(!s)return t;t+=e+"invite="+s}return t}genTitle(){if(this.title)return this.title;let t=this.messageList.find((t=>"user"==t.role))?.content;return"string"==typeof t&&(this.title=t?.slice(0,15)||""),"object"==typeof t&&(this.title=t?.find((t=>t?.text))?.text||""),this.title}fixMessageList(t){return t.map((t=>({role:t.role,content:t.content})))}nowStr(){let t=new Date;return`${t.getFullYear()}/${t.getMonth()+1}/${t.getDate()} ${t.getHours()}:${t.getMinutes()}:${t.getSeconds()}`}}export class FmodeChatCompletion{constructor(t,e){this.content="",this.contentBuffer=[],this.isCompleted=!1,this.indexOfList=Number(t.length),this.messages=t,this.model=e?.model||"fmode-4.5-128k"}sendCompletion(t={}){t.intTime=t?.intTime||50,t.isDirect=t?.isDirect||!1,t?.isDirect&&(t.intTime=1);let e={messages:this.messages,stream:!0,model:this.model,temperature:.5,presence_penalty:0,frequency_penalty:0};return new Observable((s=>{let i=RequestFmodeChatApi("/v1/chat/completions",e).subscribe((e=>{let o=String(e);if("data: [DONE]"==o&&(this.isCompleted=!0,t?.isDirect&&this.isCompleted&&(s.next({role:"assistant",content:this.content,complete:!0,createdAt:new Date}),i.unsubscribe(),t?.onComplete&&t.onComplete({role:"assistant",content:this.content,complete:!0,createdAt:new Date}),s.complete())),o.indexOf("data: {")>-1){let e=chunkToJson(o),n=e?.choices?.[0]?.delta?.content||"";this.contentBuffer.push(n),t?.isDirect&&(this.content+=n||"",this.isCompleted||s.next({role:"assistant",cid:e?.id,content:this.content,createdAt:new Date})),t?.isDirect||this.contentPusher||(this.contentPusher=setInterval((()=>{this.isCompleted&&0==this.contentBuffer?.length&&(s.next({role:"assistant",cid:e?.id,content:this.content,complete:!0,createdAt:new Date}),i.unsubscribe(),clearInterval(this.contentPusher),s.complete()),this.contentBuffer?.length>=0&&(this.contentBuffer?.length>0&&(this.content+=this.contentBuffer.shift()),s.next({role:"assistant",cid:e?.id,content:this.content,createdAt:new Date}))}),t?.intTime))}}))})).pipe(bufferTime(100),concatMap((t=>t)),delay(200))}}function chunkToJson(t){let e;try{e=JSON.parse(t.replaceAll("data: ",""))}catch(t){console.error(t)}return e||{}}function RequestFmodeChatApi(t,e,s="POST"){return new Observable((i=>{let o=API_BASE+t,n=`Bearer ${Parse.User.current()?.getSessionToken()||localStorage.getItem("FMODE_AI_TOKEN")}`;return e.token=n,e&&(e=JSON.stringify(e)),fetch(o,{headers:{"Content-Type":"text/plain","Cache-Control":"no-cache"},body:e||null,method:s,credentials:"omit",mode:"cors"}).then((t=>{let e="";{let s=t.body?.getReader();const o=new TextDecoder;let n=new ReadableStream({start(t){!function read(){s.read().then((({done:e,value:s})=>{if(e)return t.close(),void i.complete();t.enqueue(s),read()}))}()}}).getReader();n.read().then((function processStream({done:t,value:s}){if(t)return;!function processData(t){let s=(e+t).split("\n");if(s?.length>1){for(let t=0;t<s.length-1;t++){let e=s[t];i.next(e)}e=s[s.length-1]}}(o.decode(s)),n.read().then(processStream)}))}})).catch((t=>i.error(t))),()=>{}}))}function JsonToFormData(t){const e=new FormData;return function appendFormData(t,s=""){Array.isArray(t)?t.forEach(((t,e)=>{appendFormData(t,`${s}[${e}]`)})):"object"==typeof t&&null!==t?Object.keys(t).forEach((e=>{const i=s?`${s}.${e}`:e;appendFormData(t[e],i)})):e.append(s,t)}(t),e}
9
- var MODULE_PATH_NEED = `6K+l5paH5Lu25piv5pys6aG555uu55qE5LiA6YOo5YiGIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBDb21wb25lbnRzIGluIEZtb2RlIEluYy4KICAgIOeJiOadg+aJgOaciSDCqSDmnKrmnaXpo57pqawgwqkg5rGf6KW/6ISR5o6n56eR5oqA5pyJ6ZmQ5YWs5Y+4IENvcHlyaWdodCDCqSBGbW9kZSBUZWNobm9sb2d5IENvLiwgTHRkLgogICAg5L+d55WZ5omA5pyJ5p2D5YipIEFsbCBSaWdodHMgUmVzZXJ2ZWQuCiAgICDkuKXnpoHlnKjmnKrnu4/mjojmnYPnmoTmg4XlhrXkuIvvvIzpgJrov4fku7vkvZXlqpLku4vlpI3liLbmraTmlofku7YgVW5hdXRob3JpemVkIGNvcHlpbmcgb2YgdGhpcyBmaWxlLCB2aWEgYW55IG1lZGl1bSBpcyBzdHJpY3RseSBwcm9oaWJpdGVkCiAgICDor6Xmlofku7bmmK/kuJPmnInnmoTmnLrlr4bmlofku7YgUHJvcHJpZXRhcnkgYW5kIGNvbmZpZGVudGlhbAogICAKICAgIENvcHlyaWdodCAyMDIxLW5vdyBGbW9kZSBJbmMuIHN1cHBvcnRAZm1vZGUuY24uIDE4NjA3MDA3MDczLgogICAg5L+d55WZ5omA5pyJ5p2D5YipIEFsbCByaWdodHMgcmVzZXJ2ZWQuCgogICAgUEFUSDovaG9tZS9yeWFuL3dvcmtzcGFjZS9ub3ZhL25vdmEtYWRtaW4vZGlzdC9mbW9kZS1uZy9lc20yMDIyL2xpYi9haWdjL3NlcnZpY2UtZm1haS9zZXJ2aWNlLWNoYXQvY2hhdC1jbGFzcy5tanM=`
10
-
175
+ async loadTalkSystemPrompt(role) {
176
+ if (!this.isTalkMode)
177
+ return;
178
+ if (!role)
179
+ return;
180
+ // 加载声音模型:默认为晓晓
181
+ if (role?.get('gender') == '男') {
182
+ this.SSMLRoleVoice = "zh-CN-YunyeNeural";
183
+ }
184
+ else {
185
+ this.SSMLRoleVoice = "zh-CN-XiaoxiaoNeural";
186
+ }
187
+ this.SSMLRoleVoice = role?.get("voiceConfig")?.voice || this.SSMLRoleVoice;
188
+ let SSMLPromptTemplate = await agentPrompt.getFormatTpl(PromptTplTalkSSMLOutputCode, {
189
+ SSMLRoleVoice: this.SSMLRoleVoice, // SSML
190
+ });
191
+ let prompt = role.get("prompt") || "请你扮演飞码AI的人工智能专家。";
192
+ prompt += SSMLPromptTemplate;
193
+ let promptMsg = { role: "user", content: prompt, hidden: true };
194
+ // 查重
195
+ let content = this.messageList?.map(item => item?.content).join();
196
+ if (content.indexOf(prompt) > -1) {
197
+ // 提示词已经存在
198
+ return;
199
+ }
200
+ // 补全提示词
201
+ let systemIndex = this.messageList?.findIndex(item => item?.role == "system");
202
+ let insertIndex = systemIndex + 1;
203
+ this.messageList.splice(insertIndex, 0, promptMsg);
204
+ return;
205
+ }
206
+ /**
207
+ * 角色提示词
208
+ * @returns
209
+ */
210
+ loadRolePrompt() {
211
+ // 角色提示
212
+ let prompt = this.role?.get("prompt");
213
+ let promptMsg = { role: "user", content: prompt, hidden: true };
214
+ if (!prompt)
215
+ return; // 无提示词无需添加
216
+ // 内容检查
217
+ let content = this.messageList?.map(item => item?.content).join();
218
+ if (content.indexOf(prompt) > -1) {
219
+ // 提示词已经存在
220
+ return;
221
+ }
222
+ // 补全提示词
223
+ let systemIndex = this.messageList?.findIndex(item => item?.role == "system");
224
+ let insertIndex = systemIndex + 1;
225
+ this.messageList.splice(insertIndex, 0, promptMsg);
226
+ // console.log(this.messageList)
227
+ }
228
+ /**
229
+ * 发送消息
230
+ * @param message
231
+ * @param imageUrl
232
+ */
233
+ async sendMessage(message = "FmodeAiTest测试问题", imageUrl, onComplete, eventMap, voice) {
234
+ // 为消息列表补全提示词
235
+ // await this.loadTalkSystemPrompt(this.role);
236
+ this.isPromptMessageAreaShow = false; // 发送第一条消息后,关闭提示看板
237
+ this.loadRolePrompt();
238
+ // 用户输入消息,添加到历史消息清单中
239
+ if (!imageUrl) { // 纯文本
240
+ // console.log("纯文本")
241
+ let msg = {
242
+ role: "user",
243
+ content: message,
244
+ complete: true,
245
+ createdAt: new Date()
246
+ };
247
+ if (voice) {
248
+ msg.voice = { id: voice?.id, duration: voice?.duration };
249
+ }
250
+ this.messageList.push(msg);
251
+ }
252
+ else { // 带图片
253
+ let msg = {
254
+ "role": "user",
255
+ "content": [
256
+ {
257
+ "type": "image_url",
258
+ "image_url": { "url": imageUrl },
259
+ },
260
+ {
261
+ "type": "text",
262
+ "text": message
263
+ },
264
+ ],
265
+ complete: true,
266
+ createdAt: new Date()
267
+ };
268
+ if (voice) {
269
+ msg.voice = { id: voice?.id };
270
+ }
271
+ this.messageList.push({
272
+ "role": "user",
273
+ "content": [
274
+ {
275
+ "type": "image_url",
276
+ "image_url": { "url": imageUrl },
277
+ },
278
+ {
279
+ "type": "text",
280
+ "text": message
281
+ },
282
+ ],
283
+ complete: true,
284
+ createdAt: new Date()
285
+ });
286
+ }
287
+ // 创建并发起一条新的消息补全
288
+ // console.log("send",this.messageList)
289
+ let completion = new FmodeChatCompletion(this.fixMessageList(this.messageList), {
290
+ model: this.chatServ?.currentModel?.get("code") || "fmode-4.5-128k"
291
+ });
292
+ this.userInput = "";
293
+ this.userImage = "";
294
+ // console.log(this.chatServ?.currentModel?.toJSON())
295
+ // 持续更新事件推送的消息体内容至消息列表
296
+ let isDirect = this.isDirect || false;
297
+ if (this.isTalkMode) {
298
+ isDirect = true;
299
+ }
300
+ let send$ = completion.sendCompletion({
301
+ isDirect: isDirect,
302
+ onComplete: onComplete || null
303
+ }).pipe(finalize(async () => {
304
+ if (this.isTalkMode) {
305
+ let content = this.messageList[completion.indexOfList]?.content;
306
+ let voice = await this.getVoiceByContentText(content, eventMap);
307
+ eventMap?.onSSMLComplete && eventMap?.onSSMLComplete(voice);
308
+ this.messageList[completion.indexOfList].voice = voice;
309
+ this.playChatVoice(this.voiceMap[voice?.id]);
310
+ }
311
+ this.messageList[completion.indexOfList].complete = true;
312
+ })).subscribe(message => {
313
+ this.messageList[completion.indexOfList] = message;
314
+ this.latestAIResponse = this.getContentText(message?.content);
315
+ let savedList = this.chatSession?.get("messageList")?.length;
316
+ // 生命周期:会话创建后,有新消息时,创建保存会话
317
+ if (this.messageList?.length > savedList) {
318
+ // console.log("cycle新会话")
319
+ this.saveChatSession();
320
+ }
321
+ if (message?.complete) {
322
+ // 生命周期:消息发送完成后,保存聊天记录
323
+ this.saveChatSession();
324
+ send$.unsubscribe();
325
+ }
326
+ // console.log(message)
327
+ });
328
+ }
329
+ getVoiceByContentText(content, eventMap, promptEnabled = false) {
330
+ let contentText = this.getContentText(content);
331
+ let ChatVoice = Parse.Object.extend("ChatVoice");
332
+ let chatVoice = new ChatVoice();
333
+ let contentSSML = ``;
334
+ this.SSMLRoleVoice = this.role?.get("voiceConfig")?.voice || this.SSMLRoleVoice;
335
+ return new Promise(async (resolve, reject) => {
336
+ let resolveChatVoice = async () => {
337
+ chatVoice.set("content", contentText);
338
+ chatVoice.set("ssml", contentSSML);
339
+ chatVoice.set("role", "assistant");
340
+ let company = localStorage.getItem("company");
341
+ company && chatVoice.set("company", { __type: "Pointer", className: "Company", objectId: company });
342
+ Parse.User.current()?.id && chatVoice.set("user", Parse.User.current().toPointer());
343
+ this.chatSession?.id && chatVoice.set("session", this.chatSession?.toPointer());
344
+ chatVoice = await chatVoice.save();
345
+ this.voiceMap[chatVoice?.id] = chatVoice;
346
+ resolve({
347
+ id: chatVoice?.id,
348
+ });
349
+ };
350
+ /**
351
+ * 方法一:高级语音直接读文本,速度快,但细节情绪标记不足。
352
+ */
353
+ if (promptEnabled == false) {
354
+ contentSSML = `<speak xmlns="http://www.w3.org/2001/10/synthesis" xmlns:mstts="http://www.w3.org/2001/mstts" xmlns:emo="http://www.w3.org/2009/10/emotionml" version="1.0" xml:lang="zh-CN"><voice name="${this.SSMLRoleVoice}">${contentText}</voice></speak>`;
355
+ resolveChatVoice();
356
+ }
357
+ /**
358
+ * promptEnabled == true
359
+ * 方法二:通过大模型再次拼接SSML脚本,实现更优质的语音标记,但是生成时间太慢
360
+ */
361
+ if (promptEnabled == true) {
362
+ // 拼接Prompt
363
+ let TextSSMLPrompt = await agentPrompt.getFormatTpl(PromptTplTalkTextSSMLCode, {
364
+ content: contentText, // 文本内容
365
+ SSMLRoleVoice: this.SSMLRoleVoice, // SSML 演说者
366
+ });
367
+ // 生成SSML
368
+ let completion = new FmodeChatCompletion(this.fixMessageList([{
369
+ role: "user",
370
+ content: TextSSMLPrompt
371
+ }]), {
372
+ model: this.chatServ?.currentModel?.get("code") || "fmode-4.5-128k"
373
+ });
374
+ let send$ = completion.sendCompletion({
375
+ isDirect: true,
376
+ }).subscribe(async (message) => {
377
+ if (message?.complete) {
378
+ contentSSML = this.getContentText(message?.content);
379
+ resolveChatVoice();
380
+ }
381
+ });
382
+ }
383
+ });
384
+ }
385
+ getContentText(content) {
386
+ if (typeof content == "string") {
387
+ return content;
388
+ }
389
+ else {
390
+ return content?.[0]?.text || ``;
391
+ }
392
+ }
393
+ async initTTS() {
394
+ // if(this.tts) return // 待明确sts有效期和次数进行优化,避免每次重复获取
395
+ let config = await this.ncloud.apig("voice/tts/token", { company: localStorage.getItem("company") });
396
+ // 有TTS资源,使用情绪合成
397
+ console.log(config);
398
+ if (config?.token) {
399
+ let tts = new FmodeTTS(config, this.uploadServ);
400
+ this.tts = tts;
401
+ }
402
+ }
403
+ async playChatVoice(voice, eventMap) {
404
+ await this.initTTS();
405
+ // console.log(this.tts)
406
+ if (this.tts) {
407
+ try {
408
+ // console.log(textOrSSML)
409
+ // 完整的消息,通过TTS合成进行讲话
410
+ this.playAnimation("talking"); // Talking动画,暂时用wating代替
411
+ await this.tts.speakAsync(voice?.get("ssml"), voice, {
412
+ onLoaded: (audio) => {
413
+ eventMap?.onLoaded && eventMap?.onLoaded(audio); // 事件传递
414
+ },
415
+ onStop: () => {
416
+ eventMap?.onStop && eventMap?.onStop(); // 事件传递
417
+ this.playAnimation("waiting"); // Talking动画,暂时用wating代替
418
+ }
419
+ });
420
+ }
421
+ catch (ttserr) {
422
+ console.error(ttserr);
423
+ }
424
+ }
425
+ // 无TSS资源,调用Edge Speech
426
+ }
427
+ /**
428
+ * 保存单次会话
429
+ */
430
+ async saveChatSession() {
431
+ if (this.sessionId == "new") {
432
+ this.chatSession = new this.ChatSession();
433
+ }
434
+ this.chatSession.set("title", this.genTitle());
435
+ this.chatSession.set("role", this.role?.toPointer());
436
+ this.chatSession.set("messageList", this.messageList);
437
+ this.chatSession.set("user", Parse.User.current()?.toPointer());
438
+ this.chatSession = await this.chatSession.save();
439
+ this.sessionId = this.chatSession?.id;
440
+ if (this.sessionId) {
441
+ // 修改URL地址为sessionId,方便分享或切换 角色页面 => 会话页面
442
+ let newHref = `${window.location.origin}/chat/pro/chat/${this.sessionId}`;
443
+ if (window.location?.pathname?.indexOf("chat/session") > -1) {
444
+ newHref = `${window.location.origin}/chat/session/chat/${this.sessionId}`;
445
+ }
446
+ newHref = this.getInviteUrl(newHref);
447
+ window.history.replaceState(null, null, newHref + window.location.search);
448
+ // 修改最新条chatList数据
449
+ let newChat = {
450
+ sid: this.chatSession?.id,
451
+ rid: this.role?.id,
452
+ name: this.role?.get('name'),
453
+ message: this.chatSession?.get('messageList')?.[this.chatSession?.get('messageList')?.length - 1]?.content?.slice(0, 20),
454
+ latest: this.chatSession?.createdAt
455
+ };
456
+ if (!this.chatServ?.chatList?.length)
457
+ this.chatServ.chatList = [];
458
+ let index = this.chatServ?.chatList?.find(item => item?.sid == newChat?.sid);
459
+ if (index > -1) {
460
+ this.chatServ.chatList[index] = newChat;
461
+ }
462
+ else {
463
+ this.chatServ?.chatList.unshift(newChat);
464
+ }
465
+ }
466
+ }
467
+ getInviteUrl(url) {
468
+ // 判断是否有参数
469
+ let connectChar = "?";
470
+ if (url?.indexOf("?") > -1) {
471
+ connectChar = "&";
472
+ }
473
+ else {
474
+ connectChar = "?";
475
+ }
476
+ // 附加invite参数
477
+ let id = Parse.User?.current()?.id;
478
+ if (url?.indexOf("invite=" + id) == -1) {
479
+ if (!id)
480
+ return url;
481
+ url += connectChar + 'invite=' + id;
482
+ }
483
+ return url;
484
+ }
485
+ // 根据聊天内容及问题,生成标题
486
+ genTitle() {
487
+ if (this.title)
488
+ return this.title;
489
+ let content = this.messageList.find(item => item.role == "user")?.content;
490
+ if (typeof content == "string") { // 截图文本内容文字部分
491
+ this.title = content?.slice(0, 15) || "";
492
+ }
493
+ if (typeof content == "object") { // 截图复合内容文字部分
494
+ this.title = content?.find(item => item?.text)?.text || "";
495
+ }
496
+ return this.title;
497
+ }
498
+ fixMessageList(messages) {
499
+ return messages.map(msg => { return { role: msg.role, content: msg.content }; });
500
+ }
501
+ nowStr() {
502
+ let now = new Date();
503
+ return `${now.getFullYear()}/${now.getMonth() + 1}/${now.getDate()} ${now.getHours()}:${now.getMinutes()}:${now.getSeconds()}`;
504
+ }
505
+ }
506
+ /**
507
+ * FmodeChatCompletion 文本补全类
508
+ * @public
509
+ */
510
+ export class FmodeChatCompletion {
511
+ constructor(messages, options) {
512
+ this.content = ""; // 本次接收消息结果
513
+ this.contentBuffer = [];
514
+ this.isCompleted = false;
515
+ this.indexOfList = Number(messages.length);
516
+ this.messages = messages;
517
+ this.model = options?.model || "fmode-4.5-128k";
518
+ }
519
+ /**
520
+ * @param options
521
+ * @param options.isDirect 是否不等待逐字获取,直接完成内容推送
522
+ * @param options.intTime 是否不等待逐字获取,直接完成内容推送
523
+ * @returns
524
+ */
525
+ sendCompletion(options = {}) {
526
+ options.intTime = options?.intTime || 50; // 按毫秒逐字推送
527
+ options.isDirect = options?.isDirect || false;
528
+ if (options?.isDirect)
529
+ options.intTime = 1;
530
+ let that = this;
531
+ let opts = {
532
+ "messages": this.messages,
533
+ "stream": true,
534
+ "model": this.model,
535
+ "temperature": 0.5,
536
+ "presence_penalty": 0,
537
+ "frequency_penalty": 0
538
+ };
539
+ // console.log(opts)
540
+ let $messageReceiver = new Observable((observer) => {
541
+ let subscription = RequestFmodeChatApi("/v1/chat/completions", opts)
542
+ .subscribe(data => {
543
+ // Handle each chunk of data
544
+ /** Chunk文本数据格式如下:
545
+ 正常消息:
546
+ 'data: {"id":"chatcmpl-y2PLKqPDnwAFJIj2L5aqdH5TWK9Yv","object":"chat.completion.chunk","created":1696770162,"model":"gpt-3.5-turbo-0613","choices":[{"index":0,"delta":{"content":"本提示词仅用于测试。"},"finish_reason":null}]}',
547
+ 终止原因:
548
+ 'data: {"id":"chatcmpl-y2PLKqPDnwAFJIj2L5aqdH5TWK9Yv","object":"chat.completion.chunk","created":1696770162,"model":"gpt-3.5-turbo-0613","choices":[{"index":0,"delta":{},"finish_reason":"stop"}]}',
549
+ 结束消息:
550
+ 'data: [DONE]'
551
+ */
552
+ let chunk = String(data);
553
+ // Check if the completion message is received
554
+ // console.log(chunk)
555
+ if (chunk == 'data: [DONE]') {
556
+ this.isCompleted = true; // 标记完成 => 等待interval推送
557
+ // console.log(options?.isDirect,this.isCompleted)
558
+ if (options?.isDirect && this.isCompleted) {
559
+ observer.next({
560
+ role: "assistant",
561
+ // cid:chunkjson?.['id'],
562
+ content: this.content,
563
+ complete: true, // 推送完成
564
+ createdAt: new Date()
565
+ });
566
+ subscription.unsubscribe(); // Unsubscribe when done
567
+ options?.onComplete && options.onComplete({
568
+ role: "assistant",
569
+ // cid:chunkjson?.['id'],
570
+ content: this.content,
571
+ complete: true, // 推送完成
572
+ createdAt: new Date()
573
+ });
574
+ observer.complete();
575
+ }
576
+ }
577
+ // console.log(chunk)
578
+ if (chunk.indexOf("data:\ {") > -1) {
579
+ let chunkjson = chunkToJson(chunk);
580
+ // console.log(chunk)
581
+ // console.log(chunkjson?.choices?.[0]?.delta)
582
+ let words = chunkjson?.choices?.[0]?.delta?.content || "";
583
+ this.contentBuffer.push(words);
584
+ // 消息返回模式:定时器推送,模拟逐字输出
585
+ if (options?.isDirect) {
586
+ this.content += (words || "");
587
+ // 默认累加消息结果
588
+ if (!this.isCompleted) {
589
+ observer.next({
590
+ role: "assistant",
591
+ cid: chunkjson?.['id'],
592
+ content: this.content,
593
+ createdAt: new Date()
594
+ });
595
+ }
596
+ }
597
+ if (!options?.isDirect && !this.contentPusher) {
598
+ this.contentPusher = setInterval(() => {
599
+ if (this.isCompleted && this.contentBuffer?.length == 0) { // 推送完毕,清除计时器
600
+ observer.next({
601
+ role: "assistant",
602
+ cid: chunkjson?.['id'],
603
+ content: this.content,
604
+ complete: true, // 推送完成
605
+ createdAt: new Date()
606
+ });
607
+ subscription.unsubscribe(); // Unsubscribe when done
608
+ clearInterval(this.contentPusher);
609
+ observer.complete();
610
+ }
611
+ if (this.contentBuffer?.length >= 0) {
612
+ if (this.contentBuffer?.length > 0) {
613
+ this.content += this.contentBuffer.shift();
614
+ }
615
+ observer.next({
616
+ role: "assistant",
617
+ cid: chunkjson?.['id'],
618
+ content: this.content,
619
+ createdAt: new Date()
620
+ });
621
+ }
622
+ }, options?.intTime);
623
+ }
624
+ // console.log(this.content)
625
+ }
626
+ });
627
+ });
628
+ return $messageReceiver.pipe(bufferTime(100), // 每100ms收集消息
629
+ concatMap(messages => messages), // 使用 concatMap 逐个发送消息
630
+ delay(200) // 延迟200ms输出每条消息
631
+ );
632
+ }
633
+ }
634
+ function chunkToJson(chunk) {
635
+ let chunkjson;
636
+ try {
637
+ chunkjson = JSON.parse(chunk.replaceAll("data:\ ", ""));
638
+ }
639
+ catch (errdj) {
640
+ console.error(errdj);
641
+ }
642
+ return chunkjson || {};
643
+ }
644
+ function RequestFmodeChatApi(apipath, body, method = "POST") {
645
+ return new Observable((observer) => {
646
+ let url = API_BASE + apipath;
647
+ let API_TOKEN = Parse.User.current()?.getSessionToken() || localStorage.getItem("FMODE_AI_TOKEN");
648
+ // 通过body传递token参数,避免no-cors模式下Authoriztion头部无效
649
+ let AUTH_TOKEN = `Bearer ${API_TOKEN}`;
650
+ body.token = AUTH_TOKEN;
651
+ if (body)
652
+ body = JSON.stringify(body);
653
+ fetch(url, {
654
+ "headers": {
655
+ // "Authorization": AUTH_TOKEN,
656
+ "Content-Type": "text/plain",
657
+ "Cache-Control": "no-cache"
658
+ },
659
+ "body": body || null,
660
+ "method": method,
661
+ "credentials": "omit",
662
+ "mode": "cors"
663
+ }).then(response => {
664
+ let isStream = true || response.headers?.get("Content-Type")?.indexOf("text/event-stream") > -1;
665
+ let remainingData = ``;
666
+ function processData(data) {
667
+ let combinedData = remainingData + data;
668
+ let messages = combinedData.split('\n');
669
+ if (messages?.length > 1) { // 至少分割2条消息时进行处理
670
+ // 处理每个完整的消息
671
+ for (let i = 0; i < messages.length - 1; i++) {
672
+ let message = messages[i];
673
+ observer.next(message);
674
+ }
675
+ // 保存最后一个不完整的消息
676
+ remainingData = messages[messages.length - 1];
677
+ }
678
+ }
679
+ if (isStream) {
680
+ let greader = response.body?.getReader();
681
+ const decoder = new TextDecoder();
682
+ let rstream = new ReadableStream({
683
+ start(controller) {
684
+ function read() {
685
+ greader.read().then(({ done, value }) => {
686
+ if (done) {
687
+ controller.close();
688
+ observer.complete(); // Complete the observer when stream processing is done
689
+ return;
690
+ }
691
+ controller.enqueue(value);
692
+ read();
693
+ });
694
+ }
695
+ read();
696
+ }
697
+ });
698
+ let reader = rstream.getReader();
699
+ function processStream({ done, value }) {
700
+ if (done) {
701
+ return;
702
+ }
703
+ let text = decoder.decode(value);
704
+ processData(text); // Emit each chunk of data
705
+ reader.read().then(processStream);
706
+ }
707
+ reader.read().then(processStream);
708
+ }
709
+ })
710
+ .catch(error => observer.error(error)); // Handle any errors
711
+ // Return the subscription logic
712
+ return () => {
713
+ // Clean up logic, if needed
714
+ };
715
+ });
716
+ }
717
+ function JsonToFormData(json) {
718
+ const formData = new FormData();
719
+ function appendFormData(data, path = '') {
720
+ if (Array.isArray(data)) {
721
+ data.forEach((value, index) => {
722
+ appendFormData(value, `${path}[${index}]`);
723
+ });
724
+ }
725
+ else if (typeof data === 'object' && data !== null) {
726
+ Object.keys(data).forEach(key => {
727
+ const newPath = path ? `${path}.${key}` : key;
728
+ appendFormData(data[key], newPath);
729
+ });
730
+ }
731
+ else {
732
+ formData.append(path, data);
733
+ }
734
+ }
735
+ appendFormData(json);
736
+ return formData;
737
+ }
738
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-class.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/service-fmai/service-chat/chat-class.ts"],"names":[],"mappings":"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","sourcesContent":["import { bufferTime, concatMap, Observable, Observer,delay, finalize } from \"rxjs\"\n// var bufferTime:any, concatMap:any, Observable:any, Observer:any,delay:any, finalize:any,Observer:any\nimport Parse from \"parse\";\nimport { AgentPrompt } from \"../../agent\";\nimport { NavController } from \"@ionic/angular\";\nimport { FmodeTTS } from \"../../voice/tts\";\nimport { NovaCloudService } from \"../../../nova-cloud\";\nimport { NovaUploadService } from \"../../../storage/service-upload/nova-upload.service\";\nimport { PromptTemplate } from \"@langchain/core/prompts\";\n// var Parse:any = {}\n// const API_BASE:string = \"http://127.0.0.1:7337/api/apig/aigc/gpt\"\nconst API_BASE:string = \"https://server.fmode.cn/api/apig/aigc/gpt\"\n// const API_BASE:string = \"https://test.fmode.cn/api/apig/aigc/gpt\"\n\nconst agentPrompt = new AgentPrompt();\nconst PromptTplTalkSSMLOutputCode = \"talk-ssml-output-tpl\";\nconst PromptTplTalkTextSSMLCode = \"talk-text-ssml-tpl\";\nexport interface FmodeChatMessageVoice{\n    id:string,\n    duration?:number,\n    ssml?:string,\n    voiceUrl?:string,\n}\nexport interface FmodeChatEventMap {\n    // 文本补全事件\n    onComplete?(FmodeChatMessage):void;\n    // 语音合成事件\n    onSSMLComplete?(FmodeChatMessageVoice):void; // ssml脚本合成\n} \n\nexport interface FmodeChatVoiceConfig{\n    voice:string // 声音源角色代码\n    autoTalk:boolean // 自动语音回复\n    welcome:{\n        enabled:true // 开启欢迎语音\n    }\n}\n\nexport function getMessageContentText(content:any|string|Array<ChatImageContentItem>){\n    let text = \"\"\n    if(typeof content == \"string\") text = content\n    if(typeof content == \"object\") text = content?.find(item=>item?.text)?.text || \"\"\n    return text\n}\nexport function getMessageImageUrl(content:any|string|Array<ChatImageContentItem>){\n    if(typeof content == \"object\") return content?.find(item=>item?.image_url)?.image_url?.url || \"\"\n    return null\n  }\n\nexport interface ChatImageContentItem{type:string,text?:string,image_url?:{url:string}}\n\nexport interface FmodeChatMessage{\n    role:string // user 用户 assistant AI system 系统\n    content:string|Array<ChatImageContentItem> // 消息内容\n    json?:any\n    hidden?:boolean\n    createdAt?:Date // 创建时间\n    complete?:boolean // Stream流式加载完成\n    voice?:FmodeChatMessageVoice\n    /**AI回复消息字段 */\n    cid?:string // AI补全内容，包含cid，指向数据库 chatcmpl-uDfGFOwIHyi4vM9LkB6kVxcg4i1DZ\n}\n\n/**\n * FmodeChat 聊天对话类\n * @public\n */\nexport class FmodeChat{\n\n    title:string\n    sessionId:string\n    ChatSession = Parse.Object.extend(\"ChatSession\")\n    chatSession:Parse.Object\n    role:any\n    messageList:FmodeChatMessage[] = [{role:\"system\",content:\"系统提示：AI仅供参考\"}]\n    latestAIResponse:string|undefined = ``\n    chatServ:any\n    public userInput:string = ``;\n    public userImage:string = ``;\n    isDirect:boolean = false;\n\n\n    /** \n     * 虚拟形象展示状态\n     */\n    isAvatarShow:boolean = false;\n    avatarMode:string = \"\";\n    avatarConfig:any|undefined;\n    showAvatar(){\n        this.avatarConfig = this.role?.get(\"avatarConfig\")\n        if(this.avatarConfig){\n            this.isAvatarShow = true;\n            if(this.avatarConfig?.image){\n                this.avatarConfig.image.waiting = this.avatarConfig.image.waiting || this.role?.get(\"thumb\") || this.role?.get(\"avatar\")\n                this.avatarMode = \"image\"\n            }\n            if(this.avatarConfig?.video){\n                this.avatarConfig.video.waiting = this.avatarConfig.video.waiting\n                this.avatarMode = \"video\"\n            }\n        }\n    }\n\n    /**\n     * 预置提示词弹窗是否展示\n     */\n    isPromptModalOpen:boolean = false;\n    isPromptMessageAreaShow:boolean = true;\n    promptList:any = []\n\n    /**\n     * 输入按钮区域\n     */\n    navCtrl:NavController\n     leftButtons:Array<any> = [\n        // 提示 当角色配置预设提示词时 显示\n        {title:\"灵感\",icon:\"color-wand-outline\",onClick:()=>{\n          this.isPromptModalOpen = true\n        },show:()=>{\n          return this?.promptList?.length\n        }},\n        {title:\"角色\",icon:\"people-outline\",onClick:()=>{\n          this.navCtrl?.navigateRoot(\"/chat/pro/mask\");\n        },show:()=>{return true}},\n        {title:\"呼叫\",icon:\"call-outline\",onClick:()=>{\n            this.chatServ?.callRole(this.role)\n        },show:()=>{\n          return this?.role?.get('voiceConfig');\n        }},\n      ]\n\n    /**\n     * 是否开启语音消息模式（单次）\n     */\n    isVoiceInputMode:boolean = false;\n    isTexting:boolean = false;\n\n    /**\n     * 是否开启实时对话模式（实时）\n     * @desc\n     * 开启ssml system提示词\n     * 开启回答文本除xml显示模式\n     */\n    voiceConfig:FmodeChatVoiceConfig|undefined\n    isTalkMode:boolean = false;\n    SSMLRoleVoice:string = \"zh-CN-XiaoxiaoNeural\";\n\n    ncloud:NovaCloudService\n    uploadServ:NovaUploadService\n\n    constructor(\n        sessionId:string,role?:Parse.Object,chatSession?:Parse.Object,chatServ?:any,\n        navCtrl?:NavController,\n        ncloud?:NovaCloudService,\n        uploadServ?:NovaUploadService,\n        ){\n        this.chatServ = chatServ\n        this.role = role\n        this.sessionId = sessionId\n        this.navCtrl = navCtrl\n        this.ncloud = ncloud\n        this.uploadServ = uploadServ\n        if(chatSession?.id){\n            this.chatSession = chatSession;\n            this.messageList = this.chatSession.get(\"messageList\");\n            this.sessionId = chatSession?.id;\n        }\n        if(this.role?.id){\n            this.voiceConfig = this.role?.get(\"voiceConfig\")\n            if(this.voiceConfig?.autoTalk){\n                this.isTalkMode = true;\n                this.isDirect = true;\n            }\n        }\n    }\n    /**\n     * 会话Avatar控制\n     */\n    playAnimation = (animName:string)=>{\n        console.log(animName)\n        return\n    }\n    welcome = async ()=>{\n        let msglist = this.messageList?.filter(item=>item?.role==\"assistant\");\n        if(msglist?.length) return // 已有对话不开场问候\n\n        let user = Parse.User.current();\n        let person = await this.loadSelf(\"Person\",\"userVerify\")\n        let profile = await this.loadSelf(\"Profile\",\"user\")\n        let name = user?.get(\"realname\") || profile?.get(\"name\") || person?.get(\"name\") || user?.get(\"nickname\") || user?.get(\"name\");\n\n        // 问候语/首个话题\n        let tpl = this.role.get(\"voiceConfig\")?.welcome?.prompt\n        if(!tpl) return // 无模板则返回\n        let welcomeContent = await PromptTemplate.fromTemplate(tpl,{\n            templateFormat:\"mustache\"\n        }).format({\n            name:name,\n            timeOfDay:this.getTimeOfDay()\n        })\n        // let callName = name?`${name}，`:\"\";\n        // let callTime = `${this.getTimeOfDay()}好`;\n        // let welcomeContent = `${callName}${callTime}，期待聆听您的人生故事，想和我聊些什么呢？`;\n\n        // 生成ChatVoice并播放\n        let voice = await this.getVoiceByContentText(welcomeContent);\n        let message = {\n            role:\"assistant\",\n            voice: voice,\n            content:welcomeContent,\n            complete:true\n        }\n        this.voiceMap[voice?.id]\n        this.playChatVoice(this.voiceMap[voice?.id])\n        this.messageList.push(message)\n    }\n    getTimeOfDay() {\n        const now = new Date();\n        const hours = now.getHours();\n        if (hours >= 5 && hours < 12) {\n            return \"早上\";\n        } else if (hours >= 12 && hours < 14) {\n            return \"中午\";\n        } else if (hours >= 14 && hours < 18) {\n            return \"下午\";\n        } else {\n            return \"晚上\";\n        }\n    }\n    self:{\n        Person?:Parse.Object|undefined\n        Profile?:Parse.Object|undefined\n    } = {}\n    async loadSelf(className,userKey){\n        if(this.self[className]) return this.self[className]\n        let user = Parse.User.current();\n        let query = new Parse.Query(className);\n        query.equalTo(userKey,user?.id);\n        this.self[className] = await query.first();\n    }\n\n\n    /**\n     * 对话模型提示词\n     */\n     async loadTalkSystemPrompt(role:Parse.Object){\n        if(!this.isTalkMode) return\n        if(!role) return\n        // 加载声音模型：默认为晓晓\n        if(role?.get('gender')=='男'){\n          this.SSMLRoleVoice = \"zh-CN-YunyeNeural\"\n        }else{\n          this.SSMLRoleVoice = \"zh-CN-XiaoxiaoNeural\"\n        }\n        this.SSMLRoleVoice = role?.get(\"voiceConfig\")?.voice || this.SSMLRoleVoice \n        let SSMLPromptTemplate = await agentPrompt.getFormatTpl(PromptTplTalkSSMLOutputCode,{\n            SSMLRoleVoice: this.SSMLRoleVoice, // SSML \n        })\n\n        let prompt = role.get(\"prompt\") || \"请你扮演飞码AI的人工智能专家。\"\n        prompt += SSMLPromptTemplate;\n        let promptMsg:FmodeChatMessage = {role:\"user\",content:prompt,hidden:true}\n\n        // 查重\n        let content = this.messageList?.map(item=>item?.content).join()\n        if(content.indexOf(prompt)>-1){\n            // 提示词已经存在\n            return\n        }\n        // 补全提示词\n        let systemIndex = this.messageList?.findIndex(item=>item?.role==\"system\");\n        let insertIndex = systemIndex+1\n        this.messageList.splice(insertIndex,0,promptMsg)\n        return \n    }\n    /**\n     * 角色提示词\n     * @returns \n     */\n    loadRolePrompt(){\n        // 角色提示\n        let prompt = this.role?.get(\"prompt\")\n        let promptMsg:FmodeChatMessage = {role:\"user\",content:prompt,hidden:true}\n        if(!prompt) return // 无提示词无需添加\n        // 内容检查\n        let content = this.messageList?.map(item=>item?.content).join()\n        if(content.indexOf(prompt)>-1){\n            // 提示词已经存在\n            return\n        }\n        // 补全提示词\n        let systemIndex = this.messageList?.findIndex(item=>item?.role==\"system\");\n        let insertIndex = systemIndex+1\n        this.messageList.splice(insertIndex,0,promptMsg)\n        // console.log(this.messageList)\n    }\n    /**\n     * 发送消息\n     * @param message \n     * @param imageUrl \n     */\n    async sendMessage(message:string=\"FmodeAiTest测试问题\",imageUrl?:string,onComplete?:Function,eventMap?:FmodeChatEventMap,voice?:FmodeChatMessageVoice){\n        // 为消息列表补全提示词\n        // await this.loadTalkSystemPrompt(this.role);\n        this.isPromptMessageAreaShow = false; // 发送第一条消息后，关闭提示看板\n        this.loadRolePrompt();\n        // 用户输入消息，添加到历史消息清单中\n        if(!imageUrl){ // 纯文本\n            // console.log(\"纯文本\")\n            let msg:FmodeChatMessage = {\n                role:\"user\",\n                content:message,\n                complete:true,\n                createdAt:new Date()\n            }\n            if(voice){\n                msg.voice = {id:voice?.id,duration:voice?.duration}\n            }\n            this.messageList.push(msg)\n        }else{ // 带图片\n            let msg:FmodeChatMessage = {\n                \"role\": \"user\",\n                \"content\": [\n                    {\n                        \"type\": \"image_url\",\n                        \"image_url\": {\"url\":imageUrl},\n                    },\n                    {\n                        \"type\": \"text\", \n                        \"text\": message\n                    },\n                ],\n                complete:true,\n                createdAt:new Date()\n            }\n            if(voice){\n                msg.voice = {id:voice?.id}\n            }\n            this.messageList.push({\n                \"role\": \"user\",\n                \"content\": [\n                    {\n                        \"type\": \"image_url\",\n                        \"image_url\": {\"url\":imageUrl},\n                    },\n                    {\n                        \"type\": \"text\", \n                        \"text\": message\n                    },\n                ],\n                complete:true,\n                createdAt:new Date()\n            })\n        }\n        // 创建并发起一条新的消息补全\n        // console.log(\"send\",this.messageList)\n        let completion = new FmodeChatCompletion(this.fixMessageList(this.messageList),{\n            model:this.chatServ?.currentModel?.get(\"code\") || \"fmode-4.5-128k\"\n        })\n\n        this.userInput = \"\";\n        this.userImage = \"\";\n        // console.log(this.chatServ?.currentModel?.toJSON())\n        // 持续更新事件推送的消息体内容至消息列表\n        let isDirect = this.isDirect || false\n        if(this.isTalkMode){\n            isDirect = true\n        }\n        let send$ = completion.sendCompletion({\n            isDirect:isDirect,\n            onComplete:onComplete||null\n        }).pipe(finalize(async ()=>{ // 管道finalize替代了旧版的(complete)=>{}\n\n            if(this.isTalkMode){\n                let content = this.messageList[completion.indexOfList]?.content;\n                let voice = await this.getVoiceByContentText(content,eventMap);\n                eventMap?.onSSMLComplete&&eventMap?.onSSMLComplete(voice);\n                this.messageList[completion.indexOfList].voice = voice;\n                this.playChatVoice(this.voiceMap[voice?.id])\n            }\n\n            this.messageList[completion.indexOfList].complete = true\n        })).subscribe(message=>{\n            this.messageList[completion.indexOfList] = message;\n            this.latestAIResponse = this.getContentText(message?.content)\n           \n            let savedList = this.chatSession?.get(\"messageList\")?.length;\n                // 生命周期：会话创建后，有新消息时，创建保存会话\n            if(this.messageList?.length > savedList){\n                // console.log(\"cycle新会话\")\n                this.saveChatSession();\n            }\n\n            if(message?.complete){\n                // 生命周期：消息发送完成后，保存聊天记录\n                this.saveChatSession(); \n                send$.unsubscribe();\n            }\n            // console.log(message)\n        })\n    }\n\n    getVoiceByContentText(content:string|ChatImageContentItem[],eventMap?:FmodeChatEventMap,promptEnabled:boolean=false):Promise<FmodeChatMessageVoice>{\n        let contentText = this.getContentText(content);\n        let ChatVoice = Parse.Object.extend(\"ChatVoice\");\n        let chatVoice = new ChatVoice();\n        let contentSSML = ``;\n        this.SSMLRoleVoice = this.role?.get(\"voiceConfig\")?.voice || this.SSMLRoleVoice \n\n        return new Promise(async (resolve,reject)=>{\n            let resolveChatVoice = async ()=>{\n                chatVoice.set(\"content\",contentText);\n                chatVoice.set(\"ssml\",contentSSML);\n                chatVoice.set(\"role\",\"assistant\");\n                let company = localStorage.getItem(\"company\")\n                company&&chatVoice.set(\"company\",{__type:\"Pointer\",className:\"Company\",objectId:company});\n                Parse.User.current()?.id&&chatVoice.set(\"user\",Parse.User.current().toPointer());\n                this.chatSession?.id&&chatVoice.set(\"session\",this.chatSession?.toPointer());\n                chatVoice = await chatVoice.save();\n                this.voiceMap[chatVoice?.id] = chatVoice;\n                resolve({\n                    id:chatVoice?.id,\n                })\n            }\n            /**\n             * 方法一：高级语音直接读文本，速度快，但细节情绪标记不足。\n             */\n             if(promptEnabled==false){\n                contentSSML = `<speak xmlns=\"http://www.w3.org/2001/10/synthesis\" xmlns:mstts=\"http://www.w3.org/2001/mstts\" xmlns:emo=\"http://www.w3.org/2009/10/emotionml\" version=\"1.0\" xml:lang=\"zh-CN\"><voice name=\"${this.SSMLRoleVoice}\">${contentText}</voice></speak>`\n                resolveChatVoice()\n             }\n            /**\n             * promptEnabled == true\n             * 方法二：通过大模型再次拼接SSML脚本，实现更优质的语音标记，但是生成时间太慢\n             */\n            if(promptEnabled==true){\n                // 拼接Prompt\n                let TextSSMLPrompt = await agentPrompt.getFormatTpl(PromptTplTalkTextSSMLCode,{\n                    content:contentText, // 文本内容\n                    SSMLRoleVoice: this.SSMLRoleVoice, // SSML 演说者\n                })\n\n                // 生成SSML\n                let completion = new FmodeChatCompletion(this.fixMessageList([{\n                    role:\"user\",\n                    content:TextSSMLPrompt\n                }]),{\n                    model:this.chatServ?.currentModel?.get(\"code\") || \"fmode-4.5-128k\"\n                })\n                let send$ = completion.sendCompletion({\n                    isDirect:true,\n                }).subscribe(async message=>{\n                    if(message?.complete){\n                        contentSSML = this.getContentText(message?.content)\n                        resolveChatVoice()\n                    }\n                })\n            }\n            \n        })\n    }\n    getContentText(content:string|ChatImageContentItem[]){\n        if(typeof content == \"string\"){\n            return content\n        }else{\n            return content?.[0]?.text || ``\n        }\n    }\n    /**\n     * TTS - 语音合成\n     * \n     */\n    tts:FmodeTTS\n    async initTTS(){\n    // if(this.tts) return // 待明确sts有效期和次数进行优化，避免每次重复获取\n        let config:any = await this.ncloud.apig(\"voice/tts/token\",{company:localStorage.getItem(\"company\")})\n        // 有TTS资源，使用情绪合成\n        console.log(config)\n        if(config?.token){\n        let tts = new FmodeTTS(config,this.uploadServ);\n        this.tts=tts;\n        }\n    }\n    voiceMap:any = {}\n    async playChatVoice(voice:Parse.Object,eventMap?:any){\n        await this.initTTS();\n        // console.log(this.tts)\n        if(this.tts){\n            try{\n                // console.log(textOrSSML)\n                // 完整的消息，通过TTS合成进行讲话\n                this.playAnimation(\"talking\") // Talking动画，暂时用wating代替\n                await this.tts.speakAsync(voice?.get(\"ssml\"),voice,{\n                    onLoaded:(audio:any)=>{\n                        eventMap?.onLoaded&&eventMap?.onLoaded(audio); // 事件传递\n                    },\n                    onStop: ()=>{\n                        eventMap?.onStop&&eventMap?.onStop(); // 事件传递\n                        this.playAnimation(\"waiting\") // Talking动画，暂时用wating代替\n                    }\n                })\n            }catch(ttserr){\n                console.error(ttserr)\n            }\n        }\n\n    // 无TSS资源，调用Edge Speech\n    }\n    /**\n     * 保存单次会话\n     */\n    async saveChatSession(){\n        if(this.sessionId == \"new\"){\n            this.chatSession = new this.ChatSession()\n        }\n\n        this.chatSession.set(\"title\",this.genTitle())\n        this.chatSession.set(\"role\",this.role?.toPointer())\n        this.chatSession.set(\"messageList\",this.messageList)\n        this.chatSession.set(\"user\",Parse.User.current()?.toPointer())\n        this.chatSession = await this.chatSession.save();\n        this.sessionId = this.chatSession?.id\n        if(this.sessionId){ \n            // 修改URL地址为sessionId，方便分享或切换 角色页面 => 会话页面\n            let newHref = `${window.location.origin}/chat/pro/chat/${this.sessionId}`\n            if(window.location?.pathname?.indexOf(\"chat/session\")>-1){\n                newHref = `${window.location.origin}/chat/session/chat/${this.sessionId}`\n            }\n            newHref = this.getInviteUrl(newHref)\n            window.history.replaceState(null, null, newHref+window.location.search);\n            // 修改最新条chatList数据\n            let newChat = {\n                sid:this.chatSession?.id,\n                rid:this.role?.id,\n                name:this.role?.get('name'),\n                message:this.chatSession?.get('messageList')?.[this.chatSession?.get('messageList')?.length-1]?.content?.slice(0,20),\n                latest:this.chatSession?.createdAt\n            }\n            if(!this.chatServ?.chatList?.length) this.chatServ.chatList = []\n            let index = this.chatServ?.chatList?.find(item=>item?.sid==newChat?.sid)\n            if(index>-1){\n                this.chatServ.chatList[index] = newChat;\n            } else{\n                this.chatServ?.chatList.unshift(newChat)\n            }\n\n        }\n    }\n    getInviteUrl(url){\n        // 判断是否有参数\n        let connectChar = \"?\"\n        if(url?.indexOf(\"?\")>-1){\n          connectChar = \"&\"\n        }else{\n          connectChar = \"?\"\n        }\n        // 附加invite参数\n        let id = Parse.User?.current()?.id\n        if(url?.indexOf(\"invite=\"+id)==-1){\n          if(!id) return url\n          url += connectChar+ 'invite=' + id\n        }\n        return url\n      }\n    // 根据聊天内容及问题，生成标题\n    genTitle(){\n        if(this.title) return this.title\n        let content:string|Array<ChatImageContentItem> =  this.messageList.find(item=>item.role==\"user\")?.content\n        if(typeof content==\"string\"){ // 截图文本内容文字部分\n            this.title = content?.slice(0,15) || \"\"\n        }\n        if(typeof content==\"object\"){ // 截图复合内容文字部分\n            this.title = content?.find(item=>item?.text)?.text || \"\"\n        }\n        return this.title\n    }\n    fixMessageList(messages:FmodeChatMessage[]){\n        return messages.map(msg=>{return {role:msg.role,content:msg.content}})\n    }\n\n    nowStr(){\n        let now = new Date();\n        return `${now.getFullYear()}/${now.getMonth()+1}/${now.getDate()} ${now.getHours()}:${now.getMinutes()}:${now.getSeconds()}`\n    }\n    \n}\n\n/**\n * FmodeChatCompletion 文本补全类\n * @public\n */\nexport class FmodeChatCompletion{\n    indexOfList:number\n    model:string\n    messages:FmodeChatMessage[] // 补全前提示词列表\n    content:string = \"\" // 本次接收消息结果\n    contentBuffer:string[] = []\n    contentPusher:any\n    isCompleted:boolean = false;\n    constructor(\n        messages:FmodeChatMessage[],options?:{\n            model?:string\n        }\n    ){\n        this.indexOfList = Number(messages.length)\n        this.messages = messages\n        this.model = options?.model || \"fmode-4.5-128k\"\n    }\n    /**\n     * @param options\n     * @param options.isDirect 是否不等待逐字获取，直接完成内容推送\n     * @param options.intTime 是否不等待逐字获取，直接完成内容推送\n     * @returns \n     */\n    sendCompletion(options:{\n        isDirect?:boolean,\n        intTime?:number,\n        onComplete?:Function\n    }={}):Observable<FmodeChatMessage>{\n        options.intTime = options?.intTime || 50 // 按毫秒逐字推送\n        options.isDirect = options?.isDirect || false\n        if(options?.isDirect) options.intTime = 1\n\n        let that = this;\n        let opts = {\n            \"messages\":this.messages,\n            \"stream\":true,\n            \"model\":this.model,\n            \"temperature\":0.5,\n            \"presence_penalty\":0,\n            \"frequency_penalty\":0\n        }\n        // console.log(opts)\n        let $messageReceiver = new Observable((observer: Observer<FmodeChatMessage>) => {\n            let subscription = RequestFmodeChatApi(\"/v1/chat/completions\", opts)\n            .subscribe(data => {\n                // Handle each chunk of data\n                /** Chunk文本数据格式如下：\n                正常消息：\n                'data: {\"id\":\"chatcmpl-y2PLKqPDnwAFJIj2L5aqdH5TWK9Yv\",\"object\":\"chat.completion.chunk\",\"created\":1696770162,\"model\":\"gpt-3.5-turbo-0613\",\"choices\":[{\"index\":0,\"delta\":{\"content\":\"本提示词仅用于测试。\"},\"finish_reason\":null}]}',\n                终止原因：\n                'data: {\"id\":\"chatcmpl-y2PLKqPDnwAFJIj2L5aqdH5TWK9Yv\",\"object\":\"chat.completion.chunk\",\"created\":1696770162,\"model\":\"gpt-3.5-turbo-0613\",\"choices\":[{\"index\":0,\"delta\":{},\"finish_reason\":\"stop\"}]}',\n                结束消息：\n                'data: [DONE]'\n                */\n                let chunk = String(data);\n                // Check if the completion message is received\n                // console.log(chunk)\n                if (chunk == 'data: [DONE]') { \n                    this.isCompleted = true; // 标记完成 => 等待interval推送\n                    // console.log(options?.isDirect,this.isCompleted)\n                    if(options?.isDirect && this.isCompleted){\n                        observer.next({\n                            role:\"assistant\",\n                            // cid:chunkjson?.['id'],\n                            content:this.content,\n                            complete:true, // 推送完成\n                            createdAt:new Date()\n                        })\n                        subscription.unsubscribe(); // Unsubscribe when done\n                        options?.onComplete&&options.onComplete({\n                            role:\"assistant\",\n                            // cid:chunkjson?.['id'],\n                            content:this.content,\n                            complete:true, // 推送完成\n                            createdAt:new Date()\n                        })\n                        observer.complete();\n                    }\n                }\n                // console.log(chunk)\n                if(chunk.indexOf(\"data:\\ {\")>-1){\n                    let chunkjson = chunkToJson(chunk)\n                    // console.log(chunk)\n                    // console.log(chunkjson?.choices?.[0]?.delta)\n                    let words = chunkjson?.choices?.[0]?.delta?.content || \"\"\n                    this.contentBuffer.push(words);\n\n                    // 消息返回模式：定时器推送，模拟逐字输出\n                    if(options?.isDirect){\n                            this.content += (words || \"\");\n                           \n                            // 默认累加消息结果\n                            if(!this.isCompleted){\n                                observer.next({\n                                    role:\"assistant\",\n                                    cid:chunkjson?.['id'],\n                                    content:this.content,\n                                    createdAt:new Date()\n                                })\n                            }\n                    }\n                    if(!options?.isDirect&&!this.contentPusher){\n                        this.contentPusher = setInterval(()=>{\n                            if(this.isCompleted && this.contentBuffer?.length==0){ // 推送完毕，清除计时器\n                                observer.next({\n                                    role:\"assistant\",\n                                    cid:chunkjson?.['id'],\n                                    content:this.content,\n                                    complete:true, // 推送完成\n                                    createdAt:new Date()\n                                })\n                                subscription.unsubscribe(); // Unsubscribe when done\n                                clearInterval(this.contentPusher)\n                                observer.complete();\n                            }\n                            if(this.contentBuffer?.length>=0){\n                                if(this.contentBuffer?.length>0){\n                                    this.content += this.contentBuffer.shift()\n                                }\n                                observer.next({\n                                    role:\"assistant\",\n                                    cid:chunkjson?.['id'],\n                                    content:this.content,\n                                    createdAt:new Date()\n                                })\n                            }\n                        },options?.intTime)\n                    }\n                    // console.log(this.content)\n                    \n                }\n\n            });\n        })\n        return $messageReceiver.pipe(\n            bufferTime(100), // 每100ms收集消息\n            concatMap(messages => messages), // 使用 concatMap 逐个发送消息\n            delay(200) // 延迟200ms输出每条消息\n        )\n    }\n\n}\n\nfunction chunkToJson(chunk){\n    let chunkjson:any\n    try{\n        chunkjson = JSON.parse(chunk.replaceAll(\"data:\\ \",\"\"));\n    }catch(errdj){\n        console.error(errdj)\n    }\n    return chunkjson || {}\n}\nfunction RequestFmodeChatApi(apipath, body, method = \"POST\") {\n    return new Observable((observer: Observer<any>) => {\n        let url = API_BASE + apipath;\n        let API_TOKEN = Parse.User.current()?.getSessionToken() || localStorage.getItem(\"FMODE_AI_TOKEN\");\n            // 通过body传递token参数，避免no-cors模式下Authoriztion头部无效\n        let AUTH_TOKEN = `Bearer ${API_TOKEN}`\n        body.token = AUTH_TOKEN;\n        if(body) body = JSON.stringify(body)\n        fetch(url, {\n            \"headers\": {\n                // \"Authorization\": AUTH_TOKEN,\n                \"Content-Type\": \"text/plain\",\n                \"Cache-Control\": \"no-cache\"\n            },\n            \"body\": body || null,\n            \"method\": method,\n            \"credentials\":\"omit\",\n            \"mode\": \"cors\"\n        }).then(response => {\n            let isStream = true || response.headers?.get(\"Content-Type\")?.indexOf(\"text/event-stream\") > -1\n            let remainingData = ``;\n\n            function processData(data) {\n                let combinedData = remainingData + data;\n                let messages = combinedData.split('\\n');\n              \n                if(messages?.length>1){ // 至少分割2条消息时进行处理\n                    // 处理每个完整的消息\n                    for (let i = 0; i < messages.length - 1; i++) {\n                        let message = messages[i];\n                        observer.next(message)\n                    }\n\n                    // 保存最后一个不完整的消息\n                    remainingData = messages[messages.length - 1];\n                }\n              }\n\n            if (isStream) {\n                let greader = response.body?.getReader();\n                const decoder = new TextDecoder();\n\n                let rstream = new ReadableStream({\n                    start(controller) {\n                        function read() {\n                            greader.read().then(({ done, value }) => {\n                                if (done) {\n                                controller.close();\n                                observer.complete(); // Complete the observer when stream processing is done\n                                return;\n                                }\n                                controller.enqueue(value);\n                                read();\n                            });\n                        }\n\n                        read();\n                    }\n                });\n\n                let reader = rstream.getReader();\n\n                function processStream({ done, value }) {\n                    if (done) {\n                        return;\n                    }\n\n                    let text = decoder.decode(value)\n                    processData(text) // Emit each chunk of data\n                    reader.read().then(processStream);\n                }\n\n                reader.read().then(processStream);\n            }\n            })\n            .catch(error => observer.error(error)); // Handle any errors\n\n        // Return the subscription logic\n        return () => {\n            // Clean up logic, if needed\n        };\n    });\n  }\n  \n  function JsonToFormData(json) {\n    const formData = new FormData();\n  \n    function appendFormData(data, path = '') {\n      if (Array.isArray(data)) {\n        data.forEach((value, index) => {\n          appendFormData(value, `${path}[${index}]`);\n        });\n      } else if (typeof data === 'object' && data !== null) {\n        Object.keys(data).forEach(key => {\n          const newPath = path ? `${path}.${key}` : key;\n          appendFormData(data[key], newPath);\n        });\n      } else {\n        formData.append(path, data);\n      }\n    }\n  \n    appendFormData(json);\n  \n    return formData;\n  }"]}