diff --git a/src/components/Earn.tsx b/src/components/Earn.tsx index 857980a9..3acb9da2 100644 --- a/src/components/Earn.tsx +++ b/src/components/Earn.tsx @@ -2,10 +2,11 @@ import { useCallback, useEffect, useMemo, useState } from 'react' import { Formik, FormikErrors } from 'formik' import * as rb from 'react-bootstrap' import { useTranslation } from 'react-i18next' +import { TFunction } from 'i18next' import { useSettings } from '../context/SettingsContext' import { CurrentWallet, useCurrentWalletInfo, useReloadCurrentWalletInfo } from '../context/WalletContext' import { useServiceInfo, useReloadServiceInfo, Offer } from '../context/ServiceInfoContext' -import { factorToPercentage, isValidNumber, percentageToFactor } from '../utils' +import { factorToPercentage, isAbsoluteOffer, isRelativeOffer, isValidNumber, percentageToFactor } from '../utils' import * as Api from '../libs/JmWalletApi' import * as fb from './fb/utils' import Sprite from './Sprite' @@ -19,7 +20,6 @@ import { OrderbookOverlay } from './Orderbook' import Balance from './Balance' import styles from './Earn.module.css' import Accordion from './Accordion' -import { TFunction } from 'i18next' // In order to prevent state mismatch, the 'maker stop' response is delayed shortly. // Even though the API response suggests that the maker has started or stopped immediately, it seems that this is not always the case. @@ -31,14 +31,8 @@ const MAKER_STOP_RESPONSE_DELAY_MS = 2_000 // that the UTXO corresponding to the fidelity bond is correctly marked as such. const RELOAD_FIDELITY_BONDS_DELAY_MS = 2_000 -const OFFERTYPE_REL = 'sw0reloffer' -const OFFERTYPE_ABS = 'sw0absoffer' - -// can be any of ['sw0reloffer', 'swreloffer', 'reloffer'] -const isRelativeOffer = (offertype: string) => offertype.includes('reloffer') - -// can be any of ['sw0absoffer', 'swabsoffer', 'absoffer'] -const isAbsoluteOffer = (offertype: string) => offertype.includes('absoffer') +const OFFERTYPE_REL: Api.OfferType = 'sw0reloffer' +const OFFERTYPE_ABS: Api.OfferType = 'sw0absoffer' const FORM_INPUT_LOCAL_STORAGE_KEYS = { offertype: 'jm-offertype', diff --git a/src/components/Orderbook.tsx b/src/components/Orderbook.tsx index 5e192d12..10c0c58f 100644 --- a/src/components/Orderbook.tsx +++ b/src/components/Orderbook.tsx @@ -13,21 +13,13 @@ import { useSettings } from '../context/SettingsContext' import Balance from './Balance' import Sprite from './Sprite' import TablePagination from './TablePagination' +import { factorToPercentage, isAbsoluteOffer, isRelativeOffer } from '../utils' +import { isDevMode } from '../constants/debugFeatures' import styles from './Orderbook.module.css' -const SORT_KEYS = { - type: 'TYPE', - counterparty: 'COUNTERPARTY', - fee: 'FEE', - minimumSize: 'MINIMUM_SIZE', - maximumSize: 'MAXIMUM_SIZE', - minerFeeContribution: 'MINER_FEE_CONTRIBUTION', - bondValue: 'BOND_VALUE', -} - const TABLE_THEME = { Table: ` - --data-table-library_grid-template-columns: 1fr 5rem 1fr 1fr 2fr 2fr 2fr 2fr; + --data-table-library_grid-template-columns: 1fr 5rem 1fr 2fr 2fr 2fr 2fr; font-size: 0.9rem; `, BaseCell: ` @@ -78,27 +70,89 @@ const TABLE_THEME = { `, } -const withTooltip = (node: ReactElement, tooltip: string) => { +const withTooltip = (node: ReactElement, tooltip: string, overlayProps?: Partial) => { return ( - {tooltip}}>{node} + {tooltip}}> + {node} + ) } -// `TableNode` is known to have same properties as `ObwatchApi.Order`, hence prefer casting over object destructuring -const toOrder = (tableNode: TableTypes.TableNode) => tableNode as unknown as ObwatchApi.Order +// `TableNode` is known to have same properties as `OrderTableEntry`, hence prefer casting over object destructuring +const asOrderTableEntry = (tableNode: TableTypes.TableNode) => tableNode as unknown as OrderTableEntry + +const renderOrderType = (type: OrderTypeProps) => { + const elem = {type.displayValue} + return type.tooltip ? withTooltip(elem, type.tooltip) : elem +} -const renderOrderType = (val: string, t: TFunction) => { - if (val === ObwatchApi.ABSOLUTE_ORDER_TYPE_VAL) { - return withTooltip({t('orderbook.text_offer_type_absolute')}, val) +type OrderTypeProps = { + value: string // original value, example: 'sw0reloffer', 'swreloffer', 'reloffer', 'sw0absoffer', 'swabsoffer', 'absoffer' + displayValue: string // example: "absolute" or "relative" (respecting i18n) + badgeColor: 'info' | 'primary' | 'secondary' + tooltip?: 'Native SW Absolute Fee' | 'Native SW Relative Fee' | string + isAbsolute?: boolean + isRelative?: boolean +} +interface OrderTableEntry { + type: OrderTypeProps + counterparty: string // example: "J5Bv3JSxPFWm2Yjb" + orderId: string // example: "0" (not unique!) + fee: { + value: number + displayValue: string // example: "250" (abs offers) or "0.000100%" (rel offers) } - if (val === ObwatchApi.RELATIVE_ORDER_TYPE_VAL) { - return withTooltip({t('orderbook.text_offer_type_relative')}, val) + minerFeeContribution: string // example: "0" + minimumSize: string // example: "27300" + maximumSize: string // example: "237499972700" + bondValue: { + value: number + displayValue: string // example: "0" (no fb) or "114557102085.28133" + } +} + +const SORT_KEYS = { + type: 'TYPE', + counterparty: 'COUNTERPARTY', + fee: 'FEE', + minimumSize: 'MINIMUM_SIZE', + maximumSize: 'MAXIMUM_SIZE', + minerFeeContribution: 'MINER_FEE_CONTRIBUTION', + bondValue: 'BOND_VALUE', +} + +const orderTypeProps = (offer: ObwatchApi.Offer, t: TFunction): OrderTypeProps => { + if (isAbsoluteOffer(offer.ordertype)) { + return { + value: offer.ordertype, + displayValue: t('orderbook.text_offer_type_absolute'), + badgeColor: 'info', + tooltip: offer.ordertype === 'sw0absoffer' ? 'Native SW Absolute Fee' : offer.ordertype, + isAbsolute: true, + } + } + if (isRelativeOffer(offer.ordertype)) { + return { + value: offer.ordertype, + displayValue: t('orderbook.text_offer_type_relative'), + badgeColor: 'primary', + tooltip: offer.ordertype === 'sw0reloffer' ? 'Native SW Relative Fee' : offer.ordertype, + isRelative: true, + } + } + return { + value: offer.ordertype, + displayValue: offer.ordertype, + badgeColor: 'secondary', } - return {val} } const renderOrderFee = (val: string, settings: any) => { - return val.includes('%') ? <>{val} : + return val.includes('%') ? ( + {val} + ) : ( + + ) } interface OrderbookTableProps { @@ -134,39 +188,28 @@ const OrderbookTable = ({ data }: OrderbookTableProps) => { }, sortToggleType: SortToggleType.AlternateWithReset, sortFns: { - [SORT_KEYS.type]: (array) => array.sort((a, b) => a.type.localeCompare(b.type)), + [SORT_KEYS.type]: (array) => array.sort((a, b) => a.type.displayValue.localeCompare(b.type.displayValue)), [SORT_KEYS.fee]: (array) => array.sort((a, b) => { - const aOrder = toOrder(a) - const bOrder = toOrder(b) - - if (aOrder.type !== bOrder.type) { - return aOrder.type === ObwatchApi.ABSOLUTE_ORDER_TYPE_VAL ? 1 : -1 - } - - if (aOrder.type === ObwatchApi.ABSOLUTE_ORDER_TYPE_VAL) { - return +aOrder.fee - +bOrder.fee - } else { - const aIndexOfPercent = aOrder.fee.indexOf('%') - const bIndexOfPercent = bOrder.fee.indexOf('%') + const aOrder = asOrderTableEntry(a) + const bOrder = asOrderTableEntry(b) - if (aIndexOfPercent > 0 && bIndexOfPercent > 0) { - return +aOrder.fee.substring(0, aIndexOfPercent) - +bOrder.fee.substring(0, bIndexOfPercent) - } + if (aOrder.type.isAbsolute !== bOrder.type.isAbsolute) { + return aOrder.type.isAbsolute === true ? 1 : -1 } - - return 0 + return aOrder.fee.value - bOrder.fee.value }), [SORT_KEYS.minimumSize]: (array) => array.sort((a, b) => a.minimumSize - b.minimumSize), [SORT_KEYS.maximumSize]: (array) => array.sort((a, b) => a.maximumSize - b.maximumSize), [SORT_KEYS.minerFeeContribution]: (array) => array.sort((a, b) => a.minerFeeContribution - b.minerFeeContribution), + [SORT_KEYS.counterparty]: (array) => array.sort((a, b) => { const val = a.counterparty.localeCompare(b.counterparty) return val !== 0 ? val : +a.orderId - +b.orderId }), - [SORT_KEYS.bondValue]: (array) => array.sort((a, b) => a.bondValue - b.bondValue), + [SORT_KEYS.bondValue]: (array) => array.sort((a, b) => a.bondValue.value - b.bondValue.value), }, }, ) @@ -197,7 +240,7 @@ const OrderbookTable = ({ data }: OrderbookTableProps) => { {t('orderbook.table.heading_maximum_size')} - + {t('orderbook.table.heading_miner_fee_contribution')} {t('orderbook.table.heading_bond_value')} @@ -205,27 +248,27 @@ const OrderbookTable = ({ data }: OrderbookTableProps) => { {tableList.map((item) => { - const order = toOrder(item) + const order = asOrderTableEntry(item) return ( - {order.counterparty} + {order.counterparty} {order.orderId} - {renderOrderType(order.type, t)} - {renderOrderFee(order.fee, settings)} + {renderOrderType(order.type)} + {renderOrderFee(order.fee.displayValue, settings)} - + - {order.bondValue} + {order.bondValue.displayValue} ) })} @@ -240,35 +283,63 @@ const OrderbookTable = ({ data }: OrderbookTableProps) => { ) } +const offerToTableEntry = (offer: ObwatchApi.Offer, t: TFunction): OrderTableEntry => { + return { + type: orderTypeProps(offer, t), + counterparty: offer.counterparty, + orderId: String(offer.oid), + fee: + typeof offer.cjfee === 'number' + ? { + value: offer.cjfee, + displayValue: String(offer.cjfee), + } + : (() => { + const value = parseFloat(offer.cjfee) + return { + value, + displayValue: factorToPercentage(value).toFixed(4) + '%', + } + })(), + minerFeeContribution: String(offer.txfee), + minimumSize: String(offer.minsize), + maximumSize: String(offer.maxsize), + bondValue: { + value: offer.fidelity_bond_value, + displayValue: String(offer.fidelity_bond_value.toFixed(0)), + }, + } +} + interface OrderbookProps { - orders: ObwatchApi.Order[] + entries: OrderTableEntry[] refresh: (signal: AbortSignal) => Promise nickname?: string } -export function Orderbook({ orders, refresh, nickname }: OrderbookProps) { +export function Orderbook({ entries, refresh, nickname }: OrderbookProps) { const { t } = useTranslation() const settings = useSettings() const [search, setSearch] = useState('') const [isLoadingRefresh, setIsLoadingRefresh] = useState(false) const [isHighlightOwnOffers, setIsHighlightOwnOffers] = useState(false) - const [highlightedOrders, setHighlightedOrders] = useState([]) + const [highlightedOrders, setHighlightedOrders] = useState([]) const tableData: TableTypes.Data = useMemo(() => { const searchVal = search.replace('.', '').toLowerCase() const filteredOrders = searchVal === '' - ? orders - : orders.filter((order) => { + ? entries + : entries.filter((entry) => { return ( - order.type.toLowerCase().includes(searchVal) || - order.counterparty.toLowerCase().includes(searchVal) || - order.fee.replace('.', '').toLowerCase().includes(searchVal) || - order.minimumSize.replace('.', '').toLowerCase().includes(searchVal) || - order.maximumSize.replace('.', '').toLowerCase().includes(searchVal) || - order.minerFeeContribution.replace('.', '').toLowerCase().includes(searchVal) || - order.bondValue.replace('.', '').toLowerCase().includes(searchVal) || - order.orderId.toLowerCase().includes(searchVal) + entry.type.displayValue.toLowerCase().includes(searchVal) || + entry.counterparty.toLowerCase().includes(searchVal) || + entry.fee.displayValue.replace('.', '').toLowerCase().includes(searchVal) || + entry.minimumSize.replace('.', '').toLowerCase().includes(searchVal) || + entry.maximumSize.replace('.', '').toLowerCase().includes(searchVal) || + entry.minerFeeContribution.replace('.', '').toLowerCase().includes(searchVal) || + entry.bondValue.displayValue.replace('.', '').toLowerCase().includes(searchVal) || + entry.orderId.toLowerCase().includes(searchVal) ) }) const nodes = filteredOrders.map((order) => ({ @@ -278,9 +349,9 @@ export function Orderbook({ orders, refresh, nickname }: OrderbookProps) { })) return { nodes } - }, [orders, search, highlightedOrders]) + }, [entries, search, highlightedOrders]) - const counterpartyCount = useMemo(() => new Set(orders.map((it) => it.counterparty)).size, [orders]) + const counterpartyCount = useMemo(() => new Set(entries.map((it) => it.counterparty)).size, [entries]) const counterpartyCountFiltered = useMemo( () => new Set(tableData.nodes.map((it) => it.counterparty)).size, [tableData], @@ -290,9 +361,9 @@ export function Orderbook({ orders, refresh, nickname }: OrderbookProps) { if (!nickname || !isHighlightOwnOffers) { setHighlightedOrders([]) } else { - setHighlightedOrders(orders.filter((it) => it.counterparty === nickname)) + setHighlightedOrders(entries.filter((it) => it.counterparty === nickname)) } - }, [orders, nickname, isHighlightOwnOffers]) + }, [entries, nickname, isHighlightOwnOffers]) return (
@@ -323,7 +394,7 @@ export function Orderbook({ orders, refresh, nickname }: OrderbookProps) { {search === '' ? ( <> {t('orderbook.text_orderbook_summary', { - count: orders.length, + count: entries.length, counterpartyCount, })} @@ -352,7 +423,7 @@ export function Orderbook({ orders, refresh, nickname }: OrderbookProps) {
- {orders.length === 0 ? ( + {entries.length === 0 ? ( {t('orderbook.alert_empty_orderbook')} ) : ( <> @@ -385,7 +456,8 @@ export function OrderbookOverlay({ nickname, show, onHide }: OrderbookOverlayPro const [alert, setAlert] = useState() const [isInitialized, setIsInitialized] = useState(false) const [isLoading, setIsLoading] = useState(true) - const [orders, setOrders] = useState(null) + const [offers, setOffers] = useState() + const tableEntries = useMemo(() => offers && offers.map((offer) => offerToTableEntry(offer, t)), [offers, t]) const refresh = useCallback( (signal: AbortSignal) => { @@ -398,11 +470,15 @@ export function OrderbookOverlay({ nickname, show, onHide }: OrderbookOverlayPro return ObwatchApi.fetchOrderbook({ signal }) }) - .then((orders) => { + .then((orderbook) => { if (signal.aborted) return - setOrders(orders) setAlert(undefined) + setOffers(orderbook.offers || []) + + if (isDevMode()) { + console.table(orderbook.offers) + } }) .catch((e) => { if (signal.aborted) return @@ -468,10 +544,10 @@ export function OrderbookOverlay({ nickname, show, onHide }: OrderbookOverlayPro ) : ( <> {alert && {alert.message}} - {orders && ( + {tableEntries && ( - + )} diff --git a/src/libs/JmObwatchApi.ts b/src/libs/JmObwatchApi.ts index 6cff3e4e..23d00987 100644 --- a/src/libs/JmObwatchApi.ts +++ b/src/libs/JmObwatchApi.ts @@ -1,76 +1,30 @@ -import { Helper as ApiHelper } from '../libs/JmWalletApi' +import { AmountSats, Helper as ApiHelper } from '../libs/JmWalletApi' const basePath = () => `${window.JM.PUBLIC_PATH}/obwatch` -export const ABSOLUTE_ORDER_TYPE_VAL = 'Native SW Absolute Fee' -export const RELATIVE_ORDER_TYPE_VAL = 'Native SW Relative Fee' - -export interface Order { - type: string // example: "Native SW Absolute Fee" or "Native SW Relative Fee" +export interface Offer { counterparty: string // example: "J5Bv3JSxPFWm2Yjb" - orderId: string // example: "0" (not unique!) - fee: string // example: "0.00000250" (abs offers) or "0.000100%" (rel offers) - minerFeeContribution: string // example: "0.00000000" - minimumSize: string // example: "0.00027300" - maximumSize: string // example: "2374.99972700" - bondValue: string // example: "0" (no fb) or "0.0000052877962973" + oid: number // example: 0 (not unique!) + ordertype: string // example: "sw0absoffer" or "sw0reloffer" + minsize: AmountSats // example: 27300 + maxsize: AmountSats // example: 237499972700 + txfee: AmountSats // example: 0 + cjfee: AmountSats | string // example: 250 (abs offers) or "0.00017" (rel offers) + fidelity_bond_value: number // example: 0 (no fb) or 114557102085.28133 } -const ORDER_KEYS: (keyof Order)[] = [ - 'type', - 'counterparty', - 'orderId', - 'fee', - 'minerFeeContribution', - 'minimumSize', - 'maximumSize', - 'bondValue', -] - -const parseOrderbook = (res: Response): Promise => { - if (!res.ok) { - // e.g. error is raised if ob-watcher is not running - return ApiHelper.throwError(res) - } - - return res.text().then((html) => { - var parser = new DOMParser() - var doc = parser.parseFromString(html, 'text/html') - - const tables = doc.getElementsByTagName('table') - if (tables.length !== 1) { - throw new Error('Cannot find orderbook table') - } - const orderbookTable = tables[0] - const tbodies = [...orderbookTable.children].filter((child) => child.tagName.toLowerCase() === 'tbody') - if (tbodies.length !== 1) { - throw new Error('Cannot find orderbook table body') - } - - const tbody = tbodies[0] - - const orders: Order[] = [...tbody.children] - .filter((row) => row.tagName.toLowerCase() === 'tr') - .filter((row) => row.children.length > 0) - .map((row) => [...row.children].filter((child) => child.tagName.toLowerCase() === 'td')) - .filter((cols) => cols.length === ORDER_KEYS.length) - .map((cols) => { - const data: unknown = ORDER_KEYS.map((key, index) => ({ [key]: cols[index].innerHTML })).reduce( - (acc, curr) => ({ ...acc, ...curr }), - {}, - ) - return data as Order - }) +export interface OrderbookJson { + offers?: Offer[] +} - return orders +const orderbookJson = async ({ signal }: { signal: AbortSignal }) => { + return await fetch(`${basePath()}/orderbook.json`, { + signal, }) } -// TODO: why is "orderbook.json" always empty? -> Parse HTML in the meantime.. ¯\_(ツ)_/¯ -const fetchOrderbook = async ({ signal }: { signal: AbortSignal }) => { - return await fetch(`${basePath()}/`, { - signal, - }).then((res) => parseOrderbook(res)) +const fetchOrderbook = async (options: { signal: AbortSignal }): Promise => { + return orderbookJson(options).then((res) => (res.ok ? res.json() : ApiHelper.throwError(res))) } const refreshOrderbook = async ({ signal }: { signal: AbortSignal }) => { diff --git a/src/utils.ts b/src/utils.ts index 9c68b985..4cdd7eff 100644 --- a/src/utils.ts +++ b/src/utils.ts @@ -1,4 +1,4 @@ -import { WalletFileName } from './libs/JmWalletApi' +import { OfferType, WalletFileName } from './libs/JmWalletApi' const BTC_FORMATTER = new Intl.NumberFormat('en-US', { minimumIntegerDigits: 1, @@ -74,6 +74,12 @@ export const factorToPercentage = (val: number, precision = 6) => { return Number((val * 100).toFixed(precision)) } +// can be any of ['sw0reloffer', 'swreloffer', 'reloffer'] +export const isRelativeOffer = (offertype: OfferType) => offertype.includes('reloffer') + +// can be any of ['sw0absoffer', 'swabsoffer', 'absoffer'] +export const isAbsoluteOffer = (offertype: OfferType) => offertype.includes('absoffer') + export const isValidNumber = (val: number | undefined) => typeof val === 'number' && !isNaN(val) export const UNKNOWN_VERSION: SemVer = { major: 0, minor: 0, patch: 0, raw: 'unknown' }