koboprice/lib/koboprice.user.js

359 lines
33 KiB
JavaScript
Raw Normal View History

2024-07-16 21:12:40 +03:00
// ==UserScript==
// @name Kobo Price
// @namespace https://tertyshny.dev
// @description Find lowest book price on kobo.com
// @noframes
// @match https://www.kobo.com/*/*/ebook/*
// @match https://www.kobo.com/*/*/audiobook/*
// @require https://cdn.jsdelivr.net/npm/currency.js
// @require https://cdn.jsdelivr.net/npm/date-fns/cdn.min.js
// @run-at document-end
// @version 1
// @icon 
// ==/UserScript==
globalThis.l = (...m) => console.log("KOBOPRICE", ...m);
const COUNTRIES = [
{ countryCode: "ww", currencyCode: "usd" },
{ countryCode: "ca", currencyCode: "cad" },
{ countryCode: "us", currencyCode: "usd" },
{ countryCode: "in", currencyCode: "inr" },
{ countryCode: "za", currencyCode: "zar" },
{ countryCode: "au", currencyCode: "aud" },
{ countryCode: "hk", currencyCode: "hkd" },
{ countryCode: "jp", currencyCode: "jpy" },
{ countryCode: "my", currencyCode: "myr" },
{ countryCode: "nz", currencyCode: "nzd" },
{ countryCode: "ph", currencyCode: "php" },
{ countryCode: "sg", currencyCode: "sgd" },
{ countryCode: "tw", currencyCode: "twd" },
{ countryCode: "th", currencyCode: "usd" },
{ countryCode: "at", currencyCode: "eur" },
{ countryCode: "be", currencyCode: "eur" },
{ countryCode: "cy", currencyCode: "eur" },
{ countryCode: "cz", currencyCode: "czk" },
{ countryCode: "dk", currencyCode: "dkk" },
{ countryCode: "ee", currencyCode: "eur" },
{ countryCode: "fi", currencyCode: "eur" },
{ countryCode: "fr", currencyCode: "eur" },
{ countryCode: "de", currencyCode: "eur" },
{ countryCode: "gr", currencyCode: "eur" },
{ countryCode: "ie", currencyCode: "eur" },
{ countryCode: "it", currencyCode: "eur" },
{ countryCode: "lt", currencyCode: "eur" },
{ countryCode: "lu", currencyCode: "eur" },
{ countryCode: "mt", currencyCode: "eur" },
{ countryCode: "nl", currencyCode: "eur" },
{ countryCode: "no", currencyCode: "nok" },
{ countryCode: "pl", currencyCode: "pln" },
{ countryCode: "pt", currencyCode: "eur" },
{ countryCode: "ro", currencyCode: "ron" },
{ countryCode: "sk", currencyCode: "eur" },
{ countryCode: "si", currencyCode: "eur" },
{ countryCode: "es", currencyCode: "eur" },
{ countryCode: "se", currencyCode: "sek" },
{ countryCode: "ch", currencyCode: "chf" },
{ countryCode: "tr", currencyCode: "try" },
{ countryCode: "gb", currencyCode: "gbp" },
{ countryCode: "ar", currencyCode: "usd" },
{ countryCode: "br", currencyCode: "brl" },
{ countryCode: "cl", currencyCode: "clp" },
{ countryCode: "co", currencyCode: "cop" },
{ countryCode: "mx", currencyCode: "mxn" },
{ countryCode: "pe", currencyCode: "pen" },
];
const CACHE_PREFIX = "KOBOPRICE";
const initCache = () => {
if (!localStorage.getItem(CACHE_PREFIX)) {
setState({ books: {}, rates: null });
}
};
const getState = () => JSON.parse(localStorage.getItem(CACHE_PREFIX));
const setState = (s) => localStorage.setItem(CACHE_PREFIX, JSON.stringify(s));
const getRates = () => getState().rates;
const cacheRates = (rates) => setState({ ...getState(), rates });
const getBookPrice = (url) => getState().books[url];
const cacheBookPrice = (price, url) => {
const state = getState();
state.books[url] = price;
setState(state);
};
const bookUrl = (country) => {
const urlPattern = /^https:\/\/www\.kobo\.com\/../;
const newPath = `https://www.kobo.com/${country}`;
const url = window.location.href.replace(urlPattern, newPath);
l("url for country", country, url);
return url;
};
const convertPrice = async (price, curr, rate) => {
l("rate for", curr, "is", rate);
let useVedic = false;
switch (curr) {
case "inr":
case "php": {
useVedic = true;
break;
}
case "jpy": {
break;
}
case "zar": {
price = price.replace(",", ".");
break;
}
case "dkk": {
price = price.replace("kr.", "").replace(",", ".");
break;
}
case "pen": {
price = price.replace("S/.", "");
break;
}
case "clp":
case "cop":
case "twd": {
break;
}
default:
price = price.replace(/[^\d,.-]/, "").replace(",", ".");
}
const convertedPrice = currency(price, { useVedic }).divide(rate);
l("converted price for", curr, convertedPrice);
return {
intValue: convertedPrice.intValue,
formatted: convertedPrice.format(),
};
};
const timeout = (duration) => new Promise((r) => setTimeout(r, duration));
const observePriceOnPage = (page) =>
new Promise((res, rej) => {
timeout(10000).then(() => rej("price not found"));
var observer = new MutationObserver(() => {
const price = page?.querySelector(
".primary-right-container .pricing-details .active-price span",
)?.textContent;
if (price) {
l("found price", price);
observer.disconnect();
res(price);
}
});
observer.observe(page, {
attributes: true,
childList: true,
characterData: true,
subtree: true,
});
});
const extractPrice = async (url) => {
try {
l("going to", url);
const iframe = document.createElement("iframe");
iframe.src = url;
iframe.hidden = true;
document.body.append(iframe);
await new Promise((res) => (iframe.contentWindow.onload = res));
l("starting observing price on", url);
const price = await observePriceOnPage(iframe.contentDocument.body, url);
document.body.removeChild(iframe);
return price;
} catch (e) {
l("getPriceForCountry", e, url);
return "";
}
};
const baseCurrency = "usd";
const loadCurrencyRates = async () => {
try {
const url = `https://cdn.jsdelivr.net/npm/@fawazahmed0/currency-api@latest/v1/currencies/${baseCurrency}.json`;
l("loading currency rates", url);
const res = await fetch(url);
return await res.json();
} catch (e) {
l("loadCurrencyRates", e);
return null;
}
};
const getRate = async (currency) => {
const cached = getRates();
if (cached && dateFns.isToday(cached.date)) {
l("found rates in cache", cached);
return cached[baseCurrency][currency];
}
const newRates = await loadCurrencyRates();
if (!newRates) {
l("failed to download rates");
return 0;
}
l("new rates", newRates);
cacheRates(newRates);
return newRates[baseCurrency][currency];
};
const bookPriceFor = async (country) => {
l("looking price for", country.countryCode);
const url = bookUrl(country.countryCode);
const fromCache = getBookPrice(url);
if (fromCache && dateFns.isToday(fromCache.cachedAt)) {
l("found book price in cache", fromCache);
return fromCache;
}
const countryPrice = await extractPrice(url);
let convertedPrice;
if (countryPrice) {
l("found price", countryPrice, url);
const rate = await getRate(country.currencyCode);
convertedPrice = await convertPrice(
countryPrice,
country.currencyCode,
rate,
);
}
const newPrice = {
...country,
countryPrice,
convertedPrice,
url,
cachedAt: dateFns.format(new Date(), "yyyy-MM-dd"),
};
cacheBookPrice(newPrice, url);
return newPrice;
};
const sortPrices = (prices) =>
prices.sort(
(a, b) =>
(a?.convertedPrice?.intValue || Infinity) -
(b?.convertedPrice?.intValue || Infinity),
);
const getPrices = async () => {
const prices = [];
for (const country of COUNTRIES) {
// intentionally blocking execution
// to resolve sequentially.
// It should prevent DOS and triggering captcha
prices.push(await bookPriceFor(country));
}
return prices;
};
const showPrices = (container, prices) => {
container.innerText = null;
prices.forEach((price) => {
const link = document.createElement("a");
link.href = price.url;
link.target = "_blank";
link.style.marginBottom = "5px";
link.style.display = "flex";
link.style.justifyContent = "space-between";
const oldPrice = document.createElement("p");
oldPrice.innerText = `${price.countryCode.toUpperCase()}: ${price?.countryPrice || "NO PRICE"}`;
const newPrice = document.createElement("p");
newPrice.innerText = price?.convertedPrice?.formatted ?? "NO PRICE";
newPrice.style.fontWeight = "bold";
link.appendChild(oldPrice);
link.appendChild(newPrice);
container.appendChild(link);
});
};
const createPricesContainer = () => {
const pricingActionContainers = document.querySelectorAll(".pricing-details");
l("all pricing containers", pricingActionContainers);
let visible;
for (const node of pricingActionContainers) {
if (node.checkVisibility()) {
l("found visible pricing container", node);
visible = node;
break;
}
}
const container = document.createElement("div");
container.style.display = "flex";
container.style.flexDirection = "column";
container.style.border = "1px solid black";
container.style.padding = "10px";
return visible.parentNode.insertBefore(container, visible);
};
async function main() {
const container = createPricesContainer();
container.innerText = "LOADING PRICES...";
try {
initCache();
const countriesPrice = await getPrices();
l("country prices", countriesPrice);
sortPrices(countriesPrice);
l("sorted prices", countriesPrice);
showPrices(container, countriesPrice);
} catch (e) {
l("error", e);
container.innerText = "FAILED TO LOAD PRICES. CHECK CONSOLE FOR MORE INFO";
} finally {
l("done");
}
}
l("starting...");
void main();