// /api/proxy/[...path].mjs - Vercel Serverless Function (ES Module) import fetch from 'node-fetch'; import { URL } from 'url'; // 使用 Node.js 内置 URL 处理 // --- 配置 (从环境变量读取) --- const DEBUG_ENABLED = process.env.DEBUG === 'true'; const CACHE_TTL = parseInt(process.env.CACHE_TTL || '86400', 10); // 默认 24 小时 const MAX_RECURSION = parseInt(process.env.MAX_RECURSION || '5', 10); // 默认 5 层 // --- User Agent 处理 --- // 默认 User Agent 列表 let USER_AGENTS = [ 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36', 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.0 Safari/605.1.15' ]; // 尝试从环境变量读取并解析 USER_AGENTS_JSON try { const agentsJsonString = process.env.USER_AGENTS_JSON; if (agentsJsonString) { const parsedAgents = JSON.parse(agentsJsonString); // 检查解析结果是否为非空数组 if (Array.isArray(parsedAgents) && parsedAgents.length > 0) { USER_AGENTS = parsedAgents; // 使用环境变量中的数组 console.log(`[代理日志] 已从环境变量加载 ${USER_AGENTS.length} 个 User Agent。`); } else { console.warn("[代理日志] 环境变量 USER_AGENTS_JSON 不是有效的非空数组,使用默认值。"); } } else { console.log("[代理日志] 未设置环境变量 USER_AGENTS_JSON,使用默认 User Agent。"); } } catch (e) { // 如果 JSON 解析失败,记录错误并使用默认值 console.error(`[代理日志] 解析环境变量 USER_AGENTS_JSON 出错: ${e.message}。使用默认 User Agent。`); } // 广告过滤在代理中禁用,由播放器处理 const FILTER_DISCONTINUITY = false; // --- 辅助函数 --- function logDebug(message) { if (DEBUG_ENABLED) { console.log(`[代理日志] ${message}`); } } /** * 从代理请求路径中提取编码后的目标 URL。 * @param {string} encodedPath - URL 编码后的路径部分 (例如 "https%3A%2F%2F...") * @returns {string|null} 解码后的目标 URL,如果无效则返回 null。 */ function getTargetUrlFromPath(encodedPath) { if (!encodedPath) { logDebug("getTargetUrlFromPath 收到空路径。"); return null; } try { const decodedUrl = decodeURIComponent(encodedPath); // 基础检查,看是否像一个 HTTP/HTTPS URL if (decodedUrl.match(/^https?:\/\/.+/i)) { return decodedUrl; } else { logDebug(`无效的解码 URL 格式: ${decodedUrl}`); // 备选检查:原始路径是否未编码但看起来像 URL? if (encodedPath.match(/^https?:\/\/.+/i)) { logDebug(`警告: 路径未编码但看起来像 URL: ${encodedPath}`); return encodedPath; } return null; } } catch (e) { // 捕获解码错误 (例如格式错误的 URI) logDebug(`解码目标 URL 出错: ${encodedPath} - ${e.message}`); return null; } } function getBaseUrl(urlStr) { if (!urlStr) return ''; try { const parsedUrl = new URL(urlStr); // 处理根目录或只有文件名的情况 const pathSegments = parsedUrl.pathname.split('/').filter(Boolean); // 移除空字符串 if (pathSegments.length <= 1) { return `${parsedUrl.origin}/`; } pathSegments.pop(); // 移除最后一段 return `${parsedUrl.origin}/${pathSegments.join('/')}/`; } catch (e) { logDebug(`获取 BaseUrl 失败: "${urlStr}": ${e.message}`); // 备用方法:查找最后一个斜杠 const lastSlashIndex = urlStr.lastIndexOf('/'); if (lastSlashIndex > urlStr.indexOf('://') + 2) { // 确保不是协议部分的斜杠 return urlStr.substring(0, lastSlashIndex + 1); } return urlStr + '/'; // 如果没有路径,添加斜杠 } } function resolveUrl(baseUrl, relativeUrl) { if (!relativeUrl) return ''; // 处理空的 relativeUrl if (relativeUrl.match(/^https?:\/\/.+/i)) { return relativeUrl; // 已经是绝对 URL } if (!baseUrl) return relativeUrl; // 没有基础 URL 无法解析 try { // 使用 Node.js 的 URL 构造函数处理相对路径 return new URL(relativeUrl, baseUrl).toString(); } catch (e) { logDebug(`URL 解析失败: base="${baseUrl}", relative="${relativeUrl}". 错误: ${e.message}`); // 简单的备用逻辑 if (relativeUrl.startsWith('/')) { try { const baseOrigin = new URL(baseUrl).origin; return `${baseOrigin}${relativeUrl}`; } catch { return relativeUrl; } // 如果 baseUrl 也无效,返回原始相对路径 } else { // 假设相对于包含基础 URL 资源的目录 return `${baseUrl.substring(0, baseUrl.lastIndexOf('/') + 1)}${relativeUrl}`; } } } // ** 已修正:确保生成 /proxy/ 前缀的链接 ** function rewriteUrlToProxy(targetUrl) { if (!targetUrl || typeof targetUrl !== 'string') return ''; // 返回与 vercel.json 的 "source" 和前端 PROXY_URL 一致的路径 return `/proxy/${encodeURIComponent(targetUrl)}`; } function getRandomUserAgent() { return USER_AGENTS[Math.floor(Math.random() * USER_AGENTS.length)]; } async function fetchContentWithType(targetUrl, requestHeaders) { // 准备请求头 const headers = { 'User-Agent': getRandomUserAgent(), 'Accept': requestHeaders['accept'] || '*/*', // 传递原始 Accept 头(如果有) 'Accept-Language': requestHeaders['accept-language'] || 'zh-CN,zh;q=0.9,en;q=0.8', // 尝试设置一个合理的 Referer 'Referer': requestHeaders['referer'] || new URL(targetUrl).origin, }; // 清理空值的头 Object.keys(headers).forEach(key => headers[key] === undefined || headers[key] === null || headers[key] === '' ? delete headers[key] : {}); logDebug(`准备请求目标: ${targetUrl},请求头: ${JSON.stringify(headers)}`); try { // 发起 fetch 请求 const response = await fetch(targetUrl, { headers, redirect: 'follow' }); // 检查响应是否成功 if (!response.ok) { const errorBody = await response.text().catch(() => ''); // 尝试获取错误响应体 logDebug(`请求失败: ${response.status} ${response.statusText} - ${targetUrl}`); // 创建一个包含状态码的错误对象 const err = new Error(`HTTP 错误 ${response.status}: ${response.statusText}. URL: ${targetUrl}. Body: ${errorBody.substring(0, 200)}`); err.status = response.status; // 将状态码附加到错误对象 throw err; // 抛出错误 } // 读取响应内容 const content = await response.text(); const contentType = response.headers.get('content-type') || ''; logDebug(`请求成功: ${targetUrl}, Content-Type: ${contentType}, 内容长度: ${content.length}`); // 返回结果 return { content, contentType, responseHeaders: response.headers }; } catch (error) { // 捕获 fetch 本身的错误(网络、超时等)或上面抛出的 HTTP 错误 logDebug(`请求异常 ${targetUrl}: ${error.message}`); // 重新抛出,确保包含原始错误信息 throw new Error(`请求目标 URL 失败 ${targetUrl}: ${error.message}`); } } function isM3u8Content(content, contentType) { if (contentType && (contentType.includes('application/vnd.apple.mpegurl') || contentType.includes('application/x-mpegurl') || contentType.includes('audio/mpegurl'))) { return true; } return content && typeof content === 'string' && content.trim().startsWith('#EXTM3U'); } function processKeyLine(line, baseUrl) { return line.replace(/URI="([^"]+)"/, (match, uri) => { const absoluteUri = resolveUrl(baseUrl, uri); logDebug(`处理 KEY URI: 原始='${uri}', 绝对='${absoluteUri}'`); return `URI="${rewriteUrlToProxy(absoluteUri)}"`; }); } function processMapLine(line, baseUrl) { return line.replace(/URI="([^"]+)"/, (match, uri) => { const absoluteUri = resolveUrl(baseUrl, uri); logDebug(`处理 MAP URI: 原始='${uri}', 绝对='${absoluteUri}'`); return `URI="${rewriteUrlToProxy(absoluteUri)}"`; }); } function processMediaPlaylist(url, content) { const baseUrl = getBaseUrl(url); if (!baseUrl) { logDebug(`无法确定媒体列表的 Base URL: ${url},相对路径可能无法处理。`); } const lines = content.split('\n'); const output = []; for (let i = 0; i < lines.length; i++) { const line = lines[i].trim(); // 保留最后一个空行 if (!line && i === lines.length - 1) { output.push(line); continue; } if (!line) continue; // 跳过中间空行 // 广告过滤已禁用 if (line.startsWith('#EXT-X-KEY')) { output.push(processKeyLine(line, baseUrl)); continue; } if (line.startsWith('#EXT-X-MAP')) { output.push(processMapLine(line, baseUrl)); continue; } if (line.startsWith('#EXTINF')) { output.push(line); continue; } // 处理 URL 行 if (!line.startsWith('#')) { const absoluteUrl = resolveUrl(baseUrl, line); logDebug(`重写媒体片段: 原始='${line}', 解析后='${absoluteUrl}'`); output.push(rewriteUrlToProxy(absoluteUrl)); continue; } // 保留其他 M3U8 标签 output.push(line); } return output.join('\n'); } async function processM3u8Content(targetUrl, content, recursionDepth = 0) { // 判断是主列表还是媒体列表 if (content.includes('#EXT-X-STREAM-INF') || content.includes('#EXT-X-MEDIA:')) { logDebug(`检测到主播放列表: ${targetUrl} (深度: ${recursionDepth})`); return await processMasterPlaylist(targetUrl, content, recursionDepth); } logDebug(`检测到媒体播放列表: ${targetUrl} (深度: ${recursionDepth})`); return processMediaPlaylist(targetUrl, content); } async function processMasterPlaylist(url, content, recursionDepth) { // 检查递归深度 if (recursionDepth > MAX_RECURSION) { throw new Error(`处理主播放列表时,递归深度超过最大限制 (${MAX_RECURSION}): ${url}`); } const baseUrl = getBaseUrl(url); const lines = content.split('\n'); let highestBandwidth = -1; let bestVariantUrl = ''; // 查找最高带宽的流 for (let i = 0; i < lines.length; i++) { if (lines[i].startsWith('#EXT-X-STREAM-INF')) { const bandwidthMatch = lines[i].match(/BANDWIDTH=(\d+)/); const currentBandwidth = bandwidthMatch ? parseInt(bandwidthMatch[1], 10) : 0; let variantUriLine = ''; // 找到下一行的 URI for (let j = i + 1; j < lines.length; j++) { const line = lines[j].trim(); if (line && !line.startsWith('#')) { variantUriLine = line; i = j; break; } } if (variantUriLine && currentBandwidth >= highestBandwidth) { highestBandwidth = currentBandwidth; bestVariantUrl = resolveUrl(baseUrl, variantUriLine); } } } // 如果没有找到带宽信息,尝试查找第一个 .m3u8 链接 if (!bestVariantUrl) { logDebug(`主播放列表中未找到 BANDWIDTH 信息,尝试查找第一个 URI: ${url}`); for (let i = 0; i < lines.length; i++) { const line = lines[i].trim(); // 更可靠地匹配 .m3u8 链接 if (line && !line.startsWith('#') && line.match(/\.m3u8($|\?.*)/i)) { bestVariantUrl = resolveUrl(baseUrl, line); logDebug(`备选方案: 找到第一个子播放列表 URI: ${bestVariantUrl}`); break; } } } // 如果仍然没有找到子列表 URL if (!bestVariantUrl) { logDebug(`在主播放列表 ${url} 中未找到有效的子列表 URI,将其作为媒体列表处理。`); return processMediaPlaylist(url, content); } logDebug(`选择的子播放列表 (带宽: ${highestBandwidth}): ${bestVariantUrl}`); // 请求选定的子播放列表内容 (注意:这里传递 {} 作为请求头,不传递客户端的原始请求头) const { content: variantContent, contentType: variantContentType } = await fetchContentWithType(bestVariantUrl, {}); // 检查获取的内容是否是 M3U8 if (!isM3u8Content(variantContent, variantContentType)) { logDebug(`获取的子播放列表 ${bestVariantUrl} 不是 M3U8 (类型: ${variantContentType}),将其作为媒体列表处理。`); return processMediaPlaylist(bestVariantUrl, variantContent); } // 递归处理获取到的子 M3U8 内容 return await processM3u8Content(bestVariantUrl, variantContent, recursionDepth + 1); } // --- Vercel Handler 函数 --- export default async function handler(req, res) { // --- 记录请求开始 --- console.info('--- Vercel 代理请求开始 ---'); console.info('时间:', new Date().toISOString()); console.info('方法:', req.method); console.info('URL:', req.url); // 原始请求 URL (例如 /proxy/...) console.info('查询参数:', JSON.stringify(req.query)); // Vercel 解析的查询参数 // --- 提前设置 CORS 头 --- res.setHeader('Access-Control-Allow-Origin', '*'); res.setHeader('Access-Control-Allow-Methods', 'GET, HEAD, OPTIONS'); res.setHeader('Access-Control-Allow-Headers', '*'); // 允许所有请求头 // --- 处理 OPTIONS 预检请求 --- if (req.method === 'OPTIONS') { console.info("处理 OPTIONS 预检请求"); res.status(204).setHeader('Access-Control-Max-Age', '86400').end(); // 缓存预检结果 24 小时 return; } let targetUrl = null; // 初始化目标 URL try { // ---- 开始主处理逻辑的 try 块 ---- // --- 提取目标 URL (主要依赖 req.query["...path"]) --- // Vercel 将 :path* 捕获的内容(可能包含斜杠)放入 req.query["...path"] 数组 const pathData = req.query["...path"]; // 使用正确的键名 let encodedUrlPath = ''; if (pathData) { if (Array.isArray(pathData)) { encodedUrlPath = pathData.join('/'); // 重新组合 console.info(`从 req.query["...path"] (数组) 组合的编码路径: ${encodedUrlPath}`); } else if (typeof pathData === 'string') { encodedUrlPath = pathData; // 也处理 Vercel 可能只返回字符串的情况 console.info(`从 req.query["...path"] (字符串) 获取的编码路径: ${encodedUrlPath}`); } else { console.warn(`[代理警告] req.query["...path"] 类型未知: ${typeof pathData}`); } } else { console.warn(`[代理警告] req.query["...path"] 为空或未定义。`); // 备选:尝试从 req.url 提取(如果需要) if (req.url && req.url.startsWith('/proxy/')) { encodedUrlPath = req.url.substring('/proxy/'.length); console.info(`使用备选方法从 req.url 提取的编码路径: ${encodedUrlPath}`); } } // 如果仍然为空,则无法继续 if (!encodedUrlPath) { throw new Error("无法从请求中确定编码后的目标路径。"); } // 解析目标 URL targetUrl = getTargetUrlFromPath(encodedUrlPath); console.info(`解析出的目标 URL: ${targetUrl || 'null'}`); // 记录解析结果 // 检查目标 URL 是否有效 if (!targetUrl) { // 抛出包含更多上下文的错误 throw new Error(`无效的代理请求路径。无法从组合路径 "${encodedUrlPath}" 中提取有效的目标 URL。`); } console.info(`开始处理目标 URL 的代理请求: ${targetUrl}`); // --- 获取并处理目标内容 --- const { content, contentType, responseHeaders } = await fetchContentWithType(targetUrl, req.headers); // --- 如果是 M3U8,处理并返回 --- if (isM3u8Content(content, contentType)) { console.info(`正在处理 M3U8 内容: ${targetUrl}`); const processedM3u8 = await processM3u8Content(targetUrl, content); console.info(`成功处理 M3U8: ${targetUrl}`); // 发送处理后的 M3U8 响应 res.status(200) .setHeader('Content-Type', 'application/vnd.apple.mpegurl;charset=utf-8') .setHeader('Cache-Control', `public, max-age=${CACHE_TTL}`) // 移除可能导致问题的原始响应头 .removeHeader('content-encoding') // 很重要!node-fetch 已解压 .removeHeader('content-length') // 长度已改变 .send(processedM3u8); // 发送 M3U8 文本 } else { // --- 如果不是 M3U8,直接返回原始内容 --- console.info(`直接返回非 M3U8 内容: ${targetUrl}, 类型: ${contentType}`); // 设置原始响应头,但排除有问题的头和 CORS 头(已设置) responseHeaders.forEach((value, key) => { const lowerKey = key.toLowerCase(); if (!lowerKey.startsWith('access-control-') && lowerKey !== 'content-encoding' && // 很重要! lowerKey !== 'content-length') { // 很重要! res.setHeader(key, value); // 设置其他原始头 } }); // 设置我们自己的缓存策略 res.setHeader('Cache-Control', `public, max-age=${CACHE_TTL}`); // 发送原始(已解压)内容 res.status(200).send(content); } // ---- 结束主处理逻辑的 try 块 ---- } catch (error) { // ---- 捕获处理过程中的任何错误 ---- // **检查这个错误是否是 "Assignment to constant variable"** console.error(`[代理错误处理 V3] 捕获错误!目标: ${targetUrl || '解析失败'} | 错误类型: ${error.constructor.name} | 错误消息: ${error.message}`); console.error(`[代理错误堆栈 V3] ${error.stack}`); // 记录完整的错误堆栈信息 // 特别标记 "Assignment to constant variable" 错误 if (error instanceof TypeError && error.message.includes("Assignment to constant variable")) { console.error("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); console.error("捕获到 'Assignment to constant variable' 错误!"); console.error("请再次检查函数代码及所有辅助函数中,是否有 const 声明的变量被重新赋值。"); console.error("错误堆栈指向:", error.stack); console.error("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); } // 尝试从错误对象获取状态码,否则默认为 500 const statusCode = error.status || 500; // 确保在发送错误响应前没有发送过响应头 if (!res.headersSent) { res.setHeader('Content-Type', 'application/json'); // CORS 头应该已经在前面设置好了 res.status(statusCode).json({ success: false, error: `代理处理错误: ${error.message}`, // 返回错误消息给前端 targetUrl: targetUrl // 包含目标 URL 以便调试 }); } else { // 如果响应头已发送,无法再发送 JSON 错误 console.error("[代理错误处理 V3] 响应头已发送,无法发送 JSON 错误响应。"); // 尝试结束响应 if (!res.writableEnded) { res.end(); } } } finally { // 记录请求处理结束 console.info('--- Vercel 代理请求结束 ---'); } } // --- [确保所有辅助函数定义都在这里] --- // getTargetUrlFromPath, getBaseUrl, resolveUrl, rewriteUrlToProxy, getRandomUserAgent, // fetchContentWithType, isM3u8Content, processKeyLine, processMapLine, // processMediaPlaylist, processM3u8Content, processMasterPlaylist