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 { // If no detailed performance data, only return total time 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 handleUrlBlur = () => { if (!url.trim()) return; let input = url.trim(); try { // Try to parse as URL const parsedUrl = new URL(input.startsWith('http') ? input : `https://${input}`); const normalizedUrl = parsedUrl.toString(); if (normalizedUrl !== input) { setUrl(normalizedUrl); } } catch { // If parsing fails, add https:// prefix if (!input.startsWith("http://") && !input.startsWith("https://")) { setUrl(`https://${input}`); } } }; const startTest = async () => { if (!url.trim()) { toast.error("Please enter a URL"); return; } const targetUrl = url.trim(); setTesting(true); setResult(null); try { let testResult: SpeedTestResult = {}; switch (testType) { case "performance": testResult = await testPerformance(targetUrl); toast.success("Performance test completed"); break; case "download": testResult = await testDownloadSpeed(targetUrl); toast.success("Download speed test completed"); break; case "upload": testResult = await testUploadSpeed(targetUrl); toast.success("Upload speed test completed"); break; } setResult(testResult); } catch (error: unknown) { if (error instanceof Error) { toast.error(`Test failed: ${error.message}`); } else { toast.error("Test failed"); } } finally { setTesting(false); } }; const handleKeyPress = (e: React.KeyboardEvent) => { if (e.key === "Enter" && !testing) { startTest(); } }; return (
⚠️ CORS Restrictions

Due to browser CORS security policies, some websites cannot be tested directly.

Recommended websites to test:

  • Public APIs with CORS support
  • Your own websites (with configured CORS headers)
  • Using CORS proxy services
setUrl(e.target.value)} onBlur={handleUrlBlur} onKeyPress={handleKeyPress} disabled={testing} />
{result && (
Test Results:
{result.performance && (
Page Load Performance
{result.performance.dns > 0 && (
DNS Lookup:
{result.performance.dns.toFixed(2)} ms
)} {result.performance.tcp > 0 && (
TCP Connection:
{result.performance.tcp.toFixed(2)} ms
)} {result.performance.ssl > 0 && (
SSL Handshake:
{result.performance.ssl.toFixed(2)} ms
)} {result.performance.ttfb > 0 && (
Time to First Byte (TTFB):
{result.performance.ttfb.toFixed(2)} ms
)} {result.performance.download > 0 && (
Content Download:
{result.performance.download.toFixed(2)} ms
)}
Total Time:
{result.performance.total.toFixed(2)} ms
)} {result.downloadSpeed !== undefined && (
Download Speed
{result.downloadSpeed.toFixed(2)} Mbps
{(result.downloadSpeed / 8).toFixed(2)} MB/s
)} {result.uploadSpeed !== undefined && (
Upload Speed
{result.uploadSpeed.toFixed(2)} Mbps
{(result.uploadSpeed / 8).toFixed(2)} MB/s
)}
)} {testType === "download" && (
Note: Download speed test will download content from the target URL and calculate speed
)} {testType === "upload" && (
Note: Upload speed test will send 1MB of test data to the target URL
)}
); }; export default Tool;