1
0
Fork 0
mirror of https://github.com/ton-blockchain/ton synced 2025-02-15 04:32:21 +00:00
ton/crypto/func/auto-tests/legacy_tests/uni-lock-wallet/uni-lockup-wallet.fc
EmelyanenkoK 6b49d6a382
Add legacy_tester for existing funC contracts (#588)
* Add legacy_tester for existing funC contracts

* Add storage-contracts and pragma options
2023-01-12 12:33:15 +03:00

217 lines
7.1 KiB
Text

;; Restricted wallet initialized by a third party (a variant of restricted-wallet3-code.fc)
;; Allows to add more locked budget after initialization
#include "stdlib.fc";
int err:wrong_signature() asm "31 PUSHINT";
int err:wrong_config_signature() asm "32 PUSHINT";
int err:value_is_too_small() asm "33 PUSHINT";
int err:wrong_seqno() asm "34 PUSHINT";
int err:wrong_subwallet_id() asm "35 PUSHINT";
int err:replay_protection() asm "36 PUSHINT";
int err:unknown_op() asm "40 PUSHINT";
int err:unknown_cmd() asm "41 PUSHINT";
int op:rwallet_op() asm "0x82eaf9c4 PUSHINT";
int cmd:restricted_transfer() asm "0x373aa9f4 PUSHINT";
_ is_whitelisted?(addr, allowed_destinations) {
(_, _, _, int found) = allowed_destinations.pfxdict_get?(addr.slice_bits(), addr);
return found;
}
_ check_message_destination(msg, allowed_destinations) inline_ref {
var cs = msg.begin_parse();
var flags = cs~load_uint(4);
if (flags & 8) {
;; external messages are always valid
return true;
}
var (s_addr, d_addr) = (cs~load_msg_addr(), cs~load_msg_addr());
return is_whitelisted?(d_addr, allowed_destinations);
}
_ unpack_data() {
var cs = get_data().begin_parse();
var res = (
cs~load_uint(32),
cs~load_uint(32),
cs~load_uint(256),
cs~load_uint(256),
cs~load_dict(),
cs~load_grams(),
cs~load_dict(),
cs~load_grams(),
cs~load_dict()
);
cs.end_parse();
return res;
}
_ pack_data(int seqno, int subwallet_id, int public_key, int config_public_key, cell allowed_destinations, int total_locked_value, cell
locked, int total_restricted_value, cell restricted) {
return begin_cell()
.store_int(seqno, 32)
.store_int(subwallet_id, 32)
.store_uint(public_key, 256)
.store_uint(config_public_key, 256)
.store_dict(allowed_destinations)
.store_grams(total_locked_value)
.store_dict(locked)
.store_grams(total_restricted_value)
.store_dict(restricted).end_cell();
}
(cell, int) lock_grams(cell locked, int total, int ts, int value) {
total += value;
(slice found_cs, var found) = locked.udict_get?(32, ts);
if (found) {
var found_value = found_cs~load_grams();
found_cs.end_parse();
value += found_value;
}
locked~udict_set_builder(32, ts, begin_cell().store_grams(value));
return (locked, total);
}
(cell, int) unlock_grams(cell locked, int total, int now_ts) {
do {
var (locked', ts, value_cs, f) = locked.udict_delete_get_min(32);
f~touch();
if (f) {
f = ts <= now_ts;
}
if (f) {
locked = locked';
int value = value_cs~load_grams();
value_cs.end_parse();
total -= value;
}
} until (~ f);
return (locked, total);
}
() recv_internal(int msg_value, cell in_msg_cell, slice in_msg) impure {
var cs = in_msg_cell.begin_parse();
var flags = cs~load_uint(4); ;; int_msg_info$0 ihr_disabled:Bool bounce:Bool bounced:Bool
if (flags & 1) {
;; ignore all bounced messages
return ();
}
var s_addr = cs~load_msg_addr();
if (in_msg.slice_empty?()) {
return();
}
int op = in_msg~load_uint(32);
if (op <= 1) {
;; simple transfer with comment, return
return ();
}
var (stored_seqno, stored_subwallet, public_key, config_public_key, allowed_destinations, total_locked_value, locked,
total_restricted_value, restricted) = unpack_data();
if is_whitelisted?(s_addr, allowed_destinations) & (op != op:rwallet_op()) {
return ();
}
throw_unless(err:unknown_op(), op == op:rwallet_op());
throw_unless(err:value_is_too_small(), msg_value >= 1000000000);
var signature = in_msg~load_bits(512);
throw_unless(err:wrong_config_signature(), check_signature(slice_hash(in_msg), signature, config_public_key));
int cmd = in_msg~load_uint(32);
throw_unless(err:unknown_cmd(), cmd == cmd:restricted_transfer());
var (only_restrict, ts) = (in_msg~load_uint(1), in_msg~load_uint(32));
if (only_restrict) {
(restricted, total_restricted_value) = lock_grams(restricted, total_restricted_value, ts, msg_value);
} else {
(locked, total_locked_value) = lock_grams(locked, total_locked_value, ts, msg_value);
}
set_data(pack_data(stored_seqno, stored_subwallet, public_key, config_public_key, allowed_destinations, total_locked_value, locked,
total_restricted_value, restricted));
}
() recv_external(slice in_msg) impure {
var signature = in_msg~load_bits(512);
var cs = in_msg;
var (subwallet_id, valid_until, msg_seqno) = (cs~load_uint(32), cs~load_uint(32), cs~load_uint(32));
throw_if(err:replay_protection(), valid_until <= now());
var (stored_seqno, stored_subwallet, public_key, config_public_key, allowed_destinations, total_locked_value, locked,
total_restricted_value, restricted) = unpack_data();
throw_unless(err:wrong_seqno(), msg_seqno == stored_seqno);
throw_unless(err:wrong_subwallet_id(), subwallet_id == stored_subwallet);
throw_unless(err:wrong_signature(), check_signature(slice_hash(in_msg), signature, public_key));
accept_message();
(restricted, total_restricted_value) = unlock_grams(restricted, total_restricted_value, now());
(locked, total_locked_value) = unlock_grams(locked, total_locked_value, now());
int effectively_locked = total_locked_value;
int can_use_restricted = 1;
var cs_copy = cs;
while (cs_copy.slice_refs()) {
var mode = cs_copy~load_uint(8);
var msg = cs_copy~load_ref();
can_use_restricted &= check_message_destination(msg, allowed_destinations);
}
ifnot (can_use_restricted) {
effectively_locked += total_restricted_value;
}
raw_reserve(effectively_locked, 2);
cs~touch();
while (cs.slice_refs()) {
var mode = cs~load_uint(8);
var msg = cs~load_ref();
send_raw_message(msg, mode);
}
cs.end_parse();
set_data(pack_data(stored_seqno + 1, stored_subwallet, public_key, config_public_key, allowed_destinations, total_locked_value, locked,
total_restricted_value, restricted));
}
;; Get methods
int seqno() method_id {
return get_data().begin_parse().preload_uint(32);
}
int wallet_id() method_id {
var ds = get_data().begin_parse();
ds~load_uint(32);
return ds.preload_uint(32);
}
int get_public_key() method_id {
var ds = get_data().begin_parse();
ds~load_uint(32 + 32);
return ds.preload_uint(256);
}
;; the next three methods are mostly for testing
(int, int, int) get_balances_at(int time) method_id {
var (stored_seqno, stored_subwallet, public_key, config_public_key, allowed_destinations, total_locked_value, locked,
total_restricted_value, restricted) = unpack_data();
(restricted, total_restricted_value) = unlock_grams(restricted, total_restricted_value, time);
(locked, total_locked_value) = unlock_grams(locked, total_locked_value, time);
int ton_balance = get_balance().pair_first();
return ( ton_balance,
total_restricted_value,
total_locked_value );
}
(int, int, int) get_balances() method_id {
return get_balances_at(now());
}
int check_destination(slice destination) method_id {
var (stored_seqno, stored_subwallet, public_key, config_public_key, allowed_destinations, total_locked_value, locked,
total_restricted_value, restricted) = unpack_data();
return is_whitelisted?(destination, allowed_destinations);
}