mirror of
https://github.com/langgenius/dify.git
synced 2025-07-03 15:18:05 +00:00

Signed-off-by: yihong0618 <zouzou0208@gmail.com> Signed-off-by: -LAN- <laipz8200@outlook.com> Signed-off-by: xhe <xw897002528@gmail.com> Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: takatost <takatost@gmail.com> Co-authored-by: kurokobo <kuro664@gmail.com> Co-authored-by: Novice Lee <novicelee@NoviPro.local> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: AkaraChen <akarachen@outlook.com> Co-authored-by: Yi <yxiaoisme@gmail.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: JzoNg <jzongcode@gmail.com> Co-authored-by: twwu <twwu@dify.ai> Co-authored-by: Hiroshi Fujita <fujita-h@users.noreply.github.com> Co-authored-by: AkaraChen <85140972+AkaraChen@users.noreply.github.com> Co-authored-by: NFish <douxc512@gmail.com> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: 非法操作 <hjlarry@163.com> Co-authored-by: Novice <857526207@qq.com> Co-authored-by: Hiroki Nagai <82458324+nagaihiroki-git@users.noreply.github.com> Co-authored-by: Gen Sato <52241300+halogen22@users.noreply.github.com> Co-authored-by: eux <euxuuu@gmail.com> Co-authored-by: huangzhuo1949 <167434202+huangzhuo1949@users.noreply.github.com> Co-authored-by: huangzhuo <huangzhuo1@xiaomi.com> Co-authored-by: lotsik <lotsik@mail.ru> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: nite-knite <nkCoding@gmail.com> Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: gakkiyomi <gakkiyomi@aliyun.com> Co-authored-by: CN-P5 <heibai2006@gmail.com> Co-authored-by: CN-P5 <heibai2006@qq.com> Co-authored-by: Chuehnone <1897025+chuehnone@users.noreply.github.com> Co-authored-by: yihong <zouzou0208@gmail.com> Co-authored-by: Kevin9703 <51311316+Kevin9703@users.noreply.github.com> Co-authored-by: -LAN- <laipz8200@outlook.com> Co-authored-by: Boris Feld <lothiraldan@gmail.com> Co-authored-by: mbo <himabo@gmail.com> Co-authored-by: mabo <mabo@aeyes.ai> Co-authored-by: Warren Chen <warren.chen830@gmail.com> Co-authored-by: JzoNgKVO <27049666+JzoNgKVO@users.noreply.github.com> Co-authored-by: jiandanfeng <chenjh3@wangsu.com> Co-authored-by: zhu-an <70234959+xhdd123321@users.noreply.github.com> Co-authored-by: zhaoqingyu.1075 <zhaoqingyu.1075@bytedance.com> Co-authored-by: 海狸大師 <86974027+yenslife@users.noreply.github.com> Co-authored-by: Xu Song <xusong.vip@gmail.com> Co-authored-by: rayshaw001 <396301947@163.com> Co-authored-by: Ding Jiatong <dingjiatong@gmail.com> Co-authored-by: Bowen Liang <liangbowen@gf.com.cn> Co-authored-by: JasonVV <jasonwangiii@outlook.com> Co-authored-by: le0zh <newlight@qq.com> Co-authored-by: zhuxinliang <zhuxinliang@didiglobal.com> Co-authored-by: k-zaku <zaku99@outlook.jp> Co-authored-by: luckylhb90 <luckylhb90@gmail.com> Co-authored-by: hobo.l <hobo.l@binance.com> Co-authored-by: jiangbo721 <365065261@qq.com> Co-authored-by: 刘江波 <jiangbo721@163.com> Co-authored-by: Shun Miyazawa <34241526+miya@users.noreply.github.com> Co-authored-by: EricPan <30651140+Egfly@users.noreply.github.com> Co-authored-by: crazywoola <427733928@qq.com> Co-authored-by: sino <sino2322@gmail.com> Co-authored-by: Jhvcc <37662342+Jhvcc@users.noreply.github.com> Co-authored-by: lowell <lowell.hu@zkteco.in> Co-authored-by: Boris Polonsky <BorisPolonsky@users.noreply.github.com> Co-authored-by: Ademílson Tonato <ademilsonft@outlook.com> Co-authored-by: Ademílson Tonato <ademilson.tonato@refurbed.com> Co-authored-by: IWAI, Masaharu <iwaim.sub@gmail.com> Co-authored-by: Yueh-Po Peng (Yabi) <94939112+y10ab1@users.noreply.github.com> Co-authored-by: Jason <ggbbddjm@gmail.com> Co-authored-by: Xin Zhang <sjhpzx@gmail.com> Co-authored-by: yjc980121 <3898524+yjc980121@users.noreply.github.com> Co-authored-by: heyszt <36215648+hieheihei@users.noreply.github.com> Co-authored-by: Abdullah AlOsaimi <osaimiacc@gmail.com> Co-authored-by: Abdullah AlOsaimi <189027247+osaimi@users.noreply.github.com> Co-authored-by: Yingchun Lai <laiyingchun@apache.org> Co-authored-by: Hash Brown <hi@xzd.me> Co-authored-by: zuodongxu <192560071+zuodongxu@users.noreply.github.com> Co-authored-by: Masashi Tomooka <tmokmss@users.noreply.github.com> Co-authored-by: aplio <ryo.091219@gmail.com> Co-authored-by: Obada Khalili <54270856+obadakhalili@users.noreply.github.com> Co-authored-by: Nam Vu <zuzoovn@gmail.com> Co-authored-by: Kei YAMAZAKI <1715090+kei-yamazaki@users.noreply.github.com> Co-authored-by: TechnoHouse <13776377+deephbz@users.noreply.github.com> Co-authored-by: Riddhimaan-Senapati <114703025+Riddhimaan-Senapati@users.noreply.github.com> Co-authored-by: MaFee921 <31881301+2284730142@users.noreply.github.com> Co-authored-by: te-chan <t-nakanome@sakura-is.co.jp> Co-authored-by: HQidea <HQidea@users.noreply.github.com> Co-authored-by: Joshbly <36315710+Joshbly@users.noreply.github.com> Co-authored-by: xhe <xw897002528@gmail.com> Co-authored-by: weiwenyan-dev <154779315+weiwenyan-dev@users.noreply.github.com> Co-authored-by: ex_wenyan.wei <ex_wenyan.wei@tcl.com> Co-authored-by: engchina <12236799+engchina@users.noreply.github.com> Co-authored-by: engchina <atjapan2015@gmail.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: 呆萌闷油瓶 <253605712@qq.com> Co-authored-by: Kemal <kemalmeler@outlook.com> Co-authored-by: Lazy_Frog <4590648+lazyFrogLOL@users.noreply.github.com> Co-authored-by: Yi Xiao <54782454+YIXIAO0@users.noreply.github.com> Co-authored-by: Steven sun <98230804+Tuyohai@users.noreply.github.com> Co-authored-by: steven <sunzwj@digitalchina.com> Co-authored-by: Kalo Chin <91766386+fdb02983rhy@users.noreply.github.com> Co-authored-by: Katy Tao <34019945+KatyTao@users.noreply.github.com> Co-authored-by: depy <42985524+h4ckdepy@users.noreply.github.com> Co-authored-by: 胡春东 <gycm520@gmail.com> Co-authored-by: Junjie.M <118170653@qq.com> Co-authored-by: MuYu <mr.muzea@gmail.com> Co-authored-by: Naoki Takashima <39912547+takatea@users.noreply.github.com> Co-authored-by: Summer-Gu <37869445+gubinjie@users.noreply.github.com> Co-authored-by: Fei He <droxer.he@gmail.com> Co-authored-by: ybalbert001 <120714773+ybalbert001@users.noreply.github.com> Co-authored-by: Yuanbo Li <ybalbert@amazon.com> Co-authored-by: douxc <7553076+douxc@users.noreply.github.com> Co-authored-by: liuzhenghua <1090179900@qq.com> Co-authored-by: Wu Jiayang <62842862+Wu-Jiayang@users.noreply.github.com> Co-authored-by: Your Name <you@example.com> Co-authored-by: kimjion <45935338+kimjion@users.noreply.github.com> Co-authored-by: AugNSo <song.tiankai@icloud.com> Co-authored-by: llinvokerl <38915183+llinvokerl@users.noreply.github.com> Co-authored-by: liusurong.lsr <liusurong.lsr@alibaba-inc.com> Co-authored-by: Vasu Negi <vasu-negi@users.noreply.github.com> Co-authored-by: Hundredwz <1808096180@qq.com> Co-authored-by: Xiyuan Chen <52963600+GareArc@users.noreply.github.com>
537 lines
19 KiB
TypeScript
537 lines
19 KiB
TypeScript
import { API_PREFIX, IS_CE_EDITION, PUBLIC_API_PREFIX } from '@/config'
|
|
import { refreshAccessTokenOrRelogin } from './refresh-token'
|
|
import Toast from '@/app/components/base/toast'
|
|
import type { AnnotationReply, MessageEnd, MessageReplace, ThoughtItem } from '@/app/components/base/chat/chat/type'
|
|
import type { VisionFile } from '@/types/app'
|
|
import type {
|
|
AgentLogResponse,
|
|
IterationFinishedResponse,
|
|
IterationNextResponse,
|
|
IterationStartedResponse,
|
|
NodeFinishedResponse,
|
|
NodeStartedResponse,
|
|
ParallelBranchFinishedResponse,
|
|
ParallelBranchStartedResponse,
|
|
TextChunkResponse,
|
|
TextReplaceResponse,
|
|
WorkflowFinishedResponse,
|
|
WorkflowStartedResponse,
|
|
} from '@/types/workflow'
|
|
import { removeAccessToken } from '@/app/components/share/utils'
|
|
import type { FetchOptionType, ResponseError } from './fetch'
|
|
import { ContentType, base, baseOptions, getAccessToken } from './fetch'
|
|
import { asyncRunSafe } from '@/utils'
|
|
const TIME_OUT = 100000
|
|
|
|
export type IOnDataMoreInfo = {
|
|
conversationId?: string
|
|
taskId?: string
|
|
messageId: string
|
|
errorMessage?: string
|
|
errorCode?: string
|
|
}
|
|
|
|
export type IOnData = (message: string, isFirstMessage: boolean, moreInfo: IOnDataMoreInfo) => void
|
|
export type IOnThought = (though: ThoughtItem) => void
|
|
export type IOnFile = (file: VisionFile) => void
|
|
export type IOnMessageEnd = (messageEnd: MessageEnd) => void
|
|
export type IOnMessageReplace = (messageReplace: MessageReplace) => void
|
|
export type IOnAnnotationReply = (messageReplace: AnnotationReply) => void
|
|
export type IOnCompleted = (hasError?: boolean, errorMessage?: string) => void
|
|
export type IOnError = (msg: string, code?: string) => void
|
|
|
|
export type IOnWorkflowStarted = (workflowStarted: WorkflowStartedResponse) => void
|
|
export type IOnWorkflowFinished = (workflowFinished: WorkflowFinishedResponse) => void
|
|
export type IOnNodeStarted = (nodeStarted: NodeStartedResponse) => void
|
|
export type IOnNodeFinished = (nodeFinished: NodeFinishedResponse) => void
|
|
export type IOnIterationStarted = (workflowStarted: IterationStartedResponse) => void
|
|
export type IOnIterationNext = (workflowStarted: IterationNextResponse) => void
|
|
export type IOnNodeRetry = (nodeFinished: NodeFinishedResponse) => void
|
|
export type IOnIterationFinished = (workflowFinished: IterationFinishedResponse) => void
|
|
export type IOnParallelBranchStarted = (parallelBranchStarted: ParallelBranchStartedResponse) => void
|
|
export type IOnParallelBranchFinished = (parallelBranchFinished: ParallelBranchFinishedResponse) => void
|
|
export type IOnTextChunk = (textChunk: TextChunkResponse) => void
|
|
export type IOnTTSChunk = (messageId: string, audioStr: string, audioType?: string) => void
|
|
export type IOnTTSEnd = (messageId: string, audioStr: string, audioType?: string) => void
|
|
export type IOnTextReplace = (textReplace: TextReplaceResponse) => void
|
|
export type IOnAgentLog = (agentLog: AgentLogResponse) => void
|
|
|
|
export type IOtherOptions = {
|
|
isPublicAPI?: boolean
|
|
isMarketplaceAPI?: boolean
|
|
bodyStringify?: boolean
|
|
needAllResponseContent?: boolean
|
|
deleteContentType?: boolean
|
|
silent?: boolean
|
|
onData?: IOnData // for stream
|
|
onThought?: IOnThought
|
|
onFile?: IOnFile
|
|
onMessageEnd?: IOnMessageEnd
|
|
onMessageReplace?: IOnMessageReplace
|
|
onError?: IOnError
|
|
onCompleted?: IOnCompleted // for stream
|
|
getAbortController?: (abortController: AbortController) => void
|
|
|
|
onWorkflowStarted?: IOnWorkflowStarted
|
|
onWorkflowFinished?: IOnWorkflowFinished
|
|
onNodeStarted?: IOnNodeStarted
|
|
onNodeFinished?: IOnNodeFinished
|
|
onIterationStart?: IOnIterationStarted
|
|
onIterationNext?: IOnIterationNext
|
|
onIterationFinish?: IOnIterationFinished
|
|
onNodeRetry?: IOnNodeRetry
|
|
onParallelBranchStarted?: IOnParallelBranchStarted
|
|
onParallelBranchFinished?: IOnParallelBranchFinished
|
|
onTextChunk?: IOnTextChunk
|
|
onTTSChunk?: IOnTTSChunk
|
|
onTTSEnd?: IOnTTSEnd
|
|
onTextReplace?: IOnTextReplace
|
|
onAgentLog?: IOnAgentLog
|
|
}
|
|
|
|
function unicodeToChar(text: string) {
|
|
if (!text)
|
|
return ''
|
|
|
|
return text.replace(/\\u[0-9a-f]{4}/g, (_match, p1) => {
|
|
return String.fromCharCode(Number.parseInt(p1, 16))
|
|
})
|
|
}
|
|
|
|
function requiredWebSSOLogin() {
|
|
globalThis.location.href = `/webapp-signin?redirect_url=${globalThis.location.pathname}`
|
|
}
|
|
|
|
export function format(text: string) {
|
|
let res = text.trim()
|
|
if (res.startsWith('\n'))
|
|
res = res.replace('\n', '')
|
|
|
|
return res.replaceAll('\n', '<br/>').replaceAll('```', '')
|
|
}
|
|
|
|
const handleStream = (
|
|
response: Response,
|
|
onData: IOnData,
|
|
onCompleted?: IOnCompleted,
|
|
onThought?: IOnThought,
|
|
onMessageEnd?: IOnMessageEnd,
|
|
onMessageReplace?: IOnMessageReplace,
|
|
onFile?: IOnFile,
|
|
onWorkflowStarted?: IOnWorkflowStarted,
|
|
onWorkflowFinished?: IOnWorkflowFinished,
|
|
onNodeStarted?: IOnNodeStarted,
|
|
onNodeFinished?: IOnNodeFinished,
|
|
onIterationStart?: IOnIterationStarted,
|
|
onIterationNext?: IOnIterationNext,
|
|
onIterationFinish?: IOnIterationFinished,
|
|
onNodeRetry?: IOnNodeRetry,
|
|
onParallelBranchStarted?: IOnParallelBranchStarted,
|
|
onParallelBranchFinished?: IOnParallelBranchFinished,
|
|
onTextChunk?: IOnTextChunk,
|
|
onTTSChunk?: IOnTTSChunk,
|
|
onTTSEnd?: IOnTTSEnd,
|
|
onTextReplace?: IOnTextReplace,
|
|
onAgentLog?: IOnAgentLog,
|
|
) => {
|
|
if (!response.ok)
|
|
throw new Error('Network response was not ok')
|
|
|
|
const reader = response.body?.getReader()
|
|
const decoder = new TextDecoder('utf-8')
|
|
let buffer = ''
|
|
let bufferObj: Record<string, any>
|
|
let isFirstMessage = true
|
|
function read() {
|
|
let hasError = false
|
|
reader?.read().then((result: any) => {
|
|
if (result.done) {
|
|
onCompleted && onCompleted()
|
|
return
|
|
}
|
|
buffer += decoder.decode(result.value, { stream: true })
|
|
const lines = buffer.split('\n')
|
|
try {
|
|
lines.forEach((message) => {
|
|
if (message.startsWith('data: ')) { // check if it starts with data:
|
|
try {
|
|
bufferObj = JSON.parse(message.substring(6)) as Record<string, any>// remove data: and parse as json
|
|
}
|
|
catch (e) {
|
|
// mute handle message cut off
|
|
onData('', isFirstMessage, {
|
|
conversationId: bufferObj?.conversation_id,
|
|
messageId: bufferObj?.message_id,
|
|
})
|
|
return
|
|
}
|
|
if (bufferObj.status === 400 || !bufferObj.event) {
|
|
onData('', false, {
|
|
conversationId: undefined,
|
|
messageId: '',
|
|
errorMessage: bufferObj?.message,
|
|
errorCode: bufferObj?.code,
|
|
})
|
|
hasError = true
|
|
onCompleted?.(true, bufferObj?.message)
|
|
return
|
|
}
|
|
if (bufferObj.event === 'message' || bufferObj.event === 'agent_message') {
|
|
// can not use format here. Because message is splitted.
|
|
onData(unicodeToChar(bufferObj.answer), isFirstMessage, {
|
|
conversationId: bufferObj.conversation_id,
|
|
taskId: bufferObj.task_id,
|
|
messageId: bufferObj.id,
|
|
})
|
|
isFirstMessage = false
|
|
}
|
|
else if (bufferObj.event === 'agent_thought') {
|
|
onThought?.(bufferObj as ThoughtItem)
|
|
}
|
|
else if (bufferObj.event === 'message_file') {
|
|
onFile?.(bufferObj as VisionFile)
|
|
}
|
|
else if (bufferObj.event === 'message_end') {
|
|
onMessageEnd?.(bufferObj as MessageEnd)
|
|
}
|
|
else if (bufferObj.event === 'message_replace') {
|
|
onMessageReplace?.(bufferObj as MessageReplace)
|
|
}
|
|
else if (bufferObj.event === 'workflow_started') {
|
|
onWorkflowStarted?.(bufferObj as WorkflowStartedResponse)
|
|
}
|
|
else if (bufferObj.event === 'workflow_finished') {
|
|
onWorkflowFinished?.(bufferObj as WorkflowFinishedResponse)
|
|
}
|
|
else if (bufferObj.event === 'node_started') {
|
|
onNodeStarted?.(bufferObj as NodeStartedResponse)
|
|
}
|
|
else if (bufferObj.event === 'node_finished') {
|
|
onNodeFinished?.(bufferObj as NodeFinishedResponse)
|
|
}
|
|
else if (bufferObj.event === 'iteration_started') {
|
|
onIterationStart?.(bufferObj as IterationStartedResponse)
|
|
}
|
|
else if (bufferObj.event === 'iteration_next') {
|
|
onIterationNext?.(bufferObj as IterationNextResponse)
|
|
}
|
|
else if (bufferObj.event === 'iteration_completed') {
|
|
onIterationFinish?.(bufferObj as IterationFinishedResponse)
|
|
}
|
|
else if (bufferObj.event === 'node_retry') {
|
|
onNodeRetry?.(bufferObj as NodeFinishedResponse)
|
|
}
|
|
else if (bufferObj.event === 'parallel_branch_started') {
|
|
onParallelBranchStarted?.(bufferObj as ParallelBranchStartedResponse)
|
|
}
|
|
else if (bufferObj.event === 'parallel_branch_finished') {
|
|
onParallelBranchFinished?.(bufferObj as ParallelBranchFinishedResponse)
|
|
}
|
|
else if (bufferObj.event === 'text_chunk') {
|
|
onTextChunk?.(bufferObj as TextChunkResponse)
|
|
}
|
|
else if (bufferObj.event === 'text_replace') {
|
|
onTextReplace?.(bufferObj as TextReplaceResponse)
|
|
}
|
|
else if (bufferObj.event === 'agent_log') {
|
|
onAgentLog?.(bufferObj as AgentLogResponse)
|
|
}
|
|
else if (bufferObj.event === 'tts_message') {
|
|
onTTSChunk?.(bufferObj.message_id, bufferObj.audio, bufferObj.audio_type)
|
|
}
|
|
else if (bufferObj.event === 'tts_message_end') {
|
|
onTTSEnd?.(bufferObj.message_id, bufferObj.audio)
|
|
}
|
|
}
|
|
})
|
|
buffer = lines[lines.length - 1]
|
|
}
|
|
catch (e) {
|
|
onData('', false, {
|
|
conversationId: undefined,
|
|
messageId: '',
|
|
errorMessage: `${e}`,
|
|
})
|
|
hasError = true
|
|
onCompleted?.(true, e as string)
|
|
return
|
|
}
|
|
if (!hasError)
|
|
read()
|
|
})
|
|
}
|
|
read()
|
|
}
|
|
|
|
const baseFetch = base
|
|
|
|
export const upload = (options: any, isPublicAPI?: boolean, url?: string, searchParams?: string): Promise<any> => {
|
|
const urlPrefix = isPublicAPI ? PUBLIC_API_PREFIX : API_PREFIX
|
|
const token = getAccessToken(isPublicAPI)
|
|
const defaultOptions = {
|
|
method: 'POST',
|
|
url: (url ? `${urlPrefix}${url}` : `${urlPrefix}/files/upload`) + (searchParams || ''),
|
|
headers: {
|
|
Authorization: `Bearer ${token}`,
|
|
},
|
|
data: {},
|
|
}
|
|
options = {
|
|
...defaultOptions,
|
|
...options,
|
|
headers: { ...defaultOptions.headers, ...options.headers },
|
|
}
|
|
return new Promise((resolve, reject) => {
|
|
const xhr = options.xhr
|
|
xhr.open(options.method, options.url)
|
|
for (const key in options.headers)
|
|
xhr.setRequestHeader(key, options.headers[key])
|
|
|
|
xhr.withCredentials = true
|
|
xhr.responseType = 'json'
|
|
xhr.onreadystatechange = function () {
|
|
if (xhr.readyState === 4) {
|
|
if (xhr.status === 201)
|
|
resolve(xhr.response)
|
|
else
|
|
reject(xhr)
|
|
}
|
|
}
|
|
xhr.upload.onprogress = options.onprogress
|
|
xhr.send(options.data)
|
|
})
|
|
}
|
|
|
|
export const ssePost = (
|
|
url: string,
|
|
fetchOptions: FetchOptionType,
|
|
otherOptions: IOtherOptions,
|
|
) => {
|
|
const {
|
|
isPublicAPI = false,
|
|
onData,
|
|
onCompleted,
|
|
onThought,
|
|
onFile,
|
|
onMessageEnd,
|
|
onMessageReplace,
|
|
onWorkflowStarted,
|
|
onWorkflowFinished,
|
|
onNodeStarted,
|
|
onNodeFinished,
|
|
onIterationStart,
|
|
onIterationNext,
|
|
onIterationFinish,
|
|
onNodeRetry,
|
|
onParallelBranchStarted,
|
|
onParallelBranchFinished,
|
|
onTextChunk,
|
|
onTTSChunk,
|
|
onTTSEnd,
|
|
onTextReplace,
|
|
onAgentLog,
|
|
onError,
|
|
getAbortController,
|
|
} = otherOptions
|
|
const abortController = new AbortController()
|
|
|
|
const token = localStorage.getItem('console_token')
|
|
|
|
const options = Object.assign({}, baseOptions, {
|
|
method: 'POST',
|
|
signal: abortController.signal,
|
|
headers: new Headers({
|
|
Authorization: `Bearer ${token}`,
|
|
}),
|
|
} as RequestInit, fetchOptions)
|
|
|
|
const contentType = (options.headers as Headers).get('Content-Type')
|
|
if (!contentType)
|
|
(options.headers as Headers).set('Content-Type', ContentType.json)
|
|
|
|
getAbortController?.(abortController)
|
|
|
|
const urlPrefix = isPublicAPI ? PUBLIC_API_PREFIX : API_PREFIX
|
|
const urlWithPrefix = (url.startsWith('http://') || url.startsWith('https://'))
|
|
? url
|
|
: `${urlPrefix}${url.startsWith('/') ? url : `/${url}`}`
|
|
|
|
const { body } = options
|
|
if (body)
|
|
options.body = JSON.stringify(body)
|
|
|
|
const accessToken = getAccessToken(isPublicAPI)
|
|
options.headers!.set('Authorization', `Bearer ${accessToken}`)
|
|
|
|
globalThis.fetch(urlWithPrefix, options as RequestInit)
|
|
.then((res) => {
|
|
if (!/^(2|3)\d{2}$/.test(String(res.status))) {
|
|
if (res.status === 401) {
|
|
refreshAccessTokenOrRelogin(TIME_OUT).then(() => {
|
|
ssePost(url, fetchOptions, otherOptions)
|
|
}).catch(() => {
|
|
res.json().then((data: any) => {
|
|
if (isPublicAPI) {
|
|
if (data.code === 'web_sso_auth_required')
|
|
requiredWebSSOLogin()
|
|
|
|
if (data.code === 'unauthorized') {
|
|
removeAccessToken()
|
|
globalThis.location.reload()
|
|
}
|
|
}
|
|
})
|
|
})
|
|
}
|
|
else {
|
|
res.json().then((data) => {
|
|
Toast.notify({ type: 'error', message: data.message || 'Server Error' })
|
|
})
|
|
onError?.('Server Error')
|
|
}
|
|
return
|
|
}
|
|
return handleStream(res, (str: string, isFirstMessage: boolean, moreInfo: IOnDataMoreInfo) => {
|
|
if (moreInfo.errorMessage) {
|
|
onError?.(moreInfo.errorMessage, moreInfo.errorCode)
|
|
// TypeError: Cannot assign to read only property ... will happen in page leave, so it should be ignored.
|
|
if (moreInfo.errorMessage !== 'AbortError: The user aborted a request.' && !moreInfo.errorMessage.includes('TypeError: Cannot assign to read only property'))
|
|
Toast.notify({ type: 'error', message: moreInfo.errorMessage })
|
|
return
|
|
}
|
|
onData?.(str, isFirstMessage, moreInfo)
|
|
}, onCompleted, onThought, onMessageEnd, onMessageReplace, onFile, onWorkflowStarted, onWorkflowFinished, onNodeStarted, onNodeFinished, onIterationStart, onIterationNext, onIterationFinish, onNodeRetry, onParallelBranchStarted, onParallelBranchFinished, onTextChunk, onTTSChunk, onTTSEnd, onTextReplace, onAgentLog)
|
|
}).catch((e) => {
|
|
if (e.toString() !== 'AbortError: The user aborted a request.' && !e.toString().errorMessage.includes('TypeError: Cannot assign to read only property'))
|
|
Toast.notify({ type: 'error', message: e })
|
|
onError?.(e)
|
|
})
|
|
}
|
|
|
|
// base request
|
|
export const request = async<T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
try {
|
|
const otherOptionsForBaseFetch = otherOptions || {}
|
|
const [err, resp] = await asyncRunSafe<T>(baseFetch(url, options, otherOptionsForBaseFetch))
|
|
if (err === null)
|
|
return resp
|
|
const errResp: Response = err as any
|
|
if (errResp.status === 401) {
|
|
const [parseErr, errRespData] = await asyncRunSafe<ResponseError>(errResp.json())
|
|
const loginUrl = `${globalThis.location.origin}/signin`
|
|
if (parseErr) {
|
|
globalThis.location.href = loginUrl
|
|
return Promise.reject(err)
|
|
}
|
|
// special code
|
|
const { code, message } = errRespData
|
|
// webapp sso
|
|
if (code === 'web_sso_auth_required') {
|
|
requiredWebSSOLogin()
|
|
return Promise.reject(err)
|
|
}
|
|
if (code === 'unauthorized_and_force_logout') {
|
|
localStorage.removeItem('console_token')
|
|
localStorage.removeItem('refresh_token')
|
|
globalThis.location.reload()
|
|
return Promise.reject(err)
|
|
}
|
|
const {
|
|
isPublicAPI = false,
|
|
silent,
|
|
} = otherOptionsForBaseFetch
|
|
if (isPublicAPI && code === 'unauthorized') {
|
|
removeAccessToken()
|
|
globalThis.location.reload()
|
|
return Promise.reject(err)
|
|
}
|
|
if (code === 'init_validate_failed' && IS_CE_EDITION && !silent) {
|
|
Toast.notify({ type: 'error', message, duration: 4000 })
|
|
return Promise.reject(err)
|
|
}
|
|
if (code === 'not_init_validated' && IS_CE_EDITION) {
|
|
globalThis.location.href = `${globalThis.location.origin}/init`
|
|
return Promise.reject(err)
|
|
}
|
|
if (code === 'not_setup' && IS_CE_EDITION) {
|
|
globalThis.location.href = `${globalThis.location.origin}/install`
|
|
return Promise.reject(err)
|
|
}
|
|
|
|
// refresh token
|
|
const [refreshErr] = await asyncRunSafe(refreshAccessTokenOrRelogin(TIME_OUT))
|
|
if (refreshErr === null)
|
|
return baseFetch<T>(url, options, otherOptionsForBaseFetch)
|
|
if (location.pathname !== '/signin' || !IS_CE_EDITION) {
|
|
globalThis.location.href = loginUrl
|
|
return Promise.reject(err)
|
|
}
|
|
if (!silent) {
|
|
Toast.notify({ type: 'error', message })
|
|
return Promise.reject(err)
|
|
}
|
|
globalThis.location.href = loginUrl
|
|
return Promise.reject(err)
|
|
}
|
|
else {
|
|
return Promise.reject(err)
|
|
}
|
|
}
|
|
catch (error) {
|
|
console.error(error)
|
|
return Promise.reject(error)
|
|
}
|
|
}
|
|
|
|
// request methods
|
|
export const get = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request<T>(url, Object.assign({}, options, { method: 'GET' }), otherOptions)
|
|
}
|
|
|
|
// For public API
|
|
export const getPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return get<T>(url, options, { ...otherOptions, isPublicAPI: true })
|
|
}
|
|
|
|
// For Marketplace API
|
|
export const getMarketplace = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return get<T>(url, options, { ...otherOptions, isMarketplaceAPI: true })
|
|
}
|
|
|
|
export const post = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request<T>(url, Object.assign({}, options, { method: 'POST' }), otherOptions)
|
|
}
|
|
|
|
// For Marketplace API
|
|
export const postMarketplace = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return post<T>(url, options, { ...otherOptions, isMarketplaceAPI: true })
|
|
}
|
|
|
|
export const postPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return post<T>(url, options, { ...otherOptions, isPublicAPI: true })
|
|
}
|
|
|
|
export const put = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request<T>(url, Object.assign({}, options, { method: 'PUT' }), otherOptions)
|
|
}
|
|
|
|
export const putPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return put<T>(url, options, { ...otherOptions, isPublicAPI: true })
|
|
}
|
|
|
|
export const del = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request<T>(url, Object.assign({}, options, { method: 'DELETE' }), otherOptions)
|
|
}
|
|
|
|
export const delPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return del<T>(url, options, { ...otherOptions, isPublicAPI: true })
|
|
}
|
|
|
|
export const patch = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request<T>(url, Object.assign({}, options, { method: 'PATCH' }), otherOptions)
|
|
}
|
|
|
|
export const patchPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return patch<T>(url, options, { ...otherOptions, isPublicAPI: true })
|
|
}
|