// ==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();