privacy-redirect/src/pages/background/background.js

609 lines
19 KiB
JavaScript
Raw Normal View History

"use strict";
2019-10-07 13:59:31 +02:00
import commonHelper from "../../assets/javascripts/helpers/common.js";
import twitterHelper from "../../assets/javascripts/helpers/twitter.js";
import youtubeHelper from "../../assets/javascripts/helpers/youtube.js";
import instagramHelper from "../../assets/javascripts/helpers/instagram.js";
import mapsHelper from "../../assets/javascripts/helpers/google-maps.js";
import redditHelper from "../../assets/javascripts/helpers/reddit.js";
import searchHelper from "../../assets/javascripts/helpers/google-search.js";
2021-02-25 16:20:23 +01:00
import googleTranslateHelper from "../../assets/javascripts/helpers/google-translate.js";
const nitterInstances = twitterHelper.redirects;
const twitterDomains = twitterHelper.targets;
const youtubeDomains = youtubeHelper.targets;
const invidiousInstances = youtubeHelper.redirects;
const instagramDomains = instagramHelper.targets;
const bibliogramInstances = instagramHelper.redirects;
const instagramReservedPaths = instagramHelper.reservedPaths;
const bibliogramBypassPaths = instagramHelper.bypassPaths;
const osmDefault = mapsHelper.redirects[0];
const googleMapsRegex = mapsHelper.targets;
const mapCentreRegex = mapsHelper.mapCentreRegex;
const dataLatLngRegex = mapsHelper.dataLatLngRegex;
const placeRegex = mapsHelper.placeRegex;
const travelModes = mapsHelper.travelModes;
const layers = mapsHelper.layers;
const redditInstances = redditHelper.redirects;
const redditDomains = redditHelper.targets;
const redditBypassPaths = redditHelper.bypassPaths;
const redditDefault = redditHelper.redirects[0];
const googleSearchRegex = searchHelper.targets;
const searchEngineInstances = searchHelper.redirects;
2021-02-25 16:20:23 +01:00
const simplyTranslateInstances = googleTranslateHelper.redirects;
const simplyTranslateDefault = simplyTranslateInstances[0];
const googleTranslateDomains = googleTranslateHelper.targets;
2019-10-07 13:59:31 +02:00
let disableNitter;
let disableInvidious;
2020-02-01 04:17:51 +01:00
let disableBibliogram;
2020-02-23 06:19:32 +01:00
let disableOsm;
let disableReddit;
2020-11-08 15:50:55 +01:00
let disableSearchEngine;
2021-02-25 16:20:23 +01:00
let disableSimplyTranslate;
let nitterInstance;
let invidiousInstance;
let bibliogramInstance;
let osmInstance;
let redditInstance;
let searchEngineInstance;
2021-02-25 16:20:23 +01:00
let simplyTranslateInstance;
let alwaysProxy;
let onlyEmbeddedVideo;
let videoQuality;
let invidiousDarkMode;
let invidiousVolume;
let invidiousPlayerStyle;
let invidiousSubtitles;
let invidiousAutoplay;
let useFreeTube;
let nitterRandomPool;
let invidiousRandomPool;
let bibliogramRandomPool;
let exceptions;
2019-10-07 13:59:31 +02:00
2020-02-24 04:19:56 +01:00
window.browser = window.browser || window.chrome;
browser.storage.sync.get(
2020-02-01 04:17:51 +01:00
[
"nitterInstance",
"invidiousInstance",
"bibliogramInstance",
"osmInstance",
"redditInstance",
"searchEngineInstance",
2021-02-25 16:20:23 +01:00
"simplyTranslateInstance",
"disableNitter",
"disableInvidious",
"disableBibliogram",
"disableOsm",
"disableReddit",
2020-11-08 15:50:55 +01:00
"disableSearchEngine",
2021-02-25 16:20:23 +01:00
"disableSimplyTranslate",
"alwaysProxy",
"onlyEmbeddedVideo",
"videoQuality",
"invidiousDarkMode",
"invidiousVolume",
"invidiousPlayerStyle",
"invidiousSubtitles",
"invidiousAutoplay",
"useFreeTube",
"nitterRandomPool",
"invidiousRandomPool",
"bibliogramRandomPool",
"exceptions",
2020-02-01 04:17:51 +01:00
],
(result) => {
nitterInstance = result.nitterInstance;
invidiousInstance = result.invidiousInstance;
bibliogramInstance = result.bibliogramInstance;
osmInstance = result.osmInstance || osmDefault;
redditInstance = result.redditInstance || redditDefault;
searchEngineInstance = result.searchEngineInstance;
2021-04-16 02:45:15 +02:00
simplyTranslateInstance =
result.simplyTranslateInstance || simplyTranslateDefault;
2019-10-07 13:59:31 +02:00
disableNitter = result.disableNitter;
disableInvidious = result.disableInvidious;
2020-02-01 04:17:51 +01:00
disableBibliogram = result.disableBibliogram;
2020-02-23 06:19:32 +01:00
disableOsm = result.disableOsm;
disableReddit = result.disableReddit;
2020-11-08 15:50:55 +01:00
disableSearchEngine = result.disableSearchEngine;
2021-02-25 16:20:23 +01:00
disableSimplyTranslate = result.disableSimplyTranslate;
alwaysProxy = result.alwaysProxy;
onlyEmbeddedVideo = result.onlyEmbeddedVideo;
videoQuality = result.videoQuality;
invidiousDarkMode = result.invidiousDarkMode;
exceptions = result.exceptions
? result.exceptions.map((e) => {
return new RegExp(e);
})
: [];
invidiousVolume = result.invidiousVolume;
invidiousPlayerStyle = result.invidiousPlayerStyle;
invidiousSubtitles = result.invidiousSubtitles || "";
invidiousAutoplay = result.invidiousAutoplay;
useFreeTube = result.useFreeTube;
nitterRandomPool = result.nitterRandomPool
? result.nitterRandomPool.split(",")
: commonHelper.filterInstances(nitterInstances);
invidiousRandomPool = result.invidiousRandomPool
? result.invidiousRandomPool.split(",")
: commonHelper.filterInstances(invidiousInstances);
bibliogramRandomPool = result.bibliogramRandomPool
? result.bibliogramRandomPool.split(",")
: commonHelper.filterInstances(bibliogramInstances);
2019-10-07 13:59:31 +02:00
}
);
browser.storage.onChanged.addListener((changes) => {
if ("nitterInstance" in changes) {
2020-09-01 09:26:23 +02:00
nitterInstance = changes.nitterInstance.newValue;
2019-10-07 13:59:31 +02:00
}
if ("invidiousInstance" in changes) {
2020-09-01 09:26:23 +02:00
invidiousInstance = changes.invidiousInstance.newValue;
2019-10-07 13:59:31 +02:00
}
if ("bibliogramInstance" in changes) {
2020-09-01 09:26:23 +02:00
bibliogramInstance = changes.bibliogramInstance.newValue;
2020-02-01 04:17:51 +01:00
}
if ("osmInstance" in changes) {
2020-02-23 06:19:32 +01:00
osmInstance = changes.osmInstance.newValue || osmDefault;
}
2021-02-25 16:20:23 +01:00
if ("simplyTranslateInstance" in changes) {
2021-04-16 02:45:15 +02:00
simplyTranslateInstance =
changes.simplyTranslateInstance.newValue || simplyTranslateDefault;
2021-02-25 16:20:23 +01:00
}
if ("redditInstance" in changes) {
redditInstance = changes.redditInstance.newValue || redditDefault;
}
if ("searchEngineInstance" in changes) {
searchEngineInstance = changes.searchEngineInstance.newValue;
}
if ("disableNitter" in changes) {
2019-10-07 13:59:31 +02:00
disableNitter = changes.disableNitter.newValue;
}
if ("disableInvidious" in changes) {
2019-10-07 13:59:31 +02:00
disableInvidious = changes.disableInvidious.newValue;
}
if ("disableBibliogram" in changes) {
2020-02-01 04:17:51 +01:00
disableBibliogram = changes.disableBibliogram.newValue;
}
if ("disableOsm" in changes) {
2020-02-23 06:19:32 +01:00
disableOsm = changes.disableOsm.newValue;
}
if ("disableReddit" in changes) {
disableReddit = changes.disableReddit.newValue;
}
2020-11-08 15:50:55 +01:00
if ("disableSearchEngine" in changes) {
disableSearchEngine = changes.disableSearchEngine.newValue;
}
2021-02-25 16:20:23 +01:00
if ("disableSimplyTranslate" in changes) {
disableSimplyTranslate = changes.disableSimplyTranslate.newValue;
}
if ("alwaysProxy" in changes) {
alwaysProxy = changes.alwaysProxy.newValue;
}
if ("onlyEmbeddedVideo" in changes) {
onlyEmbeddedVideo = changes.onlyEmbeddedVideo.newValue;
}
if ("videoQuality" in changes) {
videoQuality = changes.videoQuality.newValue;
}
if ("invidiousDarkMode" in changes) {
invidiousDarkMode = changes.invidiousDarkMode.newValue;
}
if ("invidiousVolume" in changes) {
invidiousVolume = changes.invidiousVolume.newValue;
}
if ("invidiousPlayerStyle" in changes) {
invidiousPlayerStyle = changes.invidiousPlayerStyle.newValue;
}
if ("invidiousSubtitles" in changes) {
invidiousSubtitles = changes.invidiousSubtitles.newValue;
}
if ("invidiousAutoplay" in changes) {
invidiousAutoplay = changes.invidiousAutoplay.newValue;
}
if ("useFreeTube" in changes) {
useFreeTube = changes.useFreeTube.newValue;
}
if ("nitterRandomPool" in changes) {
nitterRandomPool = changes.nitterRandomPool.newValue.split(",");
}
if ("invidiousRandomPool" in changes) {
invidiousRandomPool = changes.invidiousRandomPool.newValue.split(",");
}
if ("bibliogramRandomPool" in changes) {
bibliogramRandomPool = changes.bibliogramRandomPool.newValue.split(",");
}
if ("exceptions" in changes) {
exceptions = changes.exceptions.newValue.map((e) => {
return new RegExp(e);
});
2020-06-07 14:08:15 +02:00
}
2019-10-07 13:59:31 +02:00
});
2019-09-20 12:45:58 +02:00
function isException(url, initiator) {
return (
exceptions.some((regex) => regex.test(url.href)) ||
(initiator && exceptions.some((regex) => regex.test(initiator.href)))
);
}
2020-06-07 14:08:15 +02:00
function isFirefox() {
return typeof InstallTrigger !== "undefined";
2020-06-07 14:08:15 +02:00
}
function redirectYouTube(url, initiator, type) {
if (disableInvidious || isException(url, initiator)) {
return null;
}
if (
initiator &&
(initiator.origin === invidiousInstance ||
2020-09-01 09:26:23 +02:00
invidiousInstances.includes(initiator.origin) ||
youtubeDomains.includes(initiator.host))
) {
return null;
}
if (url.pathname.match(/iframe_api/) || url.pathname.match(/www-widgetapi/)) {
// Don't redirect YouTube Player API.
return null;
}
if (url.host.split(".")[0] === "studio") {
// Avoid redirecting `studio.youtube.com`
return null;
}
if (onlyEmbeddedVideo && type !== "sub_frame") {
2020-06-07 14:08:15 +02:00
return null;
}
if (useFreeTube && type === "main_frame") {
return `freetube://${url}`;
}
2020-06-07 14:08:15 +02:00
// Apply settings
if (alwaysProxy) {
url.searchParams.append("local", true);
}
if (videoQuality) {
url.searchParams.append("quality", videoQuality);
}
if (invidiousDarkMode) {
url.searchParams.append("dark_mode", invidiousDarkMode);
}
if (invidiousVolume) {
url.searchParams.append("volume", invidiousVolume);
}
if (invidiousPlayerStyle) {
url.searchParams.append("player_style", invidiousPlayerStyle);
}
if (invidiousSubtitles) {
url.searchParams.append("subtitles", invidiousSubtitles);
}
url.searchParams.append("autoplay", invidiousAutoplay ? 1 : 0);
return `${
invidiousInstance || commonHelper.getRandomInstance(invidiousRandomPool)
}${url.pathname.replace("/shorts", "")}${url.search}`;
}
function redirectTwitter(url, initiator) {
if (disableNitter || isException(url, initiator)) {
return null;
}
if (url.pathname.includes("/home")) {
2020-06-07 14:08:15 +02:00
return null;
}
if (
isFirefox() &&
initiator &&
(initiator.origin === nitterInstance ||
2020-09-01 09:26:23 +02:00
nitterInstances.includes(initiator.origin) ||
twitterDomains.includes(initiator.host))
) {
browser.storage.sync.set({
redirectBypassFlag: true,
});
return null;
}
if (url.host.split(".")[0] === "pbs") {
2020-09-01 09:26:23 +02:00
return `${
nitterInstance || commonHelper.getRandomInstance(nitterRandomPool)
2020-09-01 09:26:23 +02:00
}/pic/${encodeURIComponent(url.href)}`;
} else if (url.host.split(".")[0] === "video") {
2020-09-01 09:26:23 +02:00
return `${
nitterInstance || commonHelper.getRandomInstance(nitterRandomPool)
2020-09-01 09:26:23 +02:00
}/gif/${encodeURIComponent(url.href)}`;
} else if (url.pathname.includes("tweets")) {
2020-09-01 09:26:23 +02:00
return `${
nitterInstance || commonHelper.getRandomInstance(nitterRandomPool)
2020-09-01 09:26:23 +02:00
}${url.pathname.replace("/tweets", "")}${url.search}`;
} else {
return `${
nitterInstance || commonHelper.getRandomInstance(nitterRandomPool)
}${url.pathname}${url.search}`;
}
}
2020-04-09 11:37:56 +02:00
function redirectInstagram(url, initiator, type) {
if (disableBibliogram || isException(url, initiator)) {
return null;
}
2020-04-09 11:37:56 +02:00
// Do not redirect Bibliogram view on Instagram links
if (
initiator &&
(initiator.origin === bibliogramInstance ||
2020-09-01 09:26:23 +02:00
bibliogramInstances.includes(initiator.origin) ||
instagramDomains.includes(initiator.host))
) {
return null;
}
2020-04-09 11:37:56 +02:00
// Do not redirect /accounts, /embeds.js, or anything other than main_frame
if (type !== "main_frame" || url.pathname.match(bibliogramBypassPaths)) {
2020-04-09 11:37:56 +02:00
return null;
}
if (
url.pathname === "/" ||
instagramReservedPaths.includes(url.pathname.split("/")[1])
) {
return `${
bibliogramInstance || commonHelper.getRandomInstance(bibliogramRandomPool)
}${url.pathname}${url.search}`;
2020-02-01 04:17:51 +01:00
} else {
// Likely a user profile, redirect to '/u/...'
return `${
bibliogramInstance || commonHelper.getRandomInstance(bibliogramRandomPool)
}/u${url.pathname}${url.search}`;
2020-02-01 04:17:51 +01:00
}
}
function redirectGoogleMaps(url, initiator) {
if (disableOsm || isException(url, initiator)) {
return null;
}
let redirect;
let mapCentre = "";
let params = "";
// Set map centre if present
if (url.pathname.match(mapCentreRegex)) {
const [, lat, lon, zoom] = url.pathname.match(mapCentreRegex);
2020-02-24 09:22:10 +01:00
mapCentre = `#map=${zoom}/${lat}/${lon}`;
} else if (url.search.includes("center=")) {
const [lat, lon] = url.searchParams.get("center").split(",");
mapCentre = `#map=${url.searchParams.get("zoom") || "17"}/${lat}/${lon}`;
// Set default zoom if mapCentre not present
} else {
params = "&zoom=17";
2020-02-23 06:19:32 +01:00
}
// Set map layer
params = `${params}&layers=${
layers[url.searchParams.get("layer")] || layers["none"]
}`;
// Handle Google Maps Embed API
if (url.pathname.includes("/embed")) {
let query = "";
if (url.searchParams.has("q")) {
query = url.searchParams.get("q");
} else if (url.searchParams.has("query")) {
query = url.searchParams.has("query");
} else if (url.searchParams.has("pb")) {
try {
query = url.searchParams.get("pb").split(/!2s(.*?)!/)[1];
} catch (error) {
console.error(error);
// Unable to find map marker in URL.
}
}
let marker, bbox;
mapsHelper.addressToLatLng(query, (coords, boundingbox) => {
marker = coords;
bbox = boundingbox;
});
redirect = `${osmInstance}/export/embed.html?bbox=${bbox}&layer=mapnik&marker=${marker}`;
// Handle Google Maps Directions
} else if (url.pathname.includes("/dir")) {
const travelMode =
travelModes[url.searchParams.get("travelmode")] || travelModes["driving"];
let origin;
mapsHelper.addressToLatLng(url.searchParams.get("origin"), (coords) => {
origin = coords;
});
let destination;
mapsHelper.addressToLatLng(
url.searchParams.get("destination"),
(coords) => {
destination = coords;
}
);
redirect = `${osmInstance}/directions?engine=${travelMode}&route=${origin}%3B${destination}${mapCentre}${params}`;
// Get marker from data attribute
} else if (
url.pathname.includes("data=") &&
url.pathname.match(dataLatLngRegex)
) {
2020-02-24 04:19:56 +01:00
const [mlat, mlon] = url.pathname.match(dataLatLngRegex);
redirect = `${osmInstance}/?mlat=${mlat.replace(
"!3d",
""
)}&mlon=${mlon.replace("!4d", "")}${mapCentre}${params}`;
// Get marker from ll param
} else if (url.searchParams.has("ll")) {
const [mlat, mlon] = url.searchParams.get("ll").split(",");
redirect = `${osmInstance}/?mlat=${mlat}&mlon=${mlon}${mapCentre}${params}`;
// Get marker from viewpoint param.
} else if (url.searchParams.has("viewpoint")) {
const [mlat, mlon] = url.searchParams.get("viewpoint").split(",");
redirect = `${osmInstance}/?mlat=${mlat}&mlon=${mlon}${mapCentre}${params}`;
// Use query as search if present.
2020-02-24 04:19:56 +01:00
} else {
let query;
if (url.searchParams.has("q")) {
query = url.searchParams.get("q");
} else if (url.searchParams.has("query")) {
query = url.searchParams.get("query");
} else if (url.pathname.match(placeRegex)) {
query = url.pathname.match(placeRegex)[1];
}
redirect = `${osmInstance}/${query ? "search?query=" + query : ""}${
mapCentre || "#"
}${params}`;
}
return redirect;
2020-02-23 06:19:32 +01:00
}
function redirectReddit(url, initiator, type) {
if (disableReddit || isException(url, initiator)) {
return null;
}
// Do not redirect when already on the selected view
if (
(initiator && initiator.origin === redditInstance) ||
url.origin === redditInstance
) {
return null;
}
2020-09-06 07:32:23 +02:00
// Do not redirect exclusions nor anything other than main_frame
if (type !== "main_frame" || url.pathname.match(redditBypassPaths)) {
return null;
}
if (url.host === "i.redd.it") {
if (redditInstance.includes("libredd")) {
return `${redditInstance}/img${url.pathname}${url.search}`;
} else if (redditInstance.includes("teddit")) {
// As of 2021-04-09, redirects for teddit images are nontrivial:
// - navigating to the image before ever navigating to its page causes
// 404 error (probably needs fix on teddit project)
// - some image links on teddit are very different
// Therefore, don't support redirecting image links for teddit.
return null;
} else {
return null;
}
}
return `${redditInstance}${url.pathname}${url.search}`;
}
2020-11-08 15:50:55 +01:00
function redirectSearchEngine(url, initiator) {
if (disableSearchEngine || isException(url, initiator)) {
return null;
}
2020-11-22 08:41:43 +01:00
const searchEngine =
searchEngineInstance ||
commonHelper.getRandomInstance(searchEngineInstances);
2020-11-22 08:41:43 +01:00
let search = "";
url.search
.slice(1)
.split("&")
.forEach(function (input) {
if (input.startsWith("q=")) search = input;
});
return `${searchEngine.link}${searchEngine.q}?${search}`;
2020-11-08 15:50:55 +01:00
}
2021-02-25 16:20:23 +01:00
function redirectGoogleTranslate(url, initiator) {
if (disableSimplyTranslate || isException(url, initiator)) {
return null;
}
return `${simplyTranslateInstance}/${url.search}`;
}
2020-02-24 04:19:56 +01:00
browser.webRequest.onBeforeRequest.addListener(
(details) => {
2020-02-01 04:17:51 +01:00
const url = new URL(details.url);
let initiator;
if (details.originUrl) {
initiator = new URL(details.originUrl);
} else if (details.initiator) {
initiator = new URL(details.initiator);
}
2020-01-14 10:48:37 +01:00
let redirect;
if (youtubeDomains.includes(url.host)) {
redirect = {
redirectUrl: redirectYouTube(url, initiator, details.type),
};
} else if (twitterDomains.includes(url.host)) {
redirect = {
redirectUrl: redirectTwitter(url, initiator),
};
2020-04-09 11:37:56 +02:00
} else if (instagramDomains.includes(url.host)) {
redirect = {
redirectUrl: redirectInstagram(url, initiator, details.type),
};
2020-02-23 06:19:32 +01:00
} else if (url.href.match(googleMapsRegex)) {
redirect = {
redirectUrl: redirectGoogleMaps(url, initiator),
};
} else if (
redditDomains.includes(url.host) ||
redditInstances.includes(url.origin)
) {
redirect = {
redirectUrl: redirectReddit(url, initiator, details.type),
};
2020-11-11 01:01:33 +01:00
} else if (url.href.match(googleSearchRegex)) {
2020-11-08 15:50:55 +01:00
redirect = {
redirectUrl: redirectSearchEngine(url, initiator),
};
2021-02-25 16:20:23 +01:00
} else if (googleTranslateDomains.includes(url.host)) {
redirect = {
redirectUrl: redirectGoogleTranslate(url, initiator),
2021-04-16 02:45:15 +02:00
};
2019-09-20 12:45:58 +02:00
}
if (redirect && redirect.redirectUrl) {
console.info(
"Redirecting",
`"${url.href}"`,
"=>",
`"${redirect.redirectUrl}"`
2020-02-01 04:17:51 +01:00
);
console.info("Details", details);
2020-01-14 10:48:37 +01:00
}
return redirect;
2019-09-20 12:45:58 +02:00
},
{
urls: ["<all_urls>"],
2019-09-20 12:45:58 +02:00
},
["blocking"]
2019-09-20 12:45:58 +02:00
);
2020-04-09 11:37:56 +02:00
browser.runtime.onInstalled.addListener((details) => {
2021-04-16 02:45:15 +02:00
browser.storage.sync.get(
["disableSearchEngine", "disableSimplyTranslate"],
(result) => {
if (result.disableSearchEngine === undefined) {
browser.storage.sync.set({
disableSearchEngine: true,
});
}
if (result.disableSimplyTranslate === undefined) {
browser.storage.sync.set({
disableSimplyTranslate: true,
});
}
}
2021-04-16 02:45:15 +02:00
);
2020-09-01 09:26:23 +02:00
if (details.reason === "update") {
browser.storage.sync.get(
["whitelist", "exceptions", "invidiousInstance", "disableSearchEngine"],
(result) => {
if (result.whitelist) {
let whitelist = result.whitelist.map((e) =>
e.replace(/[-\/\\^$*+?.()|[\]{}]/g, "\\$&")
);
browser.storage.sync.set({
exceptions: result.exceptions.concat(whitelist),
whitelist: null,
});
}
2020-09-01 09:26:23 +02:00
if (result.invidiousInstance === "https://invidio.us") {
browser.storage.sync.set({
2020-09-01 09:26:23 +02:00
invidiousInstance: null,
});
}
}
);
2020-04-09 11:37:56 +02:00
}
});