You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
239 lines
8.4 KiB
239 lines
8.4 KiB
;; Wrapped TON minter smart contract |
|
|
|
;; storage scheme |
|
;; storage#_ total_supply:Coins admin_address:MsgAddress next_admin_address:MsgAddress content:^Cell jetton_wallet_code:^Cell = Storage; |
|
|
|
#pragma version >=0.2.0; |
|
|
|
#include "imports/stdlib.fc"; |
|
#include "imports/op-codes.fc"; |
|
#include "imports/utils.fc"; |
|
#include "imports/discovery-params.fc"; |
|
|
|
int gas_consumption() asm "10000000 PUSHINT"; ;; 0.01 TON |
|
int withdraw_gas_consumption() asm "10000000 PUSHINT"; ;; 0.01 TON |
|
int deposit_gas_consumption() asm "35000000 PUSHINT"; ;; 0.035 TON |
|
|
|
(int, slice, slice, cell, cell) load_data() inline { |
|
slice ds = get_data().begin_parse(); |
|
return ( |
|
ds~load_coins(), ;; total_supply |
|
ds~load_msg_addr(), ;; admin_address |
|
ds~load_msg_addr(), ;; next_admin_address |
|
ds~load_ref(), ;; content |
|
ds~load_ref() ;; jetton_wallet_code |
|
); |
|
} |
|
|
|
() save_data(int total_supply, slice admin_address, slice next_admin_address, cell content, cell jetton_wallet_code) impure inline { |
|
set_data(begin_cell() |
|
.store_coins(total_supply) |
|
.store_slice(admin_address) |
|
.store_slice(next_admin_address) |
|
.store_ref(content) |
|
.store_ref(jetton_wallet_code) |
|
.end_cell() |
|
); |
|
} |
|
|
|
() mint_tokens(slice to_address, cell jetton_wallet_code, int amount) impure { |
|
cell state_init = calculate_jetton_wallet_state_init(to_address, my_address(), jetton_wallet_code); |
|
slice to_wallet_address = calculate_jetton_wallet_address(state_init); |
|
var master_msg = begin_cell() |
|
.store_uint(op::internal_transfer, 32) |
|
;; query_id |
|
.store_uint(0, 64) |
|
;; jetton_amount |
|
.store_coins(amount) |
|
;; from_address == to_address |
|
.store_slice(to_address) |
|
;; response_address |
|
.store_slice(my_address()) |
|
;; forward ton amount |
|
.store_coins(0) |
|
.end_cell(); |
|
var msg = begin_cell() |
|
.store_uint(0x18, 6) |
|
.store_slice(to_wallet_address) |
|
.store_coins(gas_consumption()) |
|
.store_uint(4 + 2 + 1, 1 + 4 + 4 + 64 + 32 + 1 + 1 + 1) |
|
.store_ref(state_init) |
|
.store_ref(master_msg); |
|
send_raw_message(msg.end_cell(), 1); ;; pay transfer fees separately, revert on errors |
|
} |
|
|
|
() burn_tokens(slice to_address, cell jetton_wallet_code, int amount) impure { |
|
cell state_init = calculate_jetton_wallet_state_init(to_address, my_address(), jetton_wallet_code); |
|
slice to_wallet_address = calculate_jetton_wallet_address(state_init); |
|
var master_msg = begin_cell() |
|
.store_uint(op::burn, 32) |
|
;; query_id |
|
.store_uint(0, 64) |
|
;; jetton_amount |
|
.store_coins(amount) |
|
;; response_address |
|
.store_slice(my_address()) |
|
;; custom_payload (currently not used) |
|
.store_dict(new_dict()) |
|
.end_cell(); |
|
var msg = begin_cell() |
|
.store_uint(0x18, 6) |
|
.store_slice(to_wallet_address) |
|
.store_coins(gas_consumption() * 3) |
|
.store_uint(4 + 2 + 1, 1 + 4 + 4 + 64 + 32 + 1 + 1 + 1) |
|
.store_ref(state_init) |
|
.store_ref(master_msg); |
|
send_raw_message(msg.end_cell(), 1); ;; pay transfer fees separately, revert on errors |
|
} |
|
|
|
() recv_internal(int my_balance, int msg_value, cell in_msg_full, slice in_msg_body) impure { |
|
if (in_msg_body.slice_empty?()) { ;; ignore empty messages |
|
return (); |
|
} |
|
slice cs = in_msg_full.begin_parse(); |
|
int flags = cs~load_uint(4); |
|
|
|
if (flags & 1) { ;; ignore all bounced messages |
|
return (); |
|
} |
|
slice sender_address = cs~load_msg_addr(); |
|
cs~load_msg_addr(); ;; skip dst |
|
cs~load_coins(); ;; skip value |
|
cs~skip_bits(1); ;; skip extracurrency collection |
|
cs~load_coins(); ;; skip ihr_fee |
|
int fwd_fee = cs~load_coins(); ;; we use message fwd_fee for estimation of provide_wallet_address cost |
|
|
|
int op = in_msg_body~load_uint(32); |
|
int query_id = in_msg_body~load_uint(64); |
|
|
|
(int total_supply, slice admin_address, slice next_admin_address, cell content, cell jetton_wallet_code) = load_data(); |
|
|
|
if (op == op::deposit) { |
|
msg_value -= deposit_gas_consumption(); |
|
throw_if(74, msg_value < 0); |
|
|
|
int ton_amount = msg_value; |
|
int jetton_amount = in_msg_body~load_coins(); |
|
throw_if(75, ton_amount < jetton_amount); |
|
mint_tokens(sender_address, jetton_wallet_code, jetton_amount); |
|
save_data(total_supply + jetton_amount, admin_address, next_admin_address, content, jetton_wallet_code); |
|
|
|
msg_value -= jetton_amount + gas_consumption(); |
|
if (msg_value > 0) { |
|
var msg = begin_cell() |
|
.store_uint(0x10, 6) ;; nobounce - int_msg_info$0 ihr_disabled:Bool bounce:Bool bounced:Bool src:MsgAddress -> 010000 |
|
.store_slice(sender_address) |
|
.store_coins(msg_value) |
|
.store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1) |
|
.store_uint(op::excesses, 32) |
|
.store_uint(query_id, 64); |
|
send_raw_message(msg.end_cell(), 2); ;; pay transfer fees together, ignore errors |
|
} |
|
|
|
return (); |
|
} |
|
|
|
if (op == op::withdraw) { |
|
msg_value -= withdraw_gas_consumption(); |
|
throw_if(74, msg_value < 0); |
|
|
|
int jetton_amount = in_msg_body~load_coins(); |
|
throw_if(75, total_supply < jetton_amount); |
|
burn_tokens(sender_address, jetton_wallet_code, jetton_amount); |
|
save_data(total_supply, admin_address, next_admin_address, content, jetton_wallet_code); |
|
|
|
return (); |
|
} |
|
|
|
if (op == op::burn_notification) { |
|
int jetton_amount = in_msg_body~load_coins(); |
|
slice from_address = in_msg_body~load_msg_addr(); |
|
throw_unless(74, |
|
equal_slices(calculate_user_jetton_wallet_address(from_address, my_address(), jetton_wallet_code), sender_address) |
|
); |
|
|
|
if (jetton_amount > 0) { |
|
var msg = begin_cell() |
|
.store_uint(0x10, 6) ;; nobounce - int_msg_info$0 ihr_disabled:Bool bounce:Bool bounced:Bool src:MsgAddress -> 010000 |
|
.store_slice(from_address) |
|
.store_coins(jetton_amount) |
|
.store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1) |
|
;; op code = 0 |
|
.store_uint(0, 32) |
|
.store_uint(query_id, 64); |
|
send_raw_message(msg.end_cell(), 1); ;; pay transfer fees seperatly, revert on errors |
|
} |
|
|
|
save_data(total_supply - jetton_amount, admin_address, next_admin_address, content, jetton_wallet_code); |
|
|
|
return (); |
|
} |
|
|
|
if (op == op::provide_wallet_address) { |
|
throw_unless(75, msg_value > fwd_fee + gas_consumption()); |
|
|
|
slice owner_address = in_msg_body~load_msg_addr(); |
|
int include_address? = in_msg_body~load_uint(1); |
|
|
|
cell included_address = include_address? |
|
? begin_cell().store_slice(owner_address).end_cell() |
|
: null(); |
|
|
|
var msg = begin_cell() |
|
.store_uint(0x18, 6) |
|
.store_slice(sender_address) |
|
.store_coins(0) |
|
.store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1) |
|
.store_uint(op::take_wallet_address, 32) |
|
.store_uint(query_id, 64); |
|
|
|
if (is_resolvable?(owner_address)) { |
|
msg = msg.store_slice(calculate_user_jetton_wallet_address(owner_address, my_address(), jetton_wallet_code)); |
|
} else { |
|
msg = msg.store_uint(0, 2); ;; addr_none |
|
} |
|
send_raw_message(msg.store_maybe_ref(included_address).end_cell(), 64); |
|
return (); |
|
} |
|
|
|
if (op == op::change_next_admin) { ;; change next admin |
|
throw_unless(73, equal_slices(sender_address, admin_address)); |
|
slice new_next_admin_address = in_msg_body~load_msg_addr(); |
|
save_data(total_supply, admin_address, new_next_admin_address, content, jetton_wallet_code); |
|
return (); |
|
} |
|
|
|
if (op == op::change_admin) { ;; change admin (only next admin can get an admin permission) |
|
throw_unless(73, equal_slices(sender_address, next_admin_address)); |
|
slice addr_none = begin_cell().store_uint(0, 2).end_cell().begin_parse(); |
|
save_data(total_supply, next_admin_address, addr_none, content, jetton_wallet_code); |
|
return (); |
|
} |
|
|
|
if (op == op::change_content) { ;; change content, delete this for immutable tokens |
|
throw_unless(73, equal_slices(sender_address, admin_address)); |
|
save_data(total_supply, admin_address, next_admin_address, in_msg_body~load_ref(), jetton_wallet_code); |
|
return (); |
|
} |
|
|
|
if (op == op::excesses) { |
|
return (); |
|
} |
|
|
|
throw(0xffff); |
|
} |
|
|
|
(int, int, slice, cell, cell) get_jetton_data() method_id { |
|
(int total_supply, slice admin_address, _, cell content, cell jetton_wallet_code) = load_data(); |
|
return (total_supply, -1, admin_address, content, jetton_wallet_code); |
|
} |
|
|
|
(slice) get_jetton_extra_data() method_id { |
|
(_, _, slice next_admin_address, _, _) = load_data(); |
|
return (next_admin_address); |
|
} |
|
|
|
slice get_wallet_address(slice owner_address) method_id { |
|
(_, _, _, _, cell jetton_wallet_code) = load_data(); |
|
return calculate_user_jetton_wallet_address(owner_address, my_address(), jetton_wallet_code); |
|
}
|
|
|