ligo/src/proto_alpha/lib_delegate/client_baking_endorsement.ml

271 lines
9.2 KiB
OCaml
Raw Normal View History

2016-09-08 21:13:10 +04:00
(**************************************************************************)
(* *)
2018-02-06 00:17:03 +04:00
(* Copyright (c) 2014 - 2018. *)
2016-09-08 21:13:10 +04:00
(* Dynamic Ledger Solutions, Inc. <contact@tezos.com> *)
(* *)
(* All rights reserved. No warranty, explicit or implicit, provided. *)
(* *)
(**************************************************************************)
open Proto_alpha
open Alpha_context
include Logging.Make(struct let name = "client.endorsement" end)
2016-09-08 21:13:10 +04:00
module State = Daemon_state.Make(struct let name = "endorsement" end)
2016-09-08 21:13:10 +04:00
let get_signing_slots cctxt ?(chain = `Main) block delegate level =
Alpha_services.Delegate.Endorsing_rights.get cctxt
~levels:[level]
~delegates:[delegate]
(chain, block) >>=? function
| [{ slots }] -> return (Some slots)
| _ -> return None
2016-09-08 21:13:10 +04:00
let inject_endorsement
(cctxt : #Proto_alpha.full)
2018-06-14 11:31:10 +04:00
?(chain = `Main) block hash level ?async
2018-05-06 04:08:39 +04:00
src_sk pkh =
2018-04-30 21:06:06 +04:00
Alpha_services.Forge.endorsement cctxt
(chain, block)
~branch:hash
~level:level
2016-09-08 21:13:10 +04:00
() >>=? fun bytes ->
2018-06-17 02:07:58 +04:00
Client_keys.append cctxt
src_sk ~watermark:Endorsement bytes >>=? fun signed_bytes ->
Shell_services.Injection.operation cctxt ?async ~chain signed_bytes >>=? fun oph ->
2018-06-18 14:34:03 +04:00
State.record cctxt pkh level >>=? fun () ->
2016-09-08 21:13:10 +04:00
return oph
let check_endorsement cctxt level pkh =
2018-06-18 14:34:03 +04:00
State.get cctxt pkh >>=? function
2016-09-08 21:13:10 +04:00
| None -> return ()
| Some recorded_level ->
if Raw_level.(level = recorded_level) then
Error_monad.failwith "Level %a already endorsed" Raw_level.pp recorded_level
else
return ()
2016-09-08 21:13:10 +04:00
2018-05-22 17:22:38 +04:00
let previously_endorsed_level cctxt pkh new_lvl =
State.get cctxt pkh >>=? function
2018-05-22 17:22:38 +04:00
| None -> return false
| Some last_lvl ->
return (Raw_level.(last_lvl >= new_lvl))
2018-05-22 17:22:38 +04:00
let forge_endorsement (cctxt : #Proto_alpha.full)
2018-05-22 17:22:38 +04:00
?(chain = `Main) block ?async
~src_sk ?slots src_pk =
let src_pkh = Signature.Public_key.hash src_pk in
Alpha_block_services.metadata cctxt
~chain ~block () >>=? fun { protocol_data = { level = { level } } } ->
2018-05-22 17:22:38 +04:00
check_endorsement cctxt level src_pkh >>=? fun () ->
previously_endorsed_level cctxt src_pkh level >>=? function
| true ->
cctxt#error "Level %a : previously endorsed."
Raw_level.pp level
| false ->
begin
match slots with
| Some slots -> return slots
| None ->
get_signing_slots
cctxt ~chain block src_pkh level >>=? function
| None -> cctxt#error "No slot found at level %a" Raw_level.pp level
| Some slots -> return slots
end >>=? fun slots ->
2018-06-14 11:31:10 +04:00
Shell_services.Blocks.hash cctxt ~chain ~block () >>=? fun hash ->
2018-05-06 04:08:39 +04:00
inject_endorsement cctxt ~chain ?async block hash level src_sk src_pkh >>=? fun oph ->
Client_keys.get_key cctxt src_pkh >>=? fun (name, _pk, _sk) ->
cctxt#message
2018-05-06 04:08:39 +04:00
"Injected endorsement level %a, contract %s '%a', slots @[<h>%a@]"
Raw_level.pp level
name
2018-05-06 04:08:39 +04:00
Operation_hash.pp_short oph
(Format.pp_print_list Format.pp_print_int) slots
>>=
fun () -> return oph
2016-09-08 21:13:10 +04:00
(** Worker *)
type state = {
delegates: public_key_hash list ;
2018-06-06 17:33:28 +04:00
delay: int64 ;
to_endorse : endorsement Signature.Public_key_hash.Table.t ;
2016-09-08 21:13:10 +04:00
}
and endorsement = {
time: Time.t ;
timeout: unit Lwt.t ;
2016-09-08 21:13:10 +04:00
delegate: public_key_hash ;
2017-11-01 15:07:33 +04:00
block: Client_baking_blocks.block_info ;
2016-09-08 21:13:10 +04:00
}
let create_state delegates delay =
{ delegates ; delay ; to_endorse = Signature.Public_key_hash.Table.create 5 }
2016-09-08 21:13:10 +04:00
let get_delegates cctxt state =
match state.delegates with
| [] ->
Client_keys.get_keys cctxt >>=? fun keys ->
2018-05-28 17:36:18 +04:00
return (List.map (fun (_, pkh, _, _) -> pkh) keys)
| _ :: _ as delegates ->
return delegates
let endorse_for_delegate cctxt { delegate ; block } =
let { Client_baking_blocks.hash ; level } = block in
let b = `Hash (hash, 0) in
Client_keys.get_key cctxt delegate >>=? fun (name, _pk, sk) ->
lwt_debug "Endorsing %a for %s (level %a)!"
Block_hash.pp_short hash name
Raw_level.pp level >>= fun () ->
inject_endorsement cctxt
2018-06-14 11:31:10 +04:00
b hash level
2018-05-06 04:08:39 +04:00
sk delegate >>=? fun oph ->
lwt_log_info
"Injected endorsement for block '%a' \
(level %a, contract %s) '%a'"
Block_hash.pp_short hash
Raw_level.pp level
name
Operation_hash.pp_short oph >>= fun () ->
2018-06-14 11:31:10 +04:00
lwt_log_info
"Injected endorsement level %a, contract %s '%a'"
Raw_level.pp level
name
Operation_hash.pp_short oph >>= fun () ->
return ()
let endorse_for cctxt table =
let done_waiting = ref [] in
Signature.Public_key_hash.Table.filter_map_inplace
(fun _ ({ timeout } as endorsement) ->
match Lwt.state timeout with
| Lwt.Return () ->
done_waiting := endorsement :: !done_waiting ;
None
| Lwt.Sleep ->
Some endorsement
| Lwt.Fail exn ->
log_error "Endorsement failure: %s" (Printexc.to_string exn) ;
None)
table ;
iter_p (endorse_for_delegate cctxt) !done_waiting
2018-06-14 08:47:42 +04:00
let allowed_to_endorse cctxt state (block: Client_baking_blocks.block_info) delegate time =
Client_keys.Public_key_hash.name cctxt delegate >>=? fun name ->
lwt_log_info "Checking if allowed to endorse block %a for %s"
Block_hash.pp_short block.hash name >>= fun () ->
let b = `Hash (block.hash, 0) in
let level = block.level in
get_signing_slots cctxt b delegate level >>=? function
| None ->
lwt_debug "No slot found for %a/%s"
Block_hash.pp_short block.hash name >>= fun () ->
return ()
| Some slots ->
lwt_debug "Found slots for %a/%s (%d)"
Block_hash.pp_short block.hash name (List.length slots) >>= fun () ->
previously_endorsed_level cctxt delegate level >>=? function
| true ->
lwt_debug "Level %a (or higher) previously endorsed: do not endorse."
Raw_level.pp level >>= fun () ->
return ()
| false ->
match Client_baking_scheduling.sleep_until time with
2018-06-14 08:47:42 +04:00
| None ->
lwt_debug "Endorsment opportunity is passed." >>= fun () ->
2018-06-14 08:47:42 +04:00
return ()
| Some timeout ->
let neu = { time ; timeout ; delegate ; block } in
Signature.Public_key_hash.Table.add state.to_endorse delegate neu ;
return ()
let prepare_endorsement (cctxt : #Proto_alpha.full) ~(max_past:int64) state bi =
get_delegates cctxt state >>=? fun delegates ->
Signature.Public_key_hash.Table.clear state.to_endorse ;
iter_p
(fun delegate ->
2018-06-11 17:51:59 +04:00
let open Client_baking_blocks in
if Time.diff (Time.now ()) bi.timestamp > max_past then
2018-06-11 17:51:59 +04:00
lwt_log_info "Ignore block %a: forged too far the past"
Block_hash.pp_short bi.hash >>= return
else
let time = Time.(add (now ()) state.delay) in
allowed_to_endorse cctxt state bi delegate time
)
delegates
2016-09-08 21:13:10 +04:00
let compute_timeout state =
match Signature.Public_key_hash.Table.fold
(fun _ v acc -> v :: acc)
state.to_endorse [] with
| [] -> Lwt_utils.never_ending ()
| to_ends ->
Lwt.choose (List.map (fun to_end -> to_end.timeout) to_ends)
let check_error f =
f >>= function
| Ok () -> Lwt.return_unit
| Error errs -> lwt_log_error "Error while endorsing:@\n%a" pp_print_error errs
2018-06-06 17:33:28 +04:00
2018-06-13 10:38:16 +04:00
let create
(cctxt: #Proto_alpha.full)
~max_past
~delay
contracts
block_stream
bi =
lwt_log_info "Preparing endorsement daemon" >>= fun () ->
2018-06-13 10:38:16 +04:00
(* statefulness setup *)
let last_get_block = ref None in
let get_block () =
match !last_get_block with
| None ->
let t = Lwt_stream.get block_stream in
last_get_block := Some t ;
t
| Some t -> t in
let state = create_state contracts (Int64.of_int delay) in
(* main loop *)
let rec worker_loop () =
begin
let timeout = compute_timeout state in
Lwt.choose [ (timeout >|= fun () -> `Timeout) ;
(get_block () >|= fun b -> `Hash b) ] >>= function
| `Hash None ->
last_get_block := None;
lwt_log_error "Connection to node lost, exiting." >>= fun () ->
exit 1
| `Hash (Some (Error _)) ->
2018-06-13 10:38:16 +04:00
last_get_block := None;
Lwt.return_unit
| `Hash (Some (Ok bi)) ->
last_get_block := None;
check_error @@ prepare_endorsement cctxt ~max_past state bi
2018-06-13 10:38:16 +04:00
| `Timeout ->
endorse_for cctxt state.to_endorse >>= function
| Ok () ->
Lwt.return_unit
| Error errs ->
lwt_log_error "Error while endorsing:@\n%a" pp_print_error errs
2018-06-13 10:38:16 +04:00
end >>= fun () ->
worker_loop () in
(* ignition *)
check_error (prepare_endorsement cctxt ~max_past state bi) >>= fun () ->
lwt_log_info "Starting endorsement daemon" >>= fun () ->
worker_loop ()
(* A wrapper around the main create function (above) to wait for the initial
block. *)
let create
(cctxt: #Proto_alpha.full)
?(max_past=110L)
~delay
contracts
(block_stream: Client_baking_blocks.block_info tzresult Lwt_stream.t) =
2018-06-14 10:24:33 +04:00
Client_baking_scheduling.wait_for_first_block
~info:lwt_log_info
2018-06-14 10:24:33 +04:00
block_stream
(create cctxt ~max_past ~delay contracts block_stream)