Skip to content

Commit

Permalink
Integrate new Seen Tx Checkpoint API
Browse files Browse the repository at this point in the history
  • Loading branch information
samholmes committed Jan 15, 2025
1 parent 60d373c commit 5267e07
Show file tree
Hide file tree
Showing 30 changed files with 218 additions and 73 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
## Unreleased

- added: Implement `updateInfoPayload` for `EdgeCurrencyEngine` to get currency info updates from info-server.
- changed: Implement new Seen Tx Checkpoint API for all currencies.
- fixed: Fixed cleaner failure for getInfo Blockbook request.

## 3.4.5 (2024-12-12)
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,7 @@
"base-x": "^4.0.0",
"chai": "^4.2.0",
"chai-as-promised": "^7.1.1",
"edge-core-js": "^2.7.0",
"edge-core-js": "^2.24.0",
"esbuild-loader": "^4.1.0",
"eslint": "^7.14.0",
"eslint-config-standard-kit": "0.15.1",
Expand Down
38 changes: 31 additions & 7 deletions src/common/plugin/EngineEmitter.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import {
EdgeCurrencyEngineCallbacks,
EdgeTransaction,
EdgeTransactionEvent,
EdgeTxidMap
} from 'edge-core-js/types'
import { EventEmitter } from 'events'
Expand All @@ -9,9 +10,17 @@ import { SubscribeAddressResponse } from '../utxobased/network/blockbookApi'

