// ==UserScript== // @name ChatGPT Degraded // @name:zh-CN ChatGPT 服务降级监控 // @namespace https://github.com/lroolle/chatgpt-degraded // @version 0.2.0 // @description Monitor ChatGPT service level, IP quality and PoW difficulty // @description:zh-CN 监控 ChatGPT 服务状态、IP 质量和 PoW 难度 // @author lroolle // @license AGPL-3.0 // @match *://chatgpt.com/* // @icon data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSI2NCIgaGVpZ2h0PSI2NCIgdmlld0JveD0iMCAwIDY0IDY0Ij4KICA8ZGVmcz4KICAgIDxsaW5lYXJHcmFkaWVudCBpZD0iZ3JhZGllbnQiIHgxPSIwJSIgeTE9IjAlIiB4Mj0iMTAwJSIgeTI9IjEwMCUiPgogICAgICA8c3RvcCBvZmZzZXQ9IjAlIiBzdHlsZT0ic3RvcC1jb2xvcjojMmE5ZDhmO3N0b3Atb3BhY2l0eToxIi8+CiAgICAgIDxzdG9wIG9mZnNldD0iMTAwJSIgc3R5bGU9InN0b3AtY29sb3I6IzJhOWQ4ZjtzdG9wLW9wYWNpdHk6MC44Ii8+CiAgICA8L2xpbmVhckdyYWRpZW50PgogIDwvZGVmcz4KICA8Zz4KICAgIDxjaXJjbGUgY3g9IjMyIiBjeT0iMzIiIHI9IjI4IiBmaWxsPSJ1cmwoI2dyYWRpZW50KSIgc3Ryb2tlPSIjZmZmIiBzdHJva2Utd2lkdGg9IjEiLz4KPCEtLU91dGVyIGNpcmNsZSBtb2RpZmllZCB0byBsb29rIGxpa2UgIkMiLS0+CiAgICA8Y2lyY2xlIGN4PSIzMiIgY3k9IjMyIiByPSIyMCIgZmlsbD0ibm9uZSIgc3Ryb2tlPSIjZmZmIiBzdHJva2Utd2lkdGg9IjEiIHN0cm9rZS1kYXNoYXJyYXk9IjEyNSA1NSIgc3Ryb2tlLWRhc2hvZmZzZXQ9IjIwIi8+CiAgICA8Y2lyY2xlIGN4PSIzMiIgY3k9IjMyIiByPSIxMiIgZmlsbD0ibm9uZSIgc3Ryb2tlPSIjZmZmIiBzdHJva2Utd2lkdGg9IjEiLz4KICAgIDxjaXJjbGUgY3g9IjMyIiBjeT0iMzIiIHI9IjQiIGZpbGw9IiNmZmYiLz4KICA8L2c+Cjwvc3ZnPg== // @grant none // @homepageURL https://github.com/lroolle/chatgpt-degraded // @supportURL https://github.com/lroolle/chatgpt-degraded/issues // @downloadURL none // ==/UserScript== // Reference: https://github.com/KoriIku/chatgpt-degrade-checker (by KoriIku) (function () { "use strict"; const displayBox = document.createElement("div"); displayBox.style.position = "fixed"; displayBox.style.bottom = "10px"; displayBox.style.right = "80px"; displayBox.style.transform = "none"; displayBox.style.width = "320px"; displayBox.style.padding = "24px"; displayBox.style.backgroundColor = "var(--surface-primary, rgb(255, 255, 255))"; displayBox.style.color = "var(--text-primary, #374151)"; displayBox.style.fontSize = "14px"; displayBox.style.borderRadius = "16px"; displayBox.style.boxShadow = "0 4px 24px rgba(0, 0, 0, 0.08)"; displayBox.style.zIndex = "10000"; displayBox.style.transition = "opacity 0.15s ease, transform 0.15s ease"; displayBox.style.display = "none"; displayBox.style.opacity = "0"; displayBox.style.transform = "translateX(10px)"; displayBox.style.border = "1px solid var(--border-light, rgba(0, 0, 0, 0.05))"; displayBox.innerHTML = `
`; document.body.appendChild(displayBox); const collapsedIndicator = document.createElement("div"); collapsedIndicator.style.position = "fixed"; collapsedIndicator.style.bottom = "10px"; collapsedIndicator.style.right = "40px"; collapsedIndicator.style.transform = "none"; collapsedIndicator.style.width = "24px"; collapsedIndicator.style.height = "24px"; collapsedIndicator.style.backgroundColor = "transparent"; collapsedIndicator.style.border = "1px solid var(--token-border-light, rgba(0, 0, 0, 0.1))"; collapsedIndicator.style.borderRadius = "50%"; collapsedIndicator.style.cursor = "pointer"; collapsedIndicator.style.zIndex = "10000"; collapsedIndicator.style.display = "flex"; collapsedIndicator.style.alignItems = "center"; collapsedIndicator.style.justifyContent = "center"; collapsedIndicator.style.transition = "all 0.3s ease"; collapsedIndicator.innerHTML = ` `; document.body.appendChild(collapsedIndicator); collapsedIndicator.addEventListener("mouseenter", () => { displayBox.style.display = "block"; requestAnimationFrame(() => { displayBox.style.opacity = "1"; displayBox.style.transform = "translateX(0)"; }); }); displayBox.addEventListener("mouseleave", () => { displayBox.style.opacity = "0"; displayBox.style.transform = "translateX(10px)"; setTimeout(() => { displayBox.style.display = "none"; }, 150); }); // Add IP masking function function maskIP(ip) { if (!ip || ip === "Unknown") return ip; // For IPv4: 192.168.1.1 -> 192.*.*.1 if (ip.includes(".")) { const parts = ip.split("."); return `${parts[0]}.*.*.${parts[3]}`; } // For IPv6: 2a09:bac5:6248:183c::26a:1a -> 2a09:*:*:183c if (ip.includes(":")) { const parts = ip.split(":"); return `${parts[0]}:*:*:${parts[3]}`; } return ip; } async function fetchIPInfo() { try { const response = await fetch("https://chatgpt.com/cdn-cgi/trace"); const text = await response.text(); const data = text.split("\n").reduce((obj, line) => { const [key, value] = line.split("="); if (key && value) obj[key.trim()] = value.trim(); return obj; }, {}); const ipElement = document.getElementById("ip-address"); const warpBadge = document.getElementById("warp-badge"); const maskedIP = maskIP(data.ip); const fullIP = data.ip || "Unknown"; const warpStatus = data.warp || "off"; ipElement.innerText = maskedIP; // Update WARP badge visibility and content if (warpStatus === "on") { warpBadge.style.display = "inline-flex"; warpBadge.innerText = "warp"; warpBadge.title = "Protected by Cloudflare WARP"; } else if (warpStatus === "plus") { warpBadge.style.display = "inline-flex"; warpBadge.innerText = "warp+"; warpBadge.title = "Protected by Cloudflare WARP+"; } else { warpBadge.style.display = "none"; } // Add tooltips ipElement.title = "Click to copy full IP address"; // Add click to copy functionality const copyHandler = async () => { try { await navigator.clipboard.writeText(fullIP); const originalText = ipElement.innerText; ipElement.innerText = "Copied!"; setTimeout(() => { ipElement.innerText = originalText; }, 1000); } catch (err) { console.error("Failed to copy:", err); } }; ipElement.removeEventListener("click", copyHandler); ipElement.addEventListener("click", copyHandler); } catch (error) { document.getElementById("ip-address").innerText = "Failed to fetch"; document.getElementById("warp-status").innerText = "unknown"; } } function getRiskColorAndLevel(difficulty) { if (!difficulty || difficulty === "N/A") { return { color: "#e63946", level: "Unknown", ipQuality: "Unknown", percentage: 0, }; } const cleanDifficulty = difficulty.replace("0x", "").replace(/^0+/, ""); const hexLength = cleanDifficulty.length; if (hexLength <= 2) { return { color: "#e63946", // Red level: "Hard", ipQuality: "High Risk", percentage: 100, }; } else if (hexLength === 3) { return { color: "#FAB12F", // Orange level: "Medium", ipQuality: "Medium", percentage: 75, }; } else if (hexLength === 4) { return { color: "#859F3D", // Light Green level: "Easy", ipQuality: "Good", percentage: 50, }; } else if (hexLength === 5) { return { color: "#2a9d8f", // Teal level: "Very Easy", ipQuality: "Excellent", percentage: 25, }; } else { return { color: "#4CAF50", // Green level: "Very Easy", ipQuality: "Excellent", percentage: 0, }; } } function updateProgressBars(difficulty) { const powBar = document.getElementById("pow-bar"); const powLevel = document.getElementById("pow-level"); const ipQualityBar = document.getElementById("ip-quality-bar"); const ipQuality = document.getElementById("ip-quality"); const difficultyElement = document.getElementById("difficulty"); const ipAddressElement = document.getElementById("ip-address"); const { color, level, ipQuality: quality, percentage, } = getRiskColorAndLevel(difficulty); // Create gradient based on risk level const gradient = `linear-gradient(90deg, ${color} ${percentage}%, rgba(255, 255, 255, 0.1) ${percentage}% )`; // Update progress bars with title for explanation setProgressBar( powBar, powLevel, percentage, level, gradient, "PoW Difficulty: Required computational work before sending messages. Lower (green) means faster responses.", ); setProgressBar( ipQualityBar, ipQuality, percentage, quality, gradient, "IP Quality: Indicates the risk associated with your IP as assessed by ChatGPT. Lower(green) is better.", ); // Sync colors with text difficultyElement.style.color = color; ipAddressElement.style.color = color; ipQuality.style.color = color; powLevel.style.color = color; // Update the icon color const gradientStops = collapsedIndicator.querySelector("#gradient"); if (gradientStops) { gradientStops.innerHTML = `