441 lines
13 KiB
TypeScript
441 lines
13 KiB
TypeScript
import { API_PREFIX } from '@/config'
|
|
import Toast from '@/app/components/base/toast'
|
|
import type { AnnotationReply, MessageEnd, MessageReplace, ThoughtItem } from '@/app/components/chat/type-mars'
|
|
import type { VisionFile } from '@/types/app'
|
|
|
|
const TIME_OUT = 100000
|
|
|
|
const ContentType = {
|
|
json: 'application/json',
|
|
stream: 'text/event-stream',
|
|
form: 'application/x-www-form-urlencoded; charset=UTF-8',
|
|
download: 'application/octet-stream', // for download
|
|
}
|
|
|
|
const baseOptions = {
|
|
method: 'GET',
|
|
mode: 'cors',
|
|
credentials: 'include', // always send cookies、HTTP Basic authentication.
|
|
headers: new Headers({
|
|
'Content-Type': ContentType.json,
|
|
}),
|
|
redirect: 'follow',
|
|
}
|
|
|
|
export type WorkflowStartedResponse = {
|
|
task_id: string
|
|
workflow_run_id: string
|
|
event: string
|
|
data: {
|
|
id: string
|
|
workflow_id: string
|
|
sequence_number: number
|
|
created_at: number
|
|
}
|
|
}
|
|
|
|
export type WorkflowFinishedResponse = {
|
|
task_id: string
|
|
workflow_run_id: string
|
|
event: string
|
|
data: {
|
|
id: string
|
|
workflow_id: string
|
|
status: string
|
|
outputs: any
|
|
error: string
|
|
elapsed_time: number
|
|
total_tokens: number
|
|
total_steps: number
|
|
created_at: number
|
|
finished_at: number
|
|
}
|
|
}
|
|
|
|
export type NodeStartedResponse = {
|
|
task_id: string
|
|
workflow_run_id: string
|
|
event: string
|
|
data: {
|
|
id: string
|
|
node_id: string
|
|
node_type: string
|
|
index: number
|
|
predecessor_node_id?: string
|
|
inputs: any
|
|
created_at: number
|
|
extras?: any
|
|
}
|
|
}
|
|
|
|
export type NodeFinishedResponse = {
|
|
task_id: string
|
|
workflow_run_id: string
|
|
event: string
|
|
data: {
|
|
id: string
|
|
node_id: string
|
|
node_type: string
|
|
index: number
|
|
predecessor_node_id?: string
|
|
inputs: any
|
|
process_data: any
|
|
outputs: any
|
|
status: string
|
|
error: string
|
|
elapsed_time: number
|
|
execution_metadata: {
|
|
total_tokens: number
|
|
total_price: number
|
|
currency: string
|
|
}
|
|
created_at: number
|
|
}
|
|
}
|
|
|
|
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) => 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
|
|
|
|
type IOtherOptions = {
|
|
isPublicAPI?: boolean
|
|
bodyStringify?: boolean
|
|
needAllResponseContent?: boolean
|
|
deleteContentType?: 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
|
|
}
|
|
|
|
function unicodeToChar(text: string) {
|
|
return text.replace(/\\u[0-9a-f]{4}/g, (_match, p1) => {
|
|
return String.fromCharCode(parseInt(p1, 16))
|
|
})
|
|
}
|
|
|
|
const handleStream = (
|
|
response: Response,
|
|
onData: IOnData,
|
|
onCompleted?: IOnCompleted,
|
|
onThought?: IOnThought,
|
|
onMessageEnd?: IOnMessageEnd,
|
|
onMessageReplace?: IOnMessageReplace,
|
|
onFile?: IOnFile,
|
|
onWorkflowStarted?: IOnWorkflowStarted,
|
|
onWorkflowFinished?: IOnWorkflowFinished,
|
|
onNodeStarted?: IOnNodeStarted,
|
|
onNodeFinished?: IOnNodeFinished,
|
|
) => {
|
|
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)
|
|
return
|
|
}
|
|
if (bufferObj.event === 'message' || bufferObj.event === 'agent_message') {
|
|
// can not use format here. Because message is splited.
|
|
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)
|
|
}
|
|
}
|
|
})
|
|
buffer = lines[lines.length - 1]
|
|
}
|
|
catch (e) {
|
|
onData('', false, {
|
|
conversationId: undefined,
|
|
messageId: '',
|
|
errorMessage: `${e}`,
|
|
})
|
|
hasError = true
|
|
onCompleted?.(true)
|
|
return
|
|
}
|
|
if (!hasError)
|
|
read()
|
|
})
|
|
}
|
|
read()
|
|
}
|
|
|
|
const baseFetch = (url: string, fetchOptions: any, { needAllResponseContent }: IOtherOptions) => {
|
|
const options = Object.assign({}, baseOptions, fetchOptions)
|
|
|
|
const urlPrefix = API_PREFIX
|
|
|
|
let urlWithPrefix = `${urlPrefix}${url.startsWith('/') ? url : `/${url}`}`
|
|
|
|
const { method, params, body } = options
|
|
// handle query
|
|
if (method === 'GET' && params) {
|
|
const paramsArray: string[] = []
|
|
Object.keys(params).forEach(key =>
|
|
paramsArray.push(`${key}=${encodeURIComponent(params[key])}`),
|
|
)
|
|
if (urlWithPrefix.search(/\?/) === -1)
|
|
urlWithPrefix += `?${paramsArray.join('&')}`
|
|
|
|
else
|
|
urlWithPrefix += `&${paramsArray.join('&')}`
|
|
|
|
delete options.params
|
|
}
|
|
|
|
if (body)
|
|
options.body = JSON.stringify(body)
|
|
|
|
// Handle timeout
|
|
return Promise.race([
|
|
new Promise((resolve, reject) => {
|
|
setTimeout(() => {
|
|
reject(new Error('request timeout'))
|
|
}, TIME_OUT)
|
|
}),
|
|
new Promise((resolve, reject) => {
|
|
globalThis.fetch(urlWithPrefix, options)
|
|
.then((res: any) => {
|
|
const resClone = res.clone()
|
|
// Error handler
|
|
if (!/^(2|3)\d{2}$/.test(res.status)) {
|
|
try {
|
|
switch (res.status) {
|
|
case 401: {
|
|
Toast.notify({ type: 'error', message: 'Invalid token' })
|
|
return
|
|
}
|
|
case 500: {
|
|
Toast.notify({ type: 'error', message: 'Server error' })
|
|
return
|
|
}
|
|
default:
|
|
// eslint-disable-next-line no-new
|
|
new Promise(() => {
|
|
const bodyJson = res.json()
|
|
bodyJson.then((data: any) => {
|
|
Toast.notify({ type: 'error', message: data.message })
|
|
})
|
|
})
|
|
}
|
|
}
|
|
catch (e) {
|
|
Toast.notify({ type: 'error', message: `${e}` })
|
|
}
|
|
|
|
return Promise.reject(resClone)
|
|
}
|
|
|
|
// handle delete api. Delete api not return content.
|
|
if (res.status === 204) {
|
|
resolve({ result: 'success' })
|
|
return
|
|
}
|
|
|
|
// return data
|
|
const data = options.headers.get('Content-type') === ContentType.download ? res.blob() : res.json()
|
|
|
|
resolve(needAllResponseContent ? resClone : data)
|
|
})
|
|
.catch((err) => {
|
|
Toast.notify({ type: 'error', message: err })
|
|
reject(err)
|
|
})
|
|
}),
|
|
])
|
|
}
|
|
|
|
export const upload = (fetchOptions: any, url?: string): Promise<any> => {
|
|
const urlPrefix = API_PREFIX
|
|
const urlWithPrefix = url ? `${urlPrefix}/${url}` : `${urlPrefix}/file-upload`
|
|
const defaultOptions = {
|
|
method: 'POST',
|
|
url: `${urlWithPrefix}`,
|
|
data: {},
|
|
}
|
|
const options = {
|
|
...defaultOptions,
|
|
...fetchOptions,
|
|
}
|
|
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.onreadystatechange = function () {
|
|
if (xhr.readyState === 4) {
|
|
if (xhr.status === 200)
|
|
resolve(JSON.parse(xhr.response))
|
|
else
|
|
reject(xhr)
|
|
}
|
|
}
|
|
xhr.upload.onprogress = options.onprogress
|
|
xhr.send(options.data)
|
|
})
|
|
}
|
|
|
|
export const ssePost = (
|
|
url: string,
|
|
fetchOptions: any,
|
|
{
|
|
onData,
|
|
onCompleted,
|
|
onThought,
|
|
onFile,
|
|
onMessageEnd,
|
|
onMessageReplace,
|
|
onWorkflowStarted,
|
|
onWorkflowFinished,
|
|
onNodeStarted,
|
|
onNodeFinished,
|
|
onError,
|
|
}: IOtherOptions,
|
|
) => {
|
|
const options = Object.assign({}, baseOptions, {
|
|
method: 'POST',
|
|
}, fetchOptions)
|
|
|
|
const urlPrefix = API_PREFIX
|
|
const urlWithPrefix = `${urlPrefix}${url.startsWith('/') ? url : `/${url}`}`
|
|
|
|
const { body } = options
|
|
if (body)
|
|
options.body = JSON.stringify(body)
|
|
|
|
globalThis.fetch(urlWithPrefix, options)
|
|
.then((res: any) => {
|
|
if (!/^(2|3)\d{2}$/.test(res.status)) {
|
|
// eslint-disable-next-line no-new
|
|
new Promise(() => {
|
|
res.json().then((data: any) => {
|
|
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) {
|
|
Toast.notify({ type: 'error', message: moreInfo.errorMessage })
|
|
return
|
|
}
|
|
onData?.(str, isFirstMessage, moreInfo)
|
|
}, () => {
|
|
onCompleted?.()
|
|
}, onThought, onMessageEnd, onMessageReplace, onFile, onWorkflowStarted, onWorkflowFinished, onNodeStarted, onNodeFinished)
|
|
}).catch((e) => {
|
|
Toast.notify({ type: 'error', message: e })
|
|
onError?.(e)
|
|
})
|
|
}
|
|
|
|
export const request = (url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return baseFetch(url, options, otherOptions || {})
|
|
}
|
|
|
|
export const get = (url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request(url, Object.assign({}, options, { method: 'GET' }), otherOptions)
|
|
}
|
|
|
|
export const post = (url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request(url, Object.assign({}, options, { method: 'POST' }), otherOptions)
|
|
}
|
|
|
|
export const put = (url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request(url, Object.assign({}, options, { method: 'PUT' }), otherOptions)
|
|
}
|
|
|
|
export const del = (url: string, options = {}, otherOptions?: IOtherOptions) => {
|
|
return request(url, Object.assign({}, options, { method: 'DELETE' }), otherOptions)
|
|
}
|