base.ts 23 KB

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