(**************************************************************************) (* *) (* Copyright (c) 2014 - 2018. *) (* Dynamic Ledger Solutions, Inc. *) (* *) (* All rights reserved. No warranty, explicit or implicit, provided. *) (* *) (**************************************************************************) open Proto_alpha open Alpha_context open Tezos_micheline open Client_proto_contracts open Client_keys let get_balance (rpc : #Proto_alpha.rpc_context) block contract = Alpha_services.Contract.balance rpc block contract let get_storage (rpc : #Proto_alpha.rpc_context) block contract = Alpha_services.Contract.storage_opt rpc block contract let parse_expression arg = Lwt.return (Micheline_parser.no_parsing_error (Michelson_v1_parser.parse_expression arg)) let append_reveal cctxt block ~source ~src_pk ops = Alpha_services.Contract.manager_key cctxt block source >>=? fun (_pkh, pk) -> let is_reveal = function | Reveal _ -> true | _ -> false in match pk with | None when not (List.exists is_reveal ops) -> return (Reveal src_pk :: ops) | _ -> return ops let transfer (cctxt : #Proto_alpha.full) block ?confirmations ?branch ~source ~src_pk ~src_sk ~destination ?arg ~amount ~fee ?(gas_limit = Z.minus_one) ?(storage_limit = -1L) () = begin match arg with | Some arg -> parse_expression arg >>=? fun { expanded = arg } -> return (Some arg) | None -> return None end >>=? fun parameters -> Alpha_services.Contract.counter cctxt block source >>=? fun pcounter -> let counter = Int32.succ pcounter in let operations = [Transaction { amount ; parameters ; destination }] in append_reveal cctxt block ~source ~src_pk operations >>=? fun operations -> let contents = Sourced_operation (Manager_operations { source ; fee ; counter ; gas_limit ; storage_limit ; operations }) in Injection.inject_operation cctxt block ?confirmations ?branch ~src_sk contents >>=? fun (_oph, _op, result as res) -> Lwt.return (Injection.originated_contracts result) >>=? fun contracts -> return (res, contracts) let reveal cctxt block ?confirmations ?branch ~source ~src_pk ~src_sk ~fee () = Alpha_services.Contract.counter cctxt block source >>=? fun pcounter -> let counter = Int32.succ pcounter in append_reveal cctxt block ~source ~src_pk [] >>=? fun operations -> match operations with | [] -> failwith "The manager key was previously revealed." | _ :: _ -> let contents = Sourced_operation (Manager_operations { source ; fee ; counter ; gas_limit = Z.zero ; storage_limit = 0L ; operations }) in Injection.inject_operation cctxt block ?confirmations ?branch ~src_sk contents >>=? fun res -> return res let originate cctxt block ?confirmations ?branch ~source ~src_pk ~src_sk ~fee ?(gas_limit = Z.minus_one) ?(storage_limit = -1L) origination = Alpha_services.Contract.counter cctxt block source >>=? fun pcounter -> let counter = Int32.succ pcounter in let operations = [origination] in append_reveal cctxt block ~source ~src_pk operations >>=? fun operations -> let contents = Sourced_operation (Manager_operations { source ; fee ; counter ; gas_limit ; storage_limit ; operations }) in Injection.inject_operation cctxt block ?confirmations ?branch ~src_sk contents >>=? fun (_oph, _op, result as res) -> Lwt.return (Injection.originated_contracts result) >>=? function | [ contract ] -> return (res, contract) | contracts -> failwith "The origination introduced %d contracts instead of one." (List.length contracts) let originate_account cctxt block ?confirmations ?branch ~source ~src_pk ~src_sk ~manager_pkh ?(delegatable = false) ?delegate ~balance ~fee () = let origination = Origination { manager = manager_pkh ; delegate ; script = None ; spendable = true ; delegatable ; credit = balance ; preorigination = None } in originate cctxt block ?confirmations ?branch ~source ~gas_limit:Z.zero~src_pk ~src_sk ~fee origination let delegate_contract cctxt block ?branch ?confirmations ~source ~src_pk ~src_sk ~fee delegate_opt = Alpha_services.Contract.counter cctxt block source >>=? fun pcounter -> let counter = Int32.succ pcounter in let operations = [Delegation delegate_opt] in append_reveal cctxt block ~source ~src_pk operations >>=? fun operations -> let contents = Sourced_operation (Manager_operations { source ; fee ; counter ; gas_limit = Z.zero ; storage_limit = 0L ; operations }) in Injection.inject_operation cctxt block ?confirmations ?branch ~src_sk contents >>=? fun res -> return res let list_contract_labels (cctxt : #Proto_alpha.full) block = Alpha_services.Contract.list cctxt block >>=? fun contracts -> map_s (fun h -> begin match Contract.is_implicit h with | Some m -> begin Public_key_hash.rev_find cctxt m >>=? function | None -> return "" | Some nm -> RawContractAlias.find_opt cctxt nm >>=? function | None -> return (" (known as " ^ nm ^ ")") | Some _ -> return (" (known as key:" ^ nm ^ ")") end | None -> begin RawContractAlias.rev_find cctxt h >>=? function | None -> return "" | Some nm -> return (" (known as " ^ nm ^ ")") end end >>=? fun nm -> let kind = match Contract.is_implicit h with | Some _ -> " (implicit)" | None -> "" in let h_b58 = Contract.to_b58check h in return (nm, h_b58, kind)) contracts let message_added_contract (cctxt : #Proto_alpha.full) name = cctxt#message "Contract memorized as %s." name let get_manager (cctxt : #Proto_alpha.full) block source = Client_proto_contracts.get_manager cctxt block source >>=? fun src_pkh -> Client_keys.get_key cctxt src_pkh >>=? fun (src_name, src_pk, src_sk) -> return (src_name, src_pkh, src_pk, src_sk) let dictate rpc_config block ?confirmations command src_sk = let contents = Sourced_operation (Dictator_operation command) in Injection.inject_operation rpc_config block ?confirmations ~src_sk contents >>=? fun res -> return res let set_delegate cctxt block ?confirmations ~fee contract ~src_pk ~manager_sk opt_delegate = delegate_contract cctxt block ?confirmations ~source:contract ~src_pk ~src_sk:manager_sk ~fee opt_delegate let register_as_delegate cctxt block ?confirmations ~fee ~manager_sk src_pk = let source = Signature.Public_key.hash src_pk in delegate_contract cctxt block ?confirmations ~source:(Contract.implicit_contract source) ~src_pk ~src_sk:manager_sk ~fee (Some source) let source_to_keys (wallet : #Proto_alpha.full) block source = get_manager wallet block source >>=? fun (_src_name, _src_pkh, src_pk, src_sk) -> return (src_pk, src_sk) let save_contract ~force cctxt alias_name contract = RawContractAlias.add ~force cctxt alias_name contract >>=? fun () -> message_added_contract cctxt alias_name >>= fun () -> return () let originate_contract (cctxt : #Proto_alpha.full) block ?confirmations ?branch ~fee ?gas_limit ?storage_limit ~delegate ?(delegatable=true) ?(spendable=false) ~initial_storage ~manager ~balance ~source ~src_pk ~src_sk ~code () = Lwt.return (Michelson_v1_parser.parse_expression initial_storage) >>= fun result -> Lwt.return (Micheline_parser.no_parsing_error result) >>=? fun { Michelson_v1_parser.expanded = storage } -> let origination = Origination { manager ; delegate ; script = Some { code ; storage } ; spendable ; delegatable ; credit = balance ; preorigination = None } in originate cctxt block ?confirmations ?branch ~source ~src_pk ~src_sk ~fee ?gas_limit ?storage_limit origination type activation_key = { pkh : Ed25519.Public_key_hash.t ; amount : Tez.t ; secret : Blinded_public_key_hash.secret ; mnemonic : string list ; password : string ; email : string ; } let activation_key_encoding = let open Data_encoding in conv (fun { pkh ; amount ; secret ; mnemonic ; password ; email } -> ( pkh, amount, secret, mnemonic, password, email )) (fun ( pkh, amount, secret, mnemonic, password, email ) -> { pkh ; amount ; secret ; mnemonic ; password ; email }) (obj6 (req "pkh" Ed25519.Public_key_hash.encoding) (req "amount" Tez.encoding) (req "secret" Blinded_public_key_hash.secret_encoding) (req "mnemonic" (list string)) (req "password" string) (req "email" string)) let read_key key = match Bip39.of_words key.mnemonic with | None -> failwith "" | Some t -> (* TODO: unicode normalization (NFKD)... *) let sk = Bip39.to_seed ~passphrase:(key.email ^ key.password) t in let sk = Cstruct.(to_bigarray (sub sk 0 32)) in let sk : Signature.Secret_key.t = Ed25519 (Data_encoding.Binary.of_bytes_exn Ed25519.Secret_key.encoding sk) in let pk = Signature.Secret_key.to_public_key sk in let pkh = Signature.Public_key.hash pk in return (pkh, pk, sk) let claim_commitment (cctxt : #Proto_alpha.full) ?(encrypted = false) ?confirmations ?force block key name = read_key key >>=? fun (pkh, pk, sk) -> fail_unless (Signature.Public_key_hash.equal pkh (Ed25519 key.pkh)) (failure "@[Inconsistent activation key:@ \ Computed pkh: %a@ \ Embedded pkh: %a @]" Signature.Public_key_hash.pp pkh Ed25519.Public_key_hash.pp key.pkh) >>=? fun () -> let contents = Anonymous_operations [ Activation { id = key.pkh ; secret = key.secret } ] in Injection.inject_operation cctxt ?confirmations block contents >>=? fun (_oph, _op, _result as res) -> let pk_uri = Tezos_signer_backends.Unencrypted.make_pk pk in begin if encrypted then Tezos_signer_backends.Encrypted.encrypt cctxt sk else return (Tezos_signer_backends.Unencrypted.make_sk sk) end >>=? fun sk_uri -> Client_keys.register_key cctxt ?force (pkh, pk_uri, sk_uri) name >>=? fun () -> begin match confirmations with | None -> return () | Some _confirmations -> Alpha_services.Contract.balance cctxt (`Head 0) (Contract.implicit_contract pkh) >>=? fun balance -> cctxt#message "Account %s (%a) created with %s%a." name Signature.Public_key_hash.pp pkh Client_proto_args.tez_sym Tez.pp balance >>= fun () -> return () end >>=? fun () -> return res