2016-09-08 21:13:10 +04:00
|
|
|
(**************************************************************************)
|
|
|
|
(* *)
|
|
|
|
(* Copyright (c) 2014 - 2016. *)
|
|
|
|
(* Dynamic Ledger Solutions, Inc. <contact@tezos.com> *)
|
|
|
|
(* *)
|
|
|
|
(* All rights reserved. No warranty, explicit or implicit, provided. *)
|
|
|
|
(* *)
|
|
|
|
(**************************************************************************)
|
|
|
|
|
|
|
|
open Logging.Updater
|
|
|
|
|
|
|
|
let (//) = Filename.concat
|
|
|
|
|
2017-10-09 12:55:12 +04:00
|
|
|
type validation_result = {
|
2017-04-14 02:07:01 +04:00
|
|
|
context: Context.t ;
|
2017-04-19 21:21:23 +04:00
|
|
|
fitness: Fitness.t ;
|
2017-04-14 02:07:01 +04:00
|
|
|
message: string option ;
|
2017-04-18 13:29:14 +04:00
|
|
|
max_operations_ttl: int ;
|
2017-04-14 02:07:01 +04:00
|
|
|
}
|
|
|
|
|
2017-10-09 12:55:12 +04:00
|
|
|
type rpc_context = {
|
2017-04-14 02:07:01 +04:00
|
|
|
block_hash: Block_hash.t ;
|
2017-04-19 21:21:23 +04:00
|
|
|
block_header: Block_header.t ;
|
2017-04-14 02:07:01 +04:00
|
|
|
operation_hashes: unit -> Operation_hash.t list list Lwt.t ;
|
2017-04-19 21:21:23 +04:00
|
|
|
operations: unit -> Operation.t list list Lwt.t ;
|
2017-04-14 02:07:01 +04:00
|
|
|
context: Context.t ;
|
|
|
|
}
|
|
|
|
|
2017-10-09 12:55:12 +04:00
|
|
|
module type RAW_PROTOCOL = sig
|
|
|
|
type error = ..
|
|
|
|
type 'a tzresult
|
|
|
|
type operation
|
|
|
|
val max_operation_data_length: int
|
|
|
|
val max_block_length: int
|
|
|
|
val max_number_of_operations: int
|
|
|
|
val parse_operation:
|
|
|
|
Operation_hash.t -> Operation.t -> operation tzresult
|
|
|
|
val compare_operations: operation -> operation -> int
|
|
|
|
type validation_state
|
|
|
|
val current_context: validation_state -> Context.t tzresult Lwt.t
|
|
|
|
val precheck_block:
|
|
|
|
ancestor_context: Context.t ->
|
|
|
|
ancestor_timestamp: Time.t ->
|
|
|
|
Block_header.t ->
|
|
|
|
unit tzresult Lwt.t
|
|
|
|
val begin_application:
|
|
|
|
predecessor_context: Context.t ->
|
|
|
|
predecessor_timestamp: Time.t ->
|
|
|
|
predecessor_fitness: Fitness.t ->
|
|
|
|
Block_header.t ->
|
|
|
|
validation_state tzresult Lwt.t
|
|
|
|
val begin_construction:
|
|
|
|
predecessor_context: Context.t ->
|
|
|
|
predecessor_timestamp: Time.t ->
|
|
|
|
predecessor_level: Int32.t ->
|
|
|
|
predecessor_fitness: Fitness.t ->
|
|
|
|
predecessor: Block_hash.t ->
|
|
|
|
timestamp: Time.t ->
|
|
|
|
?proto_header: MBytes.t ->
|
|
|
|
unit -> validation_state tzresult Lwt.t
|
|
|
|
val apply_operation:
|
|
|
|
validation_state -> operation -> validation_state tzresult Lwt.t
|
|
|
|
val finalize_block:
|
|
|
|
validation_state -> validation_result tzresult Lwt.t
|
|
|
|
val rpc_services: rpc_context RPC.directory
|
|
|
|
val configure_sandbox:
|
|
|
|
Context.t -> Data_encoding.json option -> Context.t tzresult Lwt.t
|
2017-04-19 21:21:23 +04:00
|
|
|
end
|
|
|
|
|
2016-09-08 21:13:10 +04:00
|
|
|
let activate = Context.set_protocol
|
|
|
|
let fork_test_network = Context.fork_test_network
|
|
|
|
|
|
|
|
(** Compiler *)
|
|
|
|
|
2017-01-30 22:10:16 +04:00
|
|
|
let datadir = ref None
|
|
|
|
let get_datadir () =
|
|
|
|
match !datadir with
|
2016-09-08 21:13:10 +04:00
|
|
|
| None -> fatal_error "not initialized"
|
|
|
|
| Some m -> m
|
|
|
|
|
|
|
|
let init dir =
|
2017-01-30 22:10:16 +04:00
|
|
|
datadir := Some dir
|
2016-09-08 21:13:10 +04:00
|
|
|
|
|
|
|
let create_files dir units =
|
2016-12-01 21:27:53 +04:00
|
|
|
Lwt_utils.remove_dir dir >>= fun () ->
|
|
|
|
Lwt_utils.create_dir dir >>= fun () ->
|
2016-09-08 21:13:10 +04:00
|
|
|
Lwt_list.map_s
|
2017-04-19 21:21:23 +04:00
|
|
|
(fun { Protocol.name; interface; implementation } ->
|
2016-10-25 21:00:03 +04:00
|
|
|
let name = String.lowercase_ascii name in
|
|
|
|
let ml = dir // (name ^ ".ml") in
|
|
|
|
let mli = dir // (name ^ ".mli") in
|
2016-12-01 21:27:53 +04:00
|
|
|
Lwt_utils.create_file ml implementation >>= fun () ->
|
2016-10-25 21:00:03 +04:00
|
|
|
match interface with
|
2016-09-08 21:13:10 +04:00
|
|
|
| None -> Lwt.return [ml]
|
|
|
|
| Some content ->
|
2016-12-01 21:27:53 +04:00
|
|
|
Lwt_utils.create_file mli content >>= fun () ->
|
2016-09-08 21:13:10 +04:00
|
|
|
Lwt.return [mli;ml])
|
|
|
|
units >>= fun files ->
|
|
|
|
let files = List.concat files in
|
|
|
|
Lwt.return files
|
|
|
|
|
2017-10-09 12:55:12 +04:00
|
|
|
let extract dir ?hash (p: Protocol.t) =
|
|
|
|
create_files dir p.components >>= fun _files ->
|
|
|
|
Tezos_compiler.Meta.to_file dir
|
|
|
|
?hash
|
|
|
|
~env_version:p.expected_env
|
|
|
|
(List.map (fun {Protocol.name} -> String.capitalize_ascii name) p.components) ;
|
|
|
|
Lwt.return_unit
|
|
|
|
|
|
|
|
let do_compile hash p =
|
|
|
|
assert (p.Protocol.expected_env = V1) ;
|
|
|
|
let units = p.components in
|
2017-01-30 22:10:16 +04:00
|
|
|
let datadir = get_datadir () in
|
2017-02-19 21:22:32 +04:00
|
|
|
let source_dir = datadir // Protocol_hash.to_short_b58check hash // "src" in
|
|
|
|
let log_file = datadir // Protocol_hash.to_short_b58check hash // "LOG" in
|
|
|
|
let plugin_file = datadir // Protocol_hash.to_short_b58check hash //
|
2016-10-25 21:00:03 +04:00
|
|
|
Format.asprintf "protocol_%a.cmxs" Protocol_hash.pp hash
|
|
|
|
in
|
2016-09-08 21:13:10 +04:00
|
|
|
create_files source_dir units >>= fun _files ->
|
2016-10-25 21:00:03 +04:00
|
|
|
Tezos_compiler.Meta.to_file source_dir ~hash
|
2017-04-19 21:21:23 +04:00
|
|
|
(List.map (fun {Protocol.name} -> String.capitalize_ascii name) units);
|
2016-09-08 21:13:10 +04:00
|
|
|
let compiler_command =
|
|
|
|
(Sys.executable_name,
|
|
|
|
Array.of_list [Node_compiler_main.compiler_name; plugin_file; source_dir]) in
|
|
|
|
let fd = Unix.(openfile log_file [O_WRONLY; O_CREAT; O_TRUNC] 0o644) in
|
|
|
|
let pi =
|
|
|
|
Lwt_process.exec
|
|
|
|
~stdin:`Close ~stdout:(`FD_copy fd) ~stderr:(`FD_move fd)
|
|
|
|
compiler_command in
|
|
|
|
pi >>= function
|
|
|
|
| Unix.WSIGNALED _ | Unix.WSTOPPED _ ->
|
|
|
|
log_error "INTERRUPTED COMPILATION (%s)" log_file;
|
|
|
|
Lwt.return false
|
|
|
|
| Unix.WEXITED x when x <> 0 ->
|
|
|
|
log_error "COMPILATION ERROR (%s)" log_file;
|
|
|
|
Lwt.return false
|
|
|
|
| Unix.WEXITED _ ->
|
|
|
|
try Dynlink.loadfile_private plugin_file; Lwt.return true
|
|
|
|
with Dynlink.Error err ->
|
|
|
|
log_error "Can't load plugin: %s (%s)"
|
|
|
|
(Dynlink.error_message err) plugin_file;
|
|
|
|
Lwt.return false
|
|
|
|
|
2017-10-09 12:55:12 +04:00
|
|
|
let compile hash p =
|
|
|
|
if Tezos_protocol_registerer.mem hash then
|
2016-09-08 21:13:10 +04:00
|
|
|
Lwt.return true
|
|
|
|
else begin
|
2017-10-09 12:55:12 +04:00
|
|
|
do_compile hash p >>= fun success ->
|
|
|
|
let loaded = Tezos_protocol_registerer.mem hash in
|
2016-09-08 21:13:10 +04:00
|
|
|
if success && not loaded then
|
|
|
|
log_error "Internal error while compiling %a" Protocol_hash.pp hash;
|
|
|
|
Lwt.return loaded
|
|
|
|
end
|
2017-10-09 12:55:12 +04:00
|
|
|
|
|
|
|
module type REGISTRED_PROTOCOL = sig
|
|
|
|
val hash: Protocol_hash.t
|
|
|
|
include RAW_PROTOCOL with type error := error
|
|
|
|
and type 'a tzresult := 'a tzresult
|
|
|
|
val complete_b58prefix : Context.t -> string -> string list Lwt.t
|
|
|
|
end
|
|
|
|
|
|
|
|
module WrapProtocol
|
|
|
|
(Name : sig val name: string end)
|
|
|
|
(Env : Tezos_protocol_environment_sigs_v1.T
|
|
|
|
with type Format.formatter = Format.formatter
|
|
|
|
and type 'a Data_encoding.t = 'a Data_encoding.t
|
|
|
|
and type 'a Lwt.t = 'a Lwt.t
|
|
|
|
and type ('a, 'b) Pervasives.result = ('a, 'b) Pervasives.result)
|
|
|
|
(P : Env.Updater.PROTOCOL) = struct
|
|
|
|
type proto_error = Env.Error_monad.error
|
|
|
|
type error += Ecoproto_error of proto_error list
|
|
|
|
let wrap_error = function
|
|
|
|
| Ok _ as ok -> ok
|
|
|
|
| Error errors -> Error [Ecoproto_error errors]
|
|
|
|
let () =
|
|
|
|
let id = Format.asprintf "Ecoproto.%s" Name.name in
|
|
|
|
Error_monad.register_wrapped_error_kind
|
|
|
|
(fun ecoerrors -> Env.Error_monad.classify_errors ecoerrors)
|
|
|
|
~id ~title:"Error returned by the protocol"
|
|
|
|
~description:"Wrapped error for the economic protocol."
|
|
|
|
~pp:(fun ppf ->
|
|
|
|
Format.fprintf ppf
|
|
|
|
"@[<v 2>Economic error:@ %a@]"
|
|
|
|
(Format.pp_print_list Env.Error_monad.pp))
|
|
|
|
Data_encoding.(obj1 (req "ecoproto"
|
|
|
|
(list (Env.Error_monad.error_encoding ()))))
|
|
|
|
(function Ecoproto_error ecoerrors -> Some ecoerrors
|
|
|
|
| _ -> None )
|
|
|
|
(function ecoerrors -> Ecoproto_error ecoerrors)
|
|
|
|
include P
|
|
|
|
let precheck_block
|
|
|
|
~ancestor_context ~ancestor_timestamp
|
|
|
|
raw_block =
|
|
|
|
precheck_block
|
|
|
|
~ancestor_context ~ancestor_timestamp
|
|
|
|
raw_block >|= wrap_error
|
|
|
|
let begin_application
|
|
|
|
~predecessor_context ~predecessor_timestamp
|
|
|
|
~predecessor_fitness
|
|
|
|
raw_block =
|
|
|
|
begin_application
|
|
|
|
~predecessor_context ~predecessor_timestamp
|
|
|
|
~predecessor_fitness
|
|
|
|
raw_block >|= wrap_error
|
|
|
|
let begin_construction
|
|
|
|
~predecessor_context ~predecessor_timestamp
|
|
|
|
~predecessor_level ~predecessor_fitness
|
|
|
|
~predecessor ~timestamp ?proto_header () =
|
|
|
|
begin_construction
|
|
|
|
~predecessor_context ~predecessor_timestamp
|
|
|
|
~predecessor_level ~predecessor_fitness
|
|
|
|
~predecessor ~timestamp ?proto_header () >|= wrap_error
|
|
|
|
let current_context c =
|
|
|
|
current_context c >|= wrap_error
|
|
|
|
let apply_operation c o =
|
|
|
|
apply_operation c o >|= wrap_error
|
|
|
|
let finalize_block c = finalize_block c >|= wrap_error
|
|
|
|
let parse_operation h b = parse_operation h b |> wrap_error
|
|
|
|
let configure_sandbox c j =
|
|
|
|
configure_sandbox c j >|= wrap_error
|
|
|
|
end
|