- // ==UserScript==
- // @name Proxified Links DEBUG
- // @author proxi
- // @homepageURL https://greasyfork.org/en/scripts/485274-proxified-links
- // @copyright 2023 Schimon Jehudah (http://schimon.i2p)
- // @license AGPL-3.0-only; https://www.gnu.org/licenses/agpl-3.0.en.html
- // @namespace com.proxi.proxified
- // @description Proxified hyperlinks to a proxy instance or Farside with no nonsense
- // Add or remove preferred services and instances yourself!
- //
- // Forked from Proxify Links v23.10.17, by Schimon Jehudah, and modified with prejudice and style
- // - Proxy, proxy, proxy; see and configure non-proxying frontends yourself before opening links
- // - Use a small hardcoded list of favorite proxy instances, or:
- // - Hold X key to use Farside redirection where possible
- // - Hold Z key to use the original link whenever needed
- // - Easily use first-party frontends with discretion or when instances are down
- // - Clearnet by default following Farside, add personal lists of decentralized nodes
- // - Set noreferrer on supported links, optionally all links if `ENABLE_REFERER_HIDE_PAGEWIDE`,
- // preventing referer header while browsing without breaking common ajax functionality
- // - Use GET requests on supported search engines particularly with noscript
- // (Recommend disabled if `ENABLE_REFERER_HIDE` is off)
- // @run-at document-end
- // @version 0.3.0
- // @match *://*/*
- // @icon 
- // ==/UserScript==
-
- /**
- * Basic configurations for functionality and performance
- */
- const KEY_MODIFIED = 'x'; // X key to modify clicked link to Farside, if applicable
- const KEY_BONAFIDE = 'z'; // Z key to revert clicked link to the original destination
- const TOUCH_MODIFIED = 3; // Simple 3-finger tap gesture to modify tapped link to Farside, if applicable
- const TOUCH_BONAFIDE = 5; // Simple 5-finger tap gesture to revert tapped link to the original destination
- const TOUCH_PROXIFIED = 2; // Reserved 2-finger tap gesture to reset tapped link to static proxified
- const ROTATE_SITE_PROXIFIED = true; // Optional: Rotate proxy instances instead of random to prevent adjacent duplicates
- const ENABLE_RERENDER = true; // Optional: Forcibly rerender for browser to update UI, may degrade performance
-
- // Optional keys to limit unintentional triggering on non-links
- // Set to empty '' string, proxification will happen automatically on selection
- // TODO: improve touch support
- const KEY_PROXIBITE = 'b'; // Optional B key to allow heavier elements to be proxified
- const KEY_FRAMEBITE = 'f'; // Optional B+F key combination to trigger when iframes are updated
- const TOUCH_PROXIBITE = 4; // Simple 4-finger tap gesture to allow all heavier elements to be proxified
- const TOUCH_FRAMEBITE = 4;
- // TODO: support multiple keys to whitelist specific keys that will trigger handlers, consider screenreader controls
- const KEY_NAVIGATE = ''; // Optional filter for key nav used to proxify the selected element, e.g Tab
-
- // Optional proxified iframes with supported src, proxified immediately when selected
- // Best used if iframe content is already blocked by a content blocker (e.g. ublock click2load),
- // so the original frame is never loaded automatically even if switching back from proxified (B+F+Z)
- const ENABLE_IFRAME_PROXIFIED = true;
-
- // Optional noreferer override on all proxified links
- // If disabled, only upgrade undefined or noopener to noreferrer for proxified links
- // Disable if proxified links work but bona fide (Z) links are breaking
- const ENABLE_REFERER_HIDE = true;
-
- // Optional noreferer override on all links on the page, if ENABLE_REFERER_HIDE is also on
- // Disable if unrelated links or site authentication windows are breaking
- const ENABLE_REFERER_HIDE_PAGEWIDE = true;
-
- // Optional proxified links using query string if the main url does not match a proxy
- // Useful for tracking links or old-fashioned HTML GET search, e.g. DuckDuckGo (uddg)
- // Strongly recommend used with `ENABLE_REFERER_HIDE_PAGEWIDE` to avoid exposing search engine and keywords
- // Disable if random links that shouldn't be proxified still are
- const ENABLE_QUERY_PROXIFIED = true;
- const ENABLE_QUERY_PROXIFIED_ON = []; // if empty, all query parameters will be included
- const ENABLE_QUERY_PROXIFIED_OFF = ['piurl' /* Startpage image proxy */]; // blacklisted query parameters
-
- // Optional stripping links of attributes that are necessary to proxify links (Google),
- // and/or optional removal of link trackers on sites or search engines
- const ENABLE_ATTRIBUTES_SMITE = true;
- /** @type {{ [host: string]: {attributes: string[], allowFuzzy?: boolean} }} */
- const ENABLE_ATTRIBUTES_SMITE_ON_SITE = { 'google.com': { attributes: ['data-sb'] } };
-
- // Optional search engine GET requests instead of POST for improved search navigation
- // Used with ENABLE_REFERER_HIDE and ENABLE_REFERER_HIDE_PAGEWIDE on
- // CAUTION: May be unsupported, and subject to negligent server logging query string
- const ENABLE_SEARCH_GET = true;
- /** @type {{ [host: string]: { formSelector: string } }} */
- const ENABLE_SEARCH_GET_ON_SITE = {
- 'html.duckduckgo.com': { formSelector: 'form:has(#search_form_input_homepage), form:has(.btn[type="submit"])' },
- 'startpage.com': { formSelector: '#search, form:has(.header-nav-item), form:has(button[type="submit"]' },
- };
-
- /**
- * Configure site settings or add an instance of a proxy to the site's `redirect` list
- *
- * As is, use a shortlist of useful instances and mature proxy services
- * Out of box is an opinionated, not comprehensive or up-to-date, list of useful instances,
- * not ordered alphabetically but in the order of best proxied or most commonly linked
- *
- * @typedef {{
- * redirect?: {
- * replacements?: string[],
- * suffix?: string, // optional suffix to a random replacement host from `replacements`
- * routes?: ProxyRoute[], // optional route whitelist by matching url regex, defaults to replacing url host only if undefined
- * inherit?: string // inherit any undefined properties in `redirect`
- * },
- * redirectToFarside?: {
- * replacements?: string[],
- * suffix?: string, // optional suffix to a random replacement host from `replacements`
- * routes?: ProxyRoute[], // optional route whitelist by matching url regex, defaults to replacing url host only if undefined
- * inherit?: string // inherit any undefined properties in `redirectToFarside`
- * },
- * allowFuzzy?: boolean, // allows fuzzy host matching, for subdomains
- * allowIframe?: boolean, // allows matching on iframe src
- * inherit?: string // inherit any undefined redirect rules
- * }} Proxy
- * @typedef {{
- * regex: RegExp, // route regex match
- * suffix?: string, // optional route suffix on host, appending to parent `suffix` if defined
- * }} ProxyRoute
- * @typedef {{ [host: string]: Proxy }} ProxyList
- */
-
- /**
- * @type {ProxyList}
- */
- const PROXIES = {
- // youtube.com, youtu.be, m.youtube.com, youtube-nocookie.com
- // /watch, /trending, /@, /channel/
- // Allows iframe matching too, which is useful when content blocking embeds
- // e.g. ublock ||youtube.com^$3p,frame,redirect=click2load.html
- 'youtube.com': {
- allowIframe: true,
- redirect: {
- replacements: [
- 'https://piped.video',
- 'https://piped.smnz.de',
- //'https://piped.projectsegfau.lt', // down
- 'https://piped.privacydev.net',
- //'https://piped.lunar.icu', // embedded frame blocked by x-frame-options
- 'https://piped.adminforge.de',
- //'https://pd.vern.cc', // low availability
- // The following Invidious instances not only allow video proxy but proxy by default
- //'https://iv.datura.network', // proxy off by default
- //'https://invidious.projectsegfau.lt', // down
- //'https://invidious.fdn.fr', // down
- ],
- },
- redirectToFarside: {
- // Only use Farside's Piped redirect since most Invidious instances do not proxy videos by default
- // Aside from proxy by default, Invidious is preferred for nojs, configuration, and download functionality
- // Specific Invidious instances that proxy by default are included in the instance `redirect` list
- replacements: ['https://farside.link/piped'],
- },
- },
- 'youtu.be': {
- redirect: {
- inherit: 'youtube.com',
- routes: [
- {
- regex: /^https?:\/\/(www\.)?youtu\.be\/([A-Za-z0-9_-]+)\??(.*)$/,
- suffix: '/watch?v=$2&$3', // manually add in path and params to support invidious https://github.com/iv-org/invidious/issues/3933
- },
- ],
- },
- redirectToFarside: {
- inherit: 'youtube.com',
- routes: [
- {
- regex: /^https?:\/\/(www\.)?youtu\.be\/([A-Za-z0-9_-]+)\??(.*)$/,
- suffix: '/watch?v=$2&$3', // manually add in path and params to support invidious https://github.com/iv-org/invidious/issues/3933
- },
- ],
- },
- },
- 'm.youtube.com': { inherit: 'youtube.com' },
- 'youtube-nocookie.com': { inherit: 'youtube.com' },
-
- // reddit.com
- 'reddit.com': {
- redirect: {
- replacements: [
- //'https://libreddit.projectsegfau.lt', // low availability
- //'https://libreddit.privacydev.net', // low availability
- 'https://l.opnxng.com',
- //'https://reddit.invak.id', // down
- 'https://libreddit.kavin.rocks',
- 'https://red.artemislena.eu',
- ],
- },
- redirectToFarside: {
- // teddit no longer actively maintained: https://codeberg.org/teddit/teddit
- replacements: ['https://farside.link/libreddit'],
- },
- },
- // redd.it image shortlinks
- 'i.redd.it': {
- redirect: {
- inherit: 'reddit.com',
- suffix: '/img',
- },
- redirectToFarside: {
- inherit: 'reddit.com',
- suffix: '/img',
- },
- },
- 'preview.redd.it': {
- redirect: {
- inherit: 'reddit.com',
- suffix: '/preview/pre',
- },
- redirectToFarside: {
- inherit: 'reddit.com',
- suffix: '/preview/pre',
- },
- },
- 'external-preview.redd.it': {
- redirect: {
- inherit: 'reddit.com',
- suffix: '/preview/external-pre',
- },
- redirectToFarside: {
- inherit: 'reddit.com',
- suffix: '/preview/external-pre',
- },
- },
-
- /**
- *
- * Below are less mature or partially featured services
- *
- */
-
- // stackoverflow.com, {subdomain}.stackexchange.com
- // superuser.com, serverfault.com, and other stack sites pending AnonymousOverflow support
- 'stackoverflow.com': {
- redirect: {
- replacements: [
- // 'https://ao.vern.cc', // low availability
- 'https://overflow.smnz.de',
- //'https://overflow.lunar.icu', // TODO: reenable, temp disable old version incompat w/ non-stackexchange sites
- 'https://overflow.adminforge.de',
- //'https://overflow.hostux.net', // low stability
- // 'https://overflow.projectsegfau.lt', // low availability
- ],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/anonymousoverflow'],
- },
- },
- '.stackexchange.com': {
- allowFuzzy: true, // enable matching loosely with arbitrary subdomain
- redirect: {
- inherit: 'stackoverflow.com',
- routes: [
- {
- regex: /^https?:\/\/(www\.)?([a-z]+)\..*?\//g,
- suffix: '/exchange/$2/', // suffix /exchange/{subdomain} on hostname
- },
- ],
- },
- redirectToFarside: {
- inherit: 'stackoverflow.com',
- routes: [
- {
- regex: /^https?:\/\/(www\.)?([a-z]+)\..*?\//g,
- suffix: '/exchange/$2/',
- },
- ],
- },
- },
- // non-stackexchange domains are appended to route /exchange/{uri}
- 'superuser.com': {
- redirect: {
- inherit: 'stackoverflow.com',
- routes: [
- {
- regex: /^https?:\/\/(www\.)?(.*)/g, // take entire URI in group 2
- suffix: '/exchange/$2',
- },
- ],
- },
- redirectToFarSide: {
- inherit: 'stackoverflow.com',
- routes: [
- {
- regex: /^https?:\/\/(www\.)?(.*)/g, // take entire URI in group 2
- suffix: '/exchange/$2',
- },
- ],
- },
- },
- 'serverfault.com': { inherit: 'superuser.com' },
- 'askubuntu.com': { inherit: 'superuser.com' },
- 'stackapps.com': { inherit: 'superuser.com' },
-
- // quora.com
- 'quora.com': {
- redirect: {
- replacements: ['https://quetre.iket.me', 'https://quetre.pussthecat.org', 'https://quetre.privacydev.net'],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/quetre'],
- },
- },
-
- // {artist}.bandcamp.com
- // Note: bandcamp.com/search route not supported, add above for 'bandcamp.com' if this rare link is needed is the wild
- // Note: {cdn}.bcbits.com routes not supported, add below for '.bcbits.com' if this rare link is needed in the wild
- '.bandcamp.com': {
- allowFuzzy: true,
- redirect: {
- replacements: ['https://tent.sny.sh', 'https://tn.vern.cc'],
- routes: [
- {
- // {artist}.bandcamp.com with no additional path except optional /music
- // exclude daily.bandcamp.com
- regex: /^https?:\/\/(www\.)?((?!daily\.)[a-z0-9\-]+)\.bandcamp\.com\/?(music)?$/g,
- suffix: '/artist.php?name=$2',
- },
- {
- // {artist}.bandcamp.com/{release}/{name}
- // exclude daily.bandcamp.com, e.g. daily.bandcamp.com/features/{article}
- regex: /^https?:\/\/(www\.)?((?!daily\.)[a-z0-9\-]+)\.bandcamp\.com\/([a-z]+)\/([a-z0-9\-]+)/g,
- suffix: '/release.php?artist=$2&type=$3&name=$4',
- },
- ],
- },
- },
-
- // instagram.com
- // Low feature parity
- 'instagram.com': {
- redirect: {
- replacements: ['https://ig.opnxng.com', 'https://proxigram.lunar.icu'],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/proxigram'],
- },
- },
-
- // tiktok
- // Low feature parity
- 'tiktok.com': {
- redirect: {
- replacements: [
- 'https://proxitok.pussthecat.org',
- 'https://tok.artemislena.eu',
- 'https://tok.adminforge.de',
- 'https://tik.hostux.net',
- 'https://proxitok.lunar.icu',
- ],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/proxitok'],
- },
- },
-
- // imgur.com, i.imgur.com, i.stack.imgur.com
- 'imgur.com': {
- redirect: {
- replacements: [
- 'https://rimgo.pussthecat.org',
- 'https://imgur.artemislena.eu',
- // 'https://rimgo.vern.cc', // low availability
- // 'https://rimgo.hostux.net', // down
- // 'https://rimgo.lunar.icu', // down
- 'https://rimgo.eu.projectsegfau.lt',
- ],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/rimgo'],
- },
- },
- 'i.imgur.com': { inherit: 'imgur.com' },
- 'i.stack.imgur.com': {
- inherit: 'imgur.com',
- redirect: { inherit: 'imgur.com', suffix: '/stack' },
- redirectToFarside: { inherit: 'imgur.com', suffix: '/stack' },
- },
-
- // github.com, gists.github.com
- // /explore, /{group}/{repo}, /{group}/{repo}/archive, gists.github.com -> /gists/
- // Low feature parity
- // Use only for repo landing page, downloads, and gists
- 'github.com': {
- redirect: {
- replacements: [
- 'https://gothub.lunar.icu',
- 'https://g.opnxng.com',
- //'https://gothub.projectsegfau.lt', // low availability
- 'https://gothub.dev.projectsegfau.lt',
- ],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/gothub'],
- },
- },
- // gist.github.com
- 'gist.github.com': {
- redirect: {
- inherit: 'github.com',
- suffix: '/gist/',
- routes: [{ regex: /https?:\/\/(.*?)\//g }], // replace entire domain
- },
- redirectToFarside: {
- inherit: 'github.com',
- suffix: '/gist/',
- routes: [{ regex: /https?:\/\/(.*?)\//g }],
- },
- },
-
- // imdb.com, m.imdb.com
- 'imdb.com': {
- redirect: {
- replacements: [
- 'https://libremdb.pussthecat.org',
- 'https://libremdb.iket.me',
- //'https://ld.vern.cc', // low availability
- 'https://libremdb.lunar.icu',
- ],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/libremdb'],
- },
- },
- 'm.imdb.com': { inherit: 'imdb.com' },
-
- // genius.com
- // Low feature parity
- 'genius.com': {
- redirect: {
- replacements: ['https://dm.vern.cc', 'https://dumb.lunar.icu'],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/dumb'],
- },
- },
-
- // medium.com - Uncomment to use
- // Low feature parity by design
- // Not a proxy by design, alternative frontend still requests from the official servers
- //
- // Recommend setting Medium to noscript and/or loading through more standard proxies such as TOR
- // Medium with JS disabled works as of now, but other proxy sites such as archive.org can be used if needed
- /* --- Remove this line to use --- //
- 'medium.com': {
- redirect: {
- replacements: ['https://scribe.rip', 'https://sc.vern.cc', 'https://m.opnxng.com'],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/scribe'],
- },
- },
- // ------------------------------- */
-
- // fandom.com - Uncomment to use
- // Not a full proxy, alternative frontend still requests from the official servers
- //
- // Recommend simply using a content blocker to block ads and other annoyances
- /* --- Remove this line to use --- //
- '.fandom.com': {
- allowFuzzy: true,
- redirect: {
- replacements: [
- 'https://breezewiki.com',
- 'https://antifandom.com',
- 'https://breezewiki.pussthecat.org',
- 'https://bw.projectsegfau.lt',
- 'https://breeze.hostux.net',
- 'https://bw.artemislena.eu',
- 'https://breeze.nohost.network',
- 'https://z.opnxng.com',
- ],
- routes: [
- {
- regex: /^https?:\/\/(www\.)?([a-z\-]+)\..*?\//g,
- suffix: '/$2/', // suffix /{subdomain} on hostname
- },
- ],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/breezewiki'],
- routes: [
- {
- regex: /^https?:\/\/(www\.)?([a-z\-]+)\..*?\//g,
- suffix: '/$2/',
- },
- ],
- },
- },
- // ------------------------------- */
-
- // wikipedia.org - Uncomment to use
- //
- // Recommend setting Wikipedia to noscript and/or loading through more standard proxies such as TOR
- // If absolutely needed, recommend rolling your own Wikiless instance routed through a proxy or VPN
- // Wikipedia trustworthiness and scriptless tracking is more or less equivalent to wikiless instances
- /* --- Remove this line to use --- //
- 'wikipedia.org': {
- redirect: {
- replacements: [
- 'https://wiki.adminforge.de',
- 'https://wikiless.lunar.icu',
- 'https://wikiless.org',
- 'https://wl.vern.cc',
- ],
- },
- redirectToFarside: {
- replacements: ['https://farside.link/wikiless'],
- },
- },
- // ------------------------------- */
- };
-
- /**
- * Configure site link exclusions
- *
- * The most common exclusion will be on the first-party site itself, as many proxies are not complete replacements.
- * Common unsuppoorted features and paths are excluded, though this is not intended to exhaustively track the list of
- * proxied frontends and availability or configuration of individual instances.
- *
- * excludeLinks can exclude links that match any provided rule, namely matchingPath
- * excludeOn can exclude links when host matches `self` or any provided, e.g. by matchingPath (all paths if empty)
- * @typedef {{
- * excludeLinks?: {
- * matchingPath?: (string|RegExp)[],
- * matchingHost?: (string|RegExp)[],
- * matchingUrl?: (string|RegExp)[],
- * matchingText?: (string|RegExp)[],
- * inherit?: string // inherit any undefined properties in `excludeLinks`
- * },
- * excludeOn?: {
- * matchingPath?: (string|RegExp)[],
- * matchingHost?: (string|RegExp)[],
- * matchingUrl?: (string|RegExp)[],
- * matchingBody?: (string|RegExp)[],
- * matchingHead?: (string|RegExp)[],
- * inherit?: string // inherit any undefined properties in `excludeOn`
- * },
- * allowFuzzy?: boolean, // allows fuzzy host matching, for subdomains
- * inherit?: string // inherits any undefined exclusion rules
- * }} Exclusion
- * @typedef {{ [host: string]: Exclusion }} ExclusionList
- */
-
- /**
- * Configure per-site exclusions
- * @type {ExclusionList}
- */
- const EXCLUSIONS = {
- // youtube.com, m.youtube.com, youtube-nocookie.com
- 'youtube.com': {
- excludeLinks: {
- matchingPath: ['/users/'],
- },
- excludeOn: {
- // exclude linking out from the official site
- matchingHost: ['youtube.com', 'youtube-nocookie.com'],
- matchingHead: [
- '<meta property="og:title" content="Piped">',
- /<meta property="og:site_name" content=".*Invidious">/,
- ],
- },
- },
- 'm.youtube.com': { inherit: 'youtube.com' },
- 'youtube-nocookie.com': { inherit: 'youtube.com' },
-
- // reddit.com
- 'reddit.com': {
- excludeLinks: {
- // old.reddit.com can be used almost entirely with noscript
- matchingHost: ['old.reddit.com'],
- },
- excludeOn: {
- // exclude on reddit.com but still proxy links while browsing old.reddit.com
- matchingUrl: [/^https?:\/\/(www\.)?(?!old\.)reddit.com/],
- },
- },
-
- // stackoverflow.com, {subdomain}.stackexchange.com, superuser.com, etc.
- 'stackoverflow.com': {
- excludeLinks: {
- matchingPath: ['/questions/tagged/', '/users/'],
- },
- excludeOn: {
- matchingHost: [
- 'stackoverflow.com',
- 'stackexchange.com',
- 'superuser.com',
- 'serverfault.com',
- 'askubuntu.com',
- 'stackapps.com',
- ],
- },
- },
- '.stackexchange.com': {
- allowFuzzy: true,
- inherit: 'stackoverflow.com',
- },
- 'superuser.com': { inherit: 'stackoverflow.com' },
- 'serverfault.com': { inherit: 'stackoverflow.com' },
- 'askubuntu.com': { inherit: 'stackoverflow.com' },
- 'stackapps.com': { inherit: 'stackoverflow.com' },
-
- // quora.com
- 'quora.com': { excludeOn: { matchingHost: ['quora.com'] } },
-
- // {artist}.bandcamp.com
- '.bandcamp.com': { allowFuzzy: true, excludeOn: { matchingHost: ['bandcamp.com'] } },
-
- // instagram.com
- 'instagram.com': {
- excludeOn: {
- matchingHost: ['instagram.com'],
- matchingHead: [
- /<meta property="og:title" content="[a-zA-Z0-9 _\-+=.,:;'?\/\\`!@#$%^&*()-_\[\]{}|]+? • Proxigram">/,
- ],
- },
- },
-
- // tiktok.com
- 'tiktok.com': {
- excludeOn: {
- matchingHost: ['tiktok.com'],
- // exclude on ProxiTok itself, linking out to original
- matchingHead: ['<meta property="og:site_name" content="ProxiTok">'],
- },
- },
-
- // imgur.com
- 'imgur.com': { excludeOn: { matchingHost: ['imgur.com'] } },
- 'i.imgur.com': { inherit: 'imgur.com' },
- 'i.stack.imgur.com': { inherit: 'imgur.com' },
-
- // github.com
- // a lot to blacklist, can also whitelist limited functionality instead
- 'github.com': {
- excludeLinks: {
- matchingText: [],
- // prettier-ignore
- matchingPath: ['/actions','/blame/','/codespaces/','/collections/','/commit/','/commits/','/compare/','/customer-stories','/delete/','/discussions/','/enterprise/','/events/','/features/','/graphs/','/issues','/marketplace/','/notifications/','/orgs/','/projects/','/pulls', '/pull/','/pulse','/releases','/security','/sessions/','/sponsors/','/tags','/tree/','/wiki/'],
- },
- excludeOn: {
- matchingHost: ['github.com'],
- // exclude on Gothub page itself, linking out
- matchingBody: ['<a href="https://codeberg.org/gothub/gothub">Source code</a>'],
- },
- },
- 'gist.github.com': {
- excludeLinks: {
- // gist single directory paths, e.g. users, /discover, /starred
- matchingPath: [/^\/[A-Za-z0-9_.-]+\/?$/],
- },
- inherit: 'github.com',
- },
-
- // imdb.com
- 'imdb.com': {
- excludeOn: {
- matchingHost: ['imdb.com'],
- // exclude on libremdb itself, linking out
- matchingHead: ['<meta property="og:site_name" content="libremdb">'],
- },
- },
-
- // genius.com
- // /artist and other pages may not work, but not blacklisting any paths for now
- 'genius.com': { excludeOn: { matchingHost: ['genius.com'] } },
-
- // medium.com
- 'medium.com': { excludeOn: { matchingHost: ['medium.com'] } },
-
- // fandom.com
- 'fandom.com': { excludeOn: { matchingHost: ['fandom.com'] } },
-
- // wikipedia.org
- 'wikipedia.org': { excludeOn: { matchingHost: ['wikipedia.org'] } },
- };
-
- /**
- * Begin script logic
- *
- * Do not modify below if adding or making changes to available proxies
- */
- const DEBUG = true; // Console logging enabled when true
-
- // Tag names of hovered link elements that can be proxified
- const PROXIFY_ON = ['A', 'IFRAME'];
- // Tag names of hovered elements that should also lookup to the parent node for links
- // TODO: per-tag and per-site configuration
- const PARENT_LOOKUP_ON = [
- 'BUTTON',
- 'IMG',
- 'SVG',
- 'H1',
- 'H2',
- 'H3',
- 'H4',
- 'H5',
- 'H6',
- 'B',
- 'I',
- 'EM',
- 'STRONG',
- 'SMALL',
- 'SUP',
- 'SUB',
- 'S',
- 'U',
- 'LI',
- ];
- // Default depth limit to ancestor lookup recursion
- const PARENT_LOOKUP_STEPS = 1;
- /**
- * Additional per-site parent lookup config, to fix site-specific proxification
- * TODO: further segment sites on matchesPath or other rules and per-nodeName configs
- * @type {{ [host: string]: { nodeNames: string[], steps: number } }}
- */
- const PARENT_LOOKUP_ON_SITE = {
- // google.com mobile search results div
- // TODO: improve mobile search inline media touch events swallowed
- 'google.com': { nodeNames: ['DIV'], steps: 4 },
- // duckduckgo.com search results span, hero summary SVGs and title container div
- 'duckduckgo.com': { nodeNames: ['SPAN', 'DIV', 'PATH', 'RECT'], steps: 4 },
- // startpage.com search results inline images/videos
- 'startpage.com': { nodeNames: ['DIV'], steps: 2 },
- };
-
- // Tag names of hovered elements that should also globally search all hovered elements for links
- const HOVER_LOOKUP_ON = ['P', 'SPAN', 'LI', 'LABEL', /*'DIV',*/ 'BUTTON', 'IMG', 'SVG'];
- // Additional tag names to search per-site, to fix site-specific proxificaton
- /** @type {{ [host: string]: [tags: string[]] }} */
- const HOVER_LOOKUP_ON_HOST = {
- 'duckduckgo.com': ['DIV'],
- };
-
- const CLASS_PROXIFIED = 'proxi-fied';
- const CLASS_PROXIFIED_LIVE = 'proxi-live';
- const CLASS_BODY_FARSIDE = 'proxi-side';
- const CLASS_BODY_BONAFIDE = 'proxi-fide';
- const CLASS_BODY_BITE = 'proxi-bite';
- const CLASS_BODY_FRAMEBITE = 'proxi-framebite';
- const STYLE_HIGHLIGHT_ID = 'proxi-highlite';
- const ATTR_PROXIFIED_SITE = 'proxi-site';
- const ATTR_PROXIFIED_FARSIDE = 'proxi-site-side';
- const ATTR_PROXIFIED_DENIED = 'proxi-nied';
- const ATTR_BONAFIDE_SITE = 'proxi-site-bonafide';
- const ATTR_IS_FARSIDE = 'proxi-side';
-
- // https://uibakery.io/regex-library/url
- const REGEX_URL =
- /^https?:\/\/(?:www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b(?:[-a-zA-Z0-9()@:%_\+.~#?&\/=]*)$/;
-
- // Key event modifiers for selecting destination
- document.body.addEventListener('keydown', e => handleModifierKey(e));
- document.body.addEventListener('keyup', e => handleModifierKey(e));
- document.body.addEventListener('touchstart', e => handleModifierGesture(e));
-
- /**
- * Lazy load link processing when user interacts per-element
- *
- * Handle hovering over links and completed keyboard navigation over link
- * Does not handle other basic redirects such as form action or onclick attributes
- */
- document.body.addEventListener('mouseover', handleElement); // link hover
- document.body.addEventListener('touchstart', handleElement); // link touch
- document.body.addEventListener('keyup', handleElement); // keyboard navigation
-
- /**
- * Automatic page onload and DOM modifications
- */
- window.addEventListener('load', () => {
- if (ENABLE_SEARCH_GET) {
- let searchEngine = getByHost(ENABLE_SEARCH_GET_ON_SITE, window.location.host, false);
- if (!!searchEngine && !!searchEngine.formSelector) {
- const searchEngineFormEls = document.querySelectorAll(searchEngine.formSelector);
- for (let formEl of searchEngineFormEls) {
- formEl.method = 'get';
- }
- }
- }
- });
-
- /**
- * Hover event handler to find an anchor hyperlink to check
- * @param {Event} e
- */
- function handleElement(e) {
- // Currently support specific events, and optional key nav filter
- if (
- e.type !== 'mouseover' &&
- e.type !== 'touchstart' &&
- (e.type !== 'keyup' || (!!KEY_NAVIGATE && e.key !== KEY_NAVIGATE))
- ) {
- return;
- }
-
- // Ignore target on multi-touch touch event
- if (e.type === 'touchstart' && e.touches.length !== 1) {
- return;
- }
-
- // Look for any element or elements that this event could be trying to proxify
- const proxifiableEls = getProxifiableElements(e);
- if (!proxifiableEls || !proxifiableEls.length) return;
-
- // Prioritize anchor links
- // Take the first anchor, even if it has already been processed
- const targetAnchor = proxifiableEls.find(el => el.nodeName === 'A');
- if (!!targetAnchor) {
- // Hovered anchor found, process it unless it was already processed before
- if (!isElementProxified(targetAnchor)) {
- handleAnchorEl(targetAnchor);
- } else {
- // Otherwise, trigger an update on the link element to ensure it is current
- setTimeout(() => {
- updateProxifiedElement(targetAnchor, 'href');
- }, 0);
- }
-
- // Only handle one element per event for now to avoid overeager proxification
- return;
- }
-
- // Handle iframe, if enabled
- // If both or either bite key is empty, handle on any interaction
- if (ENABLE_IFRAME_PROXIFIED) {
- const isProxibite = document.body.classList.contains(CLASS_BODY_BITE);
- const isFramebite = document.body.classList.contains(CLASS_BODY_FRAMEBITE);
- const canBite =
- (!KEY_PROXIBITE || isProxibite || e.key === KEY_PROXIBITE || e.touches?.length === TOUCH_PROXIBITE) &&
- (!KEY_FRAMEBITE || isFramebite || e.key === KEY_FRAMEBITE || e.touches?.length === TOUCH_FRAMEBITE);
- const targetIframe = canBite && proxifiableEls.find(el => el.nodeName === 'IFRAME');
- if (!!targetIframe) {
- if (!isElementProxified(targetIframe)) {
- handleIframe(targetIframe);
- } else {
- setTimeout(() => {
- updateProxifiedElement(targetIframe, 'src');
- }, 0);
- }
- }
-
- return;
- }
- }
-
- /**
- * Handle an anchor element to be proxified
- * Allows re-proxifying the anchor
- *
- * @param {HTMLAnchorElement} el Anchor element
- * @returns {void}
- */
- function handleAnchorEl(el) {
- // Perform element and url validation and returns a valid proxy
- const { url, proxy } = preproxifyElement(el, 'href') || {};
-
- // Perform any global link modifications including non-proxied links
- if (ENABLE_REFERER_HIDE && ENABLE_REFERER_HIDE_PAGEWIDE) el.rel = 'noreferrer';
- let optSmite;
- if (
- ENABLE_ATTRIBUTES_SMITE &&
- !!el.href &&
- !!(optSmite = getByHost(ENABLE_ATTRIBUTES_SMITE_ON_SITE, window.location.host, true))
- ) {
- // If enabled on the current site, strip any click event attributes from this anchor element
- // Resulting hyperlink is intended to be a primitive link with no events intercepting navigation
- // TODO: Support reverting events on bonafide
- for (const attr of optSmite.attributes) {
- el.removeAttribute(attr);
- }
- }
-
- // Proxify the link
- // Error handling will have been done on preproxification
- if (!!proxy) {
- proxifyElement(
- el,
- url,
- 'href',
- proxy,
- /** @param {HTMLAnchorElement} el */ el => {
- // Trigger an update on the link now that it has been proxified
- updateProxifiedElement(el, 'href');
-
- // Upgrade undefined or noopener relationship to norefererer
- // If `ENABLE_REFERER_HIDE`, override on all proxified links
- // TODO: Support modifier key to revert change?
- if (!el.rel || el.rel === 'noopener' || ENABLE_REFERER_HIDE) el.rel = 'noreferrer';
- }
- );
- }
- }
-
- /**
- * Handle an iframe to be proxified
- * Allows re-proxifying the frame
- *
- * @param {HTMLIFrameElement} el
- * @returns {void}
- */
- function handleIframe(el) {
- const { url, proxy } = preproxifyElement(el, 'src') || {};
- if (!!proxy)
- proxifyElement(
- el,
- url,
- 'src',
- proxy,
- /** @param {HTMLIframeElement} el */ el => {
- // Trigger an update on the link now that it has been proxified
- updateProxifiedElement(el, 'src');
- }
- );
- }
-
- /**
- * Get the element or elements that can be proxified, including those already proxified,
- * from an event where the user is hovering or making a selection
- *
- * @param {MouseEvent | KeyboardEvent | TouchEvent} e
- * @returns {Element[]?} If none, null
- */
- function getProxifiableElements(e) {
- if (!(e instanceof MouseEvent || e instanceof KeyboardEvent || e instanceof TouchEvent)) {
- error('Invalid event');
- return null;
- }
-
- if (!e.target) return null;
- const nodeName = e.target.nodeName.toUpperCase();
-
- // Anchor links get top priority if directly targeted
- if (nodeName === 'A') return [e.target];
-
- // Otherwise, build a list of candidates in the order of probable priority
- // Add in other directly targeted non-anchor links to the top
- const proxifiableEls = [];
- if (PROXIFY_ON.includes(nodeName)) proxifiableEls.push(e.target);
-
- // Parents may not receive propagated events, so use this event now to check if nested under a link
- let optLookup = getByHost(PARENT_LOOKUP_ON_SITE, window.location.host, true);
- if (PARENT_LOOKUP_ON.includes(nodeName) || optLookup?.nodeNames?.includes(nodeName)) {
- // Step up through ancestors
- // Or use target.closest() to search all ancestors
- const lookupSteps = Math.max(optLookup?.steps ? optLookup.steps : 1, PARENT_LOOKUP_STEPS);
- let currentNode = e.target;
- for (let step = 0; step < lookupSteps; step++) {
- if (!!currentNode.parentNode) {
- currentNode = currentNode.parentNode;
-
- // take processed or unprocessed proxifiable elements
- if (PROXIFY_ON.includes(currentNode.nodeName.toUpperCase())) {
- proxifiableEls.push(currentNode);
- }
- } else {
- break;
- }
- }
- }
-
- // A link may be a sibling or other relative, or not even a relative, that does not receive propagated
- // For mouse hover or bite event, find any link element that is currently being hovered
- // TODO: Skip previously lookup elements and handle updateProxifiedElement rerender separately from proxifying
-
- const isHoverLookup =
- e instanceof MouseEvent &&
- (HOVER_LOOKUP_ON.includes(nodeName) || HOVER_LOOKUP_ON_HOST[window.location.host]?.includes(nodeName));
- const isProxiBite = !!KEY_PROXIBITE && document.body.classList.contains(CLASS_BODY_BITE);
- if (isHoverLookup || isProxiBite) {
- // Select all hovered elements and take the processed or unprocessed proxifiable elements
- const hoveredEls = document.querySelectorAll(':hover');
- for (const el of hoveredEls) {
- if (PROXIFY_ON.includes(el.nodeName.toUpperCase())) {
- proxifiableEls.push(el);
- }
- }
- }
-
- return !!proxifiableEls.length ? proxifiableEls : null;
- }
-
- /**
- * Perform element and destination url validation and return a valid proxy
- *
- * @param {HTMLElement} el Target element to be validated for proxification
- * @param {string} destinationAttr Element link destination attribute name
- * @returns {{url, Proxy} | null} Returns a random proxy for the element, null if none
- */
- function preproxifyElement(el, destinationAttr) {
- const destination = el[destinationAttr];
- const text = el.outerText;
-
- // Validate anchor is intended to be a hyperlink
- if (!destination.length) {
- proxifyDenyElement(el, 'Invalid, empty, or undefined destination attribute');
- return null;
- }
- if (!REGEX_URL.test(destination)) {
- proxifyDenyElement(el, 'Hyperlink destination is not a valid absolute URL');
- return null;
- }
-
- // Clean up any proxify attrs before proxifying
- el.removeAttribute(ATTR_PROXIFIED_SITE);
- el.removeAttribute(ATTR_PROXIFIED_FARSIDE);
- el.removeAttribute(ATTR_PROXIFIED_DENIED);
- el.removeAttribute(ATTR_BONAFIDE_SITE);
-
- // Parse the destination as a URL
- let url;
- try {
- url = new URL(destination);
- } catch (ex) {
- // Exit on malformed URL
- error(`[${text}](${destination}) error parsing URL: ${ex}`);
- proxifyDenyElement(el, 'Error parsing URL');
- return null;
- }
-
- // Exit if the original link is excluded for any reason
- // Page-wide exclusions to each destination host are memoized so may save unnecessary checking
- let excludedReason;
- if ((excludedReason = getExclusionReasonForLink(url, text))) {
- proxifyDenyElement(el, excludedReason);
- return null;
- }
-
- // Get a flattened proxy rule that can be applied for this link
- let proxy = getProxyForLink(url);
-
- // If no proxy can be found for the link url, optionally search its query string
- // Find a proxy match for any query param that is whitelisted, if used, and is a valid url
- if (!proxy && ENABLE_QUERY_PROXIFIED) {
- for (const [paramKey, paramVal] of url.searchParams) {
- // skip blacklisted query params
- if (ENABLE_QUERY_PROXIFIED_OFF.includes(paramKey)) {
- continue;
- }
-
- // if query param whitelist is empty, allow all
- if (!ENABLE_QUERY_PROXIFIED_ON?.length || ENABLE_QUERY_PROXIFIED_ON.includes(paramKey)) {
- let queryStringLinkCandidate = paramVal;
- let queryStringURLCandidate, queryStringProxyCandidate;
- try {
- queryStringLinkCandidate = decodeURI(paramVal); // attempt to decode a full URI encoded to the query string
- } catch (ex) {
- warn(`Failed to decode query string param: ${ex}`);
- continue;
- }
-
- // skip param if it doesn't look like an absolute url to proxify
- if (!REGEX_URL.test(queryStringLinkCandidate)) {
- continue;
- }
-
- try {
- queryStringURLCandidate = new URL(queryStringLinkCandidate);
- } catch (ex) {
- error(`Failed to parse validated url for query param "${paramKey}"`);
- continue;
- }
-
- if (getExclusionReasonForLink(queryStringURLCandidate, text)) {
- continue;
- }
-
- queryStringProxyCandidate = getProxyForLink(queryStringURLCandidate);
-
- // if successful, break on query string and take first match
- if (!!queryStringProxyCandidate) {
- url = queryStringURLCandidate;
- proxy = queryStringProxyCandidate;
- break;
- }
- }
- }
- }
-
- // Automatic exclusion if no proxy found on this pass
- if (!proxy) {
- proxifyDenyElement(el, 'Not found in proxies list');
- return null;
- }
-
- return { url, proxy };
- }
-
- /**
- * Proxify the link element with the specified proxy settings
- * Supports anchor and iframe elements for now
- *
- * @param {HTMLAnchorElement | HTMLIFrameElement} el
- * @param {URL} url
- * @param {string} destinationAttr Element link destination attribute name
- * @param {Proxy} proxy
- * @param {Function<HTMLAnchorElement | HTMLIFrameElement>?} onProxification Optional success callback
- * @returns {void}
- */
- function proxifyElement(el, url, destinationAttr, proxy, onProxification) {
- if (!(el instanceof HTMLAnchorElement) && !(el instanceof HTMLIFrameElement)) {
- error(`${el} is not a supported element type`);
- return;
- }
-
- if (!isProxyValid(proxy)) {
- error(`Invalid proxy settings for proxifying ${url}`);
- return;
- }
-
- let proxifyDenied;
-
- // Build a url to a random instance
- if (!!proxy.redirect) {
- const instanceList = proxy.redirect.replacements;
- const instanceSuffix = proxy.redirect.suffix;
- const instanceRoutes = proxy.redirect.routes;
- const urlInstanceProxified = getProxifiedUrl(url, instanceList, instanceSuffix, instanceRoutes, 'proxy');
-
- if (urlInstanceProxified instanceof URL) {
- el.setAttribute(ATTR_PROXIFIED_SITE, urlInstanceProxified);
- } else if (typeof urlInstanceProxified === 'string') {
- proxifyDenied ||= urlInstanceProxified;
- }
- }
-
- // Build a url to a random Farside service redirect
- if (!!proxy.redirectToFarside) {
- const farsideList = proxy.redirectToFarside.replacements;
- const farsideSuffix = proxy.redirectToFarside.suffix;
- const farsideRoutes = proxy.redirectToFarside.routes;
- const urlFarsideProxified = getProxifiedUrl(url, farsideList, farsideSuffix, farsideRoutes, 'farside');
-
- if (urlFarsideProxified instanceof URL) {
- el.setAttribute(ATTR_PROXIFIED_FARSIDE, urlFarsideProxified);
- } else if (typeof urlFarsideProxified === 'string') {
- proxifyDenied ||= urlFarsideProxified;
- }
- }
-
- // Complete the proxification
- if (el.hasAttribute(ATTR_PROXIFIED_SITE) || el.hasAttribute(ATTR_PROXIFIED_FARSIDE)) {
- // Mark the link as successfully proxified
- el.classList.add(CLASS_PROXIFIED);
-
- // Store the original anchor href
- if (!el.hasAttribute(ATTR_BONAFIDE_SITE)) {
- el.setAttribute(ATTR_BONAFIDE_SITE, el[destinationAttr]);
- }
-
- // Success callback
- if (!!onProxification) {
- onProxification(el);
- }
- } else if (!!proxifyDenied) {
- // No effect proxifying, but a denial reason was given
- proxifyDenyElement(el, proxifyDenied);
- } else {
- // Log unexpected failure past validated inputs
- error(`Failure proxifying ${url}`);
- }
- }
-
- /**
- * Process but mark the link element as being invalid
- *
- * @param {HTMLAnchorElement | HTMLIFrameElement} el
- * @param {string} reason
- */
- function proxifyDenyElement(el, reason) {
- if (!(el instanceof HTMLAnchorElement) && !(el instanceof HTMLIFrameElement)) {
- error(`${el} is not a supported element type`);
- return;
- }
-
- // Set denial reason as attr value
- el.setAttribute(ATTR_PROXIFIED_DENIED, reason);
-
- // Bonafide link is still stored for posterity
- el.setAttribute(ATTR_BONAFIDE_SITE, getElementDest(el));
- }
-
- /**
- * Return a proxified URL given a list of replacement hosts and optional routes
- *
- * @param {URL} url
- * @param {string[]} replacements
- * @param {string?} suffix Optional static suffix on the replacement host
- * @param {ProxyRoute[]?} routes Optional route list specifying a regex match with its corresponding options
- * @param {string} key Optional unique key used for proxifying
- * @returns {URL | string | null} Proxified URL, string reason if denied, null if failed
- */
- const replacementLast = {};
- function getProxifiedUrl(url, replacements, suffix, routes, key = '') {
- if (!url) {
- error('URL null or undefined');
- return null;
- }
-
- if (!replacements) {
- error('Replacements list null or undefined');
- return null;
- }
-
- // Get a cleaned list of replacement strings
- const replacementList = replacements.filter(
- replacement => typeof replacement === 'string' && REGEX_URL.test(replacement)
- );
-
- // Proxify the url with one of the replacements
- if (replacementList.length > 0) {
- // get a random replacement site
- const hostKey = url.host + key;
- const replacementIndex = (replacementLast[hostKey] =
- ROTATE_SITE_PROXIFIED && Object.keys(replacementLast).includes(hostKey)
- ? (replacementLast[hostKey] + 1) % replacementList.length // rotate proxies sequentially, if enabled
- : Math.floor(Math.random() * replacementList.length));
-
- const replacementSite = replacementList[replacementIndex];
- const replacementSuffix = suffix || '';
- const replacementRoute = routes?.find(route => route?.regex instanceof RegExp && route.regex.test(url.href));
-
- const failedWhitelist = !!routes && !replacementRoute; // proxy routes whitelist defined but none matched
- let urlProxified = new URL(url);
- if (!!replacementRoute) {
- // Regex replacement
- // TODO: Support replacement function as an alternative to the replacement string route suffix
- try {
- const routeRegex = replacementRoute.regex;
- const routeSuffix = replacementRoute.suffix || '';
- const hrefProxified = url.href.replace(routeRegex, replacementSite + replacementSuffix + routeSuffix);
- urlProxified = new URL(hrefProxified);
- } catch (ex) {
- warn(`Invalid regex replaced URL for ${url}`);
- return null;
- }
- } else if (!failedWhitelist) {
- // Default regex replacement on url host and scheme
- // Skip if whitelisted routes were defined but not matched
- try {
- const hrefProxified = url.href.replace(/(https?:\/\/)(.*?)(\/.*)/, replacementSite + replacementSuffix + '$3');
- urlProxified = new URL(hrefProxified);
- } catch (ex) {
- error(`Invalid default replaced URL for ${url}`);
- return null;
- }
- }
-
- if (url.href !== urlProxified.href) {
- return urlProxified;
- } else if (failedWhitelist) {
- return 'Failed routes whitelist';
- } else {
- error(`No effect proxifying ${url}`);
- return null;
- }
- }
-
- warn(`Missing or invalid replacement URLs for ${url}`);
- return null;
- }
-
- /**
- * Returns matching proxy for the specified link url
- *
- * @param {URL} url
- * @returns {Proxy?} Proxy settings by url, null if not found
- */
- function getProxyForLink(url) {
- // Get the redirect rules from the proxy matching url host
- /** @type {Proxy?} */
- const proxy = getByHost(PROXIES, url.host, true);
-
- // Inherit any redirect rules that were left completely undefined, then inherit
- // any rule properties that were left undefined
- /** @type {Proxy?} */
- let flattenedProxyRule;
- try {
- flattenedProxyRule = flattenInheritance(proxy, PROXIES);
- } catch (ex) {
- error(`Error inheriting proxy rules for ${url}: ${ex}`);
- return false;
- }
-
- if (!!flattenedProxyRule) {
- return flattenedProxyRule;
- }
-
- return null;
- }
-
- /**
- * Returns whether link URL is supported by a listed proxy and not explicitly excluded
- *
- * @param {URL} url
- * @param {string} text
- * @returns {string|null} Reason if link is invalid, null if valid
- */
- function getExclusionReasonForLink(url, text) {
- // Exit if this URL is excluded by path or innertext, or by the current page location
- // Look for exclusion rules on both this host and inherited, if applicable
- /** @type {Exclusion?} */
- const exclusion = getByHost(EXCLUSIONS, url.host, true);
-
- /** @type {Exclusion?} */
- let flattenedExclusionRule;
- try {
- flattenedExclusionRule = flattenInheritance(exclusion, EXCLUSIONS);
- } catch (ex) {
- // Likely inheriting rule that is unexpectedly undefined, probably due to incorrect or nested inheritance
- warn(`Error inheriting exclusion rules for ${url}: ${ex}`);
- return 'Error inheriting exclusion rule';
- }
-
- // With the final exclusion rule set, check if the link url is excluded
- if (!!flattenedExclusionRule) {
- const excludedReason = isLinkExcludedByRule(url, text, flattenedExclusionRule);
- if (!!excludedReason) {
- return excludedReason;
- }
- }
-
- return null;
- }
-
- /**
- * Flatten generic object upwards with inherited data
- *
- * Only goes one level deep, both for inheritance and nested inheritors
- * @typedef {{
- * [keys: string]: Inheritor,
- * inherit?: string
- * }} Inheritor
- *
- * @param {Inheritor} source
- * @param {Object<string, Inheritor>} dictionary
- * @returns {Inheritor?}
- * @throws {Error} Exception on copying with Object.assign()
- */
- function flattenInheritance(source, dictionary) {
- // Avoid destructive shallow copies on `source` or other objects
- let root = source;
- if (!root || !(typeof root === 'object')) return null;
-
- const rootInheritance = dictionary[root.inherit];
- if (!!rootInheritance && typeof rootInheritance === 'object') {
- // Merge two rules into one
- root = Object.assign({}, rootInheritance, root);
- }
-
- // For the final state of inherited nested objects that also inherit,
- // flatten their inheritance as well (non-recursive)
- for (const key of Object.keys(root)) {
- const branch = root[key];
- if (branch?.inherit && dictionary[branch.inherit]) {
- // if inherited object also includes corresponding nested data, merge to
- const branchInheritance = dictionary[branch.inherit]?.[key];
- if (!!branchInheritance && typeof branchInheritance === 'object') {
- root[key] = Object.assign({}, branchInheritance, branch);
- }
- }
- }
-
- // Warn if it appears that recursive inheritance is configured
- // Inheritance beyond the first level of root and property data is not supported
- if (!!rootInheritance?.inherit) {
- warn(`Multi depth recursion hit ${rootInheritance.inherit} but is not supported`);
- }
-
- return root;
- }
-
- /**
- * Validate proxy rules have enough instance or Farside rules to proxify a link
- *
- * @param {Proxy} proxy
- * @returns {boolean} true if valid, false otherwise
- */
- function isProxyValid(proxy) {
- if (!proxy) return false;
-
- const isInstanceRedirectValid =
- !!proxy.redirect && proxy.redirect.replacements && proxy.redirect.replacements.length > 0;
- const isFarsideRedirectValid =
- !!proxy.redirectToFarside &&
- proxy.redirectToFarside.replacements &&
- proxy.redirectToFarside.replacements.length > 0;
- if (!isInstanceRedirectValid && !isFarsideRedirectValid) {
- // neither instance or Farside rules fully defined
- return false;
- }
-
- return true;
- }
-
- /**
- * Handle keydown and keyup event to set modifiers on proxified links
- * @param {KeyboardEvent} e
- * @returns {void}
- */
- function handleModifierKey(e) {
- if (e.type !== 'keyup' && e.type !== 'keydown') {
- error('Invalid modifier key event');
- return;
- }
-
- let isDomChanged = false;
- toggleModifierKeyState(e, KEY_MODIFIED, CLASS_BODY_FARSIDE) && (isDomChanged = true);
- toggleModifierKeyState(e, KEY_BONAFIDE, CLASS_BODY_BONAFIDE) && (isDomChanged = true);
- toggleModifierKeyState(e, KEY_PROXIBITE, CLASS_BODY_BITE) && (isDomChanged = true);
- toggleModifierKeyState(e, KEY_FRAMEBITE, CLASS_BODY_FRAMEBITE) && (isDomChanged = true);
-
- // If DOM state changed, trigger an render update on proxified elements
- if (isDomChanged) {
- setTimeout(() => {
- updateAllProxifiedAnchors(true);
- }, 0);
- }
- }
-
- /**
- * Handle multi-touch gesture event to set modifiers on proxified links, for touch-only devices
- * @param {TouchEvent} e
- * @returns {void}
- */
- function handleModifierGesture(e) {
- if (!(e instanceof TouchEvent)) {
- error('Invalid modifier touch gesture event');
- return;
- }
-
- // only support multi-touch to trigger/reset modifiers
- if (e.touches.length <= 1) {
- return;
- }
-
- let isDomChanged = false;
- toggleModifierGestureState(e, TOUCH_MODIFIED, CLASS_BODY_FARSIDE) && (isDomChanged = true);
- toggleModifierGestureState(e, TOUCH_BONAFIDE, CLASS_BODY_BONAFIDE) && (isDomChanged = true);
- toggleModifierGestureState(e, TOUCH_PROXIBITE, CLASS_BODY_BITE, true) && (isDomChanged = true);
- toggleModifierGestureState(e, TOUCH_FRAMEBITE, CLASS_BODY_FRAMEBITE, true) && (isDomChanged = true);
-
- // If DOM state changed, trigger an render update on proxified elements
- if (isDomChanged) {
- setTimeout(() => {
- updateAllProxifiedAnchors(true);
- }, 0);
- }
- }
-
- /**
- * Toggle a page-wide class by checking KeyboardEvent matches the specified toggleKey
- *
- * @param {KeyboardEvent} e
- * @param {string} toggleKey
- * @param {string} toggleClass Classname to toggle in DOM
- * @returns {boolean} True if changed, false if no change to DOM
- */
- function toggleModifierKeyState(e, toggleKey, toggleClass) {
- if (!(e instanceof KeyboardEvent)) {
- error('Event is not KeyboardEvent');
- return;
- }
-
- if (e.type !== 'keyup' && e.type !== 'keydown') {
- error('Invalid modifier key event');
- return false;
- }
-
- if (e.key === toggleKey) {
- const setModifierOn = e.type === 'keydown';
- const originalState = document.body.classList.contains(toggleClass);
-
- if (setModifierOn) {
- if (!originalState) {
- document.body.classList.add(toggleClass);
- return true;
- }
- } else {
- if (originalState) {
- document.body.classList.remove(toggleClass);
- return true;
- }
- }
- }
-
- return false;
- }
-
- /**
- * Toggle a page-wide class by checking TouchEvent matches the specified touch gesture
- *
- * @param {ToggleEvent} e
- * @param {number} toggleGesture Simple gesture based on number of fingers
- * @param {string} toggleClass Classname to toggle in DOM
- * @param {boolean} isManualOff Optional setting to require manually repeating gesture to toggle off
- * @returns {boolean} True if changed, false if no change to DOM
- */
- function toggleModifierGestureState(e, toggleGesture, toggleClass, isManualOff = false) {
- if (!(e instanceof TouchEvent)) {
- error('Event is not TouchEvent');
- return;
- }
-
- // detect gesture
- // only simple touch count gesture is supported currently
- const gestureTouchCount = toggleGesture;
- const isGestured = gestureTouchCount !== TOUCH_PROXIFIED && e.touches.length === gestureTouchCount;
-
- // handle gesture to toggle modifier
- const originalState = document.body.classList.contains(toggleClass);
- if (isGestured) {
- if (!originalState) {
- document.body.classList.add(toggleClass);
- return true;
- } else if (isManualOff) {
- // gesture off, if supported by this gesture
- document.body.classList.remove(toggleClass);
- return true;
- }
- } else if (!isManualOff) {
- // automatic off, if supported by this gesture
- if (originalState) {
- document.body.classList.remove(toggleClass);
- return true;
- }
- }
-
- return false;
- }
-
- /**
- * Rerender proxified changes to the specified link element
- *
- * @param {HTMLElement} el
- * @param {string} destinationAttr Element link destination attribute name
- * @returns {void}
- */
- function updateProxifiedElement(el, destinationAttr) {
- if (!(el instanceof HTMLElement)) {
- error(`${el} is not an HTMLElement`);
- return;
- }
-
- if (el.hasAttribute(ATTR_PROXIFIED_DENIED)) {
- return; // element has been proxified but denied
- }
-
- if (!el[destinationAttr]) {
- error(`Invalid attribute ${destinationAttr}`);
- return;
- }
-
- if (!isElementProxified(el)) {
- error(`Anchor [${destinationAttr}=${el[destinationAttr]}] is not proxified`);
- return;
- }
-
- // Check DOM modifier state
- // Instead of using internal state, follow what the rendered DOM has
- const isBonafide = document.body.classList.contains(CLASS_BODY_BONAFIDE);
- // attempt to use Farside automatically if direct site is not found
- const isFarside = document.body.classList.contains(CLASS_BODY_FARSIDE) || !el.hasAttribute(ATTR_PROXIFIED_SITE);
-
- // Reset active states before re-applying as necessary
- el.removeAttribute(ATTR_IS_FARSIDE);
- el.classList.remove(CLASS_PROXIFIED_LIVE);
-
- // Update link destination
- // The original, bonafide link takes precedence
- if (isBonafide && el.hasAttribute(ATTR_BONAFIDE_SITE)) {
- if (el[destinationAttr] !== el.getAttribute(ATTR_BONAFIDE_SITE))
- el[destinationAttr] = el.getAttribute(ATTR_BONAFIDE_SITE);
- } else if (isFarside && el.hasAttribute(ATTR_PROXIFIED_FARSIDE)) {
- if (el[destinationAttr] !== el.getAttribute(ATTR_PROXIFIED_FARSIDE))
- el[destinationAttr] = el.getAttribute(ATTR_PROXIFIED_FARSIDE);
- el.setAttribute(ATTR_IS_FARSIDE, true);
- el.classList.add(CLASS_PROXIFIED_LIVE);
- } else if (el.hasAttribute(ATTR_PROXIFIED_SITE)) {
- if (el[destinationAttr] !== el.getAttribute(ATTR_PROXIFIED_SITE))
- el[destinationAttr] = el.getAttribute(ATTR_PROXIFIED_SITE);
- el.classList.add(CLASS_PROXIFIED_LIVE);
- }
-
- // Optional: Re-render element to encourage browsers to reflect proxified link
- // TODO: Consider aria-live for screenreaders
- if (ENABLE_RERENDER) {
- const isElSelected = document.activeElement == el || el.contains(document.activeElement);
- const elStyleDisplay = el.style.display;
-
- // hide the element and replace with a placeholder clone to prevent 1-frame flash
- const CLONE_CLASSNAME = 'proxi-tied';
- const elClone = el.cloneNode(true); // deep clone to preserve DOM, at cost of perf
- elClone.classList.add(CLONE_CLASSNAME);
- elClone.style = 'color: black !important; background: black !important;';
- if (!el.parentNode?.querySelector(`.${CLONE_CLASSNAME}`) && elStyleDisplay !== 'none') {
- el.parentNode?.insertBefore(elClone, el.nextSibling);
-
- if (isElSelected) el.blur();
- el.style.display = 'none';
-
- setTimeout(() => {
- // show and select the original element
- el.style.display = elStyleDisplay;
- if (isElSelected) el.focus();
-
- if (elClone.parentNode === el.parentNode) el.parentNode?.removeChild(elClone);
- }, 0);
- }
- }
- }
-
- /**
- * Rerender state to all proxified anchor elements in the document
- *
- * @param {boolean} doHoveredOnly Only render proxified elements that are being hovered
- * @returns {void}
- */
- function updateAllProxifiedAnchors(doHoveredOnly) {
- const proxifiedEls = document.querySelectorAll(
- `a.${CLASS_PROXIFIED}${doHoveredOnly ? ':is(:hover, :focus-within)' : ''}`
- );
- for (const proxifiedEl of proxifiedEls) {
- updateProxifiedElement(proxifiedEl, 'href');
- }
- }
-
- /**
- * Add proxified link styles to indicate functionality and readiness
- *
- * Highlight proxied links yellow
- * Highlight Farside-redirected links green
- */
- // TODO: Manually apply inline element styles to avoid style-src CSP
- const isHoverOnly = true; // toggle styling on when hovering on link
- const _important = true ? '!important' : ''; // toggle overriding page styles as much as possible
- // Style anchor element and children under anchor element
- const selectorAnchor = isHoverOnly
- ? `a.${CLASS_PROXIFIED_LIVE}:is(:hover, :focus-within)`
- : `a.${CLASS_PROXIFIED_LIVE}`;
- // add additional styles at higher specificity to child containers with non-empty content
- const selectorChildren = ':is(p, span, h1, h2, h3, h4, h5, h6, label, div, button):not(:empty)';
- addPageStyle(
- `${selectorAnchor} { \
- color: black ${_important}; \
- background-color: yellow ${_important}; \
- text-shadow: none ${_important}; \
- \
- font-style: oblique ${_important}; \
- font-weight: bold ${_important}; \
- \
- ${selectorChildren} { \
- color: yellow ${_important}; \
- background-color: black ${_important}; \
- text-shadow: none ${_important}; \
- \
- font-style: oblique ${_important}; \
- font-weight: bold ${_important}; \
- } \
- :is(img) { \
- filter: sepia(1) hue-rotate(20deg) contrast(1.25) brightness(1.25); ${_important}; \
- }\
- }`
- );
- // Add optional highlighter padding at lowest specificity :where()
- addPageStyle(
- `:where(${selectorAnchor}) { \
- // padding: 0 0.3em; \
- \
- ${selectorChildren} { \
- padding: 0 0.3em; \
- } \
- }`
- );
- // Change colors when links redirect through Farside
- addPageStyle(
- `${selectorAnchor}[${ATTR_IS_FARSIDE}="true"] { \
- color: yellowgreen ${_important}; \
- background-color: black ${_important}; \
- \
- ${selectorChildren} { \
- color: black ${_important}; \
- background-color: yellowgreen ${_important}; \
- } \
- :is(img) { \
- filter: invert(1) sepia(1) hue-rotate(45deg) contrast(1.25) brightness(1.25) ${_important}; \
- } \
- }`
- );
- // Highlight all iframes when frame bite is enabled
- addPageStyle(
- `body.proxi-bite.proxi-framebite { \
- iframe { \
- border: yellow solid 0.3em ${_important}; \
- } \
- iframe[${ATTR_IS_FARSIDE}="true"] { \
- border-color: yellowgreen ${_important}; \
- } \
- }`
- );
-
- /**
- * Insert a page-level style
- * Adds a document stylesheet to write to when needed
- *
- * @param {string} css
- * @returns {void}
- */
- function addPageStyle(css) {
- const style =
- document.getElementById(STYLE_HIGHLIGHT_ID) ||
- (function () {
- const style = document.createElement('style');
- style.id = STYLE_HIGHLIGHT_ID;
- document.head.appendChild(style);
- return style;
- })();
-
- const sheet = style.sheet;
- try {
- sheet.insertRule(css, (sheet.rules || sheet.cssRules || []).length);
- } catch (ex) {
- // Likely stylesheet is null from failing to add to DOM
- // Continue on, even without styling
- warn(`Failed to apply style: ${ex}`);
- }
- }
-
- /**
- * Check given link url and text against provided exclusion rule
- *
- * @param {URL} url
- * @param {string} text
- * @param {Exclusion} rule
- * @returns {string?} First found exclusion reason, null if false
- */
- function isLinkExcludedByRule(url, text, rule) {
- if (!rule) {
- error(`Invalid flattened exclusion rule for ${url}`);
- return 'Invalid flatted exclusion rule';
- }
-
- // Return reason if excluding the current page that this link is on
- // Check first to short circuit link processing on pages proxifying is excluded
- // Results are memoized by page after being processed once, manually invalidate `_excludedOn` if needed
- const currentDirection = `${window.location.href}=>${url.host}`;
- if (!!_excludedOn[currentDirection]) {
- return _excludedOn[currentDirection];
- }
- if (!!rule.excludeOn && _excludedOn[currentDirection] === undefined) {
- /** @type {URL | null} */
- let currentUrl = null;
- try {
- currentUrl = new URL(currentDirection);
- } catch (ex) {
- // unexpected failure on parsing current URL, error and continue
- error(`Error parsing page URL: ${ex}`);
- }
-
- if (!!rule.excludeOn.matchingPath) {
- for (const m of rule.excludeOn.matchingPath) {
- if (!!currentUrl.pathname.match(m)) {
- _excludedOn[currentDirection] = `excludeOn.matchingPath[${m}]`;
- }
- }
- }
-
- if (!!rule.excludeOn.matchingHost) {
- for (const m of rule.excludeOn.matchingHost) {
- if (!!currentUrl.host.match(m)) {
- _excludedOn[currentDirection] = `excludeOn.matchingHost[${m}]`;
- }
- }
- }
-
- if (!!rule.excludeOn.matchingUrl) {
- for (const m of rule.excludeOn.matchingUrl) {
- if (!!currentUrl.href.match(m)) {
- _excludedOn[currentDirection] = `excludeOn.matchingUrl[${m}]`;
- }
- }
- }
-
- if (!!rule.excludeOn.matchingBody) {
- for (const m of rule.excludeOn.matchingBody) {
- if (document.getElementsByTagName('body')[0].innerHTML.match(m)) {
- _excludedOn[currentDirection] = `excludeOn.matchingBody[${m}]`;
- }
- }
- }
-
- if (!!rule.excludeOn.matchingHead) {
- // clean before searching
- // if the userscript extension mounts to head, may false positive on the userscript itself
- if (_documentHeadCleaned === undefined) {
- const documentHeadTemp = document.createElement('head');
- documentHeadTemp.innerHTML = document.getElementsByTagName('head')[0]?.innerHTML;
- let s;
- for (const scriptNodes = documentHeadTemp.getElementsByTagName('script'); (s = scriptNodes[0]); ) {
- // cut down scriptNodes queue
- s.parentNode.removeChild(s);
- }
- _documentHeadCleaned = documentHeadTemp.innerHTML;
- }
-
- for (const m of rule.excludeOn.matchingHead) {
- if (_documentHeadCleaned.match(m)) {
- _excludedOn[currentDirection] = `excludeOn.matchingHead[${m}]`;
- }
- }
- }
-
- // process this page url once even if no matches found for this page
- // return any non-nullish match found, or continue to find other link exclusions
- if (_excludedOn[currentDirection] === undefined) {
- _excludedOn[currentDirection] = null;
- } else if (!!_excludedOn[currentDirection]) {
- return _excludedOn[currentDirection];
- }
- }
-
- // Return reason if excluding this link
- if (!!rule.excludeLinks) {
- if (!!rule.excludeLinks.matchingPath) {
- for (const m of rule.excludeLinks.matchingPath) {
- if (!!url.pathname.match(m)) return `excludeLinks.matchingPath[${m}]`;
- }
- }
-
- if (!!rule.excludeLinks.matchingHost) {
- for (const m of rule.excludeLinks.matchingHost) {
- if (!!url.host.match(m)) return `excludeLinks.matchingHost[${m}]`;
- }
- }
-
- if (!!rule.excludeLinks.matchingUrl) {
- for (const m of rule.excludeLinks.matchingUrl) {
- if (!!url.href.match(m)) return `excludeLinks.matchingUrl[${m}]`;
- }
- }
-
- if (!!rule.excludeLinks.matchingText) {
- for (const m of rule.excludeLinks.matchingText) {
- if (!!text.match(m)) return `excludeLinks.matchingText[${m}]`;
- }
- }
- }
- }
- const _excludedOn = {}; // memoized excludeOn results
- let _documentHeadCleaned; // memoized cleaned <head>
-
- /**
- * Check if element has been fully processed already
- * @param {HTMLElement} el
- * @returns {boolean}
- */
- function isElementProxified(el) {
- // bonafide attr signals completed proxification
- // successful: all proxified attrs set
- // denied: denied + bonafide attrs set
- const elBonafide = el?.getAttribute(ATTR_BONAFIDE_SITE);
-
- // proxification is outdated if current link does not match any stored destination
- // this can occur when the same link el is reused dynamically by the site scripts and must be invalidated
- const linkDest = getElementDest(el);
- const isUpToDate =
- linkDest === elBonafide ||
- linkDest === el?.getAttribute(ATTR_PROXIFIED_SITE) ||
- linkDest === el?.getAttribute(ATTR_PROXIFIED_FARSIDE);
-
- return elBonafide && isUpToDate;
- }
-
- /**
- * Get the link destination
- * @param {HTMLElement} el
- * @returns {string?}
- */
- function getElementDest(el) {
- switch (el.constructor) {
- case HTMLAnchorElement:
- return el.href;
- case HTMLIFrameElement:
- return el.src;
- default:
- return null;
- }
- }
-
- /**
- * Returns the dictionary entry for the specified host, agnostic to WWW
- * Take explicit host match if found, but optionally try searching for SLD+TLD only (strip subdomains)
- *
- * @param {Object<string, T?>} dictionary
- * @param {string} host
- * @param {boolean} allowFuzzy
- * @returns {T?}
- */
- function getByHost(dictionary, host, allowFuzzy) {
- if (typeof dictionary !== 'object') {
- error('Invalid dictionary used to look up host');
- return null;
- }
-
- // Find the explicit entry for the host key
- const explicit = dictionary[host];
- if (!!explicit) {
- return explicit;
- }
-
- // Strip WWW and try again
- const agnosticWWW = dictionary[host.replace(/^www\./, '')];
- if (!!agnosticWWW) {
- return agnosticWWW;
- }
-
- // Non-WWW subdomains in hostname may be causing misses
- // Too many TLDs to handle easily, so allow fuzzy matching on SLD+TLD if enabled
- // Search for dictionary keys matching to the end of host and try the first hit
- if (allowFuzzy) {
- // try any dictionary key that has opted into fuzzy matching
- const candidateHostKey = Object.keys(dictionary)
- .filter(hostKey => !!dictionary[hostKey].allowFuzzy)
- .find(hostKey => {
- // try to fit the key to the end of the matching host
- const SLDTLDRegex = new RegExp(escapeRegExp(hostKey) + '$');
- return SLDTLDRegex.test(host);
- });
-
- const agnosticSubdomain = dictionary[candidateHostKey];
- if (!!agnosticSubdomain) {
- return agnosticSubdomain;
- }
- }
-
- return null;
- }
-
- /**
- * Escape regex special characters in a string
- * TC39 X-standard
- *
- * @param {string} string
- * @returns {string}
- */
- function escapeRegExp(string) {
- return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
- }
-
- /**
- * Print to console as error
- *
- * @param {...any} data
- */
- function error(...data) {
- if (DEBUG) {
- console.error(...data);
- }
- }
-
- /**
- * Print to console as warning
- *
- * @param {...any} data
- */
- function warn(...data) {
- if (DEBUG) {
- console.warn(...data);
- }
- }