import { escape } from 'lodash-es' export const sleep = (ms: number) => { return new Promise(resolve => setTimeout(resolve, ms)) } export async function asyncRunSafe(fn: Promise): Promise<[Error] | [null, T]> { try { return [null, await fn] } catch (e: any) { return [e || new Error('unknown error')] } } export const getTextWidthWithCanvas = (text: string, font?: string) => { const canvas = document.createElement('canvas') const ctx = canvas.getContext('2d') if (ctx) { ctx.font = font ?? '12px Inter, ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"' return Number(ctx.measureText(text).width.toFixed(2)) } return 0 } const chars = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ-_' export function randomString(length: number) { let result = '' for (let i = length; i > 0; --i) result += chars[Math.floor(Math.random() * chars.length)] return result } export const getPurifyHref = (href: string) => { if (!href) return '' return escape(href) } export async function fetchWithRetry(fn: Promise, retries = 3): Promise<[Error] | [null, T]> { const [error, res] = await asyncRunSafe(fn) if (error) { if (retries > 0) { const res = await fetchWithRetry(fn, retries - 1) return res } else { if (error instanceof Error) return [error] return [new Error('unknown error')] } } else { return [null, res] } } export const correctModelProvider = (provider: string) => { if (!provider) return '' if (provider.includes('/')) return provider if (['google'].includes(provider)) return 'langgenius/gemini/google' return `langgenius/${provider}/${provider}` } export const correctToolProvider = (provider: string, toolInCollectionList?: boolean) => { if (!provider) return '' if (toolInCollectionList) return provider if (provider.includes('/')) return provider if (['stepfun', 'jina', 'siliconflow', 'gitee_ai'].includes(provider)) return `langgenius/${provider}_tool/${provider}` return `langgenius/${provider}/${provider}` } export const canFindTool = (providerId: string, oldToolId?: string) => { return providerId === oldToolId || providerId === `langgenius/${oldToolId}/${oldToolId}` || providerId === `langgenius/${oldToolId}_tool/${oldToolId}` } export const removeSpecificQueryParam = (key: string | string[]) => { const url = new URL(window.location.href) if (Array.isArray(key)) key.forEach(k => url.searchParams.delete(k)) else url.searchParams.delete(key) window.history.replaceState(null, '', url.toString()) }