From 801006fa70361790b111e1bb3cd73138a1964055 Mon Sep 17 00:00:00 2001 From: gerlofvanek Date: Thu, 28 Nov 2024 09:54:06 +0100 Subject: [PATCH 1/2] ui: Offers table smart refresh button (JS), various fixes. --- basicswap/static/js/offerstable.js | 243 ++++++++++++++++++++++------- basicswap/ui/util.py | 3 +- 2 files changed, 184 insertions(+), 62 deletions(-) diff --git a/basicswap/static/js/offerstable.js b/basicswap/static/js/offerstable.js index 79f2e05..7f0eaa7 100644 --- a/basicswap/static/js/offerstable.js +++ b/basicswap/static/js/offerstable.js @@ -3,7 +3,7 @@ let lastRefreshTime = null; let newEntriesCount = 0; let nextRefreshCountdown = 60; let currentPage = 1; -const itemsPerPage = 100; +const itemsPerPage = 50; let lastAppliedFilters = {}; const CACHE_KEY = 'latestPricesCache'; @@ -22,6 +22,9 @@ const isSentOffers = window.offersTableConfig.isSentOffers; let currentSortColumn = 0; let currentSortDirection = 'desc'; +const PRICE_INIT_RETRIES = 3; +const PRICE_INIT_RETRY_DELAY = 2000; + const offerCache = { set: (key, value, customTtl = null) => { const item = { @@ -105,7 +108,6 @@ const offerCache = { } }; - const coinNameToSymbol = { 'Bitcoin': 'bitcoin', 'Particl': 'particl', @@ -355,6 +357,51 @@ function makePostRequest(url, headers = {}) { }); } +async function initializePriceData() { + console.log('Initializing price data...'); + let retryCount = 0; + let prices = null; + + while (retryCount < PRICE_INIT_RETRIES) { + try { + prices = await fetchLatestPrices(); + + if (prices && Object.keys(prices).length > 0) { + console.log('Successfully fetched initial price data:', prices); + latestPrices = prices; + + const PRICES_CACHE_KEY = 'prices_coingecko'; + offerCache.set(PRICES_CACHE_KEY, prices, CACHE_DURATION); + + return true; + } + + console.warn(`Attempt ${retryCount + 1}: Price data incomplete, retrying...`); + retryCount++; + + if (retryCount < PRICE_INIT_RETRIES) { + await new Promise(resolve => setTimeout(resolve, PRICE_INIT_RETRY_DELAY)); + } + } catch (error) { + console.error(`Error fetching prices (attempt ${retryCount + 1}):`, error); + retryCount++; + + if (retryCount < PRICE_INIT_RETRIES) { + await new Promise(resolve => setTimeout(resolve, PRICE_INIT_RETRY_DELAY)); + } + } + } + + const fallbackPrices = getFallbackPrices(); + if (fallbackPrices && Object.keys(fallbackPrices).length > 0) { + console.log('Using fallback prices:', fallbackPrices); + latestPrices = fallbackPrices; + return true; + } + + return false; +} + function loadSortPreferences() { const savedColumn = localStorage.getItem('tableSortColumn'); const savedDirection = localStorage.getItem('tableSortDirection'); @@ -778,7 +825,7 @@ async function checkExpiredAndFetchNew() { newListings = newListings.filter(offer => !isOfferExpired(offer)); originalJsonData = newListings; - cache.set(OFFERS_CACHE_KEY, newListings, CACHE_DURATION); + offerCache.set(OFFERS_CACHE_KEY, newListings, CACHE_DURATION); const currentFilters = new FormData(filterForm); const hasActiveFilters = currentFilters.get('coin_to') !== 'any' || @@ -1320,13 +1367,22 @@ async function fetchOffers(manualRefresh = false) { setRefreshButtonLoading(true); const OFFERS_CACHE_KEY = `offers_${isSentOffers ? 'sent' : 'received'}`; + if (!latestPrices || Object.keys(latestPrices).length === 0) { + console.log('No price data available, initializing...'); + const priceInitSuccess = await initializePriceData(); + if (!priceInitSuccess) { + console.error('Failed to initialize price data'); + ui.displayErrorMessage('Unable to load cryptocurrency prices. Some features may be limited.'); + } + } + if (!manualRefresh) { const cachedData = offerCache.get(OFFERS_CACHE_KEY); if (cachedData) { console.log('Using cached offers data'); jsonData = cachedData.value; originalJsonData = [...cachedData.value]; - updateOffersTable(); + await updateOffersTable(); updateJsonView(); updatePaginationInfo(); setRefreshButtonLoading(false); @@ -1338,20 +1394,10 @@ async function fetchOffers(manualRefresh = false) { const endpoint = isSentOffers ? '/json/sentoffers' : '/json/offers'; console.log(`[Debug] Fetching from endpoint: ${endpoint}`); - const response = await fetch(endpoint, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - }, - body: JSON.stringify({ - with_extra_info: true, - limit: 1000 - }) - }); - + const response = await fetch(endpoint); const data = await response.json(); - let newData = Array.isArray(data) ? data : Object.values(data); + let newData = Array.isArray(data) ? data : Object.values(data); newData = newData.map(offer => ({ ...offer, offer_id: String(offer.offer_id || ''), @@ -1379,19 +1425,19 @@ async function fetchOffers(manualRefresh = false) { originalJsonData = [...mergedData]; } - // Cache the new data offerCache.set(OFFERS_CACHE_KEY, jsonData, CACHE_DURATION); if (newEntriesCountSpan) { newEntriesCountSpan.textContent = jsonData.length; } - updateOffersTable(); + await updateOffersTable(); updateJsonView(); updatePaginationInfo(); } catch (error) { console.error('[Debug] Error fetching offers:', error); + ui.displayErrorMessage('Failed to fetch offers. Please try again later.'); } finally { setRefreshButtonLoading(false); } @@ -1532,48 +1578,55 @@ function filterAndSortData() { return filteredData; } -async function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, isOwnOffer) { - console.log(`Calculating profit/loss for ${fromAmount} ${fromCoin} to ${toAmount} ${toCoin}, isOwnOffer: ${isOwnOffer}`); +function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, isOwnOffer) { + return new Promise((resolve) => { + console.log(`Calculating profit/loss for ${fromAmount} ${fromCoin} to ${toAmount} ${toCoin}, isOwnOffer: ${isOwnOffer}`); - if (!latestPrices) { - console.error('Latest prices not available. Unable to calculate profit/loss.'); - return null; - } - - const getPriceKey = (coin) => { - const lowerCoin = coin.toLowerCase(); - if (lowerCoin === 'firo' || lowerCoin === 'zcoin') { - return 'zcoin'; + if (!latestPrices) { + console.error('Latest prices not available. Unable to calculate profit/loss.'); + resolve(null); + return; } - if (lowerCoin === 'bitcoin cash') { - return 'bitcoin-cash'; + + const getPriceKey = (coin) => { + const lowerCoin = coin.toLowerCase(); + if (lowerCoin === 'firo' || lowerCoin === 'zcoin') { + return 'zcoin'; + } + if (lowerCoin === 'bitcoin cash') { + return 'bitcoin-cash'; + } + if (lowerCoin === 'particl anon' || lowerCoin === 'particl blind') { + return 'particl'; + } + return coinNameToSymbol[coin] || lowerCoin; + }; + + const fromSymbol = getPriceKey(fromCoin); + const toSymbol = getPriceKey(toCoin); + + const fromPriceUSD = latestPrices[fromSymbol]?.usd; + const toPriceUSD = latestPrices[toSymbol]?.usd; + + if (!fromPriceUSD || !toPriceUSD) { + console.warn(`Price data missing for ${fromSymbol} (${fromPriceUSD}) or ${toSymbol} (${toPriceUSD})`); + resolve(null); + return; } - return coinNameToSymbol[coin] || lowerCoin; - }; - const fromSymbol = getPriceKey(fromCoin); - const toSymbol = getPriceKey(toCoin); + const fromValueUSD = fromAmount * fromPriceUSD; + const toValueUSD = toAmount * toPriceUSD; - const fromPriceUSD = latestPrices[fromSymbol]?.usd; - const toPriceUSD = latestPrices[toSymbol]?.usd; + let percentDiff; + if (isOwnOffer) { + percentDiff = ((toValueUSD / fromValueUSD) - 1) * 100; + } else { + percentDiff = ((fromValueUSD / toValueUSD) - 1) * 100; + } - if (!fromPriceUSD || !toPriceUSD) { - console.error(`Price data missing for ${fromSymbol} or ${toSymbol}`); - return null; - } - - const fromValueUSD = fromAmount * fromPriceUSD; - const toValueUSD = toAmount * toPriceUSD; - - let percentDiff; - if (isOwnOffer) { - percentDiff = ((toValueUSD / fromValueUSD) - 1) * 100; - } else { - percentDiff = ((fromValueUSD / toValueUSD) - 1) * 100; - } - - console.log(`Percent difference: ${percentDiff.toFixed(2)}%`); - return percentDiff; + console.log(`Percent difference: ${percentDiff.toFixed(2)}%`); + resolve(percentDiff); + }); } async function getMarketRate(fromCoin, toCoin) { @@ -1885,13 +1938,83 @@ document.addEventListener('DOMContentLoaded', () => { updateCoinFilterImages(); }); -document.getElementById('refreshOffers').addEventListener('click', () => { - console.log('🔄 Refresh button clicked'); - console.log('Clearing cache before refresh...'); - offerCache.clear(); - console.log('Fetching fresh data...'); - fetchOffers(true); + document.getElementById('refreshOffers').addEventListener('click', async () => { + console.log('🔄 Smart refresh initiated'); + setRefreshButtonLoading(true); + + try { + + const PRICES_CACHE_KEY = 'prices_coingecko'; + const cachedPrices = offerCache.get(PRICES_CACHE_KEY); + + if (!cachedPrices || !cachedPrices.remainingTime || cachedPrices.remainingTime < 60000) { + + console.log('Fetching fresh price data...'); + const newPrices = await fetchLatestPrices(); + if (newPrices) { + latestPrices = newPrices; + } + } else { + console.log('Using cached price data (still valid)'); + } + + const endpoint = isSentOffers ? '/json/sentoffers' : '/json/offers'; + const response = await fetch(endpoint); + const newData = await response.json(); + + const processedNewData = Array.isArray(newData) ? newData : Object.values(newData); + const formattedNewData = processedNewData.map(offer => ({ + ...offer, + offer_id: String(offer.offer_id || ''), + swap_type: String(offer.swap_type || 'N/A'), + addr_from: String(offer.addr_from || ''), + coin_from: String(offer.coin_from || ''), + coin_to: String(offer.coin_to || ''), + amount_from: String(offer.amount_from || '0'), + amount_to: String(offer.amount_to || '0'), + rate: String(offer.rate || '0'), + created_at: Number(offer.created_at || 0), + expire_at: Number(offer.expire_at || 0), + is_own_offer: Boolean(offer.is_own_offer), + amount_negotiable: Boolean(offer.amount_negotiable), + is_revoked: Boolean(offer.is_revoked), + unique_id: `${offer.offer_id}_${offer.created_at}_${offer.coin_from}_${offer.coin_to}` + })); + + const existingIds = new Set(jsonData.map(offer => offer.offer_id)); + const newOffers = formattedNewData.filter(offer => !existingIds.has(offer.offer_id)); + + if (newOffers.length > 0) { + console.log(`Found ${newOffers.length} new offers`); + + jsonData = mergeSentOffers(jsonData, formattedNewData); + originalJsonData = [...jsonData]; + + const OFFERS_CACHE_KEY = `offers_${isSentOffers ? 'sent' : 'received'}`; + offerCache.set(OFFERS_CACHE_KEY, jsonData, CACHE_DURATION); + } else { + console.log('No new offers found'); + } + + if (!isSentOffers) { + jsonData = jsonData.filter(offer => !isOfferExpired(offer)); + originalJsonData = [...jsonData]; + } + + await updateOffersTable(); + updateJsonView(); + updatePaginationInfo(); + + console.log('Smart refresh completed successfully'); + + } catch (error) { + console.error('Error during smart refresh:', error); + ui.displayErrorMessage('Failed to refresh offers. Please try again later.'); + } finally { + setRefreshButtonLoading(false); + } }); + toggleButton.addEventListener('click', () => { tableView.classList.toggle('hidden'); jsonView.classList.toggle('hidden'); diff --git a/basicswap/ui/util.py b/basicswap/ui/util.py index 49f05a6..087c7b1 100644 --- a/basicswap/ui/util.py +++ b/basicswap/ui/util.py @@ -33,7 +33,7 @@ from basicswap.basicswap_util import ( from basicswap.protocols.xmr_swap_1 import getChainBSplitKey, getChainBRemoteSplitKey -PAGE_LIMIT = 25 +PAGE_LIMIT = 1000 invalid_coins_from = [] known_chart_coins = [ "BTC", @@ -51,7 +51,6 @@ known_chart_coins = [ "BCH", ] - def tickerToCoinId(ticker): search_str = ticker.upper() for c in Coins: From 1bb721edf5514fb93af25c8c40f95536c6f4419d Mon Sep 17 00:00:00 2001 From: gerlofvanek Date: Thu, 28 Nov 2024 09:58:52 +0100 Subject: [PATCH 2/2] fix: LINT --- basicswap/ui/util.py | 1 + 1 file changed, 1 insertion(+) diff --git a/basicswap/ui/util.py b/basicswap/ui/util.py index 087c7b1..6bb1c44 100644 --- a/basicswap/ui/util.py +++ b/basicswap/ui/util.py @@ -51,6 +51,7 @@ known_chart_coins = [ "BCH", ] + def tickerToCoinId(ticker): search_str = ticker.upper() for c in Coins: