const axios = require("axios"); const utils = require("./utils"); const conversions = { g: { unit: "g", factor: 1 }, kg: { unit: "g", factor: 1000 }, l: { unit: "ml", factor: 1000 }, ml: { unit: "ml", factor: 1 }, St: { unit: "stk", factor: 1 }, Wl: { unit: "wg", factor: 1 }, m: { unit: "cm", factor: 100 }, mm: { unit: "cm", factor: 0.1 }, Bl: { unit: "stk", factor: 1 }, Btl: { unit: "stk", factor: 1 }, Paar: { unit: "stk", factor: 1 }, Portion: { unit: "stk", factor: 1 }, Satz: { unit: "stk", factor: 1 }, Tablette: { unit: "stk", factor: 1 }, }; exports.getCanonical = function (item, today) { let quantity = item.netQuantityContent || item.basePriceQuantity; let unit = item.contentUnit || item.basePriceUnit; return utils.convertUnit( { id: item.gtin, name: `${item.brandName} ${item.title}`, price: item.price.value, priceHistory: [{ date: today, price: item.price.value }], unit, quantity, ...((item.brandName === "dmBio" || (item.name ? item.name.startsWith("Bio ") | item.name.startsWith("Bio-") : false)) && { bio: true }), url: `https://www.dm.at/product-p${item.gtin}.html`, }, conversions, "dm" ); }; exports.fetchData = async function () { const DM_BASE_URL = `https://product-search.services.dmtech.com/at/search/crawl?pageSize=1000&`; const QUERIES = [ "allCategories.id=010000&price.value.to=2", //~500 items "allCategories.id=010000&price.value.from=2&price.value.to=3", //~600 items "allCategories.id=010000&price.value.from=3&price.value.to=4", //~500 items "allCategories.id=010000&price.value.from=4&price.value.to=7", //~800 items "allCategories.id=010000&price.value.from=7&price.value.to=10", //~900 items "allCategories.id=010000&price.value.from=10&price.value.to=15", //~900 items "allCategories.id=010000&price.value.from=15", //~300 items "allCategories.id=020000&price.value.to=2", //~600 items "allCategories.id=020000&price.value.from=2&price.value.to=3", //~550 items "allCategories.id=020000&price.value.from=3&price.value.to=4", //~600 items "allCategories.id=020000&price.value.from=4&price.value.to=6", //~800 items "allCategories.id=020000&price.value.from=6&price.value.to=10", //~850 items "allCategories.id=020000&price.value.from=10&price.value.to=18", //~900 items "allCategories.id=020000&price.value.from=18", //~960 items (!) "allCategories.id=030000&price.value.to=8", //~900 items "allCategories.id=030000&price.value.from=8", //~500 items "allCategories.id=040000&price.value.to=2", //~600 items "allCategories.id=040000&price.value.from=2&price.value.to=4", //~900 items "allCategories.id=040000&price.value.from=4", //~400 items "allCategories.id=050000&price.value.to=4", //~600 items "allCategories.id=050000&price.value.from=4", //~800 items "allCategories.id=060000&price.value.to=4", //~900 items "allCategories.id=060000&price.value.from=4", //~500 items "allCategories.id=070000", //~300 items ]; let dmItems = []; for (let query of QUERIES) { var res = await axios.get(DM_BASE_URL + query, { validateStatus: function (status) { return (status >= 200 && status < 300) || status == 429; }, }); // exponential backoff backoff = 2000; while (res.status == 429) { console.info(`DM API returned 429, retrying in ${backoff / 1000}s.`); await new Promise((resolve) => setTimeout(resolve, backoff)); backoff *= 2; res = await axios.get(DM_BASE_URL + query, { validateStatus: function (status) { return (status >= 200 && status < 300) || status == 429; }, }); } let items = res.data; if (items.count > 1000) { console.warn(`Query returned more than 1000 items! Items may be missing. Adjust queries. Query: ${query}`); } dmItems = dmItems.concat(items.products); await new Promise((resolve) => setTimeout(resolve, 1000)); } return dmItems; };