359 lines
33 KiB
JavaScript
359 lines
33 KiB
JavaScript
|
// ==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();
|