/* This file is part of TON Blockchain Library. TON Blockchain Library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. TON Blockchain Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with TON Blockchain Library. If not, see . Copyright 2017-2020 Telegram Systems LLP */ #include "WalletV3.h" #include "GenericAccount.h" #include "SmartContractCode.h" #include "vm/boc.h" #include "vm/cells/CellString.h" #include "td/utils/base64.h" #include namespace ton { td::Ref WalletV3::get_init_state(const td::Ed25519::PublicKey& public_key, td::uint32 wallet_id, td::int32 revision) noexcept { auto code = get_init_code(revision); auto data = get_init_data(public_key, wallet_id); return GenericAccount::get_init_state(std::move(code), std::move(data)); } td::optional WalletV3::guess_revision(const vm::Cell::Hash& code_hash) { for (td::int32 i = 1; i <= 2; i++) { if (get_init_code(i)->get_hash() == code_hash) { return i; } } return {}; } td::optional WalletV3::guess_revision(const block::StdAddress& address, const td::Ed25519::PublicKey& public_key, td::uint32 wallet_id) { for (td::int32 i = 1; i <= 2; i++) { if (GenericAccount::get_address(address.workchain, get_init_state(public_key, wallet_id, i)) == address) { return i; } } return {}; } td::Ref WalletV3::make_a_gift_message(const td::Ed25519::PrivateKey& private_key, td::uint32 wallet_id, td::uint32 seqno, td::uint32 valid_until, td::Span gifts) noexcept { CHECK(gifts.size() <= max_gifts_size); vm::CellBuilder cb; cb.store_long(wallet_id, 32).store_long(valid_until, 32).store_long(seqno, 32); for (auto& gift : gifts) { td::int32 send_mode = 3; auto gramms = gift.gramms; if (gramms == -1) { gramms = 0; send_mode += 128; } vm::CellBuilder cbi; GenericAccount::store_int_message(cbi, gift.destination, gramms); store_gift_message(cbi, gift); auto message_inner = cbi.finalize(); cb.store_long(send_mode, 8).store_ref(std::move(message_inner)); } auto message_outer = cb.finalize(); auto signature = private_key.sign(message_outer->get_hash().as_slice()).move_as_ok(); return vm::CellBuilder().store_bytes(signature).append_cellslice(vm::load_cell_slice(message_outer)).finalize(); } td::Ref WalletV3::get_init_code(td::int32 revision) noexcept { return SmartContractCode::wallet3(revision); } vm::CellHash WalletV3::get_init_code_hash() noexcept { return get_init_code()->get_hash(); } td::Ref WalletV3::get_init_data(const td::Ed25519::PublicKey& public_key, td::uint32 wallet_id, td::uint32 seqno) noexcept { return vm::CellBuilder() .store_long(seqno, 32) .store_long(wallet_id, 32) .store_bytes(public_key.as_octet_string()) .finalize(); } td::Result WalletV3::get_seqno() const { return TRY_VM(get_seqno_or_throw()); } td::Result WalletV3::get_seqno_or_throw() const { if (state_.data.is_null()) { return 0; } //FIXME use get method return static_cast(vm::load_cell_slice(state_.data).fetch_ulong(32)); } td::Result WalletV3::get_wallet_id() const { return TRY_VM(get_wallet_id_or_throw()); } td::Result WalletV3::get_wallet_id_or_throw() const { if (state_.data.is_null()) { return 0; } //FIXME use get method auto cs = vm::load_cell_slice(state_.data); cs.skip_first(32); return static_cast(cs.fetch_ulong(32)); } td::Result WalletV3::get_public_key() const { return TRY_VM(get_public_key_or_throw()); } td::Result WalletV3::get_public_key_or_throw() const { if (state_.data.is_null()) { return td::Status::Error("data is null"); } //FIXME use get method auto cs = vm::load_cell_slice(state_.data); cs.skip_first(64); td::SecureString res(td::Ed25519::PublicKey::LENGTH); cs.fetch_bytes(res.as_mutable_slice().ubegin(), td::narrow_cast(res.size())); return td::Ed25519::PublicKey(std::move(res)); } } // namespace ton