-
Notifications
You must be signed in to change notification settings - Fork 671
/
utxo_fetching.go
106 lines (94 loc) · 2.82 KB
/
utxo_fetching.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
// Copyright (C) 2019-2021, Ava Labs, Inc. All rights reserved.
// See the file LICENSE for licensing terms.
package avax
import (
"bytes"
"fmt"
"math"
"github.com/ava-labs/avalanchego/ids"
safemath "github.com/ava-labs/avalanchego/utils/math"
)
// GetBalance returns the current balance of [addrs]
func GetBalance(db UTXOReader, addrs ids.ShortSet) (uint64, error) {
utxos, err := GetAllUTXOs(db, addrs)
if err != nil {
return 0, fmt.Errorf("couldn't get UTXOs: %w", err)
}
balance := uint64(0)
for _, utxo := range utxos {
if out, ok := utxo.Out.(Amounter); ok {
if balance, err = safemath.Add64(out.Amount(), balance); err != nil {
return 0, err
}
}
}
return balance, nil
}
func GetAllUTXOs(db UTXOReader, addrs ids.ShortSet) ([]*UTXO, error) {
utxos, _, _, err := GetPaginatedUTXOs(
db,
addrs,
ids.ShortEmpty,
ids.Empty,
math.MaxInt,
)
return utxos, err
}
// GetPaginatedUTXOs returns UTXOs such that at least one of the addresses in
// [addrs] is referenced.
//
// Returns at most [limit] UTXOs.
//
// Only returns UTXOs associated with addresses >= [startAddr].
//
// For address [startAddr], only returns UTXOs whose IDs are greater than
// [startUTXOID].
//
// Returns:
// * The fetched UTXOs
// * The address associated with the last UTXO fetched
// * The ID of the last UTXO fetched
func GetPaginatedUTXOs(
db UTXOReader,
addrs ids.ShortSet,
lastAddr ids.ShortID,
lastUTXOID ids.ID,
limit int,
) ([]*UTXO, ids.ShortID, ids.ID, error) {
var (
utxos []*UTXO
seen ids.Set // IDs of UTXOs already in the list
searchSize = limit // the limit diminishes which can impact the expected return
addrsList = addrs.SortedList() // enforces the same ordering for pagination
)
for _, addr := range addrsList {
start := ids.Empty
if comp := bytes.Compare(addr.Bytes(), lastAddr.Bytes()); comp == -1 { // Skip addresses before [startAddr]
continue
} else if comp == 0 {
start = lastUTXOID
}
lastAddr = addr // The last address searched
utxoIDs, err := db.UTXOIDs(addr.Bytes(), start, searchSize) // Get UTXOs associated with [addr]
if err != nil {
return nil, ids.ShortID{}, ids.ID{}, fmt.Errorf("couldn't get UTXOs for address %s: %w", addr, err)
}
for _, utxoID := range utxoIDs {
lastUTXOID = utxoID // The last searched UTXO - not the last found
if seen.Contains(utxoID) { // Already have this UTXO in the list
continue
}
utxo, err := db.GetUTXO(utxoID)
if err != nil {
return nil, ids.ShortID{}, ids.ID{}, fmt.Errorf("couldn't get UTXO %s: %w", utxoID, err)
}
utxos = append(utxos, utxo)
seen.Add(utxoID)
limit--
if limit <= 0 {
return utxos, lastAddr, lastUTXOID, nil // Found [limit] utxos; stop.
}
}
}
return utxos, lastAddr, lastUTXOID, nil // Didn't reach the [limit] utxos; no more were found
}