Skip to content

Commit c142994

Browse files
feat: incentivization POC: add double-spend check for txid-based eligibility (#3264)
* add double-spend check for txid-based eligibility * Apply suggestions from code review Co-authored-by: Ivan FB <[email protected]> * split assert into two in double-spending test * remove unnecessary import Co-authored-by: Ivan FB <[email protected]> --------- Co-authored-by: Ivan FB <[email protected]>
1 parent 46747fd commit c142994

File tree

2 files changed

+26
-3
lines changed

2 files changed

+26
-3
lines changed

tests/incentivization/test_poc.nim

+18-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
{.used.}
22

33
import
4-
std/[options],
4+
std/options,
55
testutils/unittests,
66
chronos,
77
web3,
@@ -194,5 +194,22 @@ suite "Waku Incentivization PoC Eligibility Proofs":
194194

195195
assert isEligible.isOk(), isEligible.error
196196

197+
asyncTest "incentivization PoC: double-spend tx is not eligible":
198+
## Test that the same tx submitted twice is not eligible the second time
199+
200+
let eligibilityProof =
201+
EligibilityProof(proofOfPayment: some(@(txHashRightReceiverRightAmount.bytes())))
202+
203+
let isEligibleOnce = await manager.isEligibleTxId(
204+
eligibilityProof, receiverExpected, TxValueExpectedWei
205+
)
206+
207+
let isEligibleTwice = await manager.isEligibleTxId(
208+
eligibilityProof, receiverExpected, TxValueExpectedWei
209+
)
210+
211+
assert isEligibleOnce.isOk()
212+
assert isEligibleTwice.isErr(), isEligibleTwice.error
213+
197214
# Stop Anvil daemon
198215
stopAnvil(runAnvil)

waku/incentivization/eligibility_manager.nim

+8-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import std/options, chronos, web3, stew/byteutils, stint, results, chronicles
1+
import std/[options, sets], chronos, web3, stew/byteutils, stint, results, chronicles
22

33
import waku/incentivization/rpc, tests/waku_rln_relay/[utils_onchain, utils]
44

@@ -7,12 +7,13 @@ const TxReceiptQueryTimeout = 3.seconds
77

88
type EligibilityManager* = ref object # FIXME: make web3 private?
99
web3*: Web3
10+
seenTxIds*: HashSet[TxHash]
1011

1112
# Initialize the eligibilityManager with a web3 instance
1213
proc init*(
1314
T: type EligibilityManager, ethClient: string
1415
): Future[EligibilityManager] {.async.} =
15-
result = EligibilityManager(web3: await newWeb3(ethClient))
16+
return EligibilityManager(web3: await newWeb3(ethClient), seenTxIds: initHashSet[TxHash]())
1617
# TODO: handle error if web3 instance is not established
1718

1819
# Clean up the web3 instance
@@ -60,6 +61,11 @@ proc isEligibleTxId*(
6061
var tx: TransactionObject
6162
var txReceipt: ReceiptObject
6263
let txHash = TxHash.fromHex(byteutils.toHex(eligibilityProof.proofOfPayment.get()))
64+
# check that it is not a double-spend
65+
let txHashWasSeen = (txHash in eligibilityManager.seenTxIds)
66+
eligibilityManager.seenTxIds.incl(txHash)
67+
if txHashWasSeen:
68+
return err("TxHash " & $txHash & " was already checked (double-spend attempt)")
6369
try:
6470
let txAndTxReceipt = await eligibilityManager.getTxAndTxReceipt(txHash)
6571
txAndTxReceipt.isOkOr:

0 commit comments

Comments
 (0)