package octez-protocol-018-Proxford-libs

  1. Overview
  2. Docs
Octez protocol 018-Proxford libraries

Install

dune-project
 Dependency

Authors

Maintainers

Sources

tezos-18.0.tar.gz
sha256=dbc3b675aee59c2c574e5d0a771193a2ecfca31e7a5bc5aed66598080596ce1c
sha512=b97ed762b9d24744305c358af0d20f394376b64bfdd758dd4a81775326caf445caa57c4f6445da3dd6468ff492de18e4c14af6f374dfcbb7e4d64b7b720e5e2a

doc/src/octez-protocol-018-Proxford-libs.baking/block_forge.ml.html

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 Source License                                                       *)
(* Copyright (c) 2021 Nomadic Labs <contact@nomadic-labs.com>                *)
(*                                                                           *)
(* Permission is hereby granted, free of charge, to any person obtaining a   *)
(* copy of this software and associated documentation files (the "Software"),*)
(* to deal in the Software without restriction, including without limitation *)
(* the rights to use, copy, modify, merge, publish, distribute, sublicense,  *)
(* and/or sell copies of the Software, and to permit persons to whom the     *)
(* Software is furnished to do so, subject to the following conditions:      *)
(*                                                                           *)
(* The above copyright notice and this permission notice shall be included   *)
(* in all copies or substantial portions of the Software.                    *)
(*                                                                           *)
(* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR*)
(* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,  *)
(* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL   *)
(* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER*)
(* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING   *)
(* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER       *)
(* DEALINGS IN THE SOFTWARE.                                                 *)
(*                                                                           *)
(*****************************************************************************)

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

(* [forge_faked_protocol_data ?payload_hash ~payload_round ~seed_nonce_hash
   ~liquidity_baking_toggle_vote ~adaptive_issuance_vote] forges a fake [block_header_data] with
   [payload_hash] ([zero] by default), [payload_round], [seed_nonce_hash],
   [liquidity_baking_toggle_vote] and with an empty [proof_of_work_nonce] and a
   dummy [signature]. *)
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;
  }

(* [finalize_block_header] updates the [shell_header] that was created
   with dummy fields at the beginning of the block construction. It
   increments the [level] and sets the actual [operations_hash],
   [fitness], [validation_passes], and [context] (the predecessor
   resulting context hash).

   When the operations from the block have been applied, the [fitness]
   is simply retrieved from the [validation_result]. Otherwise, the
   [fitness] is computed from the [round] and [locked_round]
   arguments. *)
let finalize_block_header ~shell_header ~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 header =
    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

(* [check_protocol_changed] checks whether the protocol will change with the current
   block. This function returns true if the block is the last of an [adoption]
   period. It can also return true if an [user_activated_upgrades] is given. *)
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)

(* [filter_via_node] filters operations using
   {!Operation_selection.filter_operations_without_simulation} and then applies
   them in a block via {!Node_rpc.preapply_block}. [filter_via_node] returns a
   [shell_header], the list of operations that have been applied in the block
   and the [payload_hash] corresponding to these operations. *)
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* shell_header, preapply_result =
    Node_rpc.preapply_block
      cctxt
      ~chain
      ~head:pred_info.hash
      ~timestamp
      ~protocol_data:faked_protocol_data
      filtered_operations
  in
  (* only retain valid operations *)
  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)

(* [filter_with_context] filters operations using a local context via
   {!Operation_selection.filter_operations_with_simulation} and a fresh state
   from {!Baking_simulator.begin_construction}. [finalize_block_header] is then
   called and a [shell_header], the list of operations and the corresponding
   [payload_hash] are returned. If the block is a transition block,
   [filter_via_node] is called to return these values. *)
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
    (* Fallback to processing via node, which knows both old and new protocol. *)
    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* shell_header =
      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)

(* [apply_via_node] applies already filtered and validated operations in a block
   via {!Node_rpc.preapply_block}. A [shell_header] is recovered from this call
   and returned alongside of the list of operations and the 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* shell_header, _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)

(* [apply_with_context] is similar to [filter_with_context] but filters
   consensus operations only from an [ordered_pool] via
   {!Operation_selection.filter_consensus_operations_only}. *)
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
  (* We still need to filter attestations. Two attestations could be
     referring to the same slot. *)
  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
  (* We need to compute the final [operations_hash] before
     finalizing the block because it will be used in the cache's nonce. *)
  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
    (* Fallback to processing via node, which knows both old and new protocol. *)
    apply_via_node
      ~chain_id
      ~faked_protocol_data
      ~timestamp
      ~pred_info
      ~ordered_pool
      ~payload_hash
      cctxt
  else
    let locked_round_when_no_validation_result =
      (* [locked_round] will not be used in [finalize_block_header] if there is
         a [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* shell_header =
      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)

(* [forge] a new [unsigned_block] in accordance with [simulation_kind] and
   [simulation_mode] *)
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 ->
        (* We cannot include operations that are not live with respect
           to our predecessor otherwise the node would reject the block. *)
        let filtered_pool =
          retain_live_operations_only
            ~live_blocks:pred_live_blocks
            operation_pool
        in
        Filter filtered_pool
    | Apply _ as x -> x
  in
  let* shell_header, 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 unsigned_block_header =
    {
      Block_header.shell = shell_header;
      protocol_data = {contents; signature = Signature.zero};
    }
  in
  return {unsigned_block_header; operations}