export declare interface EngineEmitter {
emit: ((
event: EngineEvent.TRANSACTIONS_CHANGED,
transactions: EdgeTransaction[]
event: EngineEvent.SEEN_TX_CHECKPOINT,
checkpoint: string
) => boolean) &
((
event: EngineEvent.TRANSACTIONS,
transactionEvents: EdgeTransactionEvent[]
) => boolean) &
((
event: EngineEvent.TRANSACTIONS_CHANGED,
transactions: EdgeTransaction[]
) => boolean) &
((
event: EngineEvent.ADDRESS_BALANCE_CHANGED,
currencyCode: string,
Expand All @@ -36,9 +45,19 @@ export declare interface EngineEmitter {
((event: EngineEvent.TXIDS_CHANGED, txids: EdgeTxidMap) => boolean)

on: ((
event: EngineEvent.TRANSACTIONS_CHANGED,
listener: (transactions: EdgeTransaction[]) => Promise<void> | void
event: EngineEvent.SEEN_TX_CHECKPOINT,
listener: (checkpoint: string) => Promise<void> | void
) => this) &
((
event: EngineEvent.TRANSACTIONS,
listener: (
transactionEvents: EdgeTransactionEvent[]
) => Promise<void> | void
) => boolean) &
((
event: EngineEvent.TRANSACTIONS_CHANGED,
listener: (transactions: EdgeTransaction[]) => Promise<void> | void
) => this) &
((
event: EngineEvent.ADDRESS_BALANCE_CHANGED,
listener: (
Expand Down Expand Up @@ -76,6 +95,9 @@ export declare interface EngineEmitter {
export class EngineEmitter extends EventEmitter {}

export enum EngineEvent {
SEEN_TX_CHECKPOINT = 'seen:tx:checkpoint',
TRANSACTIONS = 'transactions',
/** @deprecated Use TRANSACTIONS */
TRANSACTIONS_CHANGED = 'transactions:changed',
WALLET_BALANCE_CHANGED = 'wallet:balance:changed',
ADDRESS_BALANCE_CHANGED = 'address:balance:changed',
Expand All @@ -93,16 +115,18 @@ export const makeEngineEmitter = (
): EngineEmitter => {
const emitter = new EngineEmitter()

emitter.on(EngineEvent.TRANSACTIONS_CHANGED, callbacks.onTransactionsChanged)
emitter.on(EngineEvent.WALLET_BALANCE_CHANGED, callbacks.onBalanceChanged)
emitter.on(EngineEvent.ADDRESSES_CHECKED, callbacks.onAddressesChecked)
emitter.on(
EngineEvent.BLOCK_HEIGHT_CHANGED,
(_uri: string, height: number) => {
callbacks.onBlockHeightChanged(height)
}
)
emitter.on(EngineEvent.ADDRESSES_CHECKED, callbacks.onAddressesChecked)
emitter.on(EngineEvent.SEEN_TX_CHECKPOINT, callbacks.onSeenTxCheckpoint)
emitter.on(EngineEvent.TRANSACTIONS, callbacks.onTransactions)
emitter.on(EngineEvent.TRANSACTIONS_CHANGED, callbacks.onTransactionsChanged)
emitter.on(EngineEvent.TXIDS_CHANGED, callbacks.onTxidsChanged)
emitter.on(EngineEvent.WALLET_BALANCE_CHANGED, callbacks.onBalanceChanged)

return emitter
}
6 changes: 6 additions & 0 deletions src/common/utxobased/db/Models/TransactionData.ts
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,12 @@ export const toEdgeTransaction = async (
memos: [],
nativeAmount: tx.ourAmount,
networkFee: tx.fees,
networkFees: [
{
tokenId: null,
nativeAmount: tx.fees
}
],
ourReceiveAddresses,
signedTx: tx.hex,
tokenId: null,
Expand Down
37 changes: 34 additions & 3 deletions src/common/utxobased/engine/UtxoEngine.ts
Original file line number Diff line number Diff line change
Expand Up @@ -128,10 +128,34 @@ export async function makeUtxoEngine(
// private keys.
let nonceDataLayer: DataLayer | undefined

/**
* This is a stateful function, which means it is both a setter and a getter.
* The state is the cached seenTxCheckpoint for the engine during runtime.
* It is initialized with the seenTxCheckpoint state from the core via
* `EdgeCurrencyEngineOptions`.
* It is updated by the engine's processor as the wallet syncs.
* Once the wallet fully syncs, the seenTxCheckpoint value is emitted to the
* `onSeenTxCheckpoint` callback so the core can persist this state to disk.
*/
const seenTxCheckpoint = ((state?: string) => (
value?: string
): string | undefined => {
if (value != null) state = value
return state
})()

// Initialize the seenTxCheckpoint with the value from the core.
seenTxCheckpoint(config.engineOptions.seenTxCheckpoint)

emitter.on(EngineEvent.SEEN_TX_CHECKPOINT, checkpoint => {
seenTxCheckpoint(checkpoint)
})

const engineProcessor = makeUtxoEngineProcessor({
...config,
dataLayer,
pluginState,
seenTxCheckpoint,
walletTools,
walletInfo
})
Expand Down Expand Up @@ -692,6 +716,12 @@ export async function makeUtxoEngine(
memos,
nativeAmount,
networkFee,
networkFees: [
{
tokenId: null,
nativeAmount: networkFee
}
],
otherParams,
ourReceiveAddresses,
signedTx: '',
Expand Down Expand Up @@ -740,7 +770,9 @@ export async function makeUtxoEngine(
walletId: walletInfo.id,
walletTools
})
emitter.emit(EngineEvent.TRANSACTIONS_CHANGED, [rbfEdgeTx])
emitter.emit(EngineEvent.TRANSACTIONS, [
{ isNew: false, transaction: rbfEdgeTx }
])
}
}

Expand All @@ -750,8 +782,6 @@ export async function makeUtxoEngine(
scriptPubkeys: edgeTx.otherParams?.ourScriptPubkeys
})

emitter.emit(EngineEvent.TRANSACTIONS_CHANGED, [edgeTx])

/*
Get the wallet's UTXOs from the new transaction and save them to the processsor.
*/
Expand Down Expand Up @@ -987,6 +1017,7 @@ export async function makeUtxoEngine(
gapLimit: 0
}
},
seenTxCheckpoint: () => '0',
walletTools: tmpWalletTools,
walletInfo: tmpWalletInfo
})
Expand Down
69 changes: 64 additions & 5 deletions src/common/utxobased/engine/UtxoEngineProcessor.ts
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,7 @@ export interface UtxoEngineProcessor {

export interface UtxoEngineProcessorConfig extends EngineConfig {
dataLayer: DataLayer
seenTxCheckpoint: (value?: string) => string | undefined
walletTools: UtxoWalletTools
walletInfo: SafeWalletInfo
}
Expand All @@ -96,6 +97,7 @@ export function makeUtxoEngineProcessor(
engineOptions,
pluginState,
pluginInfo,
seenTxCheckpoint,
walletInfo,
walletTools
} = config
Expand Down Expand Up @@ -170,6 +172,27 @@ export function makeUtxoEngineProcessor(
}
}

const updateSeenTxCheckpoint = (): void => {
// Only update the seenTxCheckpoint if the wallet is fully synced.
// This ensure that initial syncs without a defined seenTxCheckpoint,
// will not incorrectly update the seenTxCheckpoint in the middle of an
// initial sync.
if (processedPercent < 1) return

const seenTxCheckpoint = common.seenTxCheckpoint()
const seenTxBlockHeight =
seenTxCheckpoint != null ? parseInt(seenTxCheckpoint) : undefined

// Update the seenTxCheckpoint
if (
seenTxBlockHeight == null ||
common.maxSeenTxBlockHeight > seenTxBlockHeight
) {
const newSeenTxCheckpoint = common.maxSeenTxBlockHeight.toString()
common.emitter.emit(EngineEvent.SEEN_TX_CHECKPOINT, newSeenTxCheckpoint)
}
}

const lock = new AwaitLock()

const serverStates = makeServerStates({
Expand All @@ -189,8 +212,11 @@ export function makeUtxoEngineProcessor(
emitter,
taskCache,
updateProgressRatio,
updateSeenTxCheckpoint,
io,
log,
maxSeenTxBlockHeight: 0,
seenTxCheckpoint,
serverStates,
walletFormats,
lock
Expand Down Expand Up @@ -603,8 +629,11 @@ interface CommonParams {
emitter: EngineEmitter
taskCache: TaskCache
updateProgressRatio: () => void
updateSeenTxCheckpoint: () => void
io: EdgeIo
log: EdgeLog
maxSeenTxBlockHeight: number
seenTxCheckpoint: (value?: string) => string | undefined
serverStates: ServerStates
walletFormats: CurrencyFormat[]
lock: AwaitLock
Expand Down Expand Up @@ -960,7 +989,7 @@ export async function* pickNextTask(
hasProcessedAtLeastOnce = true
cacheItem.processing = true
removeItem(transactionUpdateCache, txId)
yield* processTransactionUpdate(common, {
yield* processCheckTransactionConfirmation(common, {
serverState,
serverUri,
txId
Expand Down Expand Up @@ -1038,7 +1067,9 @@ async function* processTransactionsSpecificUpdate(
walletId: common.walletInfo.id,
walletTools: common.walletTools
})
common.emitter.emit(EngineEvent.TRANSACTIONS_CHANGED, [edgeTx])
common.emitter.emit(EngineEvent.TRANSACTIONS, [
{ isNew: false, transaction: edgeTx }
])

// Add the txid to the server cache
serverState.txids.add(txId)
Expand All @@ -1060,7 +1091,7 @@ async function* processTransactionsSpecificUpdate(
* It updates the transaction and all of the transaction's UTXO with the
* blockHeight received from the network.
*/
async function* processTransactionUpdate(
async function* processCheckTransactionConfirmation(
common: CommonParams,
args: {
serverState: ServerState
Expand Down Expand Up @@ -1105,7 +1136,9 @@ async function* processTransactionUpdate(
walletId: common.walletInfo.id,
walletTools: common.walletTools
})
common.emitter.emit(EngineEvent.TRANSACTIONS_CHANGED, [edgeTx])
common.emitter.emit(EngineEvent.TRANSACTIONS, [
{ isNew: false, transaction: edgeTx }
])

if (needsTxSpecific(common)) {
// Add task to grab transactionSpecific payload
Expand Down Expand Up @@ -1174,8 +1207,16 @@ async function* processAddressForTransactions(
addressData.used = true
}

const seenTxCheckpoint = common.seenTxCheckpoint()
const seenTxBlockHeight =
seenTxCheckpoint != null ? parseInt(seenTxCheckpoint) : undefined

// Process and save the address's transactions
for (const txResponse of transactions) {
const [existingTx] = await common.dataLayer.fetchTransactions({
txId: txResponse.txid
})

const tx = processTransactionResponse(common, { txResponse })
const processedTx = await common.dataLayer.saveTransaction({
tx,
Expand All @@ -1188,7 +1229,22 @@ async function* processAddressForTransactions(
walletId: common.walletInfo.id,
walletTools: common.walletTools
})
common.emitter.emit(EngineEvent.TRANSACTIONS_CHANGED, [edgeTx])

// Keep track of transactions which are determined to be unseen:
const isNew =
seenTxBlockHeight != null &&
// Unseen in the DataLayer
existingTx == null &&
// The tx unconfirmed or confirmed after/at the last seenTxCheckpoint
(tx.blockHeight === 0 || tx.blockHeight > seenTxBlockHeight)

common.emitter.emit(EngineEvent.TRANSACTIONS, [
{ isNew, transaction: edgeTx }
])

if (edgeTx.blockHeight > common.maxSeenTxBlockHeight) {
common.maxSeenTxBlockHeight = edgeTx.blockHeight
}

if (needsTxSpecific(common)) {
// Add task to grab transactionSpecific payload
Expand All @@ -1198,6 +1254,9 @@ async function* processAddressForTransactions(
}
}

// Make sure to update the seenTxCheckpoint after processing the transactions
common.updateSeenTxCheckpoint()

// Halt on finishing the processing of address transaction until
// we have progressed through all of the blockbook pages
if (page < totalPages) {
Expand Down
4 changes: 3 additions & 1 deletion src/common/utxobased/info/badcoin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,9 +8,10 @@ import {
} from './commonInfo'

const currencyInfo: EdgeCurrencyInfo = {
assetDisplayName: 'Badcoin',
chainDisplayName: 'Badcoin',
currencyCode: 'BAD',
customFeeTemplate: utxoCustomFeeTemplate,
displayName: 'Badcoin',
memoOptions: utxoMemoOptions,
pluginId: 'badcoin',
walletType: 'wallet:badcoin',
Expand All @@ -32,6 +33,7 @@ const currencyInfo: EdgeCurrencyInfo = {
blockbookServers: [],
enableCustomServers: false
},
displayName: 'Badcoin',
metaTokens: []
}

Expand Down
4 changes: 3 additions & 1 deletion src/common/utxobased/info/bitcoin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,11 @@ import {
} from './commonInfo'

const currencyInfo: EdgeCurrencyInfo = {
assetDisplayName: 'Bitcoin',
canReplaceByFee: true,
chainDisplayName: 'Bitcoin',
currencyCode: 'BTC',
customFeeTemplate: utxoCustomFeeTemplate,
displayName: 'Bitcoin',
memoOptions: utxoMemoOptions,
pluginId: 'bitcoin',
walletType: 'wallet:bitcoin',
Expand Down Expand Up @@ -44,6 +45,7 @@ const currencyInfo: EdgeCurrencyInfo = {
],
enableCustomServers: false
},
displayName: 'Bitcoin',
metaTokens: []
}

Expand Down
Loading

0 comments on commit 5267e07

Please sign in to comment.