ChatGPT降级检测

由于 ChatGPT 会对某些 ip 进行无提示的服务降级,此脚本用于检测你的 ip 在 ChatGPT 数据库中的风险等级。

  1. // ==UserScript==
  2. // @name ChatGPT降级检测
  3. // @namespace https://github.com/KoriIku/chatgpt-degrade-checker
  4. // @homepage https://github.com/KoriIku/chatgpt-degrade-checker
  5. // @author KoriIku
  6. // @icon data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCA2NCA2NCI+PHBhdGggZmlsbD0iIzJjM2U1MCIgZD0iTTMyIDJDMTUuNDMyIDIgMiAxNS40MzIgMiAzMnMxMy40MzIgMzAgMzAgMzAgMzAtMTMuNDMyIDMwLTMwUzQ4LjU2OCAyIDMyIDJ6bTAgNTRjLTEzLjIzMyAwLTI0LTEwLjc2Ny0yNC0yNFMxOC43NjcgOCAzMiA4czI0IDEwLjc2NyAyNCAyNFM0NS4yMzMgNTYgMzIgNTZ6Ii8+PHBhdGggZmlsbD0iIzNkYzJmZiIgZD0iTTMyIDEyYy0xMS4wNDYgMC0yMCA4Ljk1NC0yMCAyMHM4Ljk1NCAyMCAyMCAyMCAyMC04Ljk1NCAyMC0yMFM0My4wNDYgMTIgMzIgMTJ6bTAgMzZjLTguODM3IDAtMTYtNy4xNjMtMTYtMTZzNy4xNjMtMTYgMTYtMTYgMTYgNy4xNjMgMTYgMTZTNDAuODM3IDQ4IDMyIDQ4eiIvPjxwYXRoIGZpbGw9IiMwMGZmN2YiIGQ9Ik0zMiAyMGMtNi42MjcgMC0xMiA1LjM3My0xMiAxMnM1LjM3MyAxMiAxMiAxMiAxMi01LjM3MyAxMi0xMlMzOC42MjcgMjAgMzIgMjB6bTAgMjBjLTQuNDE4IDAtOC0zLjU4Mi04LThzMy41ODItOCA4LTggOCAzLjU4MiA4IDgtMy41ODIgOC04IDh6Ii8+PGNpcmNsZSBmaWxsPSIjZmZmIiBjeD0iMzIiIGN5PSIzMiIgcj0iNCIvPjwvc3ZnPg==
  7. // @version 1.91
  8. // @description 由于 ChatGPT 会对某些 ip 进行无提示的服务降级,此脚本用于检测你的 ip 在 ChatGPT 数据库中的风险等级。
  9. // @match *://chatgpt.com/*
  10. // @grant none
  11. // @license AGPLv3
  12. // ==/UserScript==
  13.  
  14. (function() {
  15. 'use strict';
  16.  
  17. // 创建显示框
  18. const displayBox = document.createElement('div');
  19. displayBox.style.position = 'fixed';
  20. displayBox.style.top = '50%';
  21. displayBox.style.right = '20px';
  22. displayBox.style.transform = 'translateY(-50%)';
  23. displayBox.style.width = '220px';
  24. displayBox.style.padding = '10px';
  25. displayBox.style.backgroundColor = 'rgba(0, 0, 0, 0.7)';
  26. displayBox.style.color = '#fff';
  27. displayBox.style.fontSize = '14px';
  28. displayBox.style.borderRadius = '8px';
  29. displayBox.style.boxShadow = '0 4px 8px rgba(0, 0, 0, 0.3)';
  30. displayBox.style.zIndex = '10000';
  31. displayBox.style.transition = 'all 0.3s ease';
  32. displayBox.style.display = 'none';
  33.  
  34. displayBox.innerHTML = `
  35. <div style="margin-bottom: 10px;">
  36. <strong>PoW 信息</strong>
  37. </div>
  38. <div id="content">
  39. PoW难度: <span id="difficulty">N/A</span><span id="difficulty-level" style="margin-left: 3px"></span>
  40. <span id="difficulty-tooltip" style="
  41. cursor: pointer;
  42. color: #fff;
  43. font-size: 12px;
  44. display: inline-block;
  45. width: 14px;
  46. height: 14px;
  47. line-height: 14px;
  48. text-align: center;
  49. border-radius: 50%;
  50. border: 1px solid #fff;
  51. margin-left: 3px;
  52. ">?</span><br>
  53. IP质量: <span id="ip-quality">N/A</span><br>
  54. <span id="persona-container" style="display: none">用户类型: <span id="persona">N/A</span></span>
  55. </div>
  56. <div style="
  57. margin-top: 12px;
  58. padding-top: 8px;
  59. border-top: 0.5px solid rgba(255, 255, 255, 0.15);
  60. font-size: 10px;
  61. color: rgba(255, 255, 255, 0.5);
  62. text-align: center;
  63. letter-spacing: 0.3px;
  64. ">
  65. ChatGPT Degrade Checker
  66. </div>`;
  67. document.body.appendChild(displayBox);
  68.  
  69. // 创建收缩状态的指示器
  70. const collapsedIndicator = document.createElement('div');
  71. collapsedIndicator.style.position = 'fixed';
  72. collapsedIndicator.style.top = '50%';
  73. collapsedIndicator.style.right = '20px';
  74. collapsedIndicator.style.transform = 'translateY(-50%)';
  75. collapsedIndicator.style.width = '32px';
  76. collapsedIndicator.style.height = '32px';
  77. collapsedIndicator.style.backgroundColor = 'transparent';
  78. collapsedIndicator.style.borderRadius = '50%';
  79. collapsedIndicator.style.cursor = 'pointer';
  80. collapsedIndicator.style.zIndex = '10000';
  81. collapsedIndicator.style.padding = '4px';
  82. collapsedIndicator.style.display = 'flex';
  83. collapsedIndicator.style.alignItems = 'center';
  84. collapsedIndicator.style.justifyContent = 'center';
  85. collapsedIndicator.style.transition = 'all 0.3s ease';
  86.  
  87. // 使用SVG作为指示器
  88. collapsedIndicator.innerHTML = `
  89. <svg id="status-icon" width="32" height="32" viewBox="0 0 64 64" style="transition: all 0.3s ease;">
  90. <defs>
  91. <linearGradient id="gradient" x1="0%" y1="0%" x2="100%" y2="100%">
  92. <stop offset="0%" style="stop-color:#3498db;stop-opacity:1" />
  93. <stop offset="100%" style="stop-color:#2ecc71;stop-opacity:1" />
  94. </linearGradient>
  95. <filter id="glow">
  96. <feGaussianBlur stdDeviation="2" result="coloredBlur"/>
  97. <feMerge>
  98. <feMergeNode in="coloredBlur"/>
  99. <feMergeNode in="SourceGraphic"/>
  100. </feMerge>
  101. </filter>
  102. </defs>
  103. <g id="icon-group" filter="url(#glow)">
  104. <circle cx="32" cy="32" r="28" fill="url(#gradient)" stroke="#fff" stroke-width="2"/>
  105. <circle cx="32" cy="32" r="20" fill="none" stroke="#fff" stroke-width="2" stroke-dasharray="100">
  106. <animateTransform
  107. attributeName="transform"
  108. attributeType="XML"
  109. type="rotate"
  110. from="0 32 32"
  111. to="360 32 32"
  112. dur="8s"
  113. repeatCount="indefinite"/>
  114. </circle>
  115. <circle cx="32" cy="32" r="12" fill="none" stroke="#fff" stroke-width="2">
  116. <animate
  117. attributeName="r"
  118. values="12;14;12"
  119. dur="2s"
  120. repeatCount="indefinite"/>
  121. </circle>
  122. <circle id="center-dot" cx="32" cy="32" r="4" fill="#fff">
  123. <animate
  124. attributeName="r"
  125. values="4;6;4"
  126. dur="2s"
  127. repeatCount="indefinite"/>
  128. </circle>
  129. </g>
  130. </svg>`;
  131. document.body.appendChild(collapsedIndicator);
  132.  
  133. // 鼠标悬停事件
  134. collapsedIndicator.addEventListener('mouseenter', function() {
  135. displayBox.style.display = 'block';
  136. collapsedIndicator.style.opacity = '0';
  137. });
  138.  
  139. displayBox.addEventListener('mouseleave', function() {
  140. displayBox.style.display = 'none';
  141. collapsedIndicator.style.opacity = '1';
  142. });
  143.  
  144. // 创建提示框
  145. const tooltip = document.createElement('div');
  146. tooltip.id = 'tooltip';
  147. tooltip.innerText = '这个值越小,代表PoW难度越高,ChatGPT认为你的IP风险越高。';
  148. tooltip.style.position = 'fixed';
  149. tooltip.style.backgroundColor = 'rgba(0, 0, 0, 0.8)';
  150. tooltip.style.color = '#fff';
  151. tooltip.style.padding = '8px 12px';
  152. tooltip.style.borderRadius = '5px';
  153. tooltip.style.fontSize = '12px';
  154. tooltip.style.visibility = 'hidden';
  155. tooltip.style.zIndex = '10001';
  156. tooltip.style.width = '240px';
  157. tooltip.style.lineHeight = '1.4';
  158. tooltip.style.pointerEvents = 'none';
  159. document.body.appendChild(tooltip);
  160.  
  161. // 显示提示
  162. document.getElementById('difficulty-tooltip').addEventListener('mouseenter', function(event) {
  163. tooltip.style.visibility = 'visible';
  164.  
  165. const tooltipWidth = 240;
  166. const windowWidth = window.innerWidth;
  167. const mouseX = event.clientX;
  168. const mouseY = event.clientY;
  169.  
  170. let leftPosition = mouseX - tooltipWidth - 10;
  171. if (leftPosition < 10) {
  172. leftPosition = mouseX + 20;
  173. }
  174.  
  175. let topPosition = mouseY - 40;
  176.  
  177. tooltip.style.left = `${leftPosition}px`;
  178. tooltip.style.top = `${topPosition}px`;
  179. });
  180.  
  181. // 隐藏提示
  182. document.getElementById('difficulty-tooltip').addEventListener('mouseleave', function() {
  183. tooltip.style.visibility = 'hidden';
  184. });
  185.  
  186. // 更新difficulty指示器
  187. function updateDifficultyIndicator(difficulty) {
  188. const difficultyLevel = document.getElementById('difficulty-level');
  189. const ipQuality = document.getElementById('ip-quality');
  190.  
  191. if (difficulty === 'N/A') {
  192. setIconColors('#888', '#666');
  193. difficultyLevel.innerText = '';
  194. ipQuality.innerHTML = 'N/A';
  195. return;
  196. }
  197.  
  198. const cleanDifficulty = difficulty.replace('0x', '').replace(/^0+/, '');
  199. const hexLength = cleanDifficulty.length;
  200.  
  201. let color, secondaryColor, textColor, level, qualityText;
  202.  
  203. if (hexLength <= 2) {
  204. color = '#F44336';
  205. secondaryColor = '#d32f2f';
  206. textColor = '#ff6b6b';
  207. level = '(困难)';
  208. qualityText = '高风险';
  209. } else if (hexLength === 3) {
  210. color = '#FFC107';
  211. secondaryColor = '#ffa000';
  212. textColor = '#ffd700';
  213. level = '(中等)';
  214. qualityText = '中等';
  215. } else if (hexLength === 4) {
  216. color = '#8BC34A';
  217. secondaryColor = '#689f38';
  218. textColor = '#9acd32';
  219. level = '(简单)';
  220. qualityText = '良好';
  221. } else {
  222. color = '#4CAF50';
  223. secondaryColor = '#388e3c';
  224. textColor = '#98fb98';
  225. level = '(极易)';
  226. qualityText = '优秀';
  227. }
  228.  
  229. setIconColors(color, secondaryColor);
  230. difficultyLevel.innerHTML = `<span style="color: ${textColor}">${level}</span>`;
  231. ipQuality.innerHTML = `<span style="color: ${textColor}">${qualityText}</span>`;
  232. }
  233.  
  234. function setIconColors(primaryColor, secondaryColor) {
  235. const gradient = document.querySelector('#gradient');
  236. gradient.innerHTML = `
  237. <stop offset="0%" style="stop-color:${primaryColor};stop-opacity:1" />
  238. <stop offset="100%" style="stop-color:${secondaryColor};stop-opacity:1" />
  239. `;
  240. }
  241.  
  242. // 拦截 fetch 请求
  243. const originalFetch = window.fetch;
  244. window.fetch = async function(resource, options) {
  245. const response = await originalFetch(resource, options);
  246.  
  247. if ((resource.includes('/backend-api/sentinel/chat-requirements')||resource.includes('backend-anon/sentinel/chat-requirements')) && options.method === 'POST') {
  248. const clonedResponse = response.clone();
  249. clonedResponse.json().then(data => {
  250. const difficulty = data.proofofwork ? data.proofofwork.difficulty : 'N/A';
  251. const persona = data.persona || 'N/A';
  252. document.getElementById('difficulty').innerText = difficulty;
  253.  
  254. const personaContainer = document.getElementById('persona-container');
  255. if (persona && !persona.toLowerCase().includes('free')) {
  256. personaContainer.style.display = 'block';
  257. document.getElementById('persona').innerText = persona;
  258. } else {
  259. personaContainer.style.display = 'none';
  260. }
  261.  
  262. updateDifficultyIndicator(difficulty);
  263. }).catch(e => console.error('解析响应时出错:', e));
  264. }
  265. return response;
  266. };
  267. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址