Source file block_forge.ml
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
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
open Protocol
open Alpha_context
type unsigned_block = {
unsigned_block_header : Block_header.t;
operations : Tezos_base.Operation.t list list;
}
type simulation_kind =
| Filter of Operation_pool.Prioritized.t
| Apply of {
ordered_pool : Operation_pool.ordered_pool;
payload_hash : Block_payload_hash.t;
}
type simulation_mode = Local of Context.index | Node
let forge_faked_protocol_data ?(payload_hash = Block_payload_hash.zero)
~payload_round ~seed_nonce_hash ~liquidity_baking_toggle_vote
~adaptive_issuance_vote () =
Block_header.
{
contents =
{
payload_hash;
payload_round;
seed_nonce_hash;
proof_of_work_nonce = Baking_pow.empty_proof_of_work_nonce;
per_block_votes =
{
liquidity_baking_vote = liquidity_baking_toggle_vote;
adaptive_issuance_vote;
};
};
signature = Signature.zero;
}
let convert_operation (op : packed_operation) : Tezos_base.Operation.t =
{
shell = op.shell;
proto =
Data_encoding.Binary.to_bytes_exn
Alpha_context.Operation
.protocol_data_encoding_with_legacy_attestation_name
op.protocol_data;
}
let ~ ~validation_result ~operations_hash
~(pred_info : Baking_state.block_info) ~pred_resulting_context_hash ~round
~locked_round =
let open Lwt_result_syntax in
let* fitness =
match validation_result with
| Some {Tezos_protocol_environment.fitness; _} -> return fitness
| None ->
let*? level =
Environment.wrap_tzresult @@ Raw_level.of_int32
@@ Int32.succ shell_header.Tezos_base.Block_header.level
in
let*? fitness =
Environment.wrap_tzresult
@@ Fitness.create
~level
~round
~predecessor_round:pred_info.round
~locked_round
in
return (Fitness.to_raw fitness)
in
let validation_passes = List.length Main.validation_passes in
let =
Tezos_base.Block_header.
{
shell_header with
level = Int32.succ shell_header.level;
validation_passes;
operations_hash;
fitness;
context = pred_resulting_context_hash;
}
in
return header
let retain_live_operations_only ~live_blocks operation_pool =
Operation_pool.Prioritized.filter
(fun ({shell; _} : packed_operation) ->
Block_hash.Set.mem shell.branch live_blocks)
operation_pool
let check_protocol_changed ~user_activated_upgrades ~level
~(validation_result : Tezos_protocol_environment.validation_result option)
~(incremental : Baking_simulator.incremental) =
let open Lwt_result_syntax in
match
Tezos_base.Block_header.get_forced_protocol_upgrade
~user_activated_upgrades
~level
with
| None -> (
match validation_result with
| None -> (
let context = Validate.get_initial_ctxt (fst incremental.state) in
let* voting_period =
Lwt.map
Environment.wrap_tzresult
(Voting_period.get_current context)
in
match voting_period.kind with
| Voting_period.Proposal | Exploration | Cooldown | Promotion ->
return_false
| Adoption ->
Lwt.map
Environment.wrap_tzresult
(Voting_period.is_last_block context))
| Some validation_result ->
let*! next_protocol =
Context_ops.get_protocol validation_result.context
in
return Protocol_hash.(Protocol.hash <> next_protocol))
| Some next_protocol -> return Protocol_hash.(Protocol.hash <> next_protocol)
let filter_via_node ~chain_id ~fees_config ~hard_gas_limit_per_block
~faked_protocol_data ~timestamp ~(pred_info : Baking_state.block_info)
~payload_round ~operation_pool cctxt =
let open Lwt_result_syntax in
let chain = `Hash chain_id in
let filtered_operations =
Operation_selection.filter_operations_without_simulation
fees_config
~hard_gas_limit_per_block
operation_pool
in
let* , preapply_result =
Node_rpc.preapply_block
cctxt
~chain
~head:pred_info.hash
~timestamp
~protocol_data:faked_protocol_data
filtered_operations
in
let operations =
List.map (fun l -> List.map snd l.Preapply_result.applied) preapply_result
in
let payload_hash =
let operation_hashes =
Stdlib.List.tl operations |> List.flatten
|> List.map Tezos_base.Operation.hash
in
Block_payload.hash
~predecessor_hash:shell_header.predecessor
~payload_round
operation_hashes
in
return (shell_header, operations, payload_hash)
let filter_with_context ~chain_id ~fees_config ~hard_gas_limit_per_block
~faked_protocol_data ~user_activated_upgrades ~timestamp
~(pred_info : Baking_state.block_info) ~pred_resulting_context_hash
~force_apply ~round ~context_index ~payload_round ~operation_pool cctxt =
let open Lwt_result_syntax in
let* incremental =
Baking_simulator.begin_construction
~timestamp
~protocol_data:faked_protocol_data
~force_apply
~pred_resulting_context_hash
context_index
pred_info
chain_id
in
let* {Operation_selection.operations; validation_result; operations_hash; _} =
Operation_selection.filter_operations_with_simulation
incremental
fees_config
~hard_gas_limit_per_block
operation_pool
in
let* changed =
check_protocol_changed
~level:(Int32.succ pred_info.shell.level)
~user_activated_upgrades
~validation_result
~incremental
in
if changed then
filter_via_node
~chain_id
~fees_config
~hard_gas_limit_per_block
~faked_protocol_data
~timestamp
~pred_info
~payload_round
~operation_pool
cctxt
else
let* =
finalize_block_header
~shell_header:incremental.header
~validation_result
~operations_hash
~pred_info
~pred_resulting_context_hash
~round
~locked_round:None
in
let operations = List.map (List.map convert_operation) operations in
let payload_hash =
let operation_hashes =
Stdlib.List.tl operations |> List.flatten
|> List.map Tezos_base.Operation.hash
in
Block_payload.hash
~predecessor_hash:shell_header.predecessor
~payload_round
operation_hashes
in
return (shell_header, operations, payload_hash)
let apply_via_node ~chain_id ~faked_protocol_data ~timestamp
~(pred_info : Baking_state.block_info) ~ordered_pool ~payload_hash cctxt =
let open Lwt_result_syntax in
let chain = `Hash chain_id in
let operations = Operation_pool.ordered_to_list_list ordered_pool in
let* , _preapply_result =
Node_rpc.preapply_block
cctxt
~chain
~head:pred_info.hash
~timestamp
~protocol_data:faked_protocol_data
operations
in
let operations = List.map (List.map convert_operation) operations in
return (shell_header, operations, payload_hash)
let apply_with_context ~chain_id ~faked_protocol_data ~user_activated_upgrades
~timestamp ~(pred_info : Baking_state.block_info)
~pred_resulting_context_hash ~force_apply ~round ~ordered_pool
~context_index ~payload_hash cctxt =
let open Lwt_result_syntax in
let* incremental =
Baking_simulator.begin_construction
~timestamp
~protocol_data:faked_protocol_data
~force_apply
~pred_resulting_context_hash
context_index
pred_info
chain_id
in
let* incremental, ordered_pool =
Operation_selection.filter_consensus_operations_only
incremental
ordered_pool
in
let operations = Operation_pool.ordered_to_list_list ordered_pool in
let operations_hash =
Operation_list_list_hash.compute
(List.map
(fun sl ->
Operation_list_hash.compute (List.map Operation.hash_packed sl))
operations)
in
let incremental =
{incremental with header = {incremental.header with operations_hash}}
in
let* validation_result = Baking_simulator.finalize_construction incremental in
let validation_result = Option.map fst validation_result in
let* changed =
check_protocol_changed
~level:(Int32.succ pred_info.shell.level)
~user_activated_upgrades
~validation_result
~incremental
in
if changed then
apply_via_node
~chain_id
~faked_protocol_data
~timestamp
~pred_info
~ordered_pool
~payload_hash
cctxt
else
let locked_round_when_no_validation_result =
if Option.is_some validation_result then None
else
List.find_map
(fun {protocol_data = Operation_data protocol_data; _} ->
match protocol_data.contents with
| Single (Preattestation {round; _}) -> Some round
| _ -> None)
(Option.value (List.hd operations) ~default:[])
in
let* =
finalize_block_header
~shell_header:incremental.header
~validation_result
~operations_hash
~pred_info
~pred_resulting_context_hash
~round
~locked_round:locked_round_when_no_validation_result
in
let operations = List.map (List.map convert_operation) operations in
return (shell_header, operations, payload_hash)
let forge (cctxt : #Protocol_client_context.full) ~chain_id
~(pred_info : Baking_state.block_info) ~pred_resulting_context_hash
~pred_live_blocks ~timestamp ~round ~liquidity_baking_toggle_vote
~adaptive_issuance_vote ~user_activated_upgrades fees_config ~force_apply
~seed_nonce_hash ~payload_round simulation_mode simulation_kind constants =
let open Lwt_result_syntax in
let hard_gas_limit_per_block =
constants.Constants.Parametric.hard_gas_limit_per_block
in
let simulation_kind =
match simulation_kind with
| Filter operation_pool ->
let filtered_pool =
retain_live_operations_only
~live_blocks:pred_live_blocks
operation_pool
in
Filter filtered_pool
| Apply _ as x -> x
in
let* , operations, payload_hash =
match (simulation_mode, simulation_kind) with
| Baking_state.Node, Filter operation_pool ->
let faked_protocol_data =
forge_faked_protocol_data
~payload_round
~seed_nonce_hash
~liquidity_baking_toggle_vote
~adaptive_issuance_vote
()
in
filter_via_node
~chain_id
~faked_protocol_data
~fees_config
~hard_gas_limit_per_block
~timestamp
~pred_info
~payload_round
~operation_pool
cctxt
| Node, Apply {ordered_pool; payload_hash} ->
let faked_protocol_data =
forge_faked_protocol_data
~payload_hash
~payload_round
~seed_nonce_hash
~liquidity_baking_toggle_vote
~adaptive_issuance_vote
()
in
apply_via_node
~chain_id
~faked_protocol_data
~timestamp
~pred_info
~ordered_pool
~payload_hash
cctxt
| Local context_index, Filter operation_pool ->
let faked_protocol_data =
forge_faked_protocol_data
~payload_round
~seed_nonce_hash
~liquidity_baking_toggle_vote
~adaptive_issuance_vote
()
in
filter_with_context
~chain_id
~faked_protocol_data
~fees_config
~hard_gas_limit_per_block
~user_activated_upgrades
~timestamp
~pred_info
~pred_resulting_context_hash
~force_apply
~round
~context_index
~payload_round
~operation_pool
cctxt
| Local context_index, Apply {ordered_pool; payload_hash} ->
let faked_protocol_data =
forge_faked_protocol_data
~payload_hash
~payload_round
~seed_nonce_hash
~liquidity_baking_toggle_vote
~adaptive_issuance_vote
()
in
apply_with_context
~chain_id
~faked_protocol_data
~user_activated_upgrades
~timestamp
~pred_info
~pred_resulting_context_hash
~force_apply
~round
~ordered_pool
~context_index
~payload_hash
cctxt
in
let* contents =
Baking_pow.mine
~proof_of_work_threshold:constants.proof_of_work_threshold
shell_header
(fun proof_of_work_nonce ->
{
Block_header.payload_hash;
payload_round;
seed_nonce_hash;
proof_of_work_nonce;
per_block_votes =
{
liquidity_baking_vote = liquidity_baking_toggle_vote;
adaptive_issuance_vote;
};
})
in
let =
{
Block_header.shell = shell_header;
protocol_data = {contents; signature = Signature.zero};
}
in
return {unsigned_block_header; operations}