@lynker-desktop/electron-sdk 0.0.9-alpha.50 → 0.0.9-alpha.52

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.
@@ -0,0 +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 http from 'node:http';\nimport https from 'node:https';\nimport md5 from 'md5';\nimport ipc from '@lynker-desktop/electron-ipc/main';\n\n/**\n * MIME 类型到文件扩展名的映射表\n */\nconst MIME_TO_EXT: Record<string, string> = {\n 'image/png': 'png',\n 'image/jpeg': 'jpeg',\n 'image/jpg': 'jpg',\n 'image/gif': 'gif',\n 'image/webp': 'webp',\n 'image/svg+xml': 'svg',\n 'image/x-icon': 'ico',\n 'image/bmp': 'bmp',\n 'image/avif': 'avif',\n 'image/heic': 'heic',\n 'image/heif': 'heif',\n 'image/tiff': 'tiff',\n 'font/woff': 'woff',\n 'font/woff2': 'woff2',\n 'font/ttf': 'ttf',\n 'font/otf': 'otf',\n 'application/font-woff': 'woff',\n 'application/font-woff2': 'woff2',\n 'video/mp4': 'mp4',\n 'video/webm': 'webm',\n 'video/ogg': 'ogg',\n 'audio/mpeg': 'mp3',\n 'audio/wav': 'wav',\n 'text/css': 'css',\n 'text/javascript': 'js',\n 'application/javascript': 'js',\n 'application/json': 'json',\n 'text/xml': 'xml',\n 'text/plain': 'txt',\n 'application/pdf': 'pdf',\n};\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 */\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\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 ipc.mainIPC.handleRenderer('core:cache', async (options: { type: 'add' | 'delete' | 'clear' | 'stats', urls?: string[], force?: boolean }) => {\n if (options.type === 'clear') {\n return await this.clearCache();\n }\n if (options.type === 'add') {\n return await this.addCacheUrls(options.urls ?? [], options.force ?? false, true);\n }\n if (options.type === 'delete') {\n return await this.deleteCacheUrls(options.urls ?? []);\n }\n if (options.type === 'stats') {\n return await this.getCacheStats();\n }\n return undefined;\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) || '.res';\n } catch {\n // 如果 URL 解析失败(可能是 base64 data URL),使用默认扩展名\n ext = '.res';\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 * @returns Promise<void> 下载完成或失败\n */\n public downloadResourceAsync(url: string, filePath: string): Promise<void> {\n return new Promise((resolve, reject) => {\n const tempFilePath = `${filePath}.cache`;\n const lib = url.startsWith('https') ? https : http;\n const file = fs.createWriteStream(tempFilePath);\n let request: http.ClientRequest;\n\n const cleanupAndAbort = (errMsg: string, err?: any) => {\n if (err) {\n console.log(errMsg, err);\n } else {\n console.log(errMsg);\n }\n if (request) {\n request.destroy();\n }\n file.close(() => {\n // 使用 existsSync 避免在文件不存在时 unlink 抛出错误\n if (fs.existsSync(tempFilePath)) {\n fs.unlink(tempFilePath, () => {});\n }\n });\n reject(new Error(errMsg));\n };\n\n request = lib.get(url, (res: http.IncomingMessage) => {\n // 处理重定向\n if (res.statusCode === 301 || res.statusCode === 302 || res.statusCode === 307 || res.statusCode === 308) {\n const location = res.headers.location;\n if (location) {\n request.destroy();\n file.close(() => {\n if (fs.existsSync(tempFilePath)) {\n fs.unlink(tempFilePath, () => {});\n }\n });\n // 递归处理重定向\n this.downloadResourceAsync(location, filePath).then(resolve).catch(reject);\n return;\n }\n // 如果没有 location,继续处理为错误\n }\n\n if (res.statusCode !== 200) {\n res.resume(); // 消费响应数据以释放内存\n cleanupAndAbort(`下载失败,状态码: ${res.statusCode} for ${url}`);\n return;\n }\n res.pipe(file);\n });\n\n file.on('finish', () => {\n file.close((err) => {\n if (err) {\n return cleanupAndAbort(`关闭临时文件流失败: ${tempFilePath}`, err);\n }\n fs.rename(tempFilePath, filePath, (renameErr) => {\n if (renameErr) {\n cleanupAndAbort(`缓存文件重命名失败 from ${tempFilePath} to ${filePath}`, renameErr);\n } else {\n resolve();\n }\n });\n });\n });\n\n file.on('error', (err) => {\n cleanupAndAbort(`写入临时文件失败: ${tempFilePath}`, err);\n });\n\n request.on('error', (err) => {\n cleanupAndAbort(`下载资源请求失败: ${url}`, err);\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 this.downloadResourceAsync(url, filePath).catch((err) => {\n console.log('后台下载资源失败:', err);\n });\n }\n\n /**\n * 检测并处理 base64 data URL\n * @param url 资源URL\n * @returns 如果是 base64 URL,返回 true 和文件扩展名;否则返回 false\n */\n private _isBase64DataUrl(url: string): { isBase64: boolean, ext?: 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 中提取文件扩展名\n // 例如:data:image/png;base64 -> png\n // 例如:data:image/jpeg;base64 -> jpeg\n let ext = 'res';\n const mimeMatch = header.match(/data:([^;]+)/);\n if (mimeMatch && mimeMatch[1]) {\n const mimeType = mimeMatch[1];\n ext = MIME_TO_EXT[mimeType] || ext;\n }\n\n return { isBase64: true, ext, 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 * @returns Promise<{ filePath: string, hostPath: string }> 返回缓存文件路径和主机路径\n * @throws 如果 URL 不匹配缓存规则或来源不允许,会抛出错误\n */\n public async cacheUrl(url: string, force: boolean = false, ignoreOrigin: boolean = false): Promise<{ filePath: string, hostPath: string }> {\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\n // 如果缓存有效且不强制重新下载,直接返回\n if (!force && await this.isCacheValidAsync(cachePath.filePath)) {\n return cachePath;\n }\n\n // 保存 base64 数据到文件\n await this._saveBase64ToFile(base64Info.data, cachePath.filePath);\n\n return cachePath;\n }\n\n // 处理普通 URL\n const shouldCache = 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 // 如果缓存有效且不强制重新下载,直接返回\n if (!force && await this.isCacheValidAsync(cachePath.filePath)) {\n return cachePath;\n }\n\n // 下载资源\n await this.downloadResourceAsync(url, cachePath.filePath);\n\n return cachePath;\n }\n\n /**\n * 批量缓存多个 URL 的资源\n * @param urls 要缓存的资源 URL 数组\n * @param force 是否强制重新下载,即使缓存有效(默认 false)\n * @returns Promise<Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, error?: string }>> 返回每个 URL 的缓存结果\n */\n public async addCacheUrls(urls: string[], force: boolean = false, ignoreOrigin: boolean = false): Promise<Array<{ url: string, success: boolean, filePath?: string, hostPath?: string, error?: string }>> {\n const results = await Promise.allSettled(\n urls.map(url => this.cacheUrl(url, force, ignoreOrigin))\n );\n\n return results.map((result, index) => {\n const url = urls[index] || '';\n if (result.status === 'fulfilled') {\n return {\n url,\n success: true,\n filePath: result.value.filePath,\n hostPath: result.value.hostPath\n };\n } else {\n return {\n url,\n success: false,\n error: result.reason?.message || '未知错误'\n };\n }\n });\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 = fs.readdirSync(this.options.cacheDir);\n if (files.length === 0) {\n console.log('缓存目录为空,无需清理');\n return { success: 0, failed: 0, totalSize: 0 };\n }\n\n // 先统计所有文件大小(避免并行时的竞态条件)\n const fileInfos = files.map((file) => {\n const filePath = path.join(this.options.cacheDir, file);\n try {\n const stats = fs.statSync(filePath);\n return { file, filePath, size: stats.size };\n } catch (error) {\n return { file, filePath, size: 0, error };\n }\n });\n\n // 计算总大小\n const totalSize = fileInfos.reduce((sum, info) => sum + info.size, 0);\n\n // 并行删除文件,提升性能\n const deleteResults = await Promise.allSettled(\n fileInfos.map(async (info) => {\n try {\n await fs.promises.unlink(info.filePath);\n return { success: true, file: info.file };\n } catch (error) {\n console.log(`清理缓存文件失败: ${info.file}`, 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 const sizeMB = (totalSize / (1024 * 1024)).toFixed(2);\n console.log(`缓存清理完成: 成功 ${success} 个, 失败 ${failed} 个, 释放空间 ${sizeMB} MB`);\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":["ipc"],"mappings":"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"}
@@ -0,0 +1,39 @@
1
+ import type { VideoDownloadOptions, VideoProgressCallback, VideoDownloadResult } from '../common';
2
+ /**
3
+ * 生成跨平台的文件URL
4
+ * @param filePath 文件路径
5
+ * @returns file:// URL
6
+ */
7
+ export declare function getFileUrl(filePath: string): string;
8
+ /**
9
+ * 下载视频并转码
10
+ * @param options 下载和转码选项
11
+ * @param callbacks 进度回调
12
+ * @returns 转码结果
13
+ */
14
+ export declare const downloadVideo: (options: VideoDownloadOptions, callbacks?: VideoProgressCallback) => Promise<VideoDownloadResult>;
15
+ /**
16
+ * 清除指定目录的所有缓存文件
17
+ */
18
+ export declare const clearVideoCache: (outputDir: string) => void;
19
+ /**
20
+ * 获取缓存统计信息
21
+ */
22
+ export declare const getVideoCacheStats: (outputDir: string) => {
23
+ size: number;
24
+ entries: Array<{
25
+ fileName: string;
26
+ filePath: string;
27
+ fileSize: number;
28
+ mtime: number;
29
+ }>;
30
+ };
31
+ /**
32
+ * 删除指定URL的缓存文件
33
+ */
34
+ export declare const removeVideoCache: (options: VideoDownloadOptions) => boolean;
35
+ /**
36
+ * 设置FFmpeg路径
37
+ */
38
+ export declare const setFFmpegPath: (ffmpegPath: string) => void;
39
+ //# sourceMappingURL=video-downloader.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"video-downloader.d.ts","sourceRoot":"","sources":["../../src/main/video-downloader.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,oBAAoB,EAAE,qBAAqB,EAAE,mBAAmB,EAAE,MAAM,WAAW,CAAC;AAElG;;;;GAIG;AACH,wBAAgB,UAAU,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,CAmBnD;AA4fD;;;;;GAKG;AACH,eAAO,MAAM,aAAa,YACf,oBAAoB,cACjB,qBAAqB,KAChC,OAAO,CAAC,mBAAmB,CAG7B,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,eAAe,cAAe,MAAM,KAAG,IAGnD,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,kBAAkB,cAAe,MAAM,KAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,KAAK,CAAC;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC,CAAA;CAG7J,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,gBAAgB,YAAa,oBAAoB,KAAG,OAGhE,CAAC;AAEF;;GAEG;AACH,eAAO,MAAM,aAAa,eAAgB,MAAM,KAAG,IAGlD,CAAC"}
@@ -0,0 +1,505 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import http from 'node:http';
4
+ import https from 'node:https';
5
+ import { spawn } from 'node:child_process';
6
+ import { app } from 'electron';
7
+ import md5 from 'md5';
8
+
9
+ /**
10
+ * 生成跨平台的文件URL
11
+ * @param filePath 文件路径
12
+ * @returns file:// URL
13
+ */
14
+ function getFileUrl(filePath) {
15
+ if (!filePath) {
16
+ return '';
17
+ }
18
+ if (filePath.startsWith('file://')) {
19
+ return filePath;
20
+ }
21
+ // 在Windows上,需要添加驱动器字母前的斜杠
22
+ if (process.platform === 'win32') {
23
+ // 如果路径是绝对路径(如 C:\path\to\file),转换为 /C:/path/to/file
24
+ if (filePath.match(/^[A-Za-z]:/)) {
25
+ return `file:///${filePath}`;
26
+ }
27
+ // 如果已经是 /C:/path/to/file 格式,直接添加 file://
28
+ return `file://${filePath}`;
29
+ }
30
+ // macOS 和 Linux 使用 file:/// 前缀
31
+ return `file://${filePath}`;
32
+ }
33
+ /**
34
+ * 视频下载和转码类
35
+ */
36
+ class VideoDownloader {
37
+ constructor() {
38
+ this.ffmpegPath = '';
39
+ this.defaultCacheTTL = 24 * 60 * 60 * 1000; // 24小时缓存有效期
40
+ }
41
+ static getInstance() {
42
+ if (!VideoDownloader.instance) {
43
+ VideoDownloader.instance = new VideoDownloader();
44
+ }
45
+ return VideoDownloader.instance;
46
+ }
47
+ setFFmpegPath(ffmpegPath) {
48
+ this.ffmpegPath = ffmpegPath;
49
+ }
50
+ /**
51
+ * 下载视频文件
52
+ */
53
+ async downloadVideoFile(url, outputPath, onProgress) {
54
+ return new Promise((resolve, reject) => {
55
+ const lib = url.startsWith('https') ? https : http;
56
+ const file = fs.createWriteStream(outputPath);
57
+ let downloadedBytes = 0;
58
+ let totalBytes = 0;
59
+ let startTime = Date.now();
60
+ let lastTime = startTime;
61
+ let lastBytes = 0;
62
+ const cleanup = () => {
63
+ if (file) {
64
+ file.close();
65
+ fs.unlinkSync(outputPath);
66
+ }
67
+ };
68
+ const request = lib.get(url, (res) => {
69
+ if (res.statusCode !== 200) {
70
+ cleanup();
71
+ reject(new Error(`下载失败,状态码: ${res.statusCode}`));
72
+ return;
73
+ }
74
+ totalBytes = parseInt(res.headers['content-length'] || '0', 10);
75
+ res.on('data', (chunk) => {
76
+ downloadedBytes += chunk.length;
77
+ const now = Date.now();
78
+ if (onProgress && (now - lastTime > 100 || downloadedBytes === totalBytes)) {
79
+ const speed = ((downloadedBytes - lastBytes) * 1000) / (now - lastTime);
80
+ onProgress({
81
+ downloaded: downloadedBytes,
82
+ total: totalBytes,
83
+ percentage: totalBytes > 0 ? (downloadedBytes / totalBytes) * 100 : 0,
84
+ speed: speed || 0
85
+ });
86
+ lastTime = now;
87
+ lastBytes = downloadedBytes;
88
+ }
89
+ });
90
+ res.pipe(file);
91
+ file.on('finish', () => {
92
+ file.close();
93
+ resolve();
94
+ });
95
+ file.on('error', (err) => {
96
+ cleanup();
97
+ reject(err);
98
+ });
99
+ });
100
+ request.on('error', (err) => {
101
+ cleanup();
102
+ reject(err);
103
+ });
104
+ });
105
+ }
106
+ /**
107
+ * 使用FFmpeg转码视频
108
+ */
109
+ async transcodeVideo(inputPath, outputPath, options, onProgress) {
110
+ return new Promise((resolve, reject) => {
111
+ const ffmpegPath = options.ffmpegPath || this.ffmpegPath;
112
+ if (!ffmpegPath) {
113
+ reject(new Error('FFmpeg路径未设置'));
114
+ return;
115
+ }
116
+ const format = options.format || 'mp4';
117
+ const quality = options.quality || 18;
118
+ const videoCodec = options.videoCodec || 'h264';
119
+ const audioCodec = options.audioCodec || 'aac';
120
+ // 根据格式选择合适的编码器
121
+ const getCodecForFormat = (format, videoCodec, audioCodec) => {
122
+ switch (format) {
123
+ case 'webm':
124
+ return {
125
+ videoCodec: videoCodec === 'h264' ? 'libvpx-vp9' : videoCodec,
126
+ audioCodec: audioCodec === 'aac' ? 'libopus' : audioCodec
127
+ };
128
+ case 'avi':
129
+ return {
130
+ videoCodec: videoCodec === 'h265' ? 'h264' : videoCodec, // AVI不支持H265
131
+ audioCodec: audioCodec === 'opus' ? 'mp3' : audioCodec
132
+ };
133
+ case 'mov':
134
+ return {
135
+ videoCodec: videoCodec === 'vp9' ? 'h264' : videoCodec, // MOV对VP9支持有限
136
+ audioCodec: audioCodec === 'opus' ? 'aac' : audioCodec
137
+ };
138
+ default: // mp4, mkv
139
+ return { videoCodec, audioCodec };
140
+ }
141
+ };
142
+ const { videoCodec: finalVideoCodec, audioCodec: finalAudioCodec } = getCodecForFormat(format, videoCodec, audioCodec);
143
+ // 构建FFmpeg命令
144
+ const args = [
145
+ '-i', inputPath,
146
+ '-c:v', finalVideoCodec,
147
+ '-c:a', finalAudioCodec,
148
+ '-crf', quality.toString(),
149
+ '-preset', 'medium',
150
+ '-f', format, // 指定输出格式
151
+ '-y', // 覆盖输出文件
152
+ outputPath
153
+ ];
154
+ const ffmpegProcess = spawn(ffmpegPath, args);
155
+ let duration = 0;
156
+ let stderr = '';
157
+ ffmpegProcess.stderr.on('data', (data) => {
158
+ stderr += data.toString();
159
+ // 解析FFmpeg输出获取进度
160
+ if (onProgress) {
161
+ const durationMatch = stderr.match(/Duration: (\d{2}):(\d{2}):(\d{2})\.(\d{2})/);
162
+ if (durationMatch && duration === 0) {
163
+ const hours = parseInt(durationMatch[1] || '0', 10);
164
+ const minutes = parseInt(durationMatch[2] || '0', 10);
165
+ const seconds = parseInt(durationMatch[3] || '0', 10);
166
+ const centiseconds = parseInt(durationMatch[4] || '0', 10);
167
+ duration = hours * 3600 + minutes * 60 + seconds + centiseconds / 100;
168
+ }
169
+ const timeMatch = stderr.match(/time=(\d{2}):(\d{2}):(\d{2})\.(\d{2})/);
170
+ if (timeMatch && duration > 0) {
171
+ const hours = parseInt(timeMatch[1] || '0', 10);
172
+ const minutes = parseInt(timeMatch[2] || '0', 10);
173
+ const seconds = parseInt(timeMatch[3] || '0', 10);
174
+ const centiseconds = parseInt(timeMatch[4] || '0', 10);
175
+ const currentTime = hours * 3600 + minutes * 60 + seconds + centiseconds / 100;
176
+ const percentage = (currentTime / duration) * 100;
177
+ const speedMatch = stderr.match(/speed=([\d.]+)x/);
178
+ const speed = speedMatch ? parseFloat(speedMatch[1] || '1') : 1;
179
+ onProgress({
180
+ percentage: Math.min(percentage, 100),
181
+ time: currentTime,
182
+ speed: speed
183
+ });
184
+ }
185
+ }
186
+ });
187
+ ffmpegProcess.on('close', (code) => {
188
+ if (code === 0) {
189
+ resolve();
190
+ }
191
+ else {
192
+ reject(new Error(`FFmpeg转码失败,退出码: ${code}\n${stderr}`));
193
+ }
194
+ });
195
+ ffmpegProcess.on('error', (err) => {
196
+ reject(new Error(`FFmpeg执行失败: ${err.message}`));
197
+ });
198
+ });
199
+ }
200
+ /**
201
+ * 检查FFmpeg是否可用
202
+ */
203
+ async checkFFmpegAvailable(ffmpegPath) {
204
+ return new Promise((resolve) => {
205
+ const command = ffmpegPath || this.ffmpegPath;
206
+ if (!command) {
207
+ resolve(false);
208
+ return;
209
+ }
210
+ const childProcess = spawn(command, ['-version']);
211
+ childProcess.on('error', (error) => {
212
+ console.log("checkFFmpegAvailable", error);
213
+ resolve(false);
214
+ });
215
+ childProcess.on('close', (code) => {
216
+ console.log("checkFFmpegAvailable", code);
217
+ resolve(code === 0);
218
+ });
219
+ });
220
+ }
221
+ /**
222
+ * 获取视频信息
223
+ */
224
+ async getVideoInfo(inputPath, ffmpegPath) {
225
+ return new Promise((resolve, reject) => {
226
+ const ffmpegPath_cmd = ffmpegPath || this.ffmpegPath;
227
+ const args = ['-i', inputPath, '-f', 'null', '-'];
228
+ if (!ffmpegPath_cmd) {
229
+ resolve({ duration: 0, fileSize: 0 });
230
+ return;
231
+ }
232
+ const ffmpegProcess = spawn(ffmpegPath_cmd, args);
233
+ let stderr = '';
234
+ let duration = 0;
235
+ ffmpegProcess.stderr.on('data', (data) => {
236
+ stderr += data.toString();
237
+ const durationMatch = stderr.match(/Duration: (\d{2}):(\d{2}):(\d{2})\.(\d{2})/);
238
+ if (durationMatch) {
239
+ const hours = parseInt(durationMatch[1] || '0', 10);
240
+ const minutes = parseInt(durationMatch[2] || '0', 10);
241
+ const seconds = parseInt(durationMatch[3] || '0', 10);
242
+ const centiseconds = parseInt(durationMatch[4] || '0', 10);
243
+ duration = hours * 3600 + minutes * 60 + seconds + centiseconds / 100;
244
+ }
245
+ });
246
+ ffmpegProcess.on('close', () => {
247
+ const stats = fs.statSync(inputPath);
248
+ resolve({
249
+ duration,
250
+ fileSize: stats.size
251
+ });
252
+ });
253
+ ffmpegProcess.on('error', (err) => {
254
+ reject(err);
255
+ });
256
+ });
257
+ }
258
+ /**
259
+ * 生成缓存文件名(基于URL的MD5)
260
+ */
261
+ generateCacheFileName(options) {
262
+ const format = options.format || 'mp4';
263
+ const urlMd5 = md5(options.url);
264
+ return `${urlMd5}.${format}`;
265
+ }
266
+ /**
267
+ * 检查缓存文件是否存在
268
+ */
269
+ isCacheFileExists(cacheFilePath) {
270
+ console.log('检查缓存文件是否存在: ', cacheFilePath);
271
+ return fs.existsSync(cacheFilePath);
272
+ }
273
+ /**
274
+ * 清理过期缓存文件
275
+ */
276
+ cleanExpiredCacheFiles(outputDir, cacheTTL = this.defaultCacheTTL) {
277
+ try {
278
+ const files = fs.readdirSync(outputDir);
279
+ const now = Date.now();
280
+ files.forEach(file => {
281
+ const filePath = path.join(outputDir, file);
282
+ try {
283
+ const stats = fs.statSync(filePath);
284
+ // 如果文件超过缓存时间,删除它
285
+ if (now - stats.mtimeMs > cacheTTL) {
286
+ fs.unlinkSync(filePath);
287
+ console.log(`🧹 删除过期缓存文件: ${file}`);
288
+ }
289
+ }
290
+ catch (error) {
291
+ // 忽略单个文件错误
292
+ }
293
+ });
294
+ }
295
+ catch (error) {
296
+ // 忽略目录读取错误
297
+ }
298
+ }
299
+ /**
300
+ * 下载并转码视频
301
+ */
302
+ async downloadAndTranscode(options, callbacks) {
303
+ try {
304
+ const outputDir = options.outputDir || app.getPath('temp');
305
+ // 获取缓存配置
306
+ const cacheConfig = options.cache || { enabled: true, ttl: this.defaultCacheTTL };
307
+ const cacheEnabled = cacheConfig.enabled !== false;
308
+ const cacheTTL = cacheConfig.ttl || this.defaultCacheTTL;
309
+ // 确保输出目录存在
310
+ if (!fs.existsSync(outputDir)) {
311
+ fs.mkdirSync(outputDir, { recursive: true });
312
+ }
313
+ // 清理过期缓存文件
314
+ this.cleanExpiredCacheFiles(outputDir, cacheTTL);
315
+ // 生成缓存文件名
316
+ const cacheFileName = this.generateCacheFileName(options);
317
+ const cacheFilePath = path.join(outputDir, cacheFileName);
318
+ // 如果启用缓存,检查缓存文件是否存在
319
+ if (cacheEnabled && this.isCacheFileExists(cacheFilePath)) {
320
+ console.log('🎯 命中缓存,直接返回缓存文件');
321
+ // 获取文件信息
322
+ const stats = fs.statSync(cacheFilePath);
323
+ const result = {
324
+ success: true,
325
+ outputPath: getFileUrl(cacheFilePath),
326
+ fileSize: stats.size,
327
+ fromCache: true
328
+ };
329
+ callbacks?.onComplete?.({
330
+ success: true,
331
+ outputPath: result.outputPath
332
+ });
333
+ return result;
334
+ }
335
+ // 检查FFmpeg是否可用
336
+ const ffmpegAvailable = await this.checkFFmpegAvailable(options.ffmpegPath);
337
+ if (!ffmpegAvailable) {
338
+ throw new Error('FFmpeg不可用,请确保已安装FFmpeg或提供正确的路径');
339
+ }
340
+ // 生成临时文件名和最终输出路径
341
+ const fileName = options.outputName || md5(options.url);
342
+ const format = options.format || 'mp4';
343
+ const tempPath = path.join(outputDir, `${fileName}.temp`);
344
+ const finalOutputPath = path.join(outputDir, `${fileName}.${format}`);
345
+ const outputPath = cacheEnabled ? cacheFilePath : finalOutputPath;
346
+ console.log('📥 开始下载视频...');
347
+ // 下载视频
348
+ await this.downloadVideoFile(options.url, tempPath, callbacks?.onDownloadProgress);
349
+ console.log('🔄 开始转码视频...');
350
+ // 转码视频
351
+ await this.transcodeVideo(tempPath, outputPath, {
352
+ ...options,
353
+ outputDir
354
+ }, callbacks?.onTranscodeProgress);
355
+ // 如果启用了缓存,将缓存文件复制到最终输出路径
356
+ if (cacheEnabled && outputPath !== finalOutputPath) {
357
+ fs.copyFileSync(outputPath, finalOutputPath);
358
+ console.log('📋 缓存文件已复制到最终输出路径');
359
+ }
360
+ // 获取视频信息
361
+ const videoInfo = await this.getVideoInfo(outputPath, options.ffmpegPath);
362
+ // 清理临时文件
363
+ if (!options.keepOriginal) {
364
+ fs.unlinkSync(tempPath);
365
+ }
366
+ console.log('💾 文件已保存到缓存');
367
+ const result = {
368
+ success: true,
369
+ outputPath: getFileUrl(cacheEnabled ? finalOutputPath : outputPath),
370
+ originalPath: options.keepOriginal ? getFileUrl(tempPath) : undefined,
371
+ duration: videoInfo.duration,
372
+ fileSize: videoInfo.fileSize,
373
+ fromCache: false
374
+ };
375
+ callbacks?.onComplete?.({
376
+ success: true,
377
+ outputPath: result.outputPath
378
+ });
379
+ return result;
380
+ }
381
+ catch (error) {
382
+ const errorMessage = error instanceof Error ? error.message : '未知错误';
383
+ const result = {
384
+ success: false,
385
+ error: errorMessage
386
+ };
387
+ callbacks?.onComplete?.({
388
+ success: false,
389
+ error: errorMessage
390
+ });
391
+ return result;
392
+ }
393
+ }
394
+ /**
395
+ * 清除指定目录的所有缓存文件
396
+ */
397
+ clearCache(outputDir) {
398
+ try {
399
+ const files = fs.readdirSync(outputDir);
400
+ files.forEach(file => {
401
+ const filePath = path.join(outputDir, file);
402
+ try {
403
+ fs.unlinkSync(filePath);
404
+ }
405
+ catch (error) {
406
+ // 忽略单个文件删除错误
407
+ }
408
+ });
409
+ console.log('🧹 缓存文件已清空');
410
+ }
411
+ catch (error) {
412
+ console.log('⚠️ 清理缓存时发生错误:', error instanceof Error ? error.message : '未知错误');
413
+ }
414
+ }
415
+ /**
416
+ * 获取缓存统计信息
417
+ */
418
+ getCacheStats(outputDir) {
419
+ try {
420
+ const files = fs.readdirSync(outputDir);
421
+ const entries = files.map(file => {
422
+ const filePath = path.join(outputDir, file);
423
+ try {
424
+ const stats = fs.statSync(filePath);
425
+ return {
426
+ fileName: file,
427
+ filePath: filePath,
428
+ fileSize: stats.size,
429
+ mtime: stats.mtimeMs
430
+ };
431
+ }
432
+ catch {
433
+ return null;
434
+ }
435
+ }).filter((entry) => entry !== null);
436
+ return {
437
+ size: entries.length,
438
+ entries
439
+ };
440
+ }
441
+ catch (error) {
442
+ return { size: 0, entries: [] };
443
+ }
444
+ }
445
+ /**
446
+ * 删除指定URL的缓存文件
447
+ */
448
+ removeCache(options) {
449
+ try {
450
+ const cacheFileName = this.generateCacheFileName(options);
451
+ const cacheFilePath = path.join(options.outputDir, cacheFileName);
452
+ if (fs.existsSync(cacheFilePath)) {
453
+ fs.unlinkSync(cacheFilePath);
454
+ console.log(`🗑️ 删除缓存文件: ${cacheFileName}`);
455
+ return true;
456
+ }
457
+ return false;
458
+ }
459
+ catch (error) {
460
+ console.log('⚠️ 删除缓存文件时发生错误:', error instanceof Error ? error.message : '未知错误');
461
+ return false;
462
+ }
463
+ }
464
+ }
465
+ /**
466
+ * 下载视频并转码
467
+ * @param options 下载和转码选项
468
+ * @param callbacks 进度回调
469
+ * @returns 转码结果
470
+ */
471
+ const downloadVideo = async (options, callbacks) => {
472
+ const downloader = VideoDownloader.getInstance();
473
+ return downloader.downloadAndTranscode(options, callbacks);
474
+ };
475
+ /**
476
+ * 清除指定目录的所有缓存文件
477
+ */
478
+ const clearVideoCache = (outputDir) => {
479
+ const downloader = VideoDownloader.getInstance();
480
+ downloader.clearCache(outputDir);
481
+ };
482
+ /**
483
+ * 获取缓存统计信息
484
+ */
485
+ const getVideoCacheStats = (outputDir) => {
486
+ const downloader = VideoDownloader.getInstance();
487
+ return downloader.getCacheStats(outputDir);
488
+ };
489
+ /**
490
+ * 删除指定URL的缓存文件
491
+ */
492
+ const removeVideoCache = (options) => {
493
+ const downloader = VideoDownloader.getInstance();
494
+ return downloader.removeCache(options);
495
+ };
496
+ /**
497
+ * 设置FFmpeg路径
498
+ */
499
+ const setFFmpegPath = (ffmpegPath) => {
500
+ const downloader = VideoDownloader.getInstance();
501
+ downloader.setFFmpegPath(ffmpegPath);
502
+ };
503
+
504
+ export { clearVideoCache, downloadVideo, getFileUrl, getVideoCacheStats, removeVideoCache, setFFmpegPath };
505
+ //# sourceMappingURL=video-downloader.js.map