-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathgen.go
461 lines (425 loc) · 15.1 KB
/
gen.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
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
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
//go:build ignore
// +build ignore
package main
// based on https://github.com/ipld/go-ipld-prime-proto/blob/master/gen/main.go
import (
"fmt"
"os"
"github.com/ipld/go-ipld-prime"
"github.com/ipld/go-ipld-prime/schema"
gengo "github.com/ipld/go-ipld-prime/schema/gen/go"
)
const (
pkgName = "dageth"
)
func main() {
// initialize a new type system
ts := new(schema.TypeSystem)
ts.Init()
// accumulate the different types
accumulateBasicTypes(ts)
accumulateChainTypes(ts)
accumulateConvenienceTypes(ts)
accumulateStateDataStructures(ts)
// verify internal correctness of the types
if errs := ts.ValidateGraph(); errs != nil {
for _, err := range errs {
fmt.Printf("- %s\n", err)
}
os.Exit(1)
}
// generate the code
adjCfg := &gengo.AdjunctCfg{}
gengo.Generate(".", pkgName, *ts, adjCfg)
}
func accumulateBasicTypes(ts *schema.TypeSystem) {
// we could more explicitly type our links with SpawnLinkReference
ts.Accumulate(schema.SpawnLink("Link"))
ts.Accumulate(schema.SpawnBytes("Bytes"))
ts.Accumulate(schema.SpawnString("String"))
ts.Accumulate(schema.SpawnBytes("BigInt"))
ts.Accumulate(schema.SpawnBytes("Uint"))
ts.Accumulate(schema.SpawnBytes("Hash"))
ts.Accumulate(schema.SpawnBytes("Address"))
ts.Accumulate(schema.SpawnBytes("Bloom"))
ts.Accumulate(schema.SpawnBytes("Balance"))
ts.Accumulate(schema.SpawnBytes("OpCode"))
ts.Accumulate(schema.SpawnBytes("Time"))
ts.Accumulate(schema.SpawnBytes("TxType"))
ts.Accumulate(schema.SpawnBool("Bool"))
}
func accumulateChainTypes(ts *schema.TypeSystem) {
/*
type Header struct {
ParentCID &Header
UnclesCID &Uncles
Coinbase Address
StateRootCID &TrieNode
TxRootCID &TrieNode
RctRootCID &TrieNode
Bloom Bloom
Difficulty BigInt
Number BigInt
GasLimit Uint
GasUsed Uint
Time Time
Extra Bytes
MixDigest Hash
Nonce Uint
BaseFee nullable BigInt
}
*/
ts.Accumulate(schema.SpawnStruct("Header",
[]schema.StructField{
schema.SpawnStructField("ParentCID", "Link", false, false),
schema.SpawnStructField("UnclesCID", "Link", false, false),
schema.SpawnStructField("Coinbase", "Address", false, false),
schema.SpawnStructField("StateRootCID", "Link", false, false),
schema.SpawnStructField("TxRootCID", "Link", false, false),
schema.SpawnStructField("RctRootCID", "Link", false, false),
schema.SpawnStructField("Bloom", "Bloom", false, false),
schema.SpawnStructField("Difficulty", "BigInt", false, false),
schema.SpawnStructField("Number", "BigInt", false, false),
schema.SpawnStructField("GasLimit", "Uint", false, false),
schema.SpawnStructField("GasUsed", "Uint", false, false),
schema.SpawnStructField("Time", "Time", false, false),
schema.SpawnStructField("Extra", "Bytes", false, false),
schema.SpawnStructField("MixDigest", "Hash", false, false),
schema.SpawnStructField("Nonce", "Uint", false, false),
schema.SpawnStructField("BaseFee", "BigInt", false, true),
},
schema.SpawnStructRepresentationMap(nil),
))
/*
type Uncles [Header]
*/
ts.Accumulate(schema.SpawnList("Uncles", "Header", false))
/*
type StorageKeys [Hash]
type AccessElement struct {
Address Address
StorageKeys StorageKeys
}
type AccessList [AccessElement]
type Transaction struct {
TxType TxType
ChainID nullable BigInt # null unless the transaction is an EIP-2930 or EIP-1559 transaction
AccountNonce Uint
GasPrice nullable BigInt # null if the transaction is an EIP-1559 transaction
GasTipCap nullable BigInt # null unless the transaciton is an EIP-1559 transaction
GasFeeCap nullable BigInt # null unless the transaction is an EIP-1559 transaction
GasLimit Uint
Recipient nullable Address # null recipient means the tx is a contract creation tx
Amount BigInt
Data Bytes
AccessList nullable AccessList # null unless the transaction is an EIP-2930 or EIP-1559 transaction
# Signature values
V BigInt
R BigInt
S BigInt
}
type Transactions [Transaction]
*/
ts.Accumulate(schema.SpawnList("StorageKeys", "Hash", false))
ts.Accumulate(schema.SpawnStruct("AccessElement",
[]schema.StructField{
schema.SpawnStructField("Address", "Address", false, false),
schema.SpawnStructField("StorageKeys", "StorageKeys", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnList("AccessList", "AccessElement", false))
ts.Accumulate(schema.SpawnStruct("Transaction",
[]schema.StructField{
schema.SpawnStructField("TxType", "TxType", false, false),
schema.SpawnStructField("ChainID", "BigInt", false, true),
schema.SpawnStructField("AccountNonce", "Uint", false, false),
schema.SpawnStructField("GasPrice", "BigInt", false, true),
schema.SpawnStructField("GasTipCap", "BigInt", false, true),
schema.SpawnStructField("GasFeeCap", "BigInt", false, true),
schema.SpawnStructField("GasLimit", "Uint", false, false),
schema.SpawnStructField("Recipient", "Address", false, true),
schema.SpawnStructField("Amount", "BigInt", false, false),
schema.SpawnStructField("Data", "Bytes", false, false),
schema.SpawnStructField("AccessList", "AccessList", false, true),
schema.SpawnStructField("V", "BigInt", false, false),
schema.SpawnStructField("R", "BigInt", false, false),
schema.SpawnStructField("S", "BigInt", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnList("Transactions", "Transaction", false))
/*
type Topics [Hash]
type Log struct {
Address Address
Topics Topics
Data Bytes
}
type Logs [Log]
*/
ts.Accumulate(schema.SpawnList("Topics", "Hash", false))
ts.Accumulate(schema.SpawnStruct("Log",
[]schema.StructField{
schema.SpawnStructField("Address", "Address", false, false),
schema.SpawnStructField("Topics", "Topics", false, false),
schema.SpawnStructField("Data", "Bytes", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnList("Logs", "Log", false))
/*
type Receipt struct {
TxType TxType
// We could make Status an enum
Status Uint // nullable
PostState Hash // nullable
CumulativeGasUsed Uint
Bloom Bloom
Logs Logs
LogRootCID &TrieNode
}
type Receipts [Receipt]
*/
ts.Accumulate(schema.SpawnStruct("Receipt",
[]schema.StructField{
schema.SpawnStructField("TxType", "TxType", false, false),
schema.SpawnStructField("PostState", "Bytes", false, true),
schema.SpawnStructField("Status", "Uint", false, true),
schema.SpawnStructField("CumulativeGasUsed", "Uint", false, false),
schema.SpawnStructField("Bloom", "Bloom", false, false),
schema.SpawnStructField("Logs", "Logs", false, false),
schema.SpawnStructField("LogRootCID", "Link", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnList("Receipts", "Receipt", false))
}
func accumulateStateDataStructures(ts *schema.TypeSystem) {
/*
# TrieNode IPLD
# Node IPLD values are RLP encoded; node IPLD multihashes are always the KECCAK_256 hash of the RLP encoded node bytes and the codec is dependent on the type of the trie
type TrieNode union {
| TrieBranchNode "branch"
| TrieExtensionNode "extension"
| TrieLeafNode "leaf"
} representation keyed
# The below are the expanded representations for the different types of TrieNodes: branch, extension, and leaf
type TrieBranchNode struct {
Child0 nullable Child
Child1 nullable Child
Child2 nullable Child
Child3 nullable Child
Child4 nullable Child
Child5 nullable Child
Child6 nullable Child
Child7 nullable Child
Child8 nullable Child
Child9 nullable Child
ChildA nullable Child
ChildB nullable Child
ChildC nullable Child
ChildD nullable Child
ChildE nullable Child
ChildF nullable Child
Value nullable Value
}
# Value union type used to handle the different values stored in leaf nodes in the different tries
type Value union {
| Transaction "tx"
| Receipt "rct"
| Account "state"
| Bytes "storage"
| Log "log"
} representation keyed
# Child union type used to handle the case where the node is stored directly in the parent node because it is smaller
# than the hash that would otherwise reference the node
type Child union {
| Link &TrieNode
| TrieNode TrieNode
} representation kinded
type TrieExtensionNode struct {
PartialPath Bytes
Child &TrieNode
}
type TrieLeafNode struct {
PartialPath Bytes
Value Value
}
*/
ts.Accumulate(schema.SpawnUnion("Value",
[]schema.TypeName{
"Transaction",
"Receipt",
"Account",
"Bytes",
"Log",
},
schema.SpawnUnionRepresentationKeyed(map[string]schema.TypeName{
"tx": "Transaction",
"rct": "Receipt",
"state": "Account",
"storage": "Bytes",
"log": "Log",
}),
))
ts.Accumulate(schema.SpawnUnion("Child",
[]schema.TypeName{
"Link",
"TrieNode",
},
schema.SpawnUnionRepresentationKinded(map[ipld.Kind]schema.TypeName{
ipld.Kind_Link: "Link",
ipld.Kind_Map: "TrieNode",
}),
))
ts.Accumulate(schema.SpawnStruct("TrieBranchNode",
[]schema.StructField{
schema.SpawnStructField("Child0", "Child", false, true),
schema.SpawnStructField("Child1", "Child", false, true),
schema.SpawnStructField("Child2", "Child", false, true),
schema.SpawnStructField("Child3", "Child", false, true),
schema.SpawnStructField("Child4", "Child", false, true),
schema.SpawnStructField("Child5", "Child", false, true),
schema.SpawnStructField("Child6", "Child", false, true),
schema.SpawnStructField("Child7", "Child", false, true),
schema.SpawnStructField("Child8", "Child", false, true),
schema.SpawnStructField("Child9", "Child", false, true),
schema.SpawnStructField("ChildA", "Child", false, true),
schema.SpawnStructField("ChildB", "Child", false, true),
schema.SpawnStructField("ChildC", "Child", false, true),
schema.SpawnStructField("ChildD", "Child", false, true),
schema.SpawnStructField("ChildE", "Child", false, true),
schema.SpawnStructField("ChildF", "Child", false, true),
schema.SpawnStructField("Value", "Value", false, true),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnStruct("TrieExtensionNode",
[]schema.StructField{
schema.SpawnStructField("PartialPath", "Bytes", false, false),
schema.SpawnStructField("Child", "Link", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnStruct("TrieLeafNode",
[]schema.StructField{
schema.SpawnStructField("PartialPath", "Bytes", false, false),
schema.SpawnStructField("Value", "Value", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnUnion("TrieNode",
[]schema.TypeName{
"TrieBranchNode",
"TrieExtensionNode",
"TrieLeafNode",
},
schema.SpawnUnionRepresentationKeyed(map[string]schema.TypeName{
"branch": "TrieBranchNode",
"extension": "TrieExtensionNode",
"leaf": "TrieLeafNode",
}),
))
/*
type ByteCode bytes
type Account struct {
Nonce Uint
Balance Balance
StorageRootCID &TrieNode
CodeCID &ByteCode
}
*/
ts.Accumulate(schema.SpawnBytes("ByteCode"))
ts.Accumulate(schema.SpawnStruct("Account",
[]schema.StructField{
schema.SpawnStructField("Nonce", "Uint", false, false),
schema.SpawnStructField("Balance", "Balance", false, false),
schema.SpawnStructField("StorageRootCID", "Link", false, false),
schema.SpawnStructField("CodeCID", "Link", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
}
func accumulateConvenienceTypes(ts *schema.TypeSystem) {
/*
# TxTrace contains the EVM context, input, and output for each OPCODE in a transaction that was applied to a specific state
type TxTrace struct {
TxCIDs TxCIDList
# CID link to the root node of the state trie that the above transaction set was applied on top of to produce this trace
StateRootCID &StateTrieNode
Result Bytes
Frames FrameList
Gas Uint
Failed Bool
}
# TxCIDList
# List of CIDs linking to the transactions that were used to generate this trace by applying them onto the state referenced below
# If this trace was produced by the first transaction in a block then this list will contain only that one transaction
# and this trace was produced by applying it directly to the referenced state
# Otherwise, the trace is the output of the last transaction in the list applied to the state produced by
# sequentially applying the proceeding txs to the referenced state
type TxCIDList [&Transaction]
# Frame represents the EVM context, input, and output for a specific OPCODE during a transaction trace
type Frame struct {
Op OpCode
From Address
To Address
Input Bytes
Output Bytes
Gas Uint
Cost Uint
Value BigInt
}
type FrameList [Frame]
*/
ts.Accumulate(schema.SpawnList("TxCIDList", "Link", false))
ts.Accumulate(schema.SpawnStruct("Frame",
[]schema.StructField{
schema.SpawnStructField("Op", "OpCode", false, false),
schema.SpawnStructField("From", "Address", false, false),
schema.SpawnStructField("To", "Address", false, false),
schema.SpawnStructField("Input", "Bytes", false, false),
schema.SpawnStructField("Output", "Bytes", false, false),
schema.SpawnStructField("Gas", "Uint", false, false),
schema.SpawnStructField("Cost", "Uint", false, false),
schema.SpawnStructField("Value", "BigInt", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
ts.Accumulate(schema.SpawnList("FrameList", "Frame", false))
ts.Accumulate(schema.SpawnStruct("TxTrace",
[]schema.StructField{
schema.SpawnStructField("TxCIDs", "TxCIDList", false, false),
schema.SpawnStructField("StateRootCID", "Link", false, false),
schema.SpawnStructField("Result", "Bytes", false, false),
schema.SpawnStructField("Frames", "FrameList", false, false),
schema.SpawnStructField("Gas", "Uint", false, false),
schema.SpawnStructField("Failed", "Bool", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
/*
# Block represents an entire block in the Ethereum blockchain.
type Block struct {
# CID link to the header at this block
# This CID is composed of the KECCAK_256 multihash of the RLP encoded header and the EthHeader codec (0x90)
# Note that the header contains references to the uncles and tx, receipt, and state tries at this height
HeaderCID &Header
# CID link to the list of transactions at this block
# This CID is composed of the KECCAK_256 multihash of the RLP encoded list of transactions and the EthTxList codec (0x9c)
TransactionsCID &Transactions
# CID link to the list of receipts at this block
# This CID is composed of the KECCAK_256 multihash of the RLP encoded list of receipts and the EthTxReceiptList codec (0x9d)
ReceiptsCID &Receipts
}
*/
ts.Accumulate(schema.SpawnStruct("Block",
[]schema.StructField{
schema.SpawnStructField("HeaderCID", "Link", false, false),
schema.SpawnStructField("TransactionsCID", "Link", false, false),
schema.SpawnStructField("ReceiptsCID", "Link", false, false),
},
schema.SpawnStructRepresentationMap(nil),
))
}