Stop Nefarious Redirects

Block unauthorized redirects and prevent history manipulation

当前为 2024-05-28 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Stop Nefarious Redirects
  3. // @namespace http://tampermonkey.net/
  4. // @version 4.1
  5. // @description Block unauthorized redirects and prevent history manipulation
  6. // @match http://*/*
  7. // @match https://*/*
  8. // @grant GM_setValue
  9. // @grant GM_getValue
  10. // @grant GM_xmlhttpRequest
  11. // @license MIT
  12. // @run-at document-start
  13. // @icon 
  14. // ==/UserScript==
  15.  
  16. const manualBlacklist = new Set([
  17. 'getrunkhomuto.info'
  18. ]);
  19.  
  20. // List of allowed popups domains
  21. const allowedPopups = [
  22. '500px.com', 'accuweather.com', 'adobe.com', 'adulttime.com', 'alibaba.com', 'amazon.com', 'amazonaws.com',
  23. 'amd.com', 'americanexpress.com', 'anaconda.com', 'angular.io', 'ap.org', 'apache.org', 'apnews.com',
  24. 'apple.com', 'arstechnica.com', 'artstation.com', 'asana.com', 'asus.com', 'atlassian.com', 'autodesk.com',
  25. 'axios.com', 'battle.net', 'bbc.co.uk', 'bbc.com', 'behance.net', 'bestbuy.com', 'bing.com', 'bitbucket.org',
  26. 'blogger.com', 'bloomberg.com', 'bmw.com', 'boeing.com', 'booking.co.uk', 'booking.com', 'bootstrapcdn.com',
  27. 'breitbart.com', 'buzzfeed.com', 'canva.com', 'capitalone.com', 'cbsnews.com', 'character.ai', 'chase.com',
  28. 'chaturbate.com', 'cisco.com', 'citi.com', 'cnbc.com', 'cnet.com', 'cnn.com', 'codecademy.com', 'constantcontact.com',
  29. 'coursera.org', 'craigslist.org', 'dailymail.co.uk', 'dell.com', 'deviantart.com', 'discord.com', 'disney.com',
  30. 'django.com', 'docker.com', 'docusign.com', 'dribbble.com', 'dropbox.com', 'duckduckgo.com', 'duolingo.com',
  31. 'duosecurity.com', 'ebay.com', 'economist.com', 'edx.org', 'elsevier.com', 'engadget.com', 'epicgames.com',
  32. 'eporner.com', 'espn.com', 'etsy.com', 'eurogamer.net', 'expedia.com', 'facebook.com', 'fandom.com', 'fedex.com',
  33. 'figma.com', 'finance.yahoo.com', 'flickr.com', 'flipkart.com', 'forbes.com', 'foxnews.com', 'framer.com',
  34. 'freecodecamp.org', 'gamespot.com', 'gartner.com', 'gettyimages.com', 'git-scm.com', 'github.com', 'gizmodo.com',
  35. 'go.com', 'godaddy.com', 'gog.com', 'goldmansachs.com', 'google.com', 'healthline.com', 'hilton.com', 'homedepot.com',
  36. 'hp.com', 'hubspot.com', 'huffpost.com', 'hulu.com', 'humblebundle.com', 'ibm.com', 'ieee.org', 'ifixit.com',
  37. 'ign.com', 'ikea.com', 'imdb.com', 'imgur.com', 'indeed.com', 'instagram.com', 'instructure.com', 'intel.com',
  38. 'intuit.com', 'invisionapp.com', 'itch.io', 'java.com', 'jetbrains.com', 'joomla.org', 'jquery.com', 'khanacademy.org',
  39. 'kotaku.com', 'kotlinlang.org', 'laravel.com', 'lenovo.com', 'lg.com', 'lifehacker.com', 'linkedin.com', 'live.com',
  40. 'lowes.com', 'lynda.com', 'macys.com', 'mailchimp.com', 'marriott.com', 'mashable.com', 'masterclass.com',
  41. 'mcdonalds.com', 'medium.com', 'mercedes-benz.com', 'microsoft.com', 'microsoftonline.commit.edu', 'mongodb.com',
  42. 'moodle.org', 'mozilla.org', 'msn.com', 'msnbc.com', 'nasa.gov', 'nationalgeographic.com', 'nbc.com', 'nbcnews.com',
  43. 'netflix.com', 'new.reddit.com', 'nextdoor.com', 'nih.gov', 'npr.org', 'nvidia.comnypost.com', 'nytimes.com',
  44. 'office.com', 'okta.com', 'onlyfans.com', 'openai.com', 'oracle.com', 'oreilly.com', 'origin.com', 'outlook.com',
  45. 'overstock.com', 'patreon.com', 'paypal.com', 'pcgamer.com', 'pexels.com', 'php.net', 'pinterest.com', 'pixabay.com',
  46. 'pluralsight.com', 'polygon.com', 'pornhub.com', 'python.org', 'quizlet.com', 'quora.com', 'reactjs.org', 'realtor.com',
  47. 'reddit.com', 'redhat.com', 'roblox.com', 'rubyonrails.org', 'salesforce.com', 'samsung.co.kr', 'samsung.com',
  48. 'sap.com', 'sciencedirect.com', 'scopus.com', 'sears.com', 'sharepoint.com', 'shutterstock.com', 'siemens.com',
  49. 'sketch.com', 'skillshare.com', 'skype.com', 'slack.com', 'sony.com', 'soundcloud.com', 'spotify.com', 'spring.io',
  50. 'stackoverflow.com', 'steamcommunity.com', 'steampowered.com', 'surveymonkey.com', 'symantec.com', 'target.com',
  51. 'techcrunch.com', 'temu.com', 'tesla.com', 'texasinstruments.com', 'theguardian.com', 'thenextweb.com', 'theverge.com',
  52. 'tiktok.com', 'time.com', 'toyota.com', 'trello.com', 'trip.com', 'tripadvisor.com', 'tumblr.com', 'twitch.tv',
  53. 'twitter.com', 'uber.com', 'ucla.edu', 'ucsf.edu', 'udemy.com', 'unity.com', 'unsplash.com', 'ups.com', 'usatoday.com',
  54. 'usnews.com', 'usps.com', 'verizon.com', 'Vice.com', 'vice.com', 'vimeo.com', 'vk.com', 'vmware.com', 'volkswagen.com',
  55. 'vox.com', 'walmart.com', 'washingtonpost.com', 'weather.com', 'weather.gov', 'webmd.com', 'whatsapp.com', 'wikimedia.org',
  56. 'wikipedia.org', 'wired.com', 'wordpress.com', 'wsj.com', 'wunderground.com', 'x.com', 'xerox.com', 'xfinity.com',
  57. 'yahoo.com', 'yelp.com', 'youtube.com', 'zapier.com', 'zendesk.com', 'zeplin.iozillow.com', 'zoom.us'
  58. ];
  59.  
  60. (function() {
  61. 'use strict';
  62.  
  63. console.log('Script initialization started.');
  64.  
  65. /**
  66. * Get the current automated blacklist
  67. * @returns {Set} The automated blacklist
  68. */
  69. function getAutomatedBlacklist() {
  70. return new Set(GM_getValue('blacklist', []));
  71. }
  72.  
  73. /**
  74. * Add a URL to the automated blacklist
  75. * @param {string} url - The URL to add to the blacklist
  76. */
  77. function addToAutomatedBlacklist(url) {
  78. const encodedUrl = encodeURIComponent(url);
  79. const blacklist = getAutomatedBlacklist();
  80. if (!blacklist.has(encodedUrl)) {
  81. blacklist.add(encodedUrl);
  82. GM_setValue('blacklist', Array.from(blacklist));
  83. console.log('Added to automated blacklist:', url);
  84. }
  85. }
  86.  
  87. /**
  88. * Check if navigation to a URL is allowed
  89. * @param {string} url - The URL to check
  90. * @returns {boolean} True if navigation is allowed, false otherwise
  91. */
  92. function isNavigationAllowed(url) {
  93. if (!isUrlBlocked(url)) {
  94. console.log('Navigation allowed to:', url);
  95. lastKnownGoodUrl = url;
  96. return true;
  97. } else {
  98. console.error('Blocked navigation to:', url);
  99. addToAutomatedBlacklist(url);
  100. if (lastKnownGoodUrl) {
  101. window.location.replace(lastKnownGoodUrl);
  102. }
  103. return false;
  104. }
  105. }
  106.  
  107. const originalAssign = window.location.assign.bind(window.location);
  108. const originalOpen = window.open;
  109.  
  110. console.log('Original window.location.assign and window.open saved.');
  111.  
  112. window.location.assign = function(url) {
  113. console.log('Redirect attempt detected:', url);
  114. if (!allowedPopups.some(domain => url.includes(domain)) && !isNavigationAllowed(url)) {
  115. console.log('Redirect to undesired domain blocked:', url);
  116. return;
  117. }
  118. console.log('Redirect allowed to:', url);
  119. return originalAssign(url);
  120. };
  121.  
  122. console.log('window.location.assign overridden with custom logic.');
  123.  
  124. window.open = function(url, name, features) {
  125. console.log('Popup attempt detected:', url);
  126. if (allowedPopups.some(domain => url.includes(domain)) || isNavigationAllowed(url)) {
  127. console.log('Popup allowed for:', url);
  128. return originalOpen(url, name, features);
  129. }
  130. console.log('Blocked a popup from:', url);
  131. return null;
  132. };
  133.  
  134. console.log('window.open overridden with custom logic.');
  135.  
  136. let lastKnownGoodUrl = window.location.href;
  137.  
  138. const locationProxy = new Proxy(window.location, {
  139. set(target, prop, value) {
  140. if (prop === 'href' || prop === 'assign' || prop === 'replace') {
  141. if (!isNavigationAllowed(value)) {
  142. return false;
  143. }
  144. }
  145. return Reflect.set(target, prop, value);
  146. },
  147. get(target, prop) {
  148. if (prop === 'assign' || prop === 'replace') {
  149. return function(url) {
  150. if (isNavigationAllowed(url)) {
  151. target[prop](url); // Properly assign the URL
  152. }
  153. };
  154. }
  155. return Reflect.get(target, prop);
  156. }
  157. });
  158.  
  159. Object.defineProperty(window, 'location', {
  160. configurable: true,
  161. enumerable: true,
  162. get() {
  163. return locationProxy;
  164. }
  165. });
  166.  
  167. window.addEventListener('popstate', function(event) {
  168. if (!isNavigationAllowed(window.location.href)) {
  169. console.error('Blocked navigation to:', window.location.href);
  170. history.pushState(null, "", lastKnownGoodUrl);
  171. window.location.replace(lastKnownGoodUrl);
  172. event.preventDefault();
  173. }
  174. });
  175.  
  176. /**
  177. * Handle history manipulation
  178. * @param {Function} originalMethod - The original history method
  179. * @param {*} data - The state data
  180. * @param {string} title - The page title
  181. * @param {string|null} url - The URL
  182. * @returns {*} The result of calling the original method
  183. */
  184. function handleHistoryManipulation(originalMethod, data, title, url) {
  185. if (!isUrlBlocked(url)) {
  186. return originalMethod.call(history, data, title, url);
  187. }
  188. console.error('Blocked history manipulation to:', url);
  189. }
  190.  
  191. const originalPushState = history.pushState;
  192. const originalReplaceState = history.replaceState;
  193.  
  194. history.pushState = function(data, title, url) {
  195. return handleHistoryManipulation(originalPushState, data, title, url);
  196. };
  197.  
  198. history.replaceState = function(data, title, url) {
  199. return handleHistoryManipulation(originalReplaceState, data, title, url);
  200. };
  201.  
  202. /**
  203. * Check if a URL is blocked based on the blacklist
  204. * @param {string} url - The URL to check
  205. * @returns {boolean} True if the URL is blocked, false otherwise
  206. */
  207. function isUrlBlocked(url) {
  208. const encodedUrl = encodeURIComponent(url);
  209. const automatedBlacklist = getAutomatedBlacklist();
  210. const isBlocked = [...manualBlacklist, ...automatedBlacklist].some(blockedUrl => encodedUrl.includes(blockedUrl));
  211. if (isBlocked) {
  212. console.log(`Blocked URL: ${url}`);
  213. }
  214. return isBlocked;
  215. }
  216.  
  217. console.log('Redirect control script with blacklist initialized.');
  218. })();