Merge pull request #170 from gerlofvanek/offers-4

ui: Offers table smart refresh button (JS), various fixes.
This commit is contained in:
tecnovert 2024-11-28 20:18:05 +00:00 committed by GitHub
commit 5941f2952e
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 184 additions and 61 deletions

View file

@ -3,7 +3,7 @@ let lastRefreshTime = null;
let newEntriesCount = 0; let newEntriesCount = 0;
let nextRefreshCountdown = 60; let nextRefreshCountdown = 60;
let currentPage = 1; let currentPage = 1;
const itemsPerPage = 100; const itemsPerPage = 50;
let lastAppliedFilters = {}; let lastAppliedFilters = {};
const CACHE_KEY = 'latestPricesCache'; const CACHE_KEY = 'latestPricesCache';
@ -22,6 +22,9 @@ const isSentOffers = window.offersTableConfig.isSentOffers;
let currentSortColumn = 0; let currentSortColumn = 0;
let currentSortDirection = 'desc'; let currentSortDirection = 'desc';
const PRICE_INIT_RETRIES = 3;
const PRICE_INIT_RETRY_DELAY = 2000;
const offerCache = { const offerCache = {
set: (key, value, customTtl = null) => { set: (key, value, customTtl = null) => {
const item = { const item = {
@ -105,7 +108,6 @@ const offerCache = {
} }
}; };
const coinNameToSymbol = { const coinNameToSymbol = {
'Bitcoin': 'bitcoin', 'Bitcoin': 'bitcoin',
'Particl': 'particl', '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() { function loadSortPreferences() {
const savedColumn = localStorage.getItem('tableSortColumn'); const savedColumn = localStorage.getItem('tableSortColumn');
const savedDirection = localStorage.getItem('tableSortDirection'); const savedDirection = localStorage.getItem('tableSortDirection');
@ -778,7 +825,7 @@ async function checkExpiredAndFetchNew() {
newListings = newListings.filter(offer => !isOfferExpired(offer)); newListings = newListings.filter(offer => !isOfferExpired(offer));
originalJsonData = newListings; originalJsonData = newListings;
cache.set(OFFERS_CACHE_KEY, newListings, CACHE_DURATION); offerCache.set(OFFERS_CACHE_KEY, newListings, CACHE_DURATION);
const currentFilters = new FormData(filterForm); const currentFilters = new FormData(filterForm);
const hasActiveFilters = currentFilters.get('coin_to') !== 'any' || const hasActiveFilters = currentFilters.get('coin_to') !== 'any' ||
@ -1320,13 +1367,22 @@ async function fetchOffers(manualRefresh = false) {
setRefreshButtonLoading(true); setRefreshButtonLoading(true);
const OFFERS_CACHE_KEY = `offers_${isSentOffers ? 'sent' : 'received'}`; 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) { if (!manualRefresh) {
const cachedData = offerCache.get(OFFERS_CACHE_KEY); const cachedData = offerCache.get(OFFERS_CACHE_KEY);
if (cachedData) { if (cachedData) {
console.log('Using cached offers data'); console.log('Using cached offers data');
jsonData = cachedData.value; jsonData = cachedData.value;
originalJsonData = [...cachedData.value]; originalJsonData = [...cachedData.value];
updateOffersTable(); await updateOffersTable();
updateJsonView(); updateJsonView();
updatePaginationInfo(); updatePaginationInfo();
setRefreshButtonLoading(false); setRefreshButtonLoading(false);
@ -1338,20 +1394,10 @@ async function fetchOffers(manualRefresh = false) {
const endpoint = isSentOffers ? '/json/sentoffers' : '/json/offers'; const endpoint = isSentOffers ? '/json/sentoffers' : '/json/offers';
console.log(`[Debug] Fetching from endpoint: ${endpoint}`); console.log(`[Debug] Fetching from endpoint: ${endpoint}`);
const response = await fetch(endpoint, { const response = await fetch(endpoint);
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
with_extra_info: true,
limit: 1000
})
});
const data = await response.json(); 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 => ({ newData = newData.map(offer => ({
...offer, ...offer,
offer_id: String(offer.offer_id || ''), offer_id: String(offer.offer_id || ''),
@ -1379,19 +1425,19 @@ async function fetchOffers(manualRefresh = false) {
originalJsonData = [...mergedData]; originalJsonData = [...mergedData];
} }
// Cache the new data
offerCache.set(OFFERS_CACHE_KEY, jsonData, CACHE_DURATION); offerCache.set(OFFERS_CACHE_KEY, jsonData, CACHE_DURATION);
if (newEntriesCountSpan) { if (newEntriesCountSpan) {
newEntriesCountSpan.textContent = jsonData.length; newEntriesCountSpan.textContent = jsonData.length;
} }
updateOffersTable(); await updateOffersTable();
updateJsonView(); updateJsonView();
updatePaginationInfo(); updatePaginationInfo();
} catch (error) { } catch (error) {
console.error('[Debug] Error fetching offers:', error); console.error('[Debug] Error fetching offers:', error);
ui.displayErrorMessage('Failed to fetch offers. Please try again later.');
} finally { } finally {
setRefreshButtonLoading(false); setRefreshButtonLoading(false);
} }
@ -1532,12 +1578,14 @@ function filterAndSortData() {
return filteredData; return filteredData;
} }
async function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, 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}`); console.log(`Calculating profit/loss for ${fromAmount} ${fromCoin} to ${toAmount} ${toCoin}, isOwnOffer: ${isOwnOffer}`);
if (!latestPrices) { if (!latestPrices) {
console.error('Latest prices not available. Unable to calculate profit/loss.'); console.error('Latest prices not available. Unable to calculate profit/loss.');
return null; resolve(null);
return;
} }
const getPriceKey = (coin) => { const getPriceKey = (coin) => {
@ -1548,6 +1596,9 @@ async function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, isOwn
if (lowerCoin === 'bitcoin cash') { if (lowerCoin === 'bitcoin cash') {
return 'bitcoin-cash'; return 'bitcoin-cash';
} }
if (lowerCoin === 'particl anon' || lowerCoin === 'particl blind') {
return 'particl';
}
return coinNameToSymbol[coin] || lowerCoin; return coinNameToSymbol[coin] || lowerCoin;
}; };
@ -1558,8 +1609,9 @@ async function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, isOwn
const toPriceUSD = latestPrices[toSymbol]?.usd; const toPriceUSD = latestPrices[toSymbol]?.usd;
if (!fromPriceUSD || !toPriceUSD) { if (!fromPriceUSD || !toPriceUSD) {
console.error(`Price data missing for ${fromSymbol} or ${toSymbol}`); console.warn(`Price data missing for ${fromSymbol} (${fromPriceUSD}) or ${toSymbol} (${toPriceUSD})`);
return null; resolve(null);
return;
} }
const fromValueUSD = fromAmount * fromPriceUSD; const fromValueUSD = fromAmount * fromPriceUSD;
@ -1573,7 +1625,8 @@ async function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, isOwn
} }
console.log(`Percent difference: ${percentDiff.toFixed(2)}%`); console.log(`Percent difference: ${percentDiff.toFixed(2)}%`);
return percentDiff; resolve(percentDiff);
});
} }
async function getMarketRate(fromCoin, toCoin) { async function getMarketRate(fromCoin, toCoin) {
@ -1885,13 +1938,83 @@ document.addEventListener('DOMContentLoaded', () => {
updateCoinFilterImages(); updateCoinFilterImages();
}); });
document.getElementById('refreshOffers').addEventListener('click', () => { document.getElementById('refreshOffers').addEventListener('click', async () => {
console.log('🔄 Refresh button clicked'); console.log('🔄 Smart refresh initiated');
console.log('Clearing cache before refresh...'); setRefreshButtonLoading(true);
offerCache.clear();
console.log('Fetching fresh data...'); try {
fetchOffers(true);
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', () => { toggleButton.addEventListener('click', () => {
tableView.classList.toggle('hidden'); tableView.classList.toggle('hidden');
jsonView.classList.toggle('hidden'); jsonView.classList.toggle('hidden');

View file

@ -33,7 +33,7 @@ from basicswap.basicswap_util import (
from basicswap.protocols.xmr_swap_1 import getChainBSplitKey, getChainBRemoteSplitKey from basicswap.protocols.xmr_swap_1 import getChainBSplitKey, getChainBRemoteSplitKey
PAGE_LIMIT = 25 PAGE_LIMIT = 1000
invalid_coins_from = [] invalid_coins_from = []
known_chart_coins = [ known_chart_coins = [
"BTC", "BTC",