@lynker-desktop/electron-sdk 0.0.9-alpha.60 → 0.0.9-alpha.61
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm/main/downloader.d.ts +17 -5
- package/esm/main/downloader.d.ts.map +1 -1
- package/esm/main/downloader.js +35 -14
- package/esm/main/downloader.js.map +1 -1
- package/esm/main/resource-cache.d.ts +1 -1
- package/esm/main/resource-cache.d.ts.map +1 -1
- package/esm/main/resource-cache.js +36 -8
- package/esm/main/resource-cache.js.map +1 -1
- package/main/downloader.d.ts +17 -5
- package/main/downloader.d.ts.map +1 -1
- package/main/downloader.js +35 -14
- package/main/downloader.js.map +1 -1
- package/main/resource-cache.d.ts +1 -1
- package/main/resource-cache.d.ts.map +1 -1
- package/main/resource-cache.js +36 -8
- package/main/resource-cache.js.map +1 -1
- package/package.json +5 -5
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"resource-cache.js","sources":["../../src/main/resource-cache.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport md5 from 'md5';\nimport mime from 'mime-types';\nimport { ipcMain, webContents } from 'electron';\nimport { DownloaderHelper } from 'node-downloader-helper';\n\n/**\n * 默认文件扩展名\n */\nconst DEFAULT_EXT = '.res';\n\n/**\n * 默认 MIME 类型\n */\nconst DEFAULT_MIME_TYPE = 'application/octet-stream';\n\n/**\n * 最大重定向次数\n */\nconst MAX_REDIRECTS = 5;\n\n/**\n * 资源缓存配置项\n */\nexport interface ResourceCacheOptions {\n /** 缓存目录,必填 */\n cacheDir: string;\n /** 缓存有效期(毫秒),默认24小时 */\n cacheTTL?: number;\n /** 匹配需要缓存的资源,支持正则或函数 */\n match?: RegExp | ((url: string) => boolean);\n /** 允许缓存的资源来源,支持null/数组/函数 */\n allowedOrigins?: null | string[] | ((url: string) => boolean);\n}\n\n/**\n * 下载进度回调函数类型\n */\nexport type DownloadProgressCallback = (progress: {\n url: string; // 正在下载的 URL\n downloaded: number; // 已下载字节数\n total: number; // 总字节数(如果服务器提供 Content-Length)\n percentage: number; // 下载进度百分比 (0-100),如果 total 未知则为 -1\n speed: number; // 下载速度(字节/秒)\n}) => void;\n\n/**\n * 缓存进度回调函数类型\n */\nexport type CacheProgressCallback = (progress: {\n current: number; // 当前完成的 URL 数量\n total: number; // 总 URL 数量\n url: string; // 当前处理的 URL\n success: boolean; // 当前 URL 是否成功\n result?: { // 当前 URL 的处理结果\n filePath?: string;\n hostPath?: string;\n mimeType?: string;\n size?: number;\n error?: string;\n };\n percentage: number; // 进度百分比 (0-100)\n downloadProgress?: { // 下载进度(如果正在下载)\n downloaded: number;\n total: number;\n percentage: number;\n speed: number;\n };\n}) => void;\n\n/**\n * 默认配置\n */\nconst DEFAULT_OPTIONS: Required<Omit<ResourceCacheOptions, 'cacheDir'>> & { cacheDir: string } = {\n cacheDir: '',\n cacheTTL: 24 * 60 * 60 * 1000,\n // 图片格式:png, jpg/jpeg, webp, gif, svg, ico, bmp, avif, heic, heif, tiff, tif\n // 字体格式:woff, woff2, ttf, eot, otf\n // 视频格式:mp4, webm, ogg, mov, avi, mkv, flv, m4v, 3gp\n // 音频格式:mp3, wav, aac, m4a, flac, opus, wma\n // 样式和脚本:css, js, json, xml, txt\n // Web资源:wasm, map (source map)\n // 文档格式:pdf\n // 压缩文件:zip, 7z, rar, tar, gz, bz2\n match: /\\.(png|jpe?g|webp|gif|svg|ico|bmp|avif|heic|heif|tiff?|woff2?|ttf|eot|otf|mp4|webm|ogg|mov|avi|mkv|flv|m4v|3gp|mp3|wav|aac|m4a|flac|opus|wma|css|js|json|xml|txt|wasm|map|pdf|zip|7z|rar|tar|gz|bz2)(\\?.*)?$/i,\n allowedOrigins: null,\n};\n\n/**\n * 资源缓存类:拦截并缓存静态资源,提升加载性能\n */\nexport class ResourceCache {\n static scheme = 'cachefile';\n private cacheHost: string = `${ResourceCache.scheme}://-`;\n /** Electron session 实例 */\n private session: Electron.Session;\n /** 缓存配置 */\n private options: Required<ResourceCacheOptions>;\n /** 缓存的匹配函数(避免重复创建) */\n private _cachedMatchFunction?: (url: string) => boolean;\n /** 缓存的来源校验函数(避免重复创建) */\n private _cachedOriginFunction?: (url: string) => boolean;\n /** 正在下载的 URL 集合(避免重复下载) */\n private _downloadingUrls = new Set<string>();\n\n /**\n * 构造函数\n * @param session Electron session\n * @param options 缓存配置\n */\n constructor(session: Electron.Session, options: ResourceCacheOptions) {\n if (!session) throw new Error('ResourceCache: session is required');\n this.session = session;\n // 合并配置,保证类型安全\n this.options = {\n ...DEFAULT_OPTIONS,\n ...options,\n cacheDir: options.cacheDir,\n cacheTTL: options.cacheTTL ?? DEFAULT_OPTIONS.cacheTTL,\n match: options.match ?? DEFAULT_OPTIONS.match,\n allowedOrigins: options.allowedOrigins ?? DEFAULT_OPTIONS.allowedOrigins,\n };\n\n if (!this.options.cacheDir) {\n throw new Error('ResourceCache: cacheDir is required');\n }\n\n // 确保缓存目录存在\n if (!fs.existsSync(this.options.cacheDir)) {\n fs.mkdirSync(this.options.cacheDir, { recursive: true });\n }\n\n this._registerInterceptor();\n // 异步清理过期缓存,不阻塞初始化\n this._cleanOldCache().catch(err => {\n console.log('初始化时清理过期缓存失败:', err);\n });\n\n ipcMain.handle('core:cache', async (event: Electron.IpcMainInvokeEvent, options: { id: string, method: 'add' | 'delete' | 'clear' | 'stats', urls?: string | string[], force?: boolean }) => {\n try {\n switch (options.method) {\n case 'clear':\n return await this.clearCache();\n case 'add': {\n const { id } = options;\n const sender = event.sender as Electron.WebContents;\n const urls = Array.isArray(options.urls) ? options.urls : options.urls ? [options.urls] : [];\n const data = await this.addCacheUrls(urls, options.force ?? false, true, (data: any) => {\n try {\n if (sender) {\n if (typeof sender?.send === 'function') {\n sender?.send?.(`core:cache:progress`, {\n id,\n data\n });\n }\n }\n } catch (error) {\n console.log('发送缓存进度回调失败:', error);\n }\n });\n return Array.isArray(options.urls) ? data : data[0];\n }\n case 'delete': {\n const urls = Array.isArray(options.urls) ? options.urls : options.urls ? [options.urls] : [];\n const data = await this.deleteCacheUrls(urls);\n return Array.isArray(options.urls) ? data : data[0];\n }\n case 'stats':\n return await this.getCacheStats();\n default:\n return undefined;\n }\n } catch (error) {\n return { success: false, error: error instanceof Error ? error.message : '未知错误' };\n }\n });\n }\n\n /**\n * 获取缓存统计信息(异步版本,性能更好)\n */\n public async getCacheStats(): Promise<{ size: number, totalSize: number }> {\n try {\n const files = await fs.promises.readdir(this.options.cacheDir);\n if (files.length === 0) {\n return { size: 0, totalSize: 0 };\n }\n\n // 并行获取文件信息,提升性能\n const fileInfos = await Promise.allSettled(\n files.map(async (file) => {\n const filePath = path.join(this.options.cacheDir, file);\n try {\n const stats = await fs.promises.stat(filePath);\n return { file, filePath, size: stats.size };\n } catch (error) {\n return { file, filePath, size: 0, error };\n }\n })\n );\n\n const validInfos = fileInfos\n .filter((r): r is PromiseFulfilledResult<{ file: string; filePath: string; size: number; error?: any }> => r.status === 'fulfilled')\n .map(r => r.value);\n\n return {\n size: validInfos.length,\n totalSize: validInfos.reduce((sum, info) => sum + info.size, 0)\n };\n } catch (error) {\n console.log('获取缓存统计信息失败:', error);\n return { size: 0, totalSize: 0 };\n }\n }\n\n /**\n * 获取资源匹配函数(带缓存,避免重复创建)\n */\n private _getMatchFunction(): (url: string) => boolean {\n if (this._cachedMatchFunction) {\n return this._cachedMatchFunction;\n }\n\n const matcher = this.options.match;\n if (typeof matcher === 'function') {\n this._cachedMatchFunction = matcher;\n } else if (matcher instanceof RegExp) {\n this._cachedMatchFunction = (url: string) => matcher.test(url);\n } else {\n this._cachedMatchFunction = () => false;\n }\n\n return this._cachedMatchFunction;\n }\n\n /**\n * 获取来源校验函数(带缓存,避免重复创建)\n */\n private _getOriginAllowFunction(): (url: string) => boolean {\n if (this._cachedOriginFunction) {\n return this._cachedOriginFunction;\n }\n\n const origins = this.options.allowedOrigins;\n if (!origins) {\n this._cachedOriginFunction = () => true;\n } else if (typeof origins === 'function') {\n this._cachedOriginFunction = origins;\n } else {\n const prefixList = origins.map(o => o.toLowerCase());\n this._cachedOriginFunction = (url: string) => {\n try {\n const origin = new URL(url).origin.toLowerCase();\n return prefixList.some(prefix => origin.startsWith(prefix));\n } catch {\n return false;\n }\n };\n }\n\n return this._cachedOriginFunction;\n }\n\n /**\n * 获取缓存文件路径\n * @param url 资源URL\n * @param customExt 自定义文件扩展名(可选,用于 base64 URL)\n */\n public getCachedPath(url: string, customExt?: string): { filePath: string, hostPath: string } {\n const md5Str = md5(url);\n let ext = '.res';\n\n if (customExt) {\n // 如果提供了自定义扩展名,使用它\n ext = customExt.startsWith('.') ? customExt : `.${customExt}`;\n } else {\n // 尝试从 URL 中提取扩展名\n try {\n const urlObj = new URL(url);\n ext = path.extname(urlObj.pathname) || DEFAULT_EXT;\n } catch {\n // 如果 URL 解析失败(可能是 base64 data URL),使用默认扩展名\n ext = DEFAULT_EXT;\n }\n }\n\n return {\n filePath: path.join(this.options.cacheDir, `${md5Str}${ext}`),\n hostPath: `${this.cacheHost}/${md5Str}${ext}`,\n }\n }\n\n /**\n * 判断缓存是否有效(同步版本,用于拦截器)\n * @param filePath 缓存文件路径\n */\n public isCacheValid(filePath: string): boolean {\n try {\n if (!fs.existsSync(filePath)) return false;\n const stat = fs.statSync(filePath);\n return Date.now() - stat.mtimeMs < this.options.cacheTTL;\n } catch {\n return false;\n }\n }\n\n /**\n * 判断缓存是否有效(异步版本,性能更好)\n * @param filePath 缓存文件路径\n */\n public async isCacheValidAsync(filePath: string): Promise<boolean> {\n try {\n const stat = await fs.promises.stat(filePath);\n return Date.now() - stat.mtimeMs < this.options.cacheTTL;\n } catch {\n return false;\n }\n }\n\n /**\n * 下载资源到本地缓存(异步版本,返回 Promise)\n * @param url 资源URL\n * @param filePath 本地缓存路径\n * @param redirectCount 当前重定向次数(内部使用)\n * @param onProgress 下载进度回调函数(可选)\n * @returns Promise<void> 下载完成或失败\n */\n public downloadResourceAsync(\n url: string,\n filePath: string,\n redirectCount: number = 0,\n onProgress?: DownloadProgressCallback\n ): Promise<void> {\n // 检查是否正在下载,避免重复下载\n if (this._downloadingUrls.has(url)) {\n return Promise.reject(new Error(`资源正在下载中: ${url}`));\n }\n\n // 检查重定向次数限制\n if (redirectCount >= MAX_REDIRECTS) {\n return Promise.reject(new Error(`重定向次数超过限制 (${MAX_REDIRECTS}): ${url}`));\n }\n\n this._downloadingUrls.add(url);\n\n return new Promise((resolve, reject) => {\n const tempFilePath = `${filePath}.cache`;\n const downloadDir = path.dirname(tempFilePath);\n const downloadFileName = path.basename(tempFilePath);\n\n // 确保目录存在\n if (!fs.existsSync(downloadDir)) {\n fs.mkdirSync(downloadDir, { recursive: true });\n }\n\n // 创建下载器实例\n const dl = new DownloaderHelper(url, downloadDir, {\n fileName: downloadFileName,\n retry: { maxRetries: 3, delay: 1000 },\n // 超时设置:10 分钟(600000 毫秒)\n // 对于视频等大文件,需要更长的超时时间\n // 如果 10 分钟内没有任何数据传输,才会超时\n timeout: 600000,\n override: true, // 覆盖已存在的文件\n httpRequestOptions: {\n // 允许重定向\n followRedirect: true,\n maxRedirects: MAX_REDIRECTS\n }\n });\n\n // 监听下载进度\n if (onProgress) {\n dl.on('progress', (stats) => {\n try {\n // stats 包含: progress (百分比), downloaded (已下载字节), total (总字节), speed (速度)\n onProgress({\n url,\n downloaded: stats.downloaded || 0,\n total: stats.total || 0,\n percentage: stats.progress !== undefined ? Math.round(stats.progress) : -1,\n speed: stats.speed || 0\n });\n } catch (error) {\n // 忽略回调中的错误,避免影响下载\n console.error('进度回调执行失败:', error);\n }\n });\n }\n\n // 监听下载完成\n dl.on('end', () => {\n this._downloadingUrls.delete(url);\n // 将临时文件重命名为最终文件\n fs.rename(tempFilePath, filePath, (renameErr) => {\n if (renameErr) {\n reject(new Error(`缓存文件重命名失败 from ${tempFilePath} to ${filePath}: ${renameErr.message}`));\n } else {\n resolve();\n }\n });\n });\n\n // 监听下载错误\n dl.on('error', (err) => {\n this._downloadingUrls.delete(url);\n // 清理临时文件\n fs.promises.unlink(tempFilePath).catch(() => {\n // 忽略删除失败\n });\n reject(err instanceof Error ? err : new Error(`下载失败: ${url}`));\n });\n\n // 监听下载停止(取消)\n dl.on('stop', () => {\n this._downloadingUrls.delete(url);\n // 清理临时文件\n fs.promises.unlink(tempFilePath).catch(() => {\n // 忽略删除失败\n });\n reject(new Error(`下载已停止: ${url}`));\n });\n\n // 开始下载\n dl.start().catch((err) => {\n this._downloadingUrls.delete(url);\n reject(err instanceof Error ? err : new Error(`启动下载失败: ${url}`));\n });\n });\n }\n\n /**\n * 下载资源到本地缓存(同步版本,不返回 Promise,用于拦截器)\n * @param url 资源URL\n * @param filePath 本地缓存路径\n */\n public downloadResource(url: string, filePath: string): void {\n // 异步执行,不等待完成\n // 如果正在下载,跳过(避免重复下载)\n if (this._downloadingUrls.has(url)) {\n return;\n }\n this.downloadResourceAsync(url, filePath).catch(() => {\n // 静默处理错误,避免日志过多\n });\n }\n\n /**\n * 从文件扩展名获取 MIME 类型\n * @param ext 文件扩展名(带或不带点)\n * @returns MIME 类型\n */\n private _getMimeTypeFromExt(ext: string): string {\n const cleanExt = ext.replace(/^\\./, '').toLowerCase();\n const mimeType = mime.lookup(cleanExt);\n return mimeType || DEFAULT_MIME_TYPE;\n }\n\n /**\n * 检测并处理 base64 data URL\n * @param url 资源URL\n * @returns 如果是 base64 URL,返回 true、文件扩展名、MIME 类型和数据;否则返回 false\n */\n private _isBase64DataUrl(url: string): { isBase64: boolean, ext?: string, mimeType?: string, data?: string } {\n if (!url.startsWith('data:')) {\n return { isBase64: false };\n }\n\n try {\n // 解析 data URL 格式:data:[<mediatype>][;base64],<data>\n const commaIndex = url.indexOf(',');\n if (commaIndex === -1) {\n return { isBase64: false };\n }\n\n const header = url.substring(0, commaIndex);\n const data = url.substring(commaIndex + 1);\n\n // 检查是否包含 base64 标识\n if (!header.includes('base64')) {\n return { isBase64: false };\n }\n\n // 从 mediatype 中提取文件扩展名和 MIME 类型\n // 例如:data:image/png;base64 -> png, image/png\n // 例如:data:image/jpeg;base64 -> jpeg, image/jpeg\n let ext = DEFAULT_EXT.replace(/^\\./, '');\n let mimeType = DEFAULT_MIME_TYPE;\n const mimeMatch = header.match(/data:([^;]+)/);\n if (mimeMatch && mimeMatch[1]) {\n mimeType = mimeMatch[1];\n // 使用 mime-types 包从 MIME 类型获取扩展名\n const extension = mime.extension(mimeType);\n if (extension) {\n ext = extension;\n }\n }\n\n return { isBase64: true, ext, mimeType, data };\n } catch (error) {\n return { isBase64: false };\n }\n }\n\n /**\n * 保存 base64 数据到文件\n * @param base64Data base64 编码的数据\n * @param filePath 目标文件路径\n */\n private async _saveBase64ToFile(base64Data: string, filePath: string): Promise<void> {\n try {\n // 解码 base64 数据\n const buffer = Buffer.from(base64Data, 'base64');\n\n // 使用 Promise 版本的 writeFile\n // Buffer 继承自 Uint8Array,可以直接使用\n await fs.promises.writeFile(filePath, buffer as Uint8Array);\n } catch (error) {\n throw new Error(`保存 base64 文件失败: ${error instanceof Error ? error.message : '未知错误'}`);\n }\n }\n\n /**\n * 手动缓存指定 URL 的资源\n * @param url 要缓存的资源 URL(支持普通 URL 和 base64 data URL)\n * @param force 是否强制重新下载,即使缓存有效(默认 false)\n * @param ignoreOrigin 是否忽略来源检查(默认 false)\n * @param onDownloadProgress 下载进度回调函数(可选)\n * @returns Promise<{ filePath: string, hostPath: string, mimeType: string, size: number }> 返回缓存文件路径、主机路径、MIME 类型和文件大小\n * @throws 如果 URL 不匹配缓存规则或来源不允许,会抛出错误\n */\n public async cacheUrl(\n url: string,\n force: boolean = false,\n ignoreOrigin: boolean = false,\n onDownloadProgress?: DownloadProgressCallback\n ): Promise<{ filePath: string, hostPath: string, mimeType: string, size: number }> {\n // 检查是否是 base64 data URL\n const base64Info = this._isBase64DataUrl(url);\n\n if (base64Info.isBase64) {\n // 处理 base64 data URL\n if (!base64Info.data || !base64Info.ext) {\n throw new Error(`无效的 base64 data URL: ${url}`);\n }\n\n // 获取缓存路径(使用检测到的扩展名)\n const cachePath = this.getCachedPath(url, base64Info.ext);\n const mimeType = base64Info.mimeType || this._getMimeTypeFromExt(base64Info.ext);\n\n // 如果缓存有效且不强制重新下载,直接返回\n if (!force && await this.isCacheValidAsync(cachePath.filePath)) {\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n // 保存 base64 数据到文件\n await this._saveBase64ToFile(base64Info.data, cachePath.filePath);\n\n // 获取文件大小\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n // 处理普通 URL\n const shouldCache = ignoreOrigin ? () => true : this._getMatchFunction();\n const isAllowedOrigin = ignoreOrigin ? () => true : this._getOriginAllowFunction();\n\n // 检查是否匹配缓存规则\n if (!shouldCache(url)) {\n throw new Error(`URL 不匹配缓存规则: ${url}`);\n }\n\n // 检查来源是否允许\n if (!isAllowedOrigin(url)) {\n throw new Error(`URL 来源不允许缓存: ${url}`);\n }\n\n // 获取缓存路径\n const cachePath = this.getCachedPath(url);\n\n // 从文件扩展名获取 MIME 类型\n const ext = path.extname(cachePath.filePath).replace(/^\\./, '') || DEFAULT_EXT.replace(/^\\./, '');\n const mimeType = this._getMimeTypeFromExt(ext);\n\n // 如果缓存有效且不强制重新下载,直接返回\n if (!force && await this.isCacheValidAsync(cachePath.filePath)) {\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n // 下载资源\n await this.downloadResourceAsync(url, cachePath.filePath, 0, onDownloadProgress);\n\n // 获取文件大小\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n /**\n * 批量缓存多个 URL 的资源\n * @param urls 要缓存的资源 URL 数组\n * @param force 是否强制重新下载,即使缓存有效(默认 false)\n * @param ignoreOrigin 是否忽略来源检查(默认 false)\n * @param onProgress 进度回调函数(可选)\n * @returns Promise<Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, mimeType?: string, size?: number, error?: string }>> 返回每个 URL 的缓存结果\n */\n public async addCacheUrls(\n urls: string[],\n force: boolean = false,\n ignoreOrigin: boolean = false,\n onProgress?: CacheProgressCallback\n ): Promise<Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, mimeType?: string, size?: number, error?: string }>> {\n const total = urls.length;\n const results: Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, mimeType?: string, size?: number, error?: string }> = [];\n let completed = 0;\n\n // 存储每个 URL 的下载进度\n const downloadProgressMap = new Map<string, { downloaded: number; total: number; percentage: number; speed: number }>();\n\n // 进度回调的包装函数,确保线程安全\n const reportProgress = (url: string, success: boolean, result?: { filePath?: string; hostPath?: string; mimeType?: string; size?: number; error?: string }) => {\n if (onProgress) {\n completed++;\n const downloadProgress = downloadProgressMap.get(url);\n onProgress({\n current: completed,\n total,\n url,\n success,\n result,\n percentage: Math.round((completed / total) * 100),\n downloadProgress: downloadProgress ? {\n downloaded: downloadProgress.downloaded,\n total: downloadProgress.total,\n percentage: downloadProgress.percentage,\n speed: downloadProgress.speed\n } : undefined\n });\n }\n };\n\n // 并行处理所有 URL,但跟踪每个的完成状态\n const promises = urls.map(async (url, index) => {\n // 为每个 URL 创建下载进度回调\n const onDownloadProgress: DownloadProgressCallback | undefined = onProgress\n ? (progress) => {\n // 更新下载进度\n downloadProgressMap.set(url, {\n downloaded: progress.downloaded,\n total: progress.total,\n percentage: progress.percentage,\n speed: progress.speed\n });\n // 实时报告下载进度(不增加 completed 计数)\n if (onProgress) {\n onProgress({\n current: completed,\n total,\n url,\n success: true, // 下载中视为进行中\n result: undefined,\n percentage: Math.round((completed / total) * 100),\n downloadProgress: {\n downloaded: progress.downloaded,\n total: progress.total,\n percentage: progress.percentage,\n speed: progress.speed\n }\n });\n }\n }\n : undefined;\n\n try {\n const result = await this.cacheUrl(url, force, ignoreOrigin, onDownloadProgress);\n const item = {\n url,\n success: true,\n filePath: result.filePath,\n hostPath: result.hostPath,\n mimeType: result.mimeType,\n size: result.size\n };\n results[index] = item;\n\n // 清除下载进度(已完成)\n downloadProgressMap.delete(url);\n\n // 调用进度回调\n reportProgress(url, true, {\n filePath: result.filePath,\n hostPath: result.hostPath,\n mimeType: result.mimeType,\n size: result.size\n });\n\n return item;\n } catch (error) {\n const item = {\n url,\n success: false,\n error: error instanceof Error ? error.message : '未知错误'\n };\n results[index] = item;\n\n // 清除下载进度(失败)\n downloadProgressMap.delete(url);\n\n // 调用进度回调\n reportProgress(url, false, {\n error: item.error\n });\n\n return item;\n }\n });\n\n // 等待所有请求完成\n await Promise.allSettled(promises);\n\n return results;\n }\n\n /**\n * 删除多个 URL 的资源\n * @param urls 要删除的资源 URL 数组\n * @returns Promise<Array<{ url: string, success: boolean, error?: string }>> 返回每个 URL 的删除结果\n */\n public async deleteCacheUrls(urls: string[]): Promise<Array<{ url: string, success: boolean, error?: string }>> {\n const results = await Promise.allSettled(\n urls.map(url => this.deleteUrl(url))\n );\n return results.map((result, index) => {\n const url = urls[index] || '';\n if (result.status === 'fulfilled') {\n return { url, success: true };\n } else {\n return { url, success: false, error: result.reason?.message || '未知错误' };\n }\n });\n }\n\n /**\n * 删除单个 URL 的资源(异步版本,性能更好)\n * @param url 要删除的资源 URL\n * @returns Promise<{ url: string, success: boolean, error?: string }> 返回删除结果\n */\n public async deleteUrl(url: string): Promise<{ url: string, success: boolean, error?: string }> {\n try {\n const cachePath = this.getCachedPath(url);\n try {\n await fs.promises.unlink(cachePath.filePath);\n return { url, success: true };\n } catch (error) {\n if ((error as NodeJS.ErrnoException).code === 'ENOENT') {\n return { url, success: false, error: '文件不存在' };\n }\n throw error;\n }\n } catch (error) {\n return {\n url,\n success: false,\n error: error instanceof Error ? error.message : '未知错误'\n };\n }\n }\n\n /**\n * 删除单个 URL 的资源(别名,保持向后兼容)\n * @deprecated 使用 deleteUrl 代替\n */\n public async deleteCache(url: string): Promise<{ url: string, success: boolean, error?: string }> {\n return this.deleteUrl(url);\n }\n\n /**\n * 清理过期缓存文件(异步并行处理,性能更好)\n */\n private async _cleanOldCache(): Promise<void> {\n try {\n const files = await fs.promises.readdir(this.options.cacheDir);\n if (files.length === 0) return;\n\n const now = Date.now();\n // 并行处理,提升性能\n await Promise.allSettled(\n files.map(async (file) => {\n const fullPath = path.join(this.options.cacheDir, file);\n try {\n const stat = await fs.promises.stat(fullPath);\n if (now - stat.mtimeMs > this.options.cacheTTL) {\n await fs.promises.unlink(fullPath);\n }\n } catch {\n // 忽略单个文件异常\n }\n })\n );\n } catch (error) {\n // 忽略目录读取错误\n console.log('清理过期缓存时发生错误:', error);\n }\n }\n\n /**\n * 清理所有缓存文件(完全异步版本,性能更好)\n * @returns Promise<{ success: number, failed: number, totalSize: number }> 返回清理统计信息\n */\n public async clearCache(): Promise<{ success: number, failed: number, totalSize: number }> {\n try {\n const files = await fs.promises.readdir(this.options.cacheDir);\n if (files.length === 0) {\n return { success: 0, failed: 0, totalSize: 0 };\n }\n\n // 并行获取所有文件信息(避免并行时的竞态条件)\n const fileInfos = await Promise.allSettled(\n files.map(async (file) => {\n const filePath = path.join(this.options.cacheDir, file);\n try {\n const stats = await fs.promises.stat(filePath);\n return { file, filePath, size: stats.size };\n } catch (error) {\n return { file, filePath, size: 0, error };\n }\n })\n );\n\n const validInfos = fileInfos\n .filter((r): r is PromiseFulfilledResult<{ file: string; filePath: string; size: number; error?: any }> => r.status === 'fulfilled')\n .map(r => r.value);\n\n // 计算总大小\n const totalSize = validInfos.reduce((sum, info) => sum + info.size, 0);\n\n // 并行删除文件,提升性能\n const deleteResults = await Promise.allSettled(\n validInfos.map(async (info) => {\n try {\n await fs.promises.unlink(info.filePath);\n return { success: true, file: info.file };\n } catch (error) {\n throw error;\n }\n })\n );\n\n // 统计成功和失败数量\n const success = deleteResults.filter(r => r.status === 'fulfilled').length;\n const failed = deleteResults.filter(r => r.status === 'rejected').length;\n\n return { success, failed, totalSize };\n } catch (error) {\n console.log('清理缓存时发生错误:', error);\n throw error;\n }\n }\n\n /**\n * 注册 Electron 请求拦截器,实现资源缓存\n */\n private _registerInterceptor(): void {\n const shouldCache = this._getMatchFunction();\n const isAllowedOrigin = this._getOriginAllowFunction();\n\n this.session.webRequest.onBeforeRequest(\n { urls: ['http://*/*', 'https://*/*'] },\n (details: Electron.OnBeforeRequestListenerDetails, callback: (response: Electron.CallbackResponse) => void) => {\n const url = details.url;\n // 不匹配或来源不允许,直接放行\n if (details.method !== 'GET' || !shouldCache(url) || !isAllowedOrigin(url)) return callback({});\n\n const cachePath = this.getCachedPath(url);\n\n // 命中缓存,直接重定向到本地文件\n if (this.isCacheValid(cachePath.filePath)) {\n return callback({ redirectURL: cachePath.hostPath});\n }\n // 未命中则异步下载,当前请求正常放行\n this.downloadResource(url, cachePath.filePath);\n return callback({});\n }\n );\n }\n}\n\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"resource-cache.js","sources":["../../src/main/resource-cache.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport md5 from 'md5';\nimport mime from 'mime-types';\nimport { ipcMain, webContents } from 'electron';\nimport { DownloaderHelper } from 'node-downloader-helper';\n\n/**\n * 默认文件扩展名\n */\nconst DEFAULT_EXT = '.res';\n\n/**\n * 默认 MIME 类型\n */\nconst DEFAULT_MIME_TYPE = 'application/octet-stream';\n\n/**\n * 最大重定向次数\n */\nconst MAX_REDIRECTS = 5;\n\n/**\n * 资源缓存配置项\n */\nexport interface ResourceCacheOptions {\n /** 缓存目录,必填 */\n cacheDir: string;\n /** 缓存有效期(毫秒),默认24小时 */\n cacheTTL?: number;\n /** 匹配需要缓存的资源,支持正则或函数 */\n match?: RegExp | ((url: string) => boolean);\n /** 允许缓存的资源来源,支持null/数组/函数 */\n allowedOrigins?: null | string[] | ((url: string) => boolean);\n}\n\n/**\n * 下载进度回调函数类型\n */\nexport type DownloadProgressCallback = (progress: {\n url: string; // 正在下载的 URL\n downloaded: number; // 已下载字节数\n total: number; // 总字节数(如果服务器提供 Content-Length)\n percentage: number; // 下载进度百分比 (0-100),如果 total 未知则为 -1\n speed: number; // 下载速度(字节/秒)\n}) => void;\n\n/**\n * 缓存进度回调函数类型\n */\nexport type CacheProgressCallback = (progress: {\n current: number; // 当前完成的 URL 数量\n total: number; // 总 URL 数量\n url: string; // 当前处理的 URL\n success: boolean; // 当前 URL 是否成功\n result?: { // 当前 URL 的处理结果\n filePath?: string;\n hostPath?: string;\n mimeType?: string;\n size?: number;\n error?: string;\n };\n percentage: number; // 进度百分比 (0-100)\n downloadProgress?: { // 下载进度(如果正在下载)\n downloaded: number;\n total: number;\n percentage: number;\n speed: number;\n };\n}) => void;\n\n/**\n * 默认配置\n */\nconst DEFAULT_OPTIONS: Required<Omit<ResourceCacheOptions, 'cacheDir'>> & { cacheDir: string } = {\n cacheDir: '',\n cacheTTL: 24 * 60 * 60 * 1000,\n // 图片格式:png, jpg/jpeg, webp, gif, svg, ico, bmp, avif, heic, heif, tiff, tif\n // 字体格式:woff, woff2, ttf, eot, otf\n // 视频格式:mp4, webm, ogg, mov, avi, mkv, flv, m4v, 3gp\n // 音频格式:mp3, wav, aac, m4a, flac, opus, wma\n // 样式和脚本:css, js, json, xml, txt\n // Web资源:wasm, map (source map)\n // 文档格式:pdf\n // 压缩文件:zip, 7z, rar, tar, gz, bz2\n match: /\\.(png|jpe?g|webp|gif|svg|ico|bmp|avif|heic|heif|tiff?|woff2?|ttf|eot|otf|mp4|webm|ogg|mov|avi|mkv|flv|m4v|3gp|mp3|wav|aac|m4a|flac|opus|wma|css|js|json|xml|txt|wasm|map|pdf|zip|7z|rar|tar|gz|bz2)(\\?.*)?$/i,\n allowedOrigins: null,\n};\n\n/**\n * 资源缓存类:拦截并缓存静态资源,提升加载性能\n */\nexport class ResourceCache {\n static scheme = 'cachefile';\n private cacheHost: string = `${ResourceCache.scheme}://-`;\n /** Electron session 实例 */\n private session: Electron.Session;\n /** 缓存配置 */\n private options: Required<ResourceCacheOptions>;\n /** 缓存的匹配函数(避免重复创建) */\n private _cachedMatchFunction?: (url: string) => boolean;\n /** 缓存的来源校验函数(避免重复创建) */\n private _cachedOriginFunction?: (url: string) => boolean;\n /** 正在下载的 URL 和对应的下载信息(避免重复下载,允许多个请求等待同一个下载任务) */\n private _downloadingUrls = new Map<string, { promise: Promise<void>; downloader: DownloaderHelper }>();\n\n /**\n * 构造函数\n * @param session Electron session\n * @param options 缓存配置\n */\n constructor(session: Electron.Session, options: ResourceCacheOptions) {\n if (!session) throw new Error('ResourceCache: session is required');\n this.session = session;\n // 合并配置,保证类型安全\n this.options = {\n ...DEFAULT_OPTIONS,\n ...options,\n cacheDir: options.cacheDir,\n cacheTTL: options.cacheTTL ?? DEFAULT_OPTIONS.cacheTTL,\n match: options.match ?? DEFAULT_OPTIONS.match,\n allowedOrigins: options.allowedOrigins ?? DEFAULT_OPTIONS.allowedOrigins,\n };\n\n if (!this.options.cacheDir) {\n throw new Error('ResourceCache: cacheDir is required');\n }\n\n // 确保缓存目录存在\n if (!fs.existsSync(this.options.cacheDir)) {\n fs.mkdirSync(this.options.cacheDir, { recursive: true });\n }\n\n this._registerInterceptor();\n // 异步清理过期缓存,不阻塞初始化\n this._cleanOldCache().catch(err => {\n console.log('初始化时清理过期缓存失败:', err);\n });\n\n ipcMain.handle('core:cache', async (event: Electron.IpcMainInvokeEvent, options: { id: string, method: 'add' | 'delete' | 'clear' | 'stats', urls?: string | string[], force?: boolean }) => {\n try {\n switch (options.method) {\n case 'clear':\n return await this.clearCache();\n case 'add': {\n const { id } = options;\n const sender = event.sender as Electron.WebContents;\n const urls = Array.isArray(options.urls) ? options.urls : options.urls ? [options.urls] : [];\n const data = await this.addCacheUrls(urls, options.force ?? false, true, (data: any) => {\n try {\n if (sender) {\n if (typeof sender?.send === 'function') {\n sender?.send?.(`core:cache:progress`, {\n id,\n data\n });\n }\n }\n } catch (error) {\n console.log('发送缓存进度回调失败:', error);\n }\n });\n return Array.isArray(options.urls) ? data : data[0];\n }\n case 'delete': {\n const urls = Array.isArray(options.urls) ? options.urls : options.urls ? [options.urls] : [];\n const data = await this.deleteCacheUrls(urls);\n return Array.isArray(options.urls) ? data : data[0];\n }\n case 'stats':\n return await this.getCacheStats();\n default:\n return undefined;\n }\n } catch (error) {\n return { success: false, error: error instanceof Error ? error.message : '未知错误' };\n }\n });\n }\n\n /**\n * 获取缓存统计信息(异步版本,性能更好)\n */\n public async getCacheStats(): Promise<{ size: number, totalSize: number }> {\n try {\n const files = await fs.promises.readdir(this.options.cacheDir);\n if (files.length === 0) {\n return { size: 0, totalSize: 0 };\n }\n\n // 并行获取文件信息,提升性能\n const fileInfos = await Promise.allSettled(\n files.map(async (file) => {\n const filePath = path.join(this.options.cacheDir, file);\n try {\n const stats = await fs.promises.stat(filePath);\n return { file, filePath, size: stats.size };\n } catch (error) {\n return { file, filePath, size: 0, error };\n }\n })\n );\n\n const validInfos = fileInfos\n .filter((r): r is PromiseFulfilledResult<{ file: string; filePath: string; size: number; error?: any }> => r.status === 'fulfilled')\n .map(r => r.value);\n\n return {\n size: validInfos.length,\n totalSize: validInfos.reduce((sum, info) => sum + info.size, 0)\n };\n } catch (error) {\n console.log('获取缓存统计信息失败:', error);\n return { size: 0, totalSize: 0 };\n }\n }\n\n /**\n * 获取资源匹配函数(带缓存,避免重复创建)\n */\n private _getMatchFunction(): (url: string) => boolean {\n if (this._cachedMatchFunction) {\n return this._cachedMatchFunction;\n }\n\n const matcher = this.options.match;\n if (typeof matcher === 'function') {\n this._cachedMatchFunction = matcher;\n } else if (matcher instanceof RegExp) {\n this._cachedMatchFunction = (url: string) => matcher.test(url);\n } else {\n this._cachedMatchFunction = () => false;\n }\n\n return this._cachedMatchFunction;\n }\n\n /**\n * 获取来源校验函数(带缓存,避免重复创建)\n */\n private _getOriginAllowFunction(): (url: string) => boolean {\n if (this._cachedOriginFunction) {\n return this._cachedOriginFunction;\n }\n\n const origins = this.options.allowedOrigins;\n if (!origins) {\n this._cachedOriginFunction = () => true;\n } else if (typeof origins === 'function') {\n this._cachedOriginFunction = origins;\n } else {\n const prefixList = origins.map(o => o.toLowerCase());\n this._cachedOriginFunction = (url: string) => {\n try {\n const origin = new URL(url).origin.toLowerCase();\n return prefixList.some(prefix => origin.startsWith(prefix));\n } catch {\n return false;\n }\n };\n }\n\n return this._cachedOriginFunction;\n }\n\n /**\n * 获取缓存文件路径\n * @param url 资源URL\n * @param customExt 自定义文件扩展名(可选,用于 base64 URL)\n */\n public getCachedPath(url: string, customExt?: string): { filePath: string, hostPath: string } {\n const md5Str = md5(url);\n let ext = '.res';\n\n if (customExt) {\n // 如果提供了自定义扩展名,使用它\n ext = customExt.startsWith('.') ? customExt : `.${customExt}`;\n } else {\n // 尝试从 URL 中提取扩展名\n try {\n const urlObj = new URL(url);\n ext = path.extname(urlObj.pathname) || DEFAULT_EXT;\n } catch {\n // 如果 URL 解析失败(可能是 base64 data URL),使用默认扩展名\n ext = DEFAULT_EXT;\n }\n }\n\n return {\n filePath: path.join(this.options.cacheDir, `${md5Str}${ext}`),\n hostPath: `${this.cacheHost}/${md5Str}${ext}`,\n }\n }\n\n /**\n * 判断缓存是否有效(同步版本,用于拦截器)\n * @param filePath 缓存文件路径\n */\n public isCacheValid(filePath: string): boolean {\n try {\n if (!fs.existsSync(filePath)) return false;\n const stat = fs.statSync(filePath);\n return Date.now() - stat.mtimeMs < this.options.cacheTTL;\n } catch {\n return false;\n }\n }\n\n /**\n * 判断缓存是否有效(异步版本,性能更好)\n * @param filePath 缓存文件路径\n */\n public async isCacheValidAsync(filePath: string): Promise<boolean> {\n try {\n const stat = await fs.promises.stat(filePath);\n return Date.now() - stat.mtimeMs < this.options.cacheTTL;\n } catch {\n return false;\n }\n }\n\n /**\n * 下载资源到本地缓存(异步版本,返回 Promise)\n * @param url 资源URL\n * @param filePath 本地缓存路径\n * @param redirectCount 当前重定向次数(内部使用)\n * @param onProgress 下载进度回调函数(可选)\n * @returns Promise<void> 下载完成或失败\n */\n public downloadResourceAsync(\n url: string,\n filePath: string,\n redirectCount: number = 0,\n onProgress?: DownloadProgressCallback\n ): Promise<void> {\n // 检查是否正在下载,如果正在下载则返回已存在的 Promise(允许多个请求等待同一个下载任务)\n const existingDownload = this._downloadingUrls.get(url);\n if (existingDownload) {\n // 如果提供了新的进度回调,绑定到已存在的下载器上\n if (onProgress && existingDownload.downloader) {\n existingDownload.downloader.on('progress', (stats) => {\n try {\n onProgress({\n url,\n downloaded: stats.downloaded || 0,\n total: stats.total || 0,\n percentage: stats.progress !== undefined ? Math.round(stats.progress) : -1,\n speed: stats.speed || 0\n });\n } catch (error) {\n // 忽略回调中的错误,避免影响下载\n console.error('进度回调执行失败:', error);\n }\n });\n }\n return existingDownload.promise;\n }\n\n // 检查重定向次数限制\n if (redirectCount >= MAX_REDIRECTS) {\n return Promise.reject(new Error(`重定向次数超过限制 (${MAX_REDIRECTS}): ${url}`));\n }\n\n // 创建新的下载 Promise\n const downloadPromise = new Promise<void>((resolve, reject) => {\n const tempFilePath = `${filePath}.cache`;\n const downloadDir = path.dirname(tempFilePath);\n const downloadFileName = path.basename(tempFilePath);\n\n // 确保目录存在\n if (!fs.existsSync(downloadDir)) {\n fs.mkdirSync(downloadDir, { recursive: true });\n }\n\n // 创建下载器实例\n const dl = new DownloaderHelper(url, downloadDir, {\n fileName: downloadFileName,\n retry: { maxRetries: 3, delay: 1000 },\n // 超时设置:10 分钟(600000 毫秒)\n // 对于视频等大文件,需要更长的超时时间\n // 如果 10 分钟内没有任何数据传输,才会超时\n timeout: 600000,\n override: true, // 覆盖已存在的文件\n httpRequestOptions: {\n // 允许重定向\n followRedirect: true,\n maxRedirects: MAX_REDIRECTS\n }\n });\n\n // 立即将 Promise 和下载器实例存储到 Map 中,供后续相同 URL 的请求复用\n // 注意:必须在监听事件之前存储,以便后续请求可以绑定进度回调\n this._downloadingUrls.set(url, { promise: downloadPromise, downloader: dl });\n\n // 监听下载进度\n if (onProgress) {\n dl.on('progress', (stats) => {\n try {\n // stats 包含: progress (百分比), downloaded (已下载字节), total (总字节), speed (速度)\n onProgress({\n url,\n downloaded: stats.downloaded || 0,\n total: stats.total || 0,\n percentage: stats.progress !== undefined ? Math.round(stats.progress) : -1,\n speed: stats.speed || 0\n });\n } catch (error) {\n // 忽略回调中的错误,避免影响下载\n console.error('进度回调执行失败:', error);\n }\n });\n }\n\n // 监听下载完成\n dl.on('end', () => {\n // 将临时文件重命名为最终文件\n fs.rename(tempFilePath, filePath, (renameErr) => {\n // 无论成功还是失败,都要从下载列表中删除\n this._downloadingUrls.delete(url);\n if (renameErr) {\n reject(new Error(`缓存文件重命名失败 from ${tempFilePath} to ${filePath}: ${renameErr.message}`));\n } else {\n resolve();\n }\n });\n });\n\n // 监听下载错误\n dl.on('error', (err) => {\n // 从下载列表中删除\n this._downloadingUrls.delete(url);\n // 清理临时文件\n fs.promises.unlink(tempFilePath).catch(() => {\n // 忽略删除失败\n });\n reject(err instanceof Error ? err : new Error(`下载失败: ${url}`));\n });\n\n // 监听下载停止(取消)\n dl.on('stop', () => {\n // 从下载列表中删除\n this._downloadingUrls.delete(url);\n // 清理临时文件\n fs.promises.unlink(tempFilePath).catch(() => {\n // 忽略删除失败\n });\n reject(new Error(`下载已停止: ${url}`));\n });\n\n // 开始下载\n dl.start().catch((err) => {\n // 从下载列表中删除\n this._downloadingUrls.delete(url);\n reject(err instanceof Error ? err : new Error(`启动下载失败: ${url}`));\n });\n });\n\n return downloadPromise;\n }\n\n /**\n * 下载资源到本地缓存(同步版本,不返回 Promise,用于拦截器)\n * @param url 资源URL\n * @param filePath 本地缓存路径\n */\n public downloadResource(url: string, filePath: string): void {\n // 异步执行,不等待完成\n // 如果正在下载,跳过(避免重复下载)\n if (this._downloadingUrls.has(url)) {\n return;\n }\n // 调用 downloadResourceAsync,它会自动处理重复下载的情况\n this.downloadResourceAsync(url, filePath).catch(() => {\n // 静默处理错误,避免日志过多\n });\n }\n\n /**\n * 从文件扩展名获取 MIME 类型\n * @param ext 文件扩展名(带或不带点)\n * @returns MIME 类型\n */\n private _getMimeTypeFromExt(ext: string): string {\n const cleanExt = ext.replace(/^\\./, '').toLowerCase();\n const mimeType = mime.lookup(cleanExt);\n return mimeType || DEFAULT_MIME_TYPE;\n }\n\n /**\n * 检测并处理 base64 data URL\n * @param url 资源URL\n * @returns 如果是 base64 URL,返回 true、文件扩展名、MIME 类型和数据;否则返回 false\n */\n private _isBase64DataUrl(url: string): { isBase64: boolean, ext?: string, mimeType?: string, data?: string } {\n if (!url.startsWith('data:')) {\n return { isBase64: false };\n }\n\n try {\n // 解析 data URL 格式:data:[<mediatype>][;base64],<data>\n const commaIndex = url.indexOf(',');\n if (commaIndex === -1) {\n return { isBase64: false };\n }\n\n const header = url.substring(0, commaIndex);\n const data = url.substring(commaIndex + 1);\n\n // 检查是否包含 base64 标识\n if (!header.includes('base64')) {\n return { isBase64: false };\n }\n\n // 从 mediatype 中提取文件扩展名和 MIME 类型\n // 例如:data:image/png;base64 -> png, image/png\n // 例如:data:image/jpeg;base64 -> jpeg, image/jpeg\n let ext = DEFAULT_EXT.replace(/^\\./, '');\n let mimeType = DEFAULT_MIME_TYPE;\n const mimeMatch = header.match(/data:([^;]+)/);\n if (mimeMatch && mimeMatch[1]) {\n mimeType = mimeMatch[1];\n // 使用 mime-types 包从 MIME 类型获取扩展名\n const extension = mime.extension(mimeType);\n if (extension) {\n ext = extension;\n }\n }\n\n return { isBase64: true, ext, mimeType, data };\n } catch (error) {\n return { isBase64: false };\n }\n }\n\n /**\n * 保存 base64 数据到文件\n * @param base64Data base64 编码的数据\n * @param filePath 目标文件路径\n */\n private async _saveBase64ToFile(base64Data: string, filePath: string): Promise<void> {\n try {\n // 解码 base64 数据\n const buffer = Buffer.from(base64Data, 'base64');\n\n // 使用 Promise 版本的 writeFile\n // Buffer 继承自 Uint8Array,可以直接使用\n await fs.promises.writeFile(filePath, buffer as Uint8Array);\n } catch (error) {\n throw new Error(`保存 base64 文件失败: ${error instanceof Error ? error.message : '未知错误'}`);\n }\n }\n\n /**\n * 手动缓存指定 URL 的资源\n * @param url 要缓存的资源 URL(支持普通 URL 和 base64 data URL)\n * @param force 是否强制重新下载,即使缓存有效(默认 false)\n * @param ignoreOrigin 是否忽略来源检查(默认 false)\n * @param onDownloadProgress 下载进度回调函数(可选)\n * @returns Promise<{ filePath: string, hostPath: string, mimeType: string, size: number }> 返回缓存文件路径、主机路径、MIME 类型和文件大小\n * @throws 如果 URL 不匹配缓存规则或来源不允许,会抛出错误\n */\n public async cacheUrl(\n url: string,\n force: boolean = false,\n ignoreOrigin: boolean = false,\n onDownloadProgress?: DownloadProgressCallback\n ): Promise<{ filePath: string, hostPath: string, mimeType: string, size: number }> {\n // 检查是否是 base64 data URL\n const base64Info = this._isBase64DataUrl(url);\n\n if (base64Info.isBase64) {\n // 处理 base64 data URL\n if (!base64Info.data || !base64Info.ext) {\n throw new Error(`无效的 base64 data URL: ${url}`);\n }\n\n // 获取缓存路径(使用检测到的扩展名)\n const cachePath = this.getCachedPath(url, base64Info.ext);\n const mimeType = base64Info.mimeType || this._getMimeTypeFromExt(base64Info.ext);\n\n // 如果缓存有效且不强制重新下载,直接返回\n if (!force && await this.isCacheValidAsync(cachePath.filePath)) {\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n // 保存 base64 数据到文件\n await this._saveBase64ToFile(base64Info.data, cachePath.filePath);\n\n // 获取文件大小\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n // 处理普通 URL\n const shouldCache = ignoreOrigin ? () => true : this._getMatchFunction();\n const isAllowedOrigin = ignoreOrigin ? () => true : this._getOriginAllowFunction();\n\n // 检查是否匹配缓存规则\n if (!shouldCache(url)) {\n throw new Error(`URL 不匹配缓存规则: ${url}`);\n }\n\n // 检查来源是否允许\n if (!isAllowedOrigin(url)) {\n throw new Error(`URL 来源不允许缓存: ${url}`);\n }\n\n // 获取缓存路径\n const cachePath = this.getCachedPath(url);\n\n // 从文件扩展名获取 MIME 类型\n const ext = path.extname(cachePath.filePath).replace(/^\\./, '') || DEFAULT_EXT.replace(/^\\./, '');\n const mimeType = this._getMimeTypeFromExt(ext);\n\n // 如果缓存有效且不强制重新下载,直接返回\n if (!force && await this.isCacheValidAsync(cachePath.filePath)) {\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n // 下载资源\n await this.downloadResourceAsync(url, cachePath.filePath, 0, onDownloadProgress);\n\n // 获取文件大小\n const stats = await fs.promises.stat(cachePath.filePath);\n return {\n ...cachePath,\n mimeType,\n size: stats.size\n };\n }\n\n /**\n * 批量缓存多个 URL 的资源\n * @param urls 要缓存的资源 URL 数组\n * @param force 是否强制重新下载,即使缓存有效(默认 false)\n * @param ignoreOrigin 是否忽略来源检查(默认 false)\n * @param onProgress 进度回调函数(可选)\n * @returns Promise<Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, mimeType?: string, size?: number, error?: string }>> 返回每个 URL 的缓存结果\n */\n public async addCacheUrls(\n urls: string[],\n force: boolean = false,\n ignoreOrigin: boolean = false,\n onProgress?: CacheProgressCallback\n ): Promise<Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, mimeType?: string, size?: number, error?: string }>> {\n const total = urls.length;\n const results: Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, mimeType?: string, size?: number, error?: string }> = [];\n let completed = 0;\n\n // 存储每个 URL 的下载进度\n const downloadProgressMap = new Map<string, { downloaded: number; total: number; percentage: number; speed: number }>();\n\n // 进度回调的包装函数,确保线程安全\n const reportProgress = (url: string, success: boolean, result?: { filePath?: string; hostPath?: string; mimeType?: string; size?: number; error?: string }) => {\n if (onProgress) {\n completed++;\n const downloadProgress = downloadProgressMap.get(url);\n onProgress({\n current: completed,\n total,\n url,\n success,\n result,\n percentage: Math.round((completed / total) * 100),\n downloadProgress: downloadProgress ? {\n downloaded: downloadProgress.downloaded,\n total: downloadProgress.total,\n percentage: downloadProgress.percentage,\n speed: downloadProgress.speed\n } : undefined\n });\n }\n };\n\n // 并行处理所有 URL,但跟踪每个的完成状态\n const promises = urls.map(async (url, index) => {\n // 为每个 URL 创建下载进度回调\n const onDownloadProgress: DownloadProgressCallback | undefined = onProgress\n ? (progress) => {\n // 更新下载进度\n downloadProgressMap.set(url, {\n downloaded: progress.downloaded,\n total: progress.total,\n percentage: progress.percentage,\n speed: progress.speed\n });\n // 实时报告下载进度(不增加 completed 计数)\n if (onProgress) {\n onProgress({\n current: completed,\n total,\n url,\n success: true, // 下载中视为进行中\n result: undefined,\n percentage: Math.round((completed / total) * 100),\n downloadProgress: {\n downloaded: progress.downloaded,\n total: progress.total,\n percentage: progress.percentage,\n speed: progress.speed\n }\n });\n }\n }\n : undefined;\n\n try {\n const result = await this.cacheUrl(url, force, ignoreOrigin, onDownloadProgress);\n const item = {\n url,\n success: true,\n filePath: result.filePath,\n hostPath: result.hostPath,\n mimeType: result.mimeType,\n size: result.size\n };\n results[index] = item;\n\n // 清除下载进度(已完成)\n downloadProgressMap.delete(url);\n\n // 调用进度回调\n reportProgress(url, true, {\n filePath: result.filePath,\n hostPath: result.hostPath,\n mimeType: result.mimeType,\n size: result.size\n });\n\n return item;\n } catch (error) {\n const item = {\n url,\n success: false,\n error: error instanceof Error ? error.message : '未知错误'\n };\n results[index] = item;\n\n // 清除下载进度(失败)\n downloadProgressMap.delete(url);\n\n // 调用进度回调\n reportProgress(url, false, {\n error: item.error\n });\n\n return item;\n }\n });\n\n // 等待所有请求完成\n await Promise.allSettled(promises);\n\n return results;\n }\n\n /**\n * 删除多个 URL 的资源\n * @param urls 要删除的资源 URL 数组\n * @returns Promise<Array<{ url: string, success: boolean, error?: string }>> 返回每个 URL 的删除结果\n */\n public async deleteCacheUrls(urls: string[]): Promise<Array<{ url: string, success: boolean, error?: string }>> {\n const results = await Promise.allSettled(\n urls.map(url => this.deleteUrl(url))\n );\n return results.map((result, index) => {\n const url = urls[index] || '';\n if (result.status === 'fulfilled') {\n return { url, success: true };\n } else {\n return { url, success: false, error: result.reason?.message || '未知错误' };\n }\n });\n }\n\n /**\n * 删除单个 URL 的资源(异步版本,性能更好)\n * @param url 要删除的资源 URL\n * @returns Promise<{ url: string, success: boolean, error?: string }> 返回删除结果\n */\n public async deleteUrl(url: string): Promise<{ url: string, success: boolean, error?: string }> {\n try {\n const cachePath = this.getCachedPath(url);\n try {\n await fs.promises.unlink(cachePath.filePath);\n return { url, success: true };\n } catch (error) {\n if ((error as NodeJS.ErrnoException).code === 'ENOENT') {\n return { url, success: false, error: '文件不存在' };\n }\n throw error;\n }\n } catch (error) {\n return {\n url,\n success: false,\n error: error instanceof Error ? error.message : '未知错误'\n };\n }\n }\n\n /**\n * 删除单个 URL 的资源(别名,保持向后兼容)\n * @deprecated 使用 deleteUrl 代替\n */\n public async deleteCache(url: string): Promise<{ url: string, success: boolean, error?: string }> {\n return this.deleteUrl(url);\n }\n\n /**\n * 清理过期缓存文件(异步并行处理,性能更好)\n */\n private async _cleanOldCache(): Promise<void> {\n try {\n const files = await fs.promises.readdir(this.options.cacheDir);\n if (files.length === 0) return;\n\n const now = Date.now();\n // 并行处理,提升性能\n await Promise.allSettled(\n files.map(async (file) => {\n const fullPath = path.join(this.options.cacheDir, file);\n try {\n const stat = await fs.promises.stat(fullPath);\n if (now - stat.mtimeMs > this.options.cacheTTL) {\n await fs.promises.unlink(fullPath);\n }\n } catch {\n // 忽略单个文件异常\n }\n })\n );\n } catch (error) {\n // 忽略目录读取错误\n console.log('清理过期缓存时发生错误:', error);\n }\n }\n\n /**\n * 清理所有缓存文件(完全异步版本,性能更好)\n * @returns Promise<{ success: number, failed: number, totalSize: number }> 返回清理统计信息\n */\n public async clearCache(): Promise<{ success: number, failed: number, totalSize: number }> {\n try {\n const files = await fs.promises.readdir(this.options.cacheDir);\n if (files.length === 0) {\n return { success: 0, failed: 0, totalSize: 0 };\n }\n\n // 并行获取所有文件信息(避免并行时的竞态条件)\n const fileInfos = await Promise.allSettled(\n files.map(async (file) => {\n const filePath = path.join(this.options.cacheDir, file);\n try {\n const stats = await fs.promises.stat(filePath);\n return { file, filePath, size: stats.size };\n } catch (error) {\n return { file, filePath, size: 0, error };\n }\n })\n );\n\n const validInfos = fileInfos\n .filter((r): r is PromiseFulfilledResult<{ file: string; filePath: string; size: number; error?: any }> => r.status === 'fulfilled')\n .map(r => r.value);\n\n // 计算总大小\n const totalSize = validInfos.reduce((sum, info) => sum + info.size, 0);\n\n // 并行删除文件,提升性能\n const deleteResults = await Promise.allSettled(\n validInfos.map(async (info) => {\n try {\n await fs.promises.unlink(info.filePath);\n return { success: true, file: info.file };\n } catch (error) {\n throw error;\n }\n })\n );\n\n // 统计成功和失败数量\n const success = deleteResults.filter(r => r.status === 'fulfilled').length;\n const failed = deleteResults.filter(r => r.status === 'rejected').length;\n\n return { success, failed, totalSize };\n } catch (error) {\n console.log('清理缓存时发生错误:', error);\n throw error;\n }\n }\n\n /**\n * 注册 Electron 请求拦截器,实现资源缓存\n */\n private _registerInterceptor(): void {\n const shouldCache = this._getMatchFunction();\n const isAllowedOrigin = this._getOriginAllowFunction();\n\n this.session.webRequest.onBeforeRequest(\n { urls: ['http://*/*', 'https://*/*'] },\n (details: Electron.OnBeforeRequestListenerDetails, callback: (response: Electron.CallbackResponse) => void) => {\n const url = details.url;\n // 不匹配或来源不允许,直接放行\n if (details.method !== 'GET' || !shouldCache(url) || !isAllowedOrigin(url)) return callback({});\n\n const cachePath = this.getCachedPath(url);\n\n // 命中缓存,直接重定向到本地文件\n if (this.isCacheValid(cachePath.filePath)) {\n return callback({ redirectURL: cachePath.hostPath});\n }\n // 未命中则异步下载,当前请求正常放行\n this.downloadResource(url, cachePath.filePath);\n return callback({});\n }\n );\n }\n}\n\n"],"names":[],"mappings":"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"}
|
package/main/downloader.d.ts
CHANGED
|
@@ -76,12 +76,20 @@ declare class Downloader {
|
|
|
76
76
|
* @returns Promise<DownloadResult> 下载结果
|
|
77
77
|
*/
|
|
78
78
|
download(url: string, options?: DownloadOptions, onProgress?: DownloadProgressCallback): Promise<DownloadResult>;
|
|
79
|
+
/**
|
|
80
|
+
* 生成下载任务的唯一标识
|
|
81
|
+
* @param url 下载 URL
|
|
82
|
+
* @param id 可选的任务 ID
|
|
83
|
+
* @returns 唯一标识
|
|
84
|
+
*/
|
|
85
|
+
private _getDownloadKey;
|
|
79
86
|
/**
|
|
80
87
|
* 取消下载
|
|
81
88
|
* @param url 要取消的 URL
|
|
89
|
+
* @param id 可选的任务 ID
|
|
82
90
|
* @returns 是否成功取消
|
|
83
91
|
*/
|
|
84
|
-
cancel(url: string): boolean;
|
|
92
|
+
cancel(url: string, id?: string): boolean;
|
|
85
93
|
/**
|
|
86
94
|
* 取消所有正在进行的下载
|
|
87
95
|
*/
|
|
@@ -89,14 +97,18 @@ declare class Downloader {
|
|
|
89
97
|
/**
|
|
90
98
|
* 检查指定 URL 是否正在下载
|
|
91
99
|
* @param url 要检查的 URL
|
|
100
|
+
* @param id 可选的任务 ID
|
|
92
101
|
* @returns 是否正在下载
|
|
93
102
|
*/
|
|
94
|
-
isDownloading(url: string): boolean;
|
|
103
|
+
isDownloading(url: string, id?: string): boolean;
|
|
95
104
|
/**
|
|
96
|
-
*
|
|
97
|
-
* @returns
|
|
105
|
+
* 获取正在下载的任务列表
|
|
106
|
+
* @returns 正在下载的任务信息数组,包含 URL 和 ID(如果有)
|
|
98
107
|
*/
|
|
99
|
-
getDownloadingUrls():
|
|
108
|
+
getDownloadingUrls(): Array<{
|
|
109
|
+
url: string;
|
|
110
|
+
id?: string;
|
|
111
|
+
}>;
|
|
100
112
|
/**
|
|
101
113
|
* 显示保存对话框
|
|
102
114
|
* @param url 下载 URL
|
package/main/downloader.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"downloader.d.ts","sourceRoot":"","sources":["../src/main/downloader.ts"],"names":[],"mappings":"AA4CA;;GAEG;AACH,MAAM,MAAM,wBAAwB,GAAG,CAAC,QAAQ,EAAE;IAChD,GAAG,EAAE,MAAM,CAAC;IACZ,UAAU,EAAE,MAAM,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;CACf,KAAK,IAAI,CAAC;AAEX;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,sBAAsB;IACtB,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,mCAAmC;IACnC,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,4BAA4B;IAC5B,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,+BAA+B;IAC/B,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,0BAA0B;IAC1B,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,2BAA2B;IAC3B,cAAc,CAAC,EAAE,OAAO,CAAC;IACzB,sBAAsB;IACtB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,wBAAwB;IACxB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW;IACX,KAAK,CAAC,EAAE;QACN,UAAU,CAAC,EAAE,MAAM,CAAC;QACpB,KAAK,CAAC,EAAE,MAAM,CAAC;KAChB,CAAC;IACF,mBAAmB;IACnB,UAAU,CAAC,EAAE,wBAAwB,CAAC;IACtC,mBAAmB;IACnB,UAAU,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,KAAK,IAAI,CAAC;IACxC,mBAAmB;IACnB,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC;IACjC,gBAAgB;IAChB,kBAAkB,CAAC,EAAE;QACnB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;QACjC,cAAc,CAAC,EAAE,OAAO,CAAC;QACzB,YAAY,CAAC,EAAE,MAAM,CAAC;KACvB,CAAC;CACH;AAED;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,WAAW;IACX,OAAO,EAAE,OAAO,CAAC;IACjB,cAAc;IACd,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,WAAW;IACX,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,eAAe;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED;;GAEG;AACH,cAAM,UAAU;IACd,2BAA2B;IAC3B,OAAO,CAAC,gBAAgB,CAAuC;IAC/D,oBAAoB;IACpB,OAAO,CAAC,gBAAgB,CAAuB;;IAiC/C;;;;;;OAMG;IACU,QAAQ,CACnB,GAAG,EAAE,MAAM,EACX,OAAO,GAAE,eAAoB,EAC7B,UAAU,CAAC,EAAE,wBAAwB,GACpC,OAAO,CAAC,cAAc,CAAC;
|
|
1
|
+
{"version":3,"file":"downloader.d.ts","sourceRoot":"","sources":["../src/main/downloader.ts"],"names":[],"mappings":"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"}
|
package/main/downloader.js
CHANGED
|
@@ -81,9 +81,11 @@ class Downloader {
|
|
|
81
81
|
* @returns Promise<DownloadResult> 下载结果
|
|
82
82
|
*/
|
|
83
83
|
async download(url, options = {}, onProgress) {
|
|
84
|
+
// 生成唯一标识:URL + ID(如果存在)
|
|
85
|
+
const downloadKey = this._getDownloadKey(url, options.id);
|
|
84
86
|
// 检查是否正在下载,避免重复下载
|
|
85
|
-
if (this._downloadingUrls.has(
|
|
86
|
-
return this._createErrorResult(`资源正在下载中: ${url}`);
|
|
87
|
+
if (this._downloadingUrls.has(downloadKey)) {
|
|
88
|
+
return this._createErrorResult(`资源正在下载中: ${url}${options.id ? ` (ID: ${options.id})` : ''}`);
|
|
87
89
|
}
|
|
88
90
|
// 合并进度回调:优先使用传入的 onProgress,其次使用 options.onProgress
|
|
89
91
|
const progressCallback = onProgress || options.onProgress;
|
|
@@ -136,8 +138,8 @@ class Downloader {
|
|
|
136
138
|
headers: options.httpRequestOptions?.headers
|
|
137
139
|
}
|
|
138
140
|
});
|
|
139
|
-
// 记录正在下载的 URL
|
|
140
|
-
this._downloadingUrls.set(
|
|
141
|
+
// 记录正在下载的 URL(使用 URL + ID 作为 key)
|
|
142
|
+
this._downloadingUrls.set(downloadKey, dl);
|
|
141
143
|
// 监听下载进度
|
|
142
144
|
if (progressCallback) {
|
|
143
145
|
dl.on('progress', (stats) => {
|
|
@@ -158,7 +160,7 @@ class Downloader {
|
|
|
158
160
|
}
|
|
159
161
|
// 监听下载完成
|
|
160
162
|
dl.on('end', () => {
|
|
161
|
-
this._downloadingUrls.delete(
|
|
163
|
+
this._downloadingUrls.delete(downloadKey);
|
|
162
164
|
// 将临时文件重命名为最终文件
|
|
163
165
|
fs.rename(tempFilePath, outputPath, (renameErr) => {
|
|
164
166
|
if (renameErr) {
|
|
@@ -181,7 +183,7 @@ class Downloader {
|
|
|
181
183
|
});
|
|
182
184
|
// 统一的错误处理函数
|
|
183
185
|
const handleError = (error) => {
|
|
184
|
-
this._downloadingUrls.delete(
|
|
186
|
+
this._downloadingUrls.delete(downloadKey);
|
|
185
187
|
// 清理临时文件
|
|
186
188
|
fs.promises.unlink(tempFilePath).catch(() => {
|
|
187
189
|
// 忽略删除失败
|
|
@@ -210,16 +212,27 @@ class Downloader {
|
|
|
210
212
|
return this._createErrorResult(error instanceof Error ? error.message : '未知错误');
|
|
211
213
|
});
|
|
212
214
|
}
|
|
215
|
+
/**
|
|
216
|
+
* 生成下载任务的唯一标识
|
|
217
|
+
* @param url 下载 URL
|
|
218
|
+
* @param id 可选的任务 ID
|
|
219
|
+
* @returns 唯一标识
|
|
220
|
+
*/
|
|
221
|
+
_getDownloadKey(url, id) {
|
|
222
|
+
return id ? `${url}::${id}` : url;
|
|
223
|
+
}
|
|
213
224
|
/**
|
|
214
225
|
* 取消下载
|
|
215
226
|
* @param url 要取消的 URL
|
|
227
|
+
* @param id 可选的任务 ID
|
|
216
228
|
* @returns 是否成功取消
|
|
217
229
|
*/
|
|
218
|
-
cancel(url) {
|
|
219
|
-
const
|
|
230
|
+
cancel(url, id) {
|
|
231
|
+
const downloadKey = this._getDownloadKey(url, id);
|
|
232
|
+
const dl = this._downloadingUrls.get(downloadKey);
|
|
220
233
|
if (dl) {
|
|
221
234
|
dl.stop();
|
|
222
|
-
this._downloadingUrls.delete(
|
|
235
|
+
this._downloadingUrls.delete(downloadKey);
|
|
223
236
|
return true;
|
|
224
237
|
}
|
|
225
238
|
return false;
|
|
@@ -236,17 +249,25 @@ class Downloader {
|
|
|
236
249
|
/**
|
|
237
250
|
* 检查指定 URL 是否正在下载
|
|
238
251
|
* @param url 要检查的 URL
|
|
252
|
+
* @param id 可选的任务 ID
|
|
239
253
|
* @returns 是否正在下载
|
|
240
254
|
*/
|
|
241
|
-
isDownloading(url) {
|
|
242
|
-
|
|
255
|
+
isDownloading(url, id) {
|
|
256
|
+
const downloadKey = this._getDownloadKey(url, id);
|
|
257
|
+
return this._downloadingUrls.has(downloadKey);
|
|
243
258
|
}
|
|
244
259
|
/**
|
|
245
|
-
*
|
|
246
|
-
* @returns
|
|
260
|
+
* 获取正在下载的任务列表
|
|
261
|
+
* @returns 正在下载的任务信息数组,包含 URL 和 ID(如果有)
|
|
247
262
|
*/
|
|
248
263
|
getDownloadingUrls() {
|
|
249
|
-
return Array.from(this._downloadingUrls.keys())
|
|
264
|
+
return Array.from(this._downloadingUrls.keys()).map(key => {
|
|
265
|
+
const parts = key.split('::');
|
|
266
|
+
if (parts.length === 2 && parts[0] && parts[1]) {
|
|
267
|
+
return { url: parts[0], id: parts[1] };
|
|
268
|
+
}
|
|
269
|
+
return { url: key };
|
|
270
|
+
});
|
|
250
271
|
}
|
|
251
272
|
/**
|
|
252
273
|
* 显示保存对话框
|
package/main/downloader.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"downloader.js","sources":["../src/src/main/downloader.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport mime from 'mime-types';\nimport { DownloaderHelper } from 'node-downloader-helper';\nimport { ipcMain, dialog, BrowserWindow, app } from 'electron';\n\n/**\n * 最大重定向次数\n */\nconst MAX_REDIRECTS = 5;\n\n/**\n * 默认文件扩展名\n */\nconst DEFAULT_EXT = '.bin';\n\n/**\n * 默认 MIME 类型\n */\nconst DEFAULT_MIME_TYPE = 'application/octet-stream';\n\n/**\n * 默认超时时间(毫秒):10 分钟\n */\nconst DEFAULT_TIMEOUT = 600000;\n\n/**\n * 默认重试配置\n */\nconst DEFAULT_RETRY = {\n maxRetries: 3,\n delay: 1000\n};\n\n/**\n * 默认文件名\n */\nconst DEFAULT_FILENAME = 'download';\n\n/**\n * 最大文件名计数器(防止无限循环)\n */\nconst MAX_FILENAME_COUNTER = 10000;\n\n/**\n * 下载进度回调函数类型\n */\nexport type DownloadProgressCallback = (progress: {\n url: string; // 正在下载的 URL\n downloaded: number; // 已下载字节数\n total: number; // 总字节数(如果服务器提供 Content-Length)\n percentage: number; // 下载进度百分比 (0-100),如果 total 未知则为 -1\n speed: number; // 下载速度(字节/秒)\n}) => void;\n\n/**\n * 下载选项\n */\nexport interface DownloadOptions {\n /** 唯一标识(IPC 调用时必需) */\n id?: string;\n /** 输出文件路径(可选,如果不提供则使用 URL 的文件名) */\n outputPath?: string;\n /** 输出目录(可选,如果不提供则使用当前目录) */\n outputDir?: string;\n /** 输出文件名(可选,如果不提供则从 URL 提取) */\n fileName?: string;\n /** 是否覆盖已存在的文件(默认 true) */\n override?: boolean;\n /** 是否弹窗选择保存路径(默认 false) */\n showSaveDialog?: boolean;\n /** 保存对话框的默认文件名(可选) */\n defaultPath?: string;\n /** 超时时间(毫秒,默认 10 分钟) */\n timeout?: number;\n /** 重试配置 */\n retry?: {\n maxRetries?: number;\n delay?: number;\n };\n /** 下载进度回调函数(可选) */\n onProgress?: DownloadProgressCallback;\n /** 下载完成回调函数(可选) */\n onComplete?: (filePath: string) => void;\n /** 下载错误回调函数(可选) */\n onError?: (error: Error) => void;\n /** HTTP 请求选项 */\n httpRequestOptions?: {\n headers?: Record<string, string>;\n followRedirect?: boolean;\n maxRedirects?: number;\n };\n}\n\n/**\n * 下载结果\n */\nexport interface DownloadResult {\n /** 是否成功 */\n success: boolean;\n /** 下载的文件路径 */\n filePath?: string;\n /** 错误信息 */\n error?: string;\n /** 文件大小(字节) */\n size?: number;\n}\n\n/**\n * 下载器类:提供文件下载功能\n */\nclass Downloader {\n /** 正在下载的 URL 集合(避免重复下载) */\n private _downloadingUrls = new Map<string, DownloaderHelper>();\n /** 上一次选择的保存文件夹路径 */\n private _lastSelectedDir: string | null = null;\n\n constructor() {\n ipcMain.handle(`core:dowloader`, async (event: Electron.IpcMainInvokeEvent, options: DownloadOptions & { url: string; id: string }) => {\n try {\n const { id, url } = options;\n const sender = event.sender as Electron.WebContents;\n\n // 创建进度回调,通过 IPC 发送进度更新\n const progressCallback: DownloadProgressCallback = (progress) => {\n try {\n if (sender && typeof sender.send === 'function') {\n sender.send(`core:dowloader:progress`, {\n id,\n data: progress\n });\n }\n } catch (error) {\n console.error('发送下载进度失败:', error);\n }\n };\n\n const result = await this.download(url, options, progressCallback);\n return result;\n } catch (error) {\n return {\n success: false,\n error: error instanceof Error ? error.message : '未知错误'\n };\n }\n });\n }\n\n /**\n * 下载文件\n * @param url 要下载的 URL\n * @param options 下载选项\n * @param onProgress 进度回调函数(可选,优先级高于 options.onProgress)\n * @returns Promise<DownloadResult> 下载结果\n */\n public async download(\n url: string,\n options: DownloadOptions = {},\n onProgress?: DownloadProgressCallback\n ): Promise<DownloadResult> {\n // 检查是否正在下载,避免重复下载\n if (this._downloadingUrls.has(url)) {\n return this._createErrorResult(`资源正在下载中: ${url}`);\n }\n\n // 合并进度回调:优先使用传入的 onProgress,其次使用 options.onProgress\n const progressCallback = onProgress || options.onProgress;\n\n // 先检查是否是 base64 data URL(用于确定默认扩展名)\n const base64Info = this._isBase64DataUrl(url);\n const defaultExt = base64Info.isBase64 && base64Info.ext ? base64Info.ext : undefined;\n\n // 确定输出路径\n let outputPath: string;\n try {\n if (options.showSaveDialog) {\n const savePath = await this._showSaveDialog(url, options, defaultExt);\n if (!savePath) {\n return this._createErrorResult('用户取消了保存操作');\n }\n outputPath = savePath;\n } else {\n outputPath = this._resolveOutputPath(url, options, defaultExt);\n }\n } catch (error) {\n return this._createErrorResult(\n error instanceof Error ? error.message : '确定输出路径失败'\n );\n }\n\n // 准备输出路径(确保目录存在、检查文件是否已存在)\n const pathCheckResult = await this._prepareOutputPath(outputPath, options);\n if (pathCheckResult) {\n return pathCheckResult;\n }\n\n const downloadDir = path.dirname(outputPath);\n const downloadFileName = path.basename(outputPath);\n\n // 如果是 base64 data URL,使用已确定的输出路径进行处理\n if (base64Info.isBase64 && base64Info.ext && base64Info.mimeType && base64Info.data) {\n return await this._handleBase64Download(\n url,\n { ext: base64Info.ext, mimeType: base64Info.mimeType, data: base64Info.data },\n outputPath,\n options,\n progressCallback\n );\n }\n\n // 使用临时文件下载,避免下载中断时文件不完整\n const tempFilePath = `${outputPath}.cache`;\n\n return new Promise<DownloadResult>((resolve, reject) => {\n // 创建下载器实例\n const dl = new DownloaderHelper(url, downloadDir, {\n fileName: downloadFileName + '.cache',\n retry: {\n maxRetries: options.retry?.maxRetries ?? DEFAULT_RETRY.maxRetries,\n delay: options.retry?.delay ?? DEFAULT_RETRY.delay\n },\n timeout: options.timeout ?? DEFAULT_TIMEOUT,\n override: true, // 临时文件总是覆盖\n httpRequestOptions: {\n followRedirect: options.httpRequestOptions?.followRedirect ?? true,\n maxRedirects: options.httpRequestOptions?.maxRedirects ?? MAX_REDIRECTS,\n headers: options.httpRequestOptions?.headers\n }\n });\n\n // 记录正在下载的 URL\n this._downloadingUrls.set(url, dl);\n\n // 监听下载进度\n if (progressCallback) {\n dl.on('progress', (stats) => {\n try {\n progressCallback({\n url,\n downloaded: stats.downloaded || 0,\n total: stats.total || 0,\n percentage: stats.progress !== undefined ? Math.round(stats.progress) : -1,\n speed: stats.speed || 0\n });\n } catch (error) {\n // 忽略回调中的错误,避免影响下载\n console.error('进度回调执行失败:', error);\n }\n });\n }\n\n // 监听下载完成\n dl.on('end', () => {\n this._downloadingUrls.delete(url);\n // 将临时文件重命名为最终文件\n fs.rename(tempFilePath, outputPath, (renameErr) => {\n if (renameErr) {\n const error = new Error(`文件重命名失败 from ${tempFilePath} to ${outputPath}: ${renameErr.message}`);\n if (options.onError) {\n options.onError(error);\n }\n reject(error);\n } else {\n // 获取文件大小并返回结果\n const stats = fs.statSync(outputPath);\n const result = this._createSuccessResult(outputPath, stats.size);\n if (options.onComplete) {\n options.onComplete(outputPath);\n }\n resolve(result);\n }\n });\n });\n\n // 统一的错误处理函数\n const handleError = (error: Error) => {\n this._downloadingUrls.delete(url);\n // 清理临时文件\n fs.promises.unlink(tempFilePath).catch(() => {\n // 忽略删除失败\n });\n if (options.onError) {\n options.onError(error);\n }\n reject(error);\n };\n\n // 监听下载错误\n dl.on('error', (err) => {\n const error = err instanceof Error ? err : new Error(`下载失败: ${url}`);\n handleError(error);\n });\n\n // 监听下载停止(取消)\n dl.on('stop', () => {\n const error = new Error(`下载已停止: ${url}`);\n handleError(error);\n });\n\n // 开始下载\n dl.start().catch((err) => {\n const error = err instanceof Error ? err : new Error(`启动下载失败: ${url}`);\n handleError(error);\n });\n }).catch((error): DownloadResult => {\n return this._createErrorResult(error instanceof Error ? error.message : '未知错误');\n });\n }\n\n /**\n * 取消下载\n * @param url 要取消的 URL\n * @returns 是否成功取消\n */\n public cancel(url: string): boolean {\n const dl = this._downloadingUrls.get(url);\n if (dl) {\n dl.stop();\n this._downloadingUrls.delete(url);\n return true;\n }\n return false;\n }\n\n /**\n * 取消所有正在进行的下载\n */\n public cancelAll(): void {\n for (const [url, dl] of this._downloadingUrls.entries()) {\n dl.stop();\n }\n this._downloadingUrls.clear();\n }\n\n /**\n * 检查指定 URL 是否正在下载\n * @param url 要检查的 URL\n * @returns 是否正在下载\n */\n public isDownloading(url: string): boolean {\n return this._downloadingUrls.has(url);\n }\n\n /**\n * 获取正在下载的 URL 列表\n * @returns 正在下载的 URL 数组\n */\n public getDownloadingUrls(): string[] {\n return Array.from(this._downloadingUrls.keys());\n }\n\n /**\n * 显示保存对话框\n * @param url 下载 URL\n * @param options 下载选项\n * @param defaultExt 默认扩展名(可选,用于 base64 URL)\n * @returns Promise<string | null> 用户选择的保存路径,如果取消则返回 null\n */\n private async _showSaveDialog(url: string, options: DownloadOptions, defaultExt?: string): Promise<string | null> {\n // 获取默认文件名\n let defaultFileName: string;\n if (options.defaultPath) {\n defaultFileName = path.basename(options.defaultPath);\n } else if (options.fileName) {\n defaultFileName = options.fileName;\n } else {\n // 尝试从 URL 中提取文件名\n if (defaultExt) {\n // 如果提供了默认扩展名(base64 情况),使用它\n defaultFileName = `${DEFAULT_FILENAME}.${defaultExt}`;\n } else {\n defaultFileName = this._extractFileNameFromUrl(url);\n }\n }\n\n // 获取默认目录\n let defaultDir: string | undefined;\n if (options.defaultPath) {\n defaultDir = path.dirname(options.defaultPath);\n } else if (options.outputDir) {\n defaultDir = path.resolve(options.outputDir);\n } else {\n // 如果没有指定目录,使用上一次选择的文件夹,如果没有则使用下载目录\n defaultDir = this._lastSelectedDir || app.getPath('downloads');\n }\n\n // 如果指定了目录,检查文件是否存在并生成唯一文件名\n if (defaultDir) {\n defaultFileName = this._generateUniqueFileName(defaultDir, defaultFileName);\n }\n\n // 构建默认路径\n const defaultPath = defaultDir ? path.join(defaultDir, defaultFileName) : defaultFileName;\n\n // 获取当前活动的 BrowserWindow\n const focusedWindow = BrowserWindow.getFocusedWindow();\n const allWindows = BrowserWindow.getAllWindows();\n const parentWindow = focusedWindow || (allWindows.length > 0 ? allWindows[0] : undefined);\n\n // 显示保存对话框\n const dialogOptions = {\n defaultPath,\n title: '保存文件',\n buttonLabel: '保存',\n filters: [\n // 尝试从文件名推断文件类型\n ...(path.extname(defaultFileName) ? [{\n name: '所有文件',\n extensions: ['*']\n }] : [])\n ]\n };\n\n // 根据是否有父窗口调用不同的方法\n const result = parentWindow\n ? await dialog.showSaveDialog(parentWindow, dialogOptions)\n : await dialog.showSaveDialog(dialogOptions);\n\n if (result.canceled || !result.filePath) {\n return null;\n }\n\n // 记录用户选择的文件夹路径,用于下次默认使用\n const selectedDir = path.dirname(result.filePath);\n if (selectedDir) {\n this._lastSelectedDir = selectedDir;\n }\n\n return result.filePath;\n }\n\n /**\n * 检测并处理 base64 data URL\n * @param url 资源URL\n * @returns 如果是 base64 URL,返回 true、文件扩展名、MIME 类型和数据;否则返回 false\n */\n private _isBase64DataUrl(url: string): { isBase64: boolean, ext?: string, mimeType?: string, data?: string } {\n if (!url.startsWith('data:')) {\n return { isBase64: false };\n }\n\n try {\n // 解析 data URL 格式:data:[<mediatype>][;base64],<data>\n const commaIndex = url.indexOf(',');\n if (commaIndex === -1) {\n return { isBase64: false };\n }\n\n const header = url.substring(0, commaIndex);\n const data = url.substring(commaIndex + 1);\n\n // 检查是否包含 base64 标识\n if (!header.includes('base64')) {\n return { isBase64: false };\n }\n\n // 从 mediatype 中提取文件扩展名和 MIME 类型\n // 例如:data:image/png;base64 -> png, image/png\n // 例如:data:image/jpeg;base64 -> jpeg, image/jpeg\n let ext = DEFAULT_EXT.replace(/^\\./, '');\n let mimeType = DEFAULT_MIME_TYPE;\n const mimeMatch = header.match(/data:([^;]+)/);\n if (mimeMatch && mimeMatch[1]) {\n mimeType = mimeMatch[1];\n // 使用 mime-types 包从 MIME 类型获取扩展名\n const extension = mime.extension(mimeType);\n if (extension) {\n ext = extension;\n }\n }\n\n return { isBase64: true, ext, mimeType, data };\n } catch (error) {\n return { isBase64: false };\n }\n }\n\n /**\n * 保存 base64 数据到文件\n * @param base64Data base64 编码的数据\n * @param filePath 目标文件路径\n */\n private async _saveBase64ToFile(base64Data: string, filePath: string): Promise<void> {\n try {\n // 解码 base64 数据\n const buffer = Buffer.from(base64Data, 'base64');\n\n // 使用 Promise 版本的 writeFile\n // Buffer 继承自 Uint8Array,可以直接使用\n await fs.promises.writeFile(filePath, buffer as Uint8Array);\n } catch (error) {\n throw new Error(`保存 base64 文件失败: ${error instanceof Error ? error.message : '未知错误'}`);\n }\n }\n\n /**\n * 准备输出路径(确保目录存在、检查文件是否已存在)\n * @param outputPath 输出路径\n * @param options 下载选项\n * @returns 如果文件已存在且不允许覆盖,返回结果;否则返回 null\n */\n private async _prepareOutputPath(\n outputPath: string,\n options: DownloadOptions\n ): Promise<DownloadResult | null> {\n const downloadDir = path.dirname(outputPath);\n\n // 确保目录存在\n if (!fs.existsSync(downloadDir)) {\n fs.mkdirSync(downloadDir, { recursive: true });\n }\n\n // 如果文件已存在且不允许覆盖,直接返回\n if (fs.existsSync(outputPath) && !(options.override ?? true)) {\n const stats = fs.statSync(outputPath);\n return this._createSuccessResult(outputPath, stats.size);\n }\n\n return null;\n }\n\n /**\n * 创建成功结果\n * @param filePath 文件路径\n * @param size 文件大小\n * @returns DownloadResult\n */\n private _createSuccessResult(filePath: string, size: number): DownloadResult {\n return {\n success: true,\n filePath,\n size\n };\n }\n\n /**\n * 创建错误结果\n * @param error 错误信息\n * @returns DownloadResult\n */\n private _createErrorResult(error: string): DownloadResult {\n return {\n success: false,\n error\n };\n }\n\n /**\n * 处理 base64 data URL 的下载\n * @param url base64 data URL\n * @param base64Info base64 信息\n * @param outputPath 已确定的输出路径\n * @param options 下载选项\n * @param progressCallback 进度回调\n * @returns Promise<DownloadResult> 下载结果\n */\n private async _handleBase64Download(\n url: string,\n base64Info: { ext: string; mimeType: string; data: string },\n outputPath: string,\n options: DownloadOptions,\n progressCallback?: DownloadProgressCallback\n ): Promise<DownloadResult> {\n if (!base64Info.data || !base64Info.ext) {\n return this._createErrorResult(`无效的 base64 data URL: ${url}`);\n }\n\n try {\n // 模拟进度回调(base64 数据通常很小,立即完成)\n if (progressCallback) {\n const dataLength = Buffer.from(base64Info.data, 'base64').length;\n progressCallback({\n url,\n downloaded: dataLength,\n total: dataLength,\n percentage: 100,\n speed: 0\n });\n }\n\n // 保存 base64 数据到文件\n await this._saveBase64ToFile(base64Info.data, outputPath);\n\n // 获取文件大小\n const stats = fs.statSync(outputPath);\n const result = this._createSuccessResult(outputPath, stats.size);\n\n if (options.onComplete) {\n options.onComplete(outputPath);\n }\n\n return result;\n } catch (error) {\n const err = error instanceof Error ? error : new Error('保存 base64 文件失败');\n if (options.onError) {\n options.onError(err);\n }\n return this._createErrorResult(err.message);\n }\n }\n\n /**\n * 生成唯一的文件名(如果文件已存在,添加数字后缀)\n * @param dir 目录路径\n * @param fileName 原始文件名\n * @returns 唯一的文件名\n */\n private _generateUniqueFileName(dir: string, fileName: string): string {\n // 分离文件名和扩展名\n const ext = path.extname(fileName);\n const nameWithoutExt = path.basename(fileName, ext);\n\n // 检查原始文件名是否存在\n const originalPath = path.join(dir, fileName);\n if (!fs.existsSync(originalPath)) {\n return fileName;\n }\n\n // 如果存在,尝试添加数字后缀\n let counter = 1;\n let newFileName: string;\n let newPath: string;\n\n do {\n newFileName = `${nameWithoutExt}(${counter})${ext}`;\n newPath = path.join(dir, newFileName);\n counter++;\n } while (fs.existsSync(newPath) && counter < MAX_FILENAME_COUNTER);\n\n return newFileName;\n }\n\n /**\n * 从 URL 中提取文件名\n * @param url 下载 URL\n * @returns 文件名\n */\n private _extractFileNameFromUrl(url: string): string {\n try {\n const urlObj = new URL(url);\n const fileName = path.basename(urlObj.pathname) || DEFAULT_FILENAME;\n return path.extname(fileName) ? fileName : `${fileName}${DEFAULT_EXT}`;\n } catch {\n return `${DEFAULT_FILENAME}${DEFAULT_EXT}`;\n }\n }\n\n /**\n * 解析输出路径\n * @param url 下载 URL\n * @param options 下载选项\n * @param defaultExt 默认扩展名(可选,用于 base64 URL)\n * @returns 输出文件路径\n */\n private _resolveOutputPath(url: string, options: DownloadOptions, defaultExt?: string): string {\n // 如果提供了完整路径,直接使用\n if (options.outputPath) {\n return path.resolve(options.outputPath);\n }\n\n // 确定输出目录:优先使用指定的目录,否则使用 Electron 的下载目录\n const outputDir = options.outputDir\n ? path.resolve(options.outputDir)\n : app.getPath('downloads');\n\n // 确定文件名\n let fileName: string;\n if (options.fileName) {\n fileName = options.fileName;\n } else {\n // 如果提供了默认扩展名(base64 情况),使用它\n if (defaultExt) {\n fileName = `${DEFAULT_FILENAME}.${defaultExt}`;\n } else {\n // 尝试从 URL 中提取文件名\n fileName = this._extractFileNameFromUrl(url);\n }\n }\n\n return path.join(outputDir, fileName);\n }\n}\n\n/**\n * 默认下载器实例(单例)\n */\nlet defaultDownloader: Downloader | null = null;\n\n/**\n * 获取默认下载器实例\n * @returns Downloader 实例\n */\nexport function getDownloader(): Downloader {\n if (!defaultDownloader) {\n defaultDownloader = new Downloader();\n }\n return defaultDownloader;\n}\n\n/**\n * 下载文件的便捷函数\n * @param url 要下载的 URL\n * @param options 下载选项\n * @param onProgress 进度回调函数(可选)\n * @returns Promise<DownloadResult> 下载结果\n */\nexport async function downloadFile(\n url: string,\n options: DownloadOptions = {},\n onProgress?: DownloadProgressCallback\n): Promise<DownloadResult> {\n const downloader = getDownloader();\n return downloader.download(url, options, onProgress);\n}\n\n"],"names":["ipcMain","DownloaderHelper","app","BrowserWindow","dialog"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"downloader.js","sources":["../src/src/main/downloader.ts"],"sourcesContent":["import fs from 'node:fs';\nimport path from 'node:path';\nimport mime from 'mime-types';\nimport { DownloaderHelper } from 'node-downloader-helper';\nimport { ipcMain, dialog, BrowserWindow, app } from 'electron';\n\n/**\n * 最大重定向次数\n */\nconst MAX_REDIRECTS = 5;\n\n/**\n * 默认文件扩展名\n */\nconst DEFAULT_EXT = '.bin';\n\n/**\n * 默认 MIME 类型\n */\nconst DEFAULT_MIME_TYPE = 'application/octet-stream';\n\n/**\n * 默认超时时间(毫秒):10 分钟\n */\nconst DEFAULT_TIMEOUT = 600000;\n\n/**\n * 默认重试配置\n */\nconst DEFAULT_RETRY = {\n maxRetries: 3,\n delay: 1000\n};\n\n/**\n * 默认文件名\n */\nconst DEFAULT_FILENAME = 'download';\n\n/**\n * 最大文件名计数器(防止无限循环)\n */\nconst MAX_FILENAME_COUNTER = 10000;\n\n/**\n * 下载进度回调函数类型\n */\nexport type DownloadProgressCallback = (progress: {\n url: string; // 正在下载的 URL\n downloaded: number; // 已下载字节数\n total: number; // 总字节数(如果服务器提供 Content-Length)\n percentage: number; // 下载进度百分比 (0-100),如果 total 未知则为 -1\n speed: number; // 下载速度(字节/秒)\n}) => void;\n\n/**\n * 下载选项\n */\nexport interface DownloadOptions {\n /** 唯一标识(IPC 调用时必需) */\n id?: string;\n /** 输出文件路径(可选,如果不提供则使用 URL 的文件名) */\n outputPath?: string;\n /** 输出目录(可选,如果不提供则使用当前目录) */\n outputDir?: string;\n /** 输出文件名(可选,如果不提供则从 URL 提取) */\n fileName?: string;\n /** 是否覆盖已存在的文件(默认 true) */\n override?: boolean;\n /** 是否弹窗选择保存路径(默认 false) */\n showSaveDialog?: boolean;\n /** 保存对话框的默认文件名(可选) */\n defaultPath?: string;\n /** 超时时间(毫秒,默认 10 分钟) */\n timeout?: number;\n /** 重试配置 */\n retry?: {\n maxRetries?: number;\n delay?: number;\n };\n /** 下载进度回调函数(可选) */\n onProgress?: DownloadProgressCallback;\n /** 下载完成回调函数(可选) */\n onComplete?: (filePath: string) => void;\n /** 下载错误回调函数(可选) */\n onError?: (error: Error) => void;\n /** HTTP 请求选项 */\n httpRequestOptions?: {\n headers?: Record<string, string>;\n followRedirect?: boolean;\n maxRedirects?: number;\n };\n}\n\n/**\n * 下载结果\n */\nexport interface DownloadResult {\n /** 是否成功 */\n success: boolean;\n /** 下载的文件路径 */\n filePath?: string;\n /** 错误信息 */\n error?: string;\n /** 文件大小(字节) */\n size?: number;\n}\n\n/**\n * 下载器类:提供文件下载功能\n */\nclass Downloader {\n /** 正在下载的 URL 集合(避免重复下载) */\n private _downloadingUrls = new Map<string, DownloaderHelper>();\n /** 上一次选择的保存文件夹路径 */\n private _lastSelectedDir: string | null = null;\n\n constructor() {\n ipcMain.handle(`core:dowloader`, async (event: Electron.IpcMainInvokeEvent, options: DownloadOptions & { url: string; id: string }) => {\n try {\n const { id, url } = options;\n const sender = event.sender as Electron.WebContents;\n\n // 创建进度回调,通过 IPC 发送进度更新\n const progressCallback: DownloadProgressCallback = (progress) => {\n try {\n if (sender && typeof sender.send === 'function') {\n sender.send(`core:dowloader:progress`, {\n id,\n data: progress\n });\n }\n } catch (error) {\n console.error('发送下载进度失败:', error);\n }\n };\n\n const result = await this.download(url, options, progressCallback);\n return result;\n } catch (error) {\n return {\n success: false,\n error: error instanceof Error ? error.message : '未知错误'\n };\n }\n });\n }\n\n /**\n * 下载文件\n * @param url 要下载的 URL\n * @param options 下载选项\n * @param onProgress 进度回调函数(可选,优先级高于 options.onProgress)\n * @returns Promise<DownloadResult> 下载结果\n */\n public async download(\n url: string,\n options: DownloadOptions = {},\n onProgress?: DownloadProgressCallback\n ): Promise<DownloadResult> {\n // 生成唯一标识:URL + ID(如果存在)\n const downloadKey = this._getDownloadKey(url, options.id);\n\n // 检查是否正在下载,避免重复下载\n if (this._downloadingUrls.has(downloadKey)) {\n return this._createErrorResult(`资源正在下载中: ${url}${options.id ? ` (ID: ${options.id})` : ''}`);\n }\n\n // 合并进度回调:优先使用传入的 onProgress,其次使用 options.onProgress\n const progressCallback = onProgress || options.onProgress;\n\n // 先检查是否是 base64 data URL(用于确定默认扩展名)\n const base64Info = this._isBase64DataUrl(url);\n const defaultExt = base64Info.isBase64 && base64Info.ext ? base64Info.ext : undefined;\n\n // 确定输出路径\n let outputPath: string;\n try {\n if (options.showSaveDialog) {\n const savePath = await this._showSaveDialog(url, options, defaultExt);\n if (!savePath) {\n return this._createErrorResult('用户取消了保存操作');\n }\n outputPath = savePath;\n } else {\n outputPath = this._resolveOutputPath(url, options, defaultExt);\n }\n } catch (error) {\n return this._createErrorResult(\n error instanceof Error ? error.message : '确定输出路径失败'\n );\n }\n\n // 准备输出路径(确保目录存在、检查文件是否已存在)\n const pathCheckResult = await this._prepareOutputPath(outputPath, options);\n if (pathCheckResult) {\n return pathCheckResult;\n }\n\n const downloadDir = path.dirname(outputPath);\n const downloadFileName = path.basename(outputPath);\n\n // 如果是 base64 data URL,使用已确定的输出路径进行处理\n if (base64Info.isBase64 && base64Info.ext && base64Info.mimeType && base64Info.data) {\n return await this._handleBase64Download(\n url,\n { ext: base64Info.ext, mimeType: base64Info.mimeType, data: base64Info.data },\n outputPath,\n options,\n progressCallback\n );\n }\n\n // 使用临时文件下载,避免下载中断时文件不完整\n const tempFilePath = `${outputPath}.cache`;\n\n return new Promise<DownloadResult>((resolve, reject) => {\n // 创建下载器实例\n const dl = new DownloaderHelper(url, downloadDir, {\n fileName: downloadFileName + '.cache',\n retry: {\n maxRetries: options.retry?.maxRetries ?? DEFAULT_RETRY.maxRetries,\n delay: options.retry?.delay ?? DEFAULT_RETRY.delay\n },\n timeout: options.timeout ?? DEFAULT_TIMEOUT,\n override: true, // 临时文件总是覆盖\n httpRequestOptions: {\n followRedirect: options.httpRequestOptions?.followRedirect ?? true,\n maxRedirects: options.httpRequestOptions?.maxRedirects ?? MAX_REDIRECTS,\n headers: options.httpRequestOptions?.headers\n }\n });\n\n // 记录正在下载的 URL(使用 URL + ID 作为 key)\n this._downloadingUrls.set(downloadKey, dl);\n\n // 监听下载进度\n if (progressCallback) {\n dl.on('progress', (stats) => {\n try {\n progressCallback({\n url,\n downloaded: stats.downloaded || 0,\n total: stats.total || 0,\n percentage: stats.progress !== undefined ? Math.round(stats.progress) : -1,\n speed: stats.speed || 0\n });\n } catch (error) {\n // 忽略回调中的错误,避免影响下载\n console.error('进度回调执行失败:', error);\n }\n });\n }\n\n // 监听下载完成\n dl.on('end', () => {\n this._downloadingUrls.delete(downloadKey);\n // 将临时文件重命名为最终文件\n fs.rename(tempFilePath, outputPath, (renameErr) => {\n if (renameErr) {\n const error = new Error(`文件重命名失败 from ${tempFilePath} to ${outputPath}: ${renameErr.message}`);\n if (options.onError) {\n options.onError(error);\n }\n reject(error);\n } else {\n // 获取文件大小并返回结果\n const stats = fs.statSync(outputPath);\n const result = this._createSuccessResult(outputPath, stats.size);\n if (options.onComplete) {\n options.onComplete(outputPath);\n }\n resolve(result);\n }\n });\n });\n\n // 统一的错误处理函数\n const handleError = (error: Error) => {\n this._downloadingUrls.delete(downloadKey);\n // 清理临时文件\n fs.promises.unlink(tempFilePath).catch(() => {\n // 忽略删除失败\n });\n if (options.onError) {\n options.onError(error);\n }\n reject(error);\n };\n\n // 监听下载错误\n dl.on('error', (err) => {\n const error = err instanceof Error ? err : new Error(`下载失败: ${url}`);\n handleError(error);\n });\n\n // 监听下载停止(取消)\n dl.on('stop', () => {\n const error = new Error(`下载已停止: ${url}`);\n handleError(error);\n });\n\n // 开始下载\n dl.start().catch((err) => {\n const error = err instanceof Error ? err : new Error(`启动下载失败: ${url}`);\n handleError(error);\n });\n }).catch((error): DownloadResult => {\n return this._createErrorResult(error instanceof Error ? error.message : '未知错误');\n });\n }\n\n /**\n * 生成下载任务的唯一标识\n * @param url 下载 URL\n * @param id 可选的任务 ID\n * @returns 唯一标识\n */\n private _getDownloadKey(url: string, id?: string): string {\n return id ? `${url}::${id}` : url;\n }\n\n /**\n * 取消下载\n * @param url 要取消的 URL\n * @param id 可选的任务 ID\n * @returns 是否成功取消\n */\n public cancel(url: string, id?: string): boolean {\n const downloadKey = this._getDownloadKey(url, id);\n const dl = this._downloadingUrls.get(downloadKey);\n if (dl) {\n dl.stop();\n this._downloadingUrls.delete(downloadKey);\n return true;\n }\n return false;\n }\n\n /**\n * 取消所有正在进行的下载\n */\n public cancelAll(): void {\n for (const [url, dl] of this._downloadingUrls.entries()) {\n dl.stop();\n }\n this._downloadingUrls.clear();\n }\n\n /**\n * 检查指定 URL 是否正在下载\n * @param url 要检查的 URL\n * @param id 可选的任务 ID\n * @returns 是否正在下载\n */\n public isDownloading(url: string, id?: string): boolean {\n const downloadKey = this._getDownloadKey(url, id);\n return this._downloadingUrls.has(downloadKey);\n }\n\n /**\n * 获取正在下载的任务列表\n * @returns 正在下载的任务信息数组,包含 URL 和 ID(如果有)\n */\n public getDownloadingUrls(): Array<{ url: string; id?: string }> {\n return Array.from(this._downloadingUrls.keys()).map(key => {\n const parts = key.split('::');\n if (parts.length === 2 && parts[0] && parts[1]) {\n return { url: parts[0], id: parts[1] };\n }\n return { url: key };\n });\n }\n\n /**\n * 显示保存对话框\n * @param url 下载 URL\n * @param options 下载选项\n * @param defaultExt 默认扩展名(可选,用于 base64 URL)\n * @returns Promise<string | null> 用户选择的保存路径,如果取消则返回 null\n */\n private async _showSaveDialog(url: string, options: DownloadOptions, defaultExt?: string): Promise<string | null> {\n // 获取默认文件名\n let defaultFileName: string;\n if (options.defaultPath) {\n defaultFileName = path.basename(options.defaultPath);\n } else if (options.fileName) {\n defaultFileName = options.fileName;\n } else {\n // 尝试从 URL 中提取文件名\n if (defaultExt) {\n // 如果提供了默认扩展名(base64 情况),使用它\n defaultFileName = `${DEFAULT_FILENAME}.${defaultExt}`;\n } else {\n defaultFileName = this._extractFileNameFromUrl(url);\n }\n }\n\n // 获取默认目录\n let defaultDir: string | undefined;\n if (options.defaultPath) {\n defaultDir = path.dirname(options.defaultPath);\n } else if (options.outputDir) {\n defaultDir = path.resolve(options.outputDir);\n } else {\n // 如果没有指定目录,使用上一次选择的文件夹,如果没有则使用下载目录\n defaultDir = this._lastSelectedDir || app.getPath('downloads');\n }\n\n // 如果指定了目录,检查文件是否存在并生成唯一文件名\n if (defaultDir) {\n defaultFileName = this._generateUniqueFileName(defaultDir, defaultFileName);\n }\n\n // 构建默认路径\n const defaultPath = defaultDir ? path.join(defaultDir, defaultFileName) : defaultFileName;\n\n // 获取当前活动的 BrowserWindow\n const focusedWindow = BrowserWindow.getFocusedWindow();\n const allWindows = BrowserWindow.getAllWindows();\n const parentWindow = focusedWindow || (allWindows.length > 0 ? allWindows[0] : undefined);\n\n // 显示保存对话框\n const dialogOptions = {\n defaultPath,\n title: '保存文件',\n buttonLabel: '保存',\n filters: [\n // 尝试从文件名推断文件类型\n ...(path.extname(defaultFileName) ? [{\n name: '所有文件',\n extensions: ['*']\n }] : [])\n ]\n };\n\n // 根据是否有父窗口调用不同的方法\n const result = parentWindow\n ? await dialog.showSaveDialog(parentWindow, dialogOptions)\n : await dialog.showSaveDialog(dialogOptions);\n\n if (result.canceled || !result.filePath) {\n return null;\n }\n\n // 记录用户选择的文件夹路径,用于下次默认使用\n const selectedDir = path.dirname(result.filePath);\n if (selectedDir) {\n this._lastSelectedDir = selectedDir;\n }\n\n return result.filePath;\n }\n\n /**\n * 检测并处理 base64 data URL\n * @param url 资源URL\n * @returns 如果是 base64 URL,返回 true、文件扩展名、MIME 类型和数据;否则返回 false\n */\n private _isBase64DataUrl(url: string): { isBase64: boolean, ext?: string, mimeType?: string, data?: string } {\n if (!url.startsWith('data:')) {\n return { isBase64: false };\n }\n\n try {\n // 解析 data URL 格式:data:[<mediatype>][;base64],<data>\n const commaIndex = url.indexOf(',');\n if (commaIndex === -1) {\n return { isBase64: false };\n }\n\n const header = url.substring(0, commaIndex);\n const data = url.substring(commaIndex + 1);\n\n // 检查是否包含 base64 标识\n if (!header.includes('base64')) {\n return { isBase64: false };\n }\n\n // 从 mediatype 中提取文件扩展名和 MIME 类型\n // 例如:data:image/png;base64 -> png, image/png\n // 例如:data:image/jpeg;base64 -> jpeg, image/jpeg\n let ext = DEFAULT_EXT.replace(/^\\./, '');\n let mimeType = DEFAULT_MIME_TYPE;\n const mimeMatch = header.match(/data:([^;]+)/);\n if (mimeMatch && mimeMatch[1]) {\n mimeType = mimeMatch[1];\n // 使用 mime-types 包从 MIME 类型获取扩展名\n const extension = mime.extension(mimeType);\n if (extension) {\n ext = extension;\n }\n }\n\n return { isBase64: true, ext, mimeType, data };\n } catch (error) {\n return { isBase64: false };\n }\n }\n\n /**\n * 保存 base64 数据到文件\n * @param base64Data base64 编码的数据\n * @param filePath 目标文件路径\n */\n private async _saveBase64ToFile(base64Data: string, filePath: string): Promise<void> {\n try {\n // 解码 base64 数据\n const buffer = Buffer.from(base64Data, 'base64');\n\n // 使用 Promise 版本的 writeFile\n // Buffer 继承自 Uint8Array,可以直接使用\n await fs.promises.writeFile(filePath, buffer as Uint8Array);\n } catch (error) {\n throw new Error(`保存 base64 文件失败: ${error instanceof Error ? error.message : '未知错误'}`);\n }\n }\n\n /**\n * 准备输出路径(确保目录存在、检查文件是否已存在)\n * @param outputPath 输出路径\n * @param options 下载选项\n * @returns 如果文件已存在且不允许覆盖,返回结果;否则返回 null\n */\n private async _prepareOutputPath(\n outputPath: string,\n options: DownloadOptions\n ): Promise<DownloadResult | null> {\n const downloadDir = path.dirname(outputPath);\n\n // 确保目录存在\n if (!fs.existsSync(downloadDir)) {\n fs.mkdirSync(downloadDir, { recursive: true });\n }\n\n // 如果文件已存在且不允许覆盖,直接返回\n if (fs.existsSync(outputPath) && !(options.override ?? true)) {\n const stats = fs.statSync(outputPath);\n return this._createSuccessResult(outputPath, stats.size);\n }\n\n return null;\n }\n\n /**\n * 创建成功结果\n * @param filePath 文件路径\n * @param size 文件大小\n * @returns DownloadResult\n */\n private _createSuccessResult(filePath: string, size: number): DownloadResult {\n return {\n success: true,\n filePath,\n size\n };\n }\n\n /**\n * 创建错误结果\n * @param error 错误信息\n * @returns DownloadResult\n */\n private _createErrorResult(error: string): DownloadResult {\n return {\n success: false,\n error\n };\n }\n\n /**\n * 处理 base64 data URL 的下载\n * @param url base64 data URL\n * @param base64Info base64 信息\n * @param outputPath 已确定的输出路径\n * @param options 下载选项\n * @param progressCallback 进度回调\n * @returns Promise<DownloadResult> 下载结果\n */\n private async _handleBase64Download(\n url: string,\n base64Info: { ext: string; mimeType: string; data: string },\n outputPath: string,\n options: DownloadOptions,\n progressCallback?: DownloadProgressCallback\n ): Promise<DownloadResult> {\n if (!base64Info.data || !base64Info.ext) {\n return this._createErrorResult(`无效的 base64 data URL: ${url}`);\n }\n\n try {\n // 模拟进度回调(base64 数据通常很小,立即完成)\n if (progressCallback) {\n const dataLength = Buffer.from(base64Info.data, 'base64').length;\n progressCallback({\n url,\n downloaded: dataLength,\n total: dataLength,\n percentage: 100,\n speed: 0\n });\n }\n\n // 保存 base64 数据到文件\n await this._saveBase64ToFile(base64Info.data, outputPath);\n\n // 获取文件大小\n const stats = fs.statSync(outputPath);\n const result = this._createSuccessResult(outputPath, stats.size);\n\n if (options.onComplete) {\n options.onComplete(outputPath);\n }\n\n return result;\n } catch (error) {\n const err = error instanceof Error ? error : new Error('保存 base64 文件失败');\n if (options.onError) {\n options.onError(err);\n }\n return this._createErrorResult(err.message);\n }\n }\n\n /**\n * 生成唯一的文件名(如果文件已存在,添加数字后缀)\n * @param dir 目录路径\n * @param fileName 原始文件名\n * @returns 唯一的文件名\n */\n private _generateUniqueFileName(dir: string, fileName: string): string {\n // 分离文件名和扩展名\n const ext = path.extname(fileName);\n const nameWithoutExt = path.basename(fileName, ext);\n\n // 检查原始文件名是否存在\n const originalPath = path.join(dir, fileName);\n if (!fs.existsSync(originalPath)) {\n return fileName;\n }\n\n // 如果存在,尝试添加数字后缀\n let counter = 1;\n let newFileName: string;\n let newPath: string;\n\n do {\n newFileName = `${nameWithoutExt}(${counter})${ext}`;\n newPath = path.join(dir, newFileName);\n counter++;\n } while (fs.existsSync(newPath) && counter < MAX_FILENAME_COUNTER);\n\n return newFileName;\n }\n\n /**\n * 从 URL 中提取文件名\n * @param url 下载 URL\n * @returns 文件名\n */\n private _extractFileNameFromUrl(url: string): string {\n try {\n const urlObj = new URL(url);\n const fileName = path.basename(urlObj.pathname) || DEFAULT_FILENAME;\n return path.extname(fileName) ? fileName : `${fileName}${DEFAULT_EXT}`;\n } catch {\n return `${DEFAULT_FILENAME}${DEFAULT_EXT}`;\n }\n }\n\n /**\n * 解析输出路径\n * @param url 下载 URL\n * @param options 下载选项\n * @param defaultExt 默认扩展名(可选,用于 base64 URL)\n * @returns 输出文件路径\n */\n private _resolveOutputPath(url: string, options: DownloadOptions, defaultExt?: string): string {\n // 如果提供了完整路径,直接使用\n if (options.outputPath) {\n return path.resolve(options.outputPath);\n }\n\n // 确定输出目录:优先使用指定的目录,否则使用 Electron 的下载目录\n const outputDir = options.outputDir\n ? path.resolve(options.outputDir)\n : app.getPath('downloads');\n\n // 确定文件名\n let fileName: string;\n if (options.fileName) {\n fileName = options.fileName;\n } else {\n // 如果提供了默认扩展名(base64 情况),使用它\n if (defaultExt) {\n fileName = `${DEFAULT_FILENAME}.${defaultExt}`;\n } else {\n // 尝试从 URL 中提取文件名\n fileName = this._extractFileNameFromUrl(url);\n }\n }\n\n return path.join(outputDir, fileName);\n }\n}\n\n/**\n * 默认下载器实例(单例)\n */\nlet defaultDownloader: Downloader | null = null;\n\n/**\n * 获取默认下载器实例\n * @returns Downloader 实例\n */\nexport function getDownloader(): Downloader {\n if (!defaultDownloader) {\n defaultDownloader = new Downloader();\n }\n return defaultDownloader;\n}\n\n/**\n * 下载文件的便捷函数\n * @param url 要下载的 URL\n * @param options 下载选项\n * @param onProgress 进度回调函数(可选)\n * @returns Promise<DownloadResult> 下载结果\n */\nexport async function downloadFile(\n url: string,\n options: DownloadOptions = {},\n onProgress?: DownloadProgressCallback\n): Promise<DownloadResult> {\n const downloader = getDownloader();\n return downloader.download(url, options, onProgress);\n}\n\n"],"names":["ipcMain","DownloaderHelper","app","BrowserWindow","dialog"],"mappings":"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"}
|