base.ts 22 KB

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