2018-04-22 12:50:34 +04:00
|
|
|
(**************************************************************************)
|
|
|
|
(* *)
|
|
|
|
(* Copyright (c) 2014 - 2018. *)
|
|
|
|
(* Dynamic Ledger Solutions, Inc. <contact@tezos.com> *)
|
|
|
|
(* *)
|
|
|
|
(* All rights reserved. No warranty, explicit or implicit, provided. *)
|
|
|
|
(* *)
|
|
|
|
(**************************************************************************)
|
|
|
|
|
|
|
|
open Proto_alpha
|
|
|
|
open Alpha_context
|
|
|
|
open Apply_operation_result
|
|
|
|
|
|
|
|
let get_branch rpc_config (block : Block_services.block) branch =
|
|
|
|
let branch = Option.unopt ~default:0 branch in (* TODO export parameter *)
|
|
|
|
begin
|
|
|
|
match block with
|
|
|
|
| `Head n -> return (`Head (n+branch))
|
|
|
|
| `Test_head n -> return (`Test_head (n+branch))
|
|
|
|
| `Hash (h,n) -> return (`Hash (h,n+branch))
|
|
|
|
| `Genesis -> return `Genesis
|
|
|
|
end >>=? fun block ->
|
|
|
|
Block_services.hash rpc_config block >>=? fun hash ->
|
|
|
|
return hash
|
|
|
|
|
|
|
|
type result = Operation_hash.t * operation * operation_result
|
|
|
|
|
|
|
|
let preapply
|
|
|
|
cctxt block
|
|
|
|
?branch ?src_sk contents =
|
|
|
|
get_branch cctxt block branch >>=? fun branch ->
|
|
|
|
let bytes =
|
|
|
|
Data_encoding.Binary.to_bytes_exn
|
|
|
|
Operation.unsigned_operation_encoding
|
|
|
|
({ branch }, contents) in
|
|
|
|
let watermark =
|
|
|
|
match contents with
|
2018-04-16 02:44:19 +04:00
|
|
|
| Sourced_operation (Consensus_operation (Endorsements _)) ->
|
2018-04-22 12:50:34 +04:00
|
|
|
Signature.Endorsement
|
|
|
|
| _ ->
|
|
|
|
Signature.Generic_operation in
|
|
|
|
begin
|
|
|
|
match src_sk with
|
|
|
|
| None -> return None
|
|
|
|
| Some src_sk ->
|
|
|
|
Client_keys.sign
|
|
|
|
~watermark src_sk bytes >>=? fun signature ->
|
|
|
|
return (Some signature)
|
|
|
|
end >>=? fun signature ->
|
|
|
|
let op =
|
|
|
|
{ shell = { branch } ;
|
|
|
|
contents ;
|
|
|
|
signature } in
|
|
|
|
let oph = Operation.hash op in
|
|
|
|
Block_services.hash cctxt block >>=? fun bh ->
|
|
|
|
Alpha_services.Helpers.apply_operation cctxt
|
|
|
|
block bh oph bytes signature >>=? fun result ->
|
|
|
|
return (oph, op, result)
|
|
|
|
|
|
|
|
let estimated_gas = function
|
|
|
|
| Sourced_operation_result (Manager_operations_result { operation_results }) ->
|
|
|
|
List.fold_left
|
|
|
|
(fun acc (_, r) -> acc >>? fun acc ->
|
|
|
|
match r with
|
|
|
|
| Applied (Transaction_result { consumed_gas }
|
|
|
|
| Origination_result { consumed_gas }) ->
|
|
|
|
Ok (Z.add consumed_gas acc)
|
|
|
|
| Applied Reveal_result -> Ok acc
|
|
|
|
| Applied Delegation_result -> Ok acc
|
|
|
|
| Skipped -> assert false
|
|
|
|
| Failed errs -> Alpha_environment.wrap_error (Error errs))
|
|
|
|
(Ok Z.zero) operation_results
|
|
|
|
| _ -> Ok Z.zero
|
|
|
|
|
|
|
|
let originated_contracts = function
|
|
|
|
| Sourced_operation_result (Manager_operations_result { operation_results }) ->
|
|
|
|
List.fold_left
|
|
|
|
(fun acc (_, r) -> acc >>? fun acc ->
|
|
|
|
match r with
|
|
|
|
| Applied (Transaction_result { originated_contracts }
|
|
|
|
| Origination_result { originated_contracts }) ->
|
|
|
|
Ok (originated_contracts @ acc)
|
|
|
|
| Applied Reveal_result -> Ok acc
|
|
|
|
| Applied Delegation_result -> Ok acc
|
|
|
|
| Skipped -> assert false
|
|
|
|
| Failed errs -> Alpha_environment.wrap_error (Error errs))
|
|
|
|
(Ok []) operation_results
|
|
|
|
| _ -> Ok []
|
|
|
|
|
|
|
|
let may_patch_gas_limit
|
|
|
|
(cctxt : #Proto_alpha.full) block ?branch
|
|
|
|
?src_sk contents =
|
|
|
|
Alpha_services.Constants.hard_gas_limits cctxt block >>=? fun (_, gas_limit) ->
|
|
|
|
match contents with
|
2018-04-16 02:44:19 +04:00
|
|
|
| Sourced_operation (Manager_operations c)
|
2018-04-22 12:50:34 +04:00
|
|
|
when c.gas_limit < Z.zero || gas_limit < c.gas_limit ->
|
|
|
|
let contents =
|
2018-04-16 02:44:19 +04:00
|
|
|
Sourced_operation (Manager_operations { c with gas_limit }) in
|
2018-04-22 12:50:34 +04:00
|
|
|
preapply cctxt block ?branch ?src_sk contents >>=? fun (_, _, result) ->
|
|
|
|
Lwt.return (estimated_gas result) >>=? fun gas ->
|
|
|
|
begin
|
|
|
|
if Z.equal gas Z.zero then
|
|
|
|
cctxt#message "Estimated gas: none" >>= fun () ->
|
|
|
|
return Z.zero
|
|
|
|
else
|
|
|
|
cctxt#message
|
|
|
|
"Estimated gas: %s units (will add 100 for safety)"
|
|
|
|
(Z.to_string gas) >>= fun () ->
|
|
|
|
return (Z.add gas (Z.of_int 100))
|
|
|
|
end >>=? fun gas_limit ->
|
2018-04-16 02:44:19 +04:00
|
|
|
return (Sourced_operation (Manager_operations { c with gas_limit }))
|
2018-04-22 12:50:34 +04:00
|
|
|
| op -> return op
|
|
|
|
|
|
|
|
let inject_operation
|
|
|
|
cctxt block
|
|
|
|
?confirmations ?branch ?src_sk contents =
|
|
|
|
may_patch_gas_limit
|
|
|
|
cctxt block ?branch ?src_sk contents >>=? fun contents ->
|
|
|
|
preapply cctxt block
|
|
|
|
?branch ?src_sk contents >>=? fun (_oph, op, result) ->
|
|
|
|
let bytes = Data_encoding.Binary.to_bytes_exn Operation.encoding op in
|
|
|
|
Block_services.chain_id cctxt block >>=? fun chain_id ->
|
|
|
|
Shell_services.inject_operation cctxt ~chain_id bytes >>=? fun oph ->
|
|
|
|
cctxt#message "Operation successfully injected in the node." >>= fun () ->
|
|
|
|
cctxt#message "Operation hash is '%a'." Operation_hash.pp oph >>= fun () ->
|
|
|
|
begin
|
|
|
|
match confirmations with
|
|
|
|
| None -> return ()
|
|
|
|
| Some confirmations ->
|
|
|
|
cctxt#message "Waiting for the operation to be included..." >>= fun () ->
|
|
|
|
Client_confirmations.wait_for_operation_inclusion
|
|
|
|
~confirmations cctxt oph >>=? fun () ->
|
|
|
|
return ()
|
|
|
|
end >>=? fun () ->
|
|
|
|
cctxt#message
|
|
|
|
"@[<v 2>This sequence of operations was run:@,%a@]"
|
|
|
|
Operation_result.pp_operation_result (op, result) >>= fun () ->
|
|
|
|
Lwt.return (originated_contracts result) >>=? fun contracts ->
|
|
|
|
Lwt_list.iter_s
|
|
|
|
(fun c ->
|
|
|
|
cctxt#message
|
|
|
|
"New contract %a originated."
|
|
|
|
Contract.pp c)
|
|
|
|
contracts >>= fun () ->
|
|
|
|
return (oph, op, result)
|