window.browser = window.browser || window.chrome;
import twitterHelper from "./twitter.js";
import youtubeHelper from "./youtube/youtube.js";
import instagramHelper from "./instagram.js";
import mediumHelper from "./medium.js";
import redditHelper from "./reddit.js";
import searchHelper from "./search.js";
import translateHelper from "./translate/translate.js";
import wikipediaHelper from "./wikipedia.js";
import peertubeHelper from "./peertube.js";
import lbryHelper from "./lbry.js";
import sendTargetsHelper from "./sendTargets.js";
import tiktokHelper from "./tiktok.js";
import quoraHelper from "./quora.js"
import libremdbHelper from "./imdb.js";
import imgurHelper from "./imgur.js";
import localise from './localise.js'
function getRandomInstance(instances) {
return instances[~~(instances.length * Math.random())];
}
let cloudflareBlackList = [];
let authenticateBlackList = [];
async function initcloudflareBlackList() {
return new Promise(resolve => {
fetch('/instances/blacklist.json').then(response => response.text()).then(data => {
cloudflareBlackList = JSON.parse(data).cloudflare;
authenticateBlackList = JSON.parse(data).authenticate;
resolve();
})
});
}
function updateInstances() {
return new Promise(async resolve => {
let http = new XMLHttpRequest();
http.open('GET', 'https://raw.githubusercontent.com/libredirect/libredirect/master/src/instances/data.json', false);
http.send(null);
if (http.status === 200) {
await initcloudflareBlackList();
const instances = JSON.parse(http.responseText);
youtubeHelper.setRedirects({ 'invidious': instances.invidious, 'piped': instances.piped, })
twitterHelper.setRedirects(instances.nitter);
instagramHelper.setRedirects(instances.bibliogram);
redditHelper.setRedirects({ 'libreddit': instances.libreddit, 'teddit': instances.teddit });
translateHelper.setRedirects({ "simplyTranslate": instances.simplyTranslate, "lingva": instances.lingva });
searchHelper.setRedirects({ 'searx': instances.searx, 'searxng': instances.searxng, 'whoogle': instances.whoogle, 'librex': instances.librex });
wikipediaHelper.setRedirects(instances.wikiless);
mediumHelper.setRedirects(instances.scribe);
quoraHelper.setRedirects(instances.quetre);
libremdbHelper.setRedirects(instances.libremdb);
sendTargetsHelper.setRedirects(instances.send);
tiktokHelper.setRedirects(instances.proxiTok);
console.info("Successfully updated Instances");
resolve(true); return;
}
resolve()
})
}
function protocolHost(url) {
if (url.username && url.password) return `${url.protocol}//${url.username}:${url.password}@${url.host}`;
return `${url.protocol}//${url.host}`;
}
async function processDefaultCustomInstances(target, name, protocol, document) {
function camelCase(str) {
return str.charAt(0).toUpperCase() + str.slice(1);
}
let latencyKey = `${name}Latency`;
let instancesLatency;
let nameProtocolElement = document.getElementById(name).getElementsByClassName(protocol)[0];
let nameCustomInstances = [];
let nameCheckListElement = nameProtocolElement.getElementsByClassName('checklist')[0];
await initcloudflareBlackList();
let nameDefaultRedirects;
let redirectsChecks = `${name}${camelCase(protocol)}RedirectsChecks`;
let customRedirects = `${name}${camelCase(protocol)}CustomRedirects`;
let redirectsKey = `${target}Redirects`;
let redirects;
async function getFromStorage() {
return new Promise(async resolve => {
browser.storage.local.get(
[
redirectsChecks,
customRedirects,
redirectsKey,
latencyKey
],
r => {
nameDefaultRedirects = r[redirectsChecks];
nameCustomInstances = r[customRedirects];
instancesLatency = r[latencyKey] ?? [];
redirects = r[redirectsKey];
resolve();
}
)
})
}
await getFromStorage();
function calcNameCheckBoxes() {
let isTrue = true;
for (const item of redirects[name][protocol])
if (!nameDefaultRedirects.includes(item)) {
isTrue = false;
break;
}
for (const element of nameCheckListElement.getElementsByTagName('input'))
element.checked = nameDefaultRedirects.includes(element.className)
if (nameDefaultRedirects.length == 0) isTrue = false;
nameProtocolElement.getElementsByClassName('toogle-all')[0].checked = isTrue;
}
nameCheckListElement.innerHTML =
[
`
Toggle All
`,
...redirects[name][protocol].map(
x => {
const cloudflare = cloudflareBlackList.includes(x) ? ' cloudflare' : '';
const authenticate = authenticateBlackList.includes(x) ? ' authenticate' : '';
let ms = instancesLatency[x];
let latencyColor = (ms <= 1000 ? "green" : ms <= 2000 ? "orange" : "red");
let latencyLimit;
if (ms == 5000) latencyLimit = '5000ms+';
else if (ms > 5000) latencyLimit = `ERROR: ${ms - 5000}`;
else latencyLimit = ms + 'ms';
const latency = x in instancesLatency ? '' + latencyLimit + '' : '';
let warnings = [cloudflare, authenticate, latency].join(' ');
return ``;
}
),
].join('\n
\n');
localise.localisePage();
calcNameCheckBoxes();
nameProtocolElement.getElementsByClassName('toogle-all')[0].addEventListener("change", async event => {
if (event.target.checked)
nameDefaultRedirects = [...redirects[name][protocol]];
else
nameDefaultRedirects = [];
browser.storage.local.set({ [redirectsChecks]: nameDefaultRedirects });
calcNameCheckBoxes();
});
for (let element of nameCheckListElement.getElementsByTagName('input')) {
if (element.className != 'toogle-all')
nameProtocolElement.getElementsByClassName(element.className)[0].addEventListener("change", async event => {
if (event.target.checked)
nameDefaultRedirects.push(element.className)
else {
let index = nameDefaultRedirects.indexOf(element.className);
if (index > -1) nameDefaultRedirects.splice(index, 1);
}
browser.storage.local.set({ [redirectsChecks]: nameDefaultRedirects });
calcNameCheckBoxes();
});
}
function calcNameCustomInstances() {
nameProtocolElement.getElementsByClassName('custom-checklist')[0].innerHTML =
nameCustomInstances.map(
x => `
`
).join('\n');
for (const item of nameCustomInstances) {
nameProtocolElement.getElementsByClassName(`clear-${item}`)[0].addEventListener("click", async () => {
let index = nameCustomInstances.indexOf(item);
if (index > -1) nameCustomInstances.splice(index, 1);
browser.storage.local.set({ [customRedirects]: nameCustomInstances });
calcNameCustomInstances();
});
}
}
calcNameCustomInstances();
nameProtocolElement.getElementsByClassName('custom-instance-form')[0].addEventListener("submit", async event => {
event.preventDefault();
let nameCustomInstanceInput = nameProtocolElement.getElementsByClassName('custom-instance')[0];
let url = new URL(nameCustomInstanceInput.value);
let protocolHostVar = protocolHost(url);
if (nameCustomInstanceInput.validity.valid && !redirects[name][protocol].includes(protocolHostVar)) {
if (!nameCustomInstances.includes(protocolHostVar)) {
nameCustomInstances.push(protocolHostVar)
browser.storage.local.set({ [customRedirects]: nameCustomInstances });
nameCustomInstanceInput.value = '';
}
calcNameCustomInstances();
}
})
}
async function ping(href) {
return new Promise(async resolve => {
let http = new XMLHttpRequest();
http.open("GET", `${href}?_=${new Date().getTime()}`, /*async*/true);
http.timeout = 5000;
let started = new Date().getTime();
http.onreadystatechange = () => {
if (http.readyState == 2) {
if (http.status == 200) {
let ended = new Date().getTime();
http.abort();
resolve(ended - started);
}
else
resolve(5000 + http.status)
}
};
http.ontimeout = () => resolve(5000)
http.onerror = () => resolve()
try {
http.send(null);
} catch (exception) {
resolve()
}
});
}
async function testLatency(element, instances) {
return new Promise(async resolve => {
let myList = {};
for (const href of instances) await ping(href).then(time => {
if (time) {
myList[href] = time;
let color;
if (time <= 1000) color = "green"
else if (time <= 2000) color = "orange"
else color = "red";
let text;
if (time == 5000) text = '5000ms+'
else if (time > 5000) text = `ERROR: ${time - 5000}`;
else text = `${time}ms`;
element.innerHTML = `${href}: ${text}`;
}
})
resolve(myList);
})
}
function copyCookie(frontend, targetUrl, urls, name) {
return new Promise(resolve => {
browser.storage.local.get('firstPartyIsolate', r => {
let query;
if (!r.firstPartyIsolate) query = { url: protocolHost(targetUrl), name: name }
else query = { url: protocolHost(targetUrl), name: name, firstPartyDomain: null }
browser.cookies.getAll(
query,
async cookies => {
for (const cookie of cookies)
if (cookie.name == name) {
for (const url of urls) {
const setQuery =
r.firstPartyIsolate ?
{
url: url, name: name, value: cookie.value, secure: true,
firstPartyDomain: new URL(url).hostname,
}
:
{
url: url, name: name, value: cookie.value, secure: true,
expirationDate: cookie.expirationDate,
};
browser.cookies.set(setQuery, () =>
browser.storage.local.set({ [`${frontend}_${name}`]: cookie }, () => resolve())
);
}
break;
}
resolve();
}
);
})
})
}
function getCookiesFromStorage(frontend, urls, name) {
let key = `${frontend}_${name}`;
browser.storage.local.get([key, 'firstPartyIsolate'], r => {
const cookie = r[key];
if (cookie === undefined) return;
for (const url of urls) {
let query =
r.firstPartyIsolate ?
{
url: url, name: cookie.name, value: cookie.value, secure: true,
expirationDate: null,
firstPartyDomain: new URL(url).hostname,
}
:
{
url: url, name: cookie.name, value: cookie.value, secure: true,
expirationDate: cookie.expirationDate,
};
browser.cookies.set(query)
}
})
}
function copyRaw(test, copyRawElement) {
return new Promise(resolve => {
browser.tabs.query(
{ active: true, currentWindow: true }, async tabs => {
let currTab = tabs[0];
if (currTab) {
let url;
try { url = new URL(currTab.url); }
catch { resolve(); return; }
let newUrl = await youtubeHelper.reverse(url);
if (!newUrl) newUrl = await twitterHelper.reverse(url);
if (!newUrl) newUrl = await instagramHelper.reverse(url);
if (!newUrl) newUrl = await tiktokHelper.reverse(url);
if (!newUrl) newUrl = await quoraHelper.reverse(url);
if (!newUrl) newUrl = await libremdbHelper.reverse(url);
if (!newUrl) newUrl = await imgurHelper.reverse(url);
if (newUrl) {
resolve(newUrl);
if (test) return;
navigator.clipboard.writeText(newUrl);
if (copyRawElement) {
const textElement = copyRawElement.getElementsByTagName('h4')[0]
const oldHtml = textElement.innerHTML;
textElement.innerHTML = browser.i18n.getMessage('copied');
setTimeout(() => textElement.innerHTML = oldHtml, 1000);
}
}
}
resolve()
}
)
})
}
function unify(test) {
return new Promise(resolve => {
browser.tabs.query(
{ active: true, currentWindow: true },
async tabs => {
let currTab = tabs[0]
if (currTab) {
let url;
try { url = new URL(currTab.url); }
catch { resolve(); return; }
if (currTab.incognito) { resolve(); return; }
let result = await youtubeHelper.copyPasteInvidiousCookies(test, url);
if (!result) result = await youtubeHelper.copyPastePipedLocalStorage(test, url, currTab.id);
if (!result) result = await youtubeHelper.copyPastePipedMaterialLocalStorage(test, url, currTab.id);
if (!result) result = await twitterHelper.initNitterCookies(test, url);
if (!result) result = await redditHelper.initLibredditCookies(test, url);
if (!result) result = await redditHelper.initTedditCookies(test, url);
if (!result) result = await searchHelper.initSearxCookies(test, url);
if (!result) result = await searchHelper.initSearxngCookies(test, url);
if (!result) result = await searchHelper.initLibrexCookies(test, url);
if (!result) result = await tiktokHelper.initProxiTokCookies(test, url);
if (!result) result = await wikipediaHelper.initWikilessCookies(test, url);
if (!result) result = await translateHelper.copyPasteSimplyTranslateCookies(test, url);
if (!result) result = await translateHelper.copyPasteLingvaLocalStorage(test, url);
resolve(result);
}
}
)
})
}
function switchInstance(test) {
return new Promise(resolve => {
browser.tabs.query({ active: true, currentWindow: true }, async tabs => {
let currTab = tabs[0];
if (currTab) {
let url;
try { url = new URL(currTab.url); }
catch { resolve(); return };
let newUrl = await youtubeHelper.switchInstance(url);
if (!newUrl) newUrl = await twitterHelper.switchInstance(url);
if (!newUrl) newUrl = await instagramHelper.switchInstance(url);
if (!newUrl) newUrl = await redditHelper.switchInstance(url);
if (!newUrl) newUrl = await searchHelper.switchInstance(url);
if (!newUrl) newUrl = await translateHelper.switchInstance(url);
if (!newUrl) newUrl = await mediumHelper.switchInstance(url);
if (!newUrl) newUrl = await quoraHelper.switchInstance(url);
if (!newUrl) newUrl = await libremdbHelper.switchInstance(url);
if (!newUrl) newUrl = await tiktokHelper.switchInstance(url);
if (!newUrl) newUrl = await sendTargetsHelper.switchInstance(url);
if (!newUrl) newUrl = await peertubeHelper.switchInstance(url);
if (!newUrl) newUrl = await lbryHelper.switchInstance(url);
if (!newUrl) newUrl = await imgurHelper.switchInstance(url);
if (!newUrl) newUrl = await wikipediaHelper.switchInstance(url);
if (newUrl) {
if (!test)
browser.tabs.update({ url: newUrl });
resolve(true)
} else resolve()
}
})
})
}
function latency(name, frontend, document, location) {
let latencyElement = document.getElementById(`latency-${frontend}`);
let latencyLabel = document.getElementById(`latency-${frontend}-label`);
latencyElement.addEventListener("click",
async () => {
let reloadWindow = () => location.reload();
latencyElement.addEventListener("click", reloadWindow);
let key = `${name}Redirects`
browser.storage.local.get(
key,
r => {
let redirects = r[key];
const oldHtml = latencyLabel.innerHTML;
latencyLabel.innerHTML = '...';
testLatency(latencyLabel, redirects[frontend].normal).then(r => {
browser.storage.local.set({ [`${frontend}Latency`]: r });
latencyLabel.innerHTML = oldHtml;
processDefaultCustomInstances(name, frontend, 'normal', document);
latencyElement.removeEventListener("click", reloadWindow)
});
}
)
}
);
}
export default {
getRandomInstance,
updateInstances,
protocolHost,
processDefaultCustomInstances,
latency,
copyCookie,
getCookiesFromStorage,
switchInstance,
copyRaw,
unify,
}