base.ts 20 KB

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