base.ts 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600
  1. import { API_PREFIX, IS_CE_EDITION, PUBLIC_API_PREFIX } from '@/config'
  2. import Toast from '@/app/components/base/toast'
  3. import type { AnnotationReply, MessageEnd, MessageReplace, ThoughtItem } from '@/app/components/base/chat/chat/type'
  4. import type { VisionFile } from '@/types/app'
  5. import type {
  6. IterationFinishedResponse,
  7. IterationNextedResponse,
  8. IterationStartedResponse,
  9. NodeFinishedResponse,
  10. NodeStartedResponse,
  11. TextChunkResponse,
  12. TextReplaceResponse,
  13. WorkflowFinishedResponse,
  14. WorkflowStartedResponse,
  15. } from '@/types/workflow'
  16. import { removeAccessToken } from '@/app/components/share/utils'
  17. const TIME_OUT = 100000
  18. const ContentType = {
  19. json: 'application/json',
  20. stream: 'text/event-stream',
  21. audio: 'audio/mpeg',
  22. form: 'application/x-www-form-urlencoded; charset=UTF-8',
  23. download: 'application/octet-stream', // for download
  24. upload: 'multipart/form-data', // for upload
  25. }
  26. const baseOptions = {
  27. method: 'GET',
  28. mode: 'cors',
  29. credentials: 'include', // always send cookies、HTTP Basic authentication.
  30. headers: new Headers({
  31. 'Content-Type': ContentType.json,
  32. }),
  33. redirect: 'follow',
  34. }
  35. export type IOnDataMoreInfo = {
  36. conversationId?: string
  37. taskId?: string
  38. messageId: string
  39. errorMessage?: string
  40. errorCode?: string
  41. }
  42. export type IOnData = (message: string, isFirstMessage: boolean, moreInfo: IOnDataMoreInfo) => void
  43. export type IOnThought = (though: ThoughtItem) => void
  44. export type IOnFile = (file: VisionFile) => void
  45. export type IOnMessageEnd = (messageEnd: MessageEnd) => void
  46. export type IOnMessageReplace = (messageReplace: MessageReplace) => void
  47. export type IOnAnnotationReply = (messageReplace: AnnotationReply) => void
  48. export type IOnCompleted = (hasError?: boolean, errorMessage?: string) => void
  49. export type IOnError = (msg: string, code?: string) => void
  50. export type IOnWorkflowStarted = (workflowStarted: WorkflowStartedResponse) => void
  51. export type IOnWorkflowFinished = (workflowFinished: WorkflowFinishedResponse) => void
  52. export type IOnNodeStarted = (nodeStarted: NodeStartedResponse) => void
  53. export type IOnNodeFinished = (nodeFinished: NodeFinishedResponse) => void
  54. export type IOnIterationStarted = (workflowStarted: IterationStartedResponse) => void
  55. export type IOnIterationNexted = (workflowStarted: IterationNextedResponse) => void
  56. export type IOnIterationFinished = (workflowFinished: IterationFinishedResponse) => void
  57. export type IOnTextChunk = (textChunk: TextChunkResponse) => void
  58. export type IOnTTSChunk = (messageId: string, audioStr: string, audioType?: string) => void
  59. export type IOnTTSEnd = (messageId: string, audioStr: string, audioType?: string) => void
  60. export type IOnTextReplace = (textReplace: TextReplaceResponse) => void
  61. export type IOtherOptions = {
  62. isPublicAPI?: boolean
  63. bodyStringify?: boolean
  64. needAllResponseContent?: boolean
  65. deleteContentType?: boolean
  66. silent?: boolean
  67. onData?: IOnData // for stream
  68. onThought?: IOnThought
  69. onFile?: IOnFile
  70. onMessageEnd?: IOnMessageEnd
  71. onMessageReplace?: IOnMessageReplace
  72. onError?: IOnError
  73. onCompleted?: IOnCompleted // for stream
  74. getAbortController?: (abortController: AbortController) => void
  75. onWorkflowStarted?: IOnWorkflowStarted
  76. onWorkflowFinished?: IOnWorkflowFinished
  77. onNodeStarted?: IOnNodeStarted
  78. onNodeFinished?: IOnNodeFinished
  79. onIterationStart?: IOnIterationStarted
  80. onIterationNext?: IOnIterationNexted
  81. onIterationFinish?: IOnIterationFinished
  82. onTextChunk?: IOnTextChunk
  83. onTTSChunk?: IOnTTSChunk
  84. onTTSEnd?: IOnTTSEnd
  85. onTextReplace?: IOnTextReplace
  86. }
  87. type ResponseError = {
  88. code: string
  89. message: string
  90. status: number
  91. }
  92. type FetchOptionType = Omit<RequestInit, 'body'> & {
  93. params?: Record<string, any>
  94. body?: BodyInit | Record<string, any> | null
  95. }
  96. function unicodeToChar(text: string) {
  97. if (!text)
  98. return ''
  99. return text.replace(/\\u[0-9a-f]{4}/g, (_match, p1) => {
  100. return String.fromCharCode(parseInt(p1, 16))
  101. })
  102. }
  103. function requiredWebSSOLogin() {
  104. globalThis.location.href = `/webapp-signin?redirect_url=${globalThis.location.pathname}`
  105. }
  106. export function format(text: string) {
  107. let res = text.trim()
  108. if (res.startsWith('\n'))
  109. res = res.replace('\n', '')
  110. return res.replaceAll('\n', '<br/>').replaceAll('```', '')
  111. }
  112. const handleStream = (
  113. response: Response,
  114. onData: IOnData,
  115. onCompleted?: IOnCompleted,
  116. onThought?: IOnThought,
  117. onMessageEnd?: IOnMessageEnd,
  118. onMessageReplace?: IOnMessageReplace,
  119. onFile?: IOnFile,
  120. onWorkflowStarted?: IOnWorkflowStarted,
  121. onWorkflowFinished?: IOnWorkflowFinished,
  122. onNodeStarted?: IOnNodeStarted,
  123. onNodeFinished?: IOnNodeFinished,
  124. onIterationStart?: IOnIterationStarted,
  125. onIterationNext?: IOnIterationNexted,
  126. onIterationFinish?: IOnIterationFinished,
  127. onTextChunk?: IOnTextChunk,
  128. onTTSChunk?: IOnTTSChunk,
  129. onTTSEnd?: IOnTTSEnd,
  130. onTextReplace?: IOnTextReplace,
  131. ) => {
  132. if (!response.ok)
  133. throw new Error('Network response was not ok')
  134. const reader = response.body?.getReader()
  135. const decoder = new TextDecoder('utf-8')
  136. let buffer = ''
  137. let bufferObj: Record<string, any>
  138. let isFirstMessage = true
  139. function read() {
  140. let hasError = false
  141. reader?.read().then((result: any) => {
  142. if (result.done) {
  143. onCompleted && onCompleted()
  144. return
  145. }
  146. buffer += decoder.decode(result.value, { stream: true })
  147. const lines = buffer.split('\n')
  148. try {
  149. lines.forEach((message) => {
  150. if (message.startsWith('data: ')) { // check if it starts with data:
  151. try {
  152. bufferObj = JSON.parse(message.substring(6)) as Record<string, any>// remove data: and parse as json
  153. }
  154. catch (e) {
  155. // mute handle message cut off
  156. onData('', isFirstMessage, {
  157. conversationId: bufferObj?.conversation_id,
  158. messageId: bufferObj?.message_id,
  159. })
  160. return
  161. }
  162. if (bufferObj.status === 400 || !bufferObj.event) {
  163. onData('', false, {
  164. conversationId: undefined,
  165. messageId: '',
  166. errorMessage: bufferObj?.message,
  167. errorCode: bufferObj?.code,
  168. })
  169. hasError = true
  170. onCompleted?.(true, bufferObj?.message)
  171. return
  172. }
  173. if (bufferObj.event === 'message' || bufferObj.event === 'agent_message') {
  174. // can not use format here. Because message is splited.
  175. onData(unicodeToChar(bufferObj.answer), isFirstMessage, {
  176. conversationId: bufferObj.conversation_id,
  177. taskId: bufferObj.task_id,
  178. messageId: bufferObj.id,
  179. })
  180. isFirstMessage = false
  181. }
  182. else if (bufferObj.event === 'agent_thought') {
  183. onThought?.(bufferObj as ThoughtItem)
  184. }
  185. else if (bufferObj.event === 'message_file') {
  186. onFile?.(bufferObj as VisionFile)
  187. }
  188. else if (bufferObj.event === 'message_end') {
  189. onMessageEnd?.(bufferObj as MessageEnd)
  190. }
  191. else if (bufferObj.event === 'message_replace') {
  192. onMessageReplace?.(bufferObj as MessageReplace)
  193. }
  194. else if (bufferObj.event === 'workflow_started') {
  195. onWorkflowStarted?.(bufferObj as WorkflowStartedResponse)
  196. }
  197. else if (bufferObj.event === 'workflow_finished') {
  198. onWorkflowFinished?.(bufferObj as WorkflowFinishedResponse)
  199. }
  200. else if (bufferObj.event === 'node_started') {
  201. onNodeStarted?.(bufferObj as NodeStartedResponse)
  202. }
  203. else if (bufferObj.event === 'node_finished') {
  204. onNodeFinished?.(bufferObj as NodeFinishedResponse)
  205. }
  206. else if (bufferObj.event === 'iteration_started') {
  207. onIterationStart?.(bufferObj as IterationStartedResponse)
  208. }
  209. else if (bufferObj.event === 'iteration_next') {
  210. onIterationNext?.(bufferObj as IterationNextedResponse)
  211. }
  212. else if (bufferObj.event === 'iteration_completed') {
  213. onIterationFinish?.(bufferObj as IterationFinishedResponse)
  214. }
  215. else if (bufferObj.event === 'text_chunk') {
  216. onTextChunk?.(bufferObj as TextChunkResponse)
  217. }
  218. else if (bufferObj.event === 'text_replace') {
  219. onTextReplace?.(bufferObj as TextReplaceResponse)
  220. }
  221. else if (bufferObj.event === 'tts_message') {
  222. onTTSChunk?.(bufferObj.message_id, bufferObj.audio, bufferObj.audio_type)
  223. }
  224. else if (bufferObj.event === 'tts_message_end') {
  225. onTTSEnd?.(bufferObj.message_id, bufferObj.audio)
  226. }
  227. }
  228. })
  229. buffer = lines[lines.length - 1]
  230. }
  231. catch (e) {
  232. onData('', false, {
  233. conversationId: undefined,
  234. messageId: '',
  235. errorMessage: `${e}`,
  236. })
  237. hasError = true
  238. onCompleted?.(true, e as string)
  239. return
  240. }
  241. if (!hasError)
  242. read()
  243. })
  244. }
  245. read()
  246. }
  247. const baseFetch = <T>(
  248. url: string,
  249. fetchOptions: FetchOptionType,
  250. {
  251. isPublicAPI = false,
  252. bodyStringify = true,
  253. needAllResponseContent,
  254. deleteContentType,
  255. getAbortController,
  256. silent,
  257. }: IOtherOptions,
  258. ): Promise<T> => {
  259. const options: typeof baseOptions & FetchOptionType = Object.assign({}, baseOptions, fetchOptions)
  260. if (getAbortController) {
  261. const abortController = new AbortController()
  262. getAbortController(abortController)
  263. options.signal = abortController.signal
  264. }
  265. if (isPublicAPI) {
  266. const sharedToken = globalThis.location.pathname.split('/').slice(-1)[0]
  267. const accessToken = localStorage.getItem('token') || JSON.stringify({ [sharedToken]: '' })
  268. let accessTokenJson = { [sharedToken]: '' }
  269. try {
  270. accessTokenJson = JSON.parse(accessToken)
  271. }
  272. catch (e) {
  273. }
  274. options.headers.set('Authorization', `Bearer ${accessTokenJson[sharedToken]}`)
  275. }
  276. else {
  277. const accessToken = localStorage.getItem('console_token') || ''
  278. options.headers.set('Authorization', `Bearer ${accessToken}`)
  279. }
  280. if (deleteContentType) {
  281. options.headers.delete('Content-Type')
  282. }
  283. else {
  284. const contentType = options.headers.get('Content-Type')
  285. if (!contentType)
  286. options.headers.set('Content-Type', ContentType.json)
  287. }
  288. const urlPrefix = isPublicAPI ? PUBLIC_API_PREFIX : API_PREFIX
  289. let urlWithPrefix = `${urlPrefix}${url.startsWith('/') ? url : `/${url}`}`
  290. const { method, params, body } = options
  291. // handle query
  292. if (method === 'GET' && params) {
  293. const paramsArray: string[] = []
  294. Object.keys(params).forEach(key =>
  295. paramsArray.push(`${key}=${encodeURIComponent(params[key])}`),
  296. )
  297. if (urlWithPrefix.search(/\?/) === -1)
  298. urlWithPrefix += `?${paramsArray.join('&')}`
  299. else
  300. urlWithPrefix += `&${paramsArray.join('&')}`
  301. delete options.params
  302. }
  303. if (body && bodyStringify)
  304. options.body = JSON.stringify(body)
  305. // Handle timeout
  306. return Promise.race([
  307. new Promise((resolve, reject) => {
  308. setTimeout(() => {
  309. reject(new Error('request timeout'))
  310. }, TIME_OUT)
  311. }),
  312. new Promise((resolve, reject) => {
  313. globalThis.fetch(urlWithPrefix, options as RequestInit)
  314. .then((res) => {
  315. const resClone = res.clone()
  316. // Error handler
  317. if (!/^(2|3)\d{2}$/.test(String(res.status))) {
  318. const bodyJson = res.json()
  319. switch (res.status) {
  320. case 401: {
  321. if (isPublicAPI) {
  322. return bodyJson.then((data: ResponseError) => {
  323. if (!silent)
  324. Toast.notify({ type: 'error', message: data.message })
  325. if (data.code === 'web_sso_auth_required')
  326. requiredWebSSOLogin()
  327. if (data.code === 'unauthorized') {
  328. removeAccessToken()
  329. globalThis.location.reload()
  330. }
  331. return Promise.reject(data)
  332. })
  333. }
  334. const loginUrl = `${globalThis.location.origin}/signin`
  335. bodyJson.then((data: ResponseError) => {
  336. if (data.code === 'init_validate_failed' && IS_CE_EDITION && !silent)
  337. Toast.notify({ type: 'error', message: data.message, duration: 4000 })
  338. else if (data.code === 'not_init_validated' && IS_CE_EDITION)
  339. globalThis.location.href = `${globalThis.location.origin}/init`
  340. else if (data.code === 'not_setup' && IS_CE_EDITION)
  341. globalThis.location.href = `${globalThis.location.origin}/install`
  342. else if (location.pathname !== '/signin' || !IS_CE_EDITION)
  343. globalThis.location.href = loginUrl
  344. else if (!silent)
  345. Toast.notify({ type: 'error', message: data.message })
  346. }).catch(() => {
  347. // Handle any other errors
  348. globalThis.location.href = loginUrl
  349. })
  350. break
  351. }
  352. case 403:
  353. bodyJson.then((data: ResponseError) => {
  354. if (!silent)
  355. Toast.notify({ type: 'error', message: data.message })
  356. if (data.code === 'already_setup')
  357. globalThis.location.href = `${globalThis.location.origin}/signin`
  358. })
  359. break
  360. // fall through
  361. default:
  362. bodyJson.then((data: ResponseError) => {
  363. if (!silent)
  364. Toast.notify({ type: 'error', message: data.message })
  365. })
  366. }
  367. return Promise.reject(resClone)
  368. }
  369. // handle delete api. Delete api not return content.
  370. if (res.status === 204) {
  371. resolve({ result: 'success' })
  372. return
  373. }
  374. // return data
  375. if (options.headers.get('Content-type') === ContentType.download || options.headers.get('Content-type') === ContentType.audio)
  376. resolve(needAllResponseContent ? resClone : res.blob())
  377. else resolve(needAllResponseContent ? resClone : res.json())
  378. })
  379. .catch((err) => {
  380. if (!silent)
  381. Toast.notify({ type: 'error', message: err })
  382. reject(err)
  383. })
  384. }),
  385. ]) as Promise<T>
  386. }
  387. export const upload = (options: any, isPublicAPI?: boolean, url?: string, searchParams?: string): Promise<any> => {
  388. const urlPrefix = isPublicAPI ? PUBLIC_API_PREFIX : API_PREFIX
  389. let token = ''
  390. if (isPublicAPI) {
  391. const sharedToken = globalThis.location.pathname.split('/').slice(-1)[0]
  392. const accessToken = localStorage.getItem('token') || JSON.stringify({ [sharedToken]: '' })
  393. let accessTokenJson = { [sharedToken]: '' }
  394. try {
  395. accessTokenJson = JSON.parse(accessToken)
  396. }
  397. catch (e) {
  398. }
  399. token = accessTokenJson[sharedToken]
  400. }
  401. else {
  402. const accessToken = localStorage.getItem('console_token') || ''
  403. token = accessToken
  404. }
  405. const defaultOptions = {
  406. method: 'POST',
  407. url: (url ? `${urlPrefix}${url}` : `${urlPrefix}/files/upload`) + (searchParams || ''),
  408. headers: {
  409. Authorization: `Bearer ${token}`,
  410. },
  411. data: {},
  412. }
  413. options = {
  414. ...defaultOptions,
  415. ...options,
  416. headers: { ...defaultOptions.headers, ...options.headers },
  417. }
  418. return new Promise((resolve, reject) => {
  419. const xhr = options.xhr
  420. xhr.open(options.method, options.url)
  421. for (const key in options.headers)
  422. xhr.setRequestHeader(key, options.headers[key])
  423. xhr.withCredentials = true
  424. xhr.responseType = 'json'
  425. xhr.onreadystatechange = function () {
  426. if (xhr.readyState === 4) {
  427. if (xhr.status === 201)
  428. resolve(xhr.response)
  429. else
  430. reject(xhr)
  431. }
  432. }
  433. xhr.upload.onprogress = options.onprogress
  434. xhr.send(options.data)
  435. })
  436. }
  437. export const ssePost = (
  438. url: string,
  439. fetchOptions: FetchOptionType,
  440. {
  441. isPublicAPI = false,
  442. onData,
  443. onCompleted,
  444. onThought,
  445. onFile,
  446. onMessageEnd,
  447. onMessageReplace,
  448. onWorkflowStarted,
  449. onWorkflowFinished,
  450. onNodeStarted,
  451. onNodeFinished,
  452. onIterationStart,
  453. onIterationNext,
  454. onIterationFinish,
  455. onTextChunk,
  456. onTTSChunk,
  457. onTTSEnd,
  458. onTextReplace,
  459. onError,
  460. getAbortController,
  461. }: IOtherOptions,
  462. ) => {
  463. const abortController = new AbortController()
  464. const options = Object.assign({}, baseOptions, {
  465. method: 'POST',
  466. signal: abortController.signal,
  467. }, fetchOptions)
  468. const contentType = options.headers.get('Content-Type')
  469. if (!contentType)
  470. options.headers.set('Content-Type', ContentType.json)
  471. getAbortController?.(abortController)
  472. const urlPrefix = isPublicAPI ? PUBLIC_API_PREFIX : API_PREFIX
  473. const urlWithPrefix = `${urlPrefix}${url.startsWith('/') ? url : `/${url}`}`
  474. const { body } = options
  475. if (body)
  476. options.body = JSON.stringify(body)
  477. globalThis.fetch(urlWithPrefix, options as RequestInit)
  478. .then((res) => {
  479. if (!/^(2|3)\d{2}$/.test(String(res.status))) {
  480. res.json().then((data: any) => {
  481. Toast.notify({ type: 'error', message: data.message || 'Server Error' })
  482. if (isPublicAPI) {
  483. if (data.code === 'web_sso_auth_required')
  484. requiredWebSSOLogin()
  485. if (data.code === 'unauthorized') {
  486. removeAccessToken()
  487. globalThis.location.reload()
  488. }
  489. }
  490. })
  491. onError?.('Server Error')
  492. return
  493. }
  494. return handleStream(res, (str: string, isFirstMessage: boolean, moreInfo: IOnDataMoreInfo) => {
  495. if (moreInfo.errorMessage) {
  496. onError?.(moreInfo.errorMessage, moreInfo.errorCode)
  497. if (moreInfo.errorMessage !== 'AbortError: The user aborted a request.')
  498. Toast.notify({ type: 'error', message: moreInfo.errorMessage })
  499. return
  500. }
  501. onData?.(str, isFirstMessage, moreInfo)
  502. }, onCompleted, onThought, onMessageEnd, onMessageReplace, onFile, onWorkflowStarted, onWorkflowFinished, onNodeStarted, onNodeFinished, onIterationStart, onIterationNext, onIterationFinish, onTextChunk, onTTSChunk, onTTSEnd, onTextReplace)
  503. }).catch((e) => {
  504. if (e.toString() !== 'AbortError: The user aborted a request.')
  505. Toast.notify({ type: 'error', message: e })
  506. onError?.(e)
  507. })
  508. }
  509. // base request
  510. export const request = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  511. return baseFetch<T>(url, options, otherOptions || {})
  512. }
  513. // request methods
  514. export const get = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  515. return request<T>(url, Object.assign({}, options, { method: 'GET' }), otherOptions)
  516. }
  517. // For public API
  518. export const getPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  519. return get<T>(url, options, { ...otherOptions, isPublicAPI: true })
  520. }
  521. export const post = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  522. return request<T>(url, Object.assign({}, options, { method: 'POST' }), otherOptions)
  523. }
  524. export const postPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  525. return post<T>(url, options, { ...otherOptions, isPublicAPI: true })
  526. }
  527. export const put = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  528. return request<T>(url, Object.assign({}, options, { method: 'PUT' }), otherOptions)
  529. }
  530. export const putPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  531. return put<T>(url, options, { ...otherOptions, isPublicAPI: true })
  532. }
  533. export const del = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  534. return request<T>(url, Object.assign({}, options, { method: 'DELETE' }), otherOptions)
  535. }
  536. export const delPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  537. return del<T>(url, options, { ...otherOptions, isPublicAPI: true })
  538. }
  539. export const patch = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  540. return request<T>(url, Object.assign({}, options, { method: 'PATCH' }), otherOptions)
  541. }
  542. export const patchPublic = <T>(url: string, options = {}, otherOptions?: IOtherOptions) => {
  543. return patch<T>(url, options, { ...otherOptions, isPublicAPI: true })
  544. }