import { useState, type FC } from "react"; import { Button } from "@/components/ui/button"; import { Input } from "@/components/ui/input"; import { toast } from "sonner"; import { Loader2 } from "lucide-react"; interface PerformanceMetrics { dns: number; tcp: number; ssl: number; ttfb: number; download: number; total: number; } interface SpeedTestResult { downloadSpeed?: number; uploadSpeed?: number; performance?: PerformanceMetrics; } const Tool: FC = () => { const [url, setUrl] = useState(""); const [testType, setTestType] = useState<"performance" | "download" | "upload">( "performance" ); const [testing, setTesting] = useState(false); const [result, setResult] = useState(null); const testPerformance = async (targetUrl: string) => { // 清除之前的性能数据 performance.clearResourceTimings(); const startTime = performance.now(); try { const response = await fetch(targetUrl, { cache: "no-cache", }); if (!response.ok) { throw new Error(`HTTP ${response.status}: ${response.statusText}`); } // 等待内容加载完成 await response.blob(); const endTime = performance.now(); // 获取性能数据 const perfEntries = performance.getEntriesByType( "resource" ) as PerformanceResourceTiming[]; const entry = perfEntries.find((e) => e.name === targetUrl); if (entry) { const metrics: PerformanceMetrics = { dns: entry.domainLookupEnd - entry.domainLookupStart, tcp: entry.connectEnd - entry.connectStart, ssl: entry.secureConnectionStart > 0 ? entry.connectEnd - entry.secureConnectionStart : 0, ttfb: entry.responseStart - entry.requestStart, download: entry.responseEnd - entry.responseStart, total: entry.responseEnd - entry.startTime, }; return { performance: metrics }; } else { // 如果没有详细的性能数据,只返回总时间 return { performance: { dns: 0, tcp: 0, ssl: 0, ttfb: 0, download: 0, total: endTime - startTime, }, }; } } catch (error) { throw error; } }; const testDownloadSpeed = async (targetUrl: string) => { const startTime = performance.now(); try { const response = await fetch(targetUrl, { cache: "no-cache", }); if (!response.ok) { throw new Error(`HTTP ${response.status}: ${response.statusText}`); } const blob = await response.blob(); const endTime = performance.now(); const fileSizeBytes = blob.size; const durationSeconds = (endTime - startTime) / 1000; const speedMbps = (fileSizeBytes * 8) / (durationSeconds * 1000000); return { downloadSpeed: speedMbps }; } catch (error) { throw error; } }; const testUploadSpeed = async (targetUrl: string) => { // 生成 1MB 的测试数据 const testData = new Uint8Array(1024 * 1024); for (let i = 0; i < testData.length; i++) { testData[i] = Math.floor(Math.random() * 256); } const startTime = performance.now(); try { const response = await fetch(targetUrl, { method: "POST", body: testData, cache: "no-cache", }); const endTime = performance.now(); if (!response.ok) { throw new Error(`HTTP ${response.status}: ${response.statusText}`); } const fileSizeBytes = testData.length; const durationSeconds = (endTime - startTime) / 1000; const speedMbps = (fileSizeBytes * 8) / (durationSeconds * 1000000); return { uploadSpeed: speedMbps }; } catch (error) { throw error; } }; const startTest = async () => { if (!url.trim()) { toast.error("请输入 URL"); return; } let targetUrl = url.trim(); // 如果没有协议前缀,默认使用 https:// if (!targetUrl.startsWith("http://") && !targetUrl.startsWith("https://")) { targetUrl = `https://${targetUrl}`; } setTesting(true); setResult(null); try { let testResult: SpeedTestResult = {}; switch (testType) { case "performance": testResult = await testPerformance(targetUrl); toast.success("性能测试完成"); break; case "download": testResult = await testDownloadSpeed(targetUrl); toast.success("下载速度测试完成"); break; case "upload": testResult = await testUploadSpeed(targetUrl); toast.success("上传速度测试完成"); break; } setResult(testResult); } catch (error: unknown) { if (error instanceof Error) { toast.error(`测试失败: ${error.message}`); } else { toast.error("测试失败"); } } finally { setTesting(false); } }; const handleKeyPress = (e: React.KeyboardEvent) => { if (e.key === "Enter" && !testing) { startTest(); } }; return (
⚠️ CORS 限制说明

由于浏览器的 CORS 安全策略,部分网站可能无法直接测试。

建议测试以下类型的网站:

  • 支持 CORS 的公共 API
  • 您自己控制的网站(可配置 CORS 头)
  • 使用 CORS 代理服务
setUrl(e.target.value)} onKeyPress={handleKeyPress} disabled={testing} />
{result && (
测试结果:
{result.performance && (
页面加载性能
{result.performance.dns > 0 && (
DNS 查询:
{result.performance.dns.toFixed(2)} ms
)} {result.performance.tcp > 0 && (
TCP 连接:
{result.performance.tcp.toFixed(2)} ms
)} {result.performance.ssl > 0 && (
SSL 握手:
{result.performance.ssl.toFixed(2)} ms
)} {result.performance.ttfb > 0 && (
首字节时间 (TTFB):
{result.performance.ttfb.toFixed(2)} ms
)} {result.performance.download > 0 && (
内容下载:
{result.performance.download.toFixed(2)} ms
)}
总时间:
{result.performance.total.toFixed(2)} ms
)} {result.downloadSpeed !== undefined && (
下载速度
{result.downloadSpeed.toFixed(2)} Mbps
{(result.downloadSpeed / 8).toFixed(2)} MB/s
)} {result.uploadSpeed !== undefined && (
上传速度
{result.uploadSpeed.toFixed(2)} Mbps
{(result.uploadSpeed / 8).toFixed(2)} MB/s
)}
)} {testType === "download" && (
提示: 下载速度测试会下载目标 URL 的内容并计算速度
)} {testType === "upload" && (
提示: 上传速度测试会向目标 URL 发送 1MB 测试数据
)}
); }; export default Tool;