-
Notifications
You must be signed in to change notification settings - Fork 28
/
track-ram-usage.ts
164 lines (140 loc) · 5.64 KB
/
track-ram-usage.ts
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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
import { DFUSE_API_KEY, runMain, DFUSE_API_NETWORK } from "../../config"
import {
createDfuseClient,
SearchTransactionRow,
RamOp,
DfuseClient,
flattenActionTraces,
waitFor,
} from "@dfuse/client"
const account = "eoscanadacom"
const resultPerPage = 50
const maxResults = 500
/**
* In this example, we showcase how to use a cursor to paginate through
* multiple pages of search results.
*
* At the same time, we go over how to work with the search result rows
* to obtain only the matching actions of the transaction, instead of dealing
* with all the actions in the transaction.
*
* This is usually what most people expect to have from the API (the
* actions of the transaction that matched the search criteria).
*/
async function main(): Promise<void> {
const client = createDfuseClient({ apiKey: DFUSE_API_KEY, network: DFUSE_API_NETWORK })
const query = `(ram.released:${account} OR ram.consumed:${account})`
try {
let resultCount = 0
let runningTotal = 0
let cursor = ""
let pageCount = 0
while (resultCount <= maxResults) {
const page = await fetchPage(client, query, cursor)
pageCount++
resultCount += page.transactions.length
cursor = page.cursor
page.transactions.forEach((result: SearchTransactionRow) => {
console.log(`- https://${inferEosqHost()}/tx/${result.lifecycle.id}`)
/**
* A transaction is composed of a deep trace of action traces
* representing the execution of the various transaction's actions.
*
* In dfuse API, there is sometimes the need to assiocate some data
* with a particular action trace. For example, a database operation
* backlink through an `action_idx` property to the actual action
* trace that generates this operation.
*
* The `action_idx` is not easy to work with at the transaction
* level since the dfuse API consumer needs to perform a depth-first
* traversal of the execution tree to determine the actual action.
*
* To ease that process, the helper `flattenActionTraces` can be used
* to get a flattened list of action traces, where each index of the list
* map to the correct `action_idx` value.
*
* This can than later be used to easily find the action representing
* a given `action_idx` value. See below for usage with RAM op.
*/
const flattenedActionTraces = flattenActionTraces(result.lifecycle)
// /**
// * Using dfuse Search API, you received the full transaction as a result.
// * However a transaction may contain 10 different actions,
// * while only 2 out of the 10 actually matches the query.
// *
// * The `matchingActionTraces` helper can easily be used to extract
// * only the matching action traces out of a `SearchTransactionRow`
// * result.
// */
// const actionTraces = matchingActionTraces(result)
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
result.lifecycle.ramops!.forEach((ramOp: RamOp) => {
// FIXME: Right logic for RAM op!
runningTotal += ramOp.delta
const ramText = `${ramOp.op} ${ramOp.delta}`
const action = flattenedActionTraces[ramOp.action_idx]
const actionText = `${action.receipt.receiver}/${action.act.account}:${action.act.name}`
console.log(` * ${ramText} @ ${actionText} (${runningTotal})`)
})
console.log()
})
/**
* To determine the end of current pages through dfuse Search API, you
* must conditionally check either if the cursor returns empty or
* if the search result count of this request is lower than our
* expected results per page.
*
* **Caveat** In an ascending search, the dfuse Search stops at
* at current time Irreversible Block or Head Block, but more blocks
* might come in the future. This means that even if there is less
* results than expected per page, future blocks might add more
* results, hence the `cursor` not returning as empty.
*
* **Note** Doing a descending search will yield an empty
* string cursor at some point because you will reach the Genesis
* Block of the chain (Block #1).
*/
if (page.cursor === "" || page.transactions.length < resultPerPage) {
// No more pages, stop page fetch
break
}
console.log(`RAM Running Total (${resultCount} transactions included) is ${runningTotal}`)
console.log(`Fetching next page (#${pageCount + 1}) in 5s ...`)
await waitFor(5000)
}
console.log(`Running total is ${runningTotal}`)
console.log(`Completed after reading ${pageCount} page(s)`)
} catch (error) {
console.log("An error occurred", error)
}
client.release()
}
type Page = {
cursor: string
transactions: SearchTransactionRow[]
}
/**
* Fetches a single page of results for a particular query, starting
* back at `cursor` if present and returning the matching actions out
* of our results.
*/
async function fetchPage(client: DfuseClient, query: string, cursor?: string): Promise<Page> {
const searchResult = await client.searchTransactions(query, {
limit: resultPerPage,
cursor,
})
return {
cursor: searchResult.cursor,
transactions: searchResult.transactions || [],
}
}
function inferEosqHost(): string {
if (DFUSE_API_NETWORK === "mainnet") {
return "eosq.app"
}
if (["jungle", "kylin", "worbli"].includes(DFUSE_API_NETWORK)) {
return `${DFUSE_API_NETWORK}.eosq.app`
}
return `${DFUSE_API_NETWORK}`
}
runMain(main)