mirror of
https://github.com/CherryHQ/cherry-studio.git
synced 2026-01-08 22:39:36 +08:00
* fix: prevent OOM when handling large base64 image data - Add memory-safe parseDataUrl utility using string operations instead of regex - Truncate large base64 data in ErrorBlock detail modal to prevent freezing - Update ImageViewer, FileStorage, messageConverter to use shared parseDataUrl - Deprecate parseDataUrlMediaType in favor of shared utility - Add GB support to formatFileSize - Add comprehensive unit tests for parseDataUrl (18 tests) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> * refactor: simplify parseDataUrl API to return DataUrlParts | null - Change return type from discriminated union to simple nullable type - Update all call sites to use optional chaining (?.) - Update tests to use toBeNull() for failure cases - More idiomatic and consistent with codebase patterns (e.g., parseJSON) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> --------- Co-authored-by: Claude <noreply@anthropic.com>
169 lines
5.4 KiB
TypeScript
169 lines
5.4 KiB
TypeScript
export const defaultAppHeaders = () => {
|
|
return {
|
|
'HTTP-Referer': 'https://cherry-ai.com',
|
|
'X-Title': 'Cherry Studio'
|
|
}
|
|
}
|
|
|
|
// Following two function are not being used for now.
|
|
// I may use them in the future, so just keep them commented. - by eurfelux
|
|
|
|
/**
|
|
* Converts an `undefined` value to `null`, otherwise returns the value as-is.
|
|
* @param value - The value to check
|
|
* @returns `null` if the input is `undefined`; otherwise the input value
|
|
*/
|
|
|
|
// export function toNullIfUndefined<T>(value: T | undefined): T | null {
|
|
// if (value === undefined) {
|
|
// return null
|
|
// } else {
|
|
// return value
|
|
// }
|
|
// }
|
|
|
|
/**
|
|
* Converts a `null` value to `undefined`, otherwise returns the value as-is.
|
|
* @param value - The value to check
|
|
* @returns `undefined` if the input is `null`; otherwise the input value
|
|
*/
|
|
|
|
// export function toUndefinedIfNull<T>(value: T | null): T | undefined {
|
|
// if (value === null) {
|
|
// return undefined
|
|
// } else {
|
|
// return value
|
|
// }
|
|
// }
|
|
|
|
/**
|
|
* Extracts the trailing API version segment from a URL path.
|
|
*
|
|
* This function extracts API version patterns (e.g., `v1`, `v2beta`) from the end of a URL.
|
|
* Only versions at the end of the path are extracted, not versions in the middle.
|
|
* The returned version string does not include leading or trailing slashes.
|
|
*
|
|
* @param {string} url - The URL string to parse.
|
|
* @returns {string | undefined} The trailing API version found (e.g., 'v1', 'v2beta'), or undefined if none found.
|
|
*
|
|
* @example
|
|
* getTrailingApiVersion('https://api.example.com/v1') // 'v1'
|
|
* getTrailingApiVersion('https://api.example.com/v2beta/') // 'v2beta'
|
|
* getTrailingApiVersion('https://api.example.com/v1/chat') // undefined (version not at end)
|
|
* getTrailingApiVersion('https://gateway.ai.cloudflare.com/v1/xxx/v1beta') // 'v1beta'
|
|
* getTrailingApiVersion('https://api.example.com') // undefined
|
|
*/
|
|
export function getTrailingApiVersion(url: string): string | undefined {
|
|
const match = url.match(TRAILING_VERSION_REGEX)
|
|
|
|
if (match) {
|
|
// Extract version without leading slash and trailing slash
|
|
return match[0].replace(/^\//, '').replace(/\/$/, '')
|
|
}
|
|
|
|
return undefined
|
|
}
|
|
|
|
/**
|
|
* Matches an API version at the end of a URL (with optional trailing slash).
|
|
* Used to detect and extract versions only from the trailing position.
|
|
*/
|
|
const TRAILING_VERSION_REGEX = /\/v\d+(?:alpha|beta)?\/?$/i
|
|
|
|
/**
|
|
* Removes the trailing API version segment from a URL path.
|
|
*
|
|
* This function removes API version patterns (e.g., `/v1`, `/v2beta`) from the end of a URL.
|
|
* Only versions at the end of the path are removed, not versions in the middle.
|
|
*
|
|
* @param {string} url - The URL string to process.
|
|
* @returns {string} The URL with the trailing API version removed, or the original URL if no trailing version found.
|
|
*
|
|
* @example
|
|
* withoutTrailingApiVersion('https://api.example.com/v1') // 'https://api.example.com'
|
|
* withoutTrailingApiVersion('https://api.example.com/v2beta/') // 'https://api.example.com'
|
|
* withoutTrailingApiVersion('https://api.example.com/v1/chat') // 'https://api.example.com/v1/chat' (no change)
|
|
* withoutTrailingApiVersion('https://api.example.com') // 'https://api.example.com'
|
|
*/
|
|
export function withoutTrailingApiVersion(url: string): string {
|
|
return url.replace(TRAILING_VERSION_REGEX, '')
|
|
}
|
|
|
|
export interface DataUrlParts {
|
|
/** The media type (e.g., 'image/png', 'text/plain') */
|
|
mediaType?: string
|
|
/** Whether the data is base64 encoded */
|
|
isBase64: boolean
|
|
/** The data portion (everything after the comma). This is the raw string, not decoded. */
|
|
data: string
|
|
}
|
|
|
|
/**
|
|
* Parses a data URL into its component parts without using regex on the data portion.
|
|
* This is memory-safe for large data URLs (e.g., 4K images) as it uses indexOf instead of regex.
|
|
*
|
|
* Data URL format: data:[<mediatype>][;base64],<data>
|
|
*
|
|
* @param url - The data URL string to parse
|
|
* @returns DataUrlParts if valid, null if invalid
|
|
*
|
|
* @example
|
|
* parseDataUrl('...')
|
|
* // { mediaType: 'image/png', isBase64: true, data: 'iVBORw0KGgo...' }
|
|
*
|
|
* parseDataUrl('data:text/plain,Hello')
|
|
* // { mediaType: 'text/plain', isBase64: false, data: 'Hello' }
|
|
*
|
|
* parseDataUrl('invalid-url')
|
|
* // null
|
|
*/
|
|
export function parseDataUrl(url: string): DataUrlParts | null {
|
|
if (!url.startsWith('data:')) {
|
|
return null
|
|
}
|
|
|
|
const commaIndex = url.indexOf(',')
|
|
if (commaIndex === -1) {
|
|
return null
|
|
}
|
|
|
|
const header = url.slice(5, commaIndex)
|
|
|
|
const isBase64 = header.includes(';base64')
|
|
|
|
const semicolonIndex = header.indexOf(';')
|
|
const mediaType = (semicolonIndex === -1 ? header : header.slice(0, semicolonIndex)).trim() || undefined
|
|
|
|
const data = url.slice(commaIndex + 1)
|
|
|
|
return { mediaType, isBase64, data }
|
|
}
|
|
|
|
/**
|
|
* Checks if a string is a data URL.
|
|
*
|
|
* @param url - The string to check
|
|
* @returns true if the string is a valid data URL
|
|
*/
|
|
export function isDataUrl(url: string): boolean {
|
|
return url.startsWith('data:') && url.includes(',')
|
|
}
|
|
|
|
/**
|
|
* Checks if a data URL contains base64-encoded image data.
|
|
*
|
|
* @param url - The data URL to check
|
|
* @returns true if the URL is a base64-encoded image data URL
|
|
*/
|
|
export function isBase64ImageDataUrl(url: string): boolean {
|
|
if (!url.startsWith('data:image/')) {
|
|
return false
|
|
}
|
|
const commaIndex = url.indexOf(',')
|
|
if (commaIndex === -1) {
|
|
return false
|
|
}
|
|
const header = url.slice(5, commaIndex)
|
|
return header.includes(';base64')
|
|
}
|