Ton

Форк
0
/
restricted-wallet3-code.fc 
103 строки · 3.0 Кб
1
;; Restricted wallet initialized by a third party (a variant of restricted-wallet2-code.fc)
2
;; restricts access to parts of balance until certain dates
3

4
() recv_internal(slice in_msg) impure {
5
  ;; do nothing for internal messages
6
}
7

8
_ seconds_passed(int start_at, int utime) inline_ref {
9
  ifnot (start_at) {
10
    var p = config_param(-13);
11
    start_at = null?(p) ? 0 : begin_parse(p).preload_uint(32);
12
  }
13
  return start_at ? utime - start_at : -1;
14
}
15

16
() recv_external(slice in_msg) impure {
17
  var signature = in_msg~load_bits(512);
18
  var cs = in_msg;
19
  var (subwallet_id, valid_until, msg_seqno) = (cs~load_uint(32), cs~load_uint(32), cs~load_uint(32));
20
  throw_if(35, valid_until <= now());
21
  var ds = get_data().begin_parse();
22
  var (stored_seqno, stored_subwallet, public_key) = (ds~load_uint(32), ds~load_uint(32), ds~load_uint(256));
23
  throw_unless(33, msg_seqno == stored_seqno);
24
  throw_unless(34, subwallet_id == stored_subwallet);
25
  throw_unless(36, check_signature(slice_hash(in_msg), signature, public_key));
26
  ifnot (msg_seqno) {
27
    public_key = ds~load_uint(256);  ;; load "final" public key
28
    ds.end_parse();
29
    cs~touch();
30
    var (start_at, rdict) = (cs~load_uint(32), cs~load_dict());
31
    cs.end_parse();
32
    accept_message();
33
    set_data(begin_cell()
34
              .store_uint(stored_seqno + 1, 32)
35
              .store_uint(stored_subwallet, 32)
36
              .store_uint(public_key, 256)
37
              .store_uint(start_at, 32)
38
              .store_dict(rdict)
39
            .end_cell());
40
    return ();
41
  }
42
  var (start_at, rdict) = (ds~load_uint(32), ds~load_dict());
43
  ds.end_parse();
44
  accept_message();
45
  var ts = seconds_passed(start_at, now());
46
  var (_, value, found) = rdict.idict_get_preveq?(32, ts);
47
  if (found) {
48
    raw_reserve(value~load_grams(), 2);
49
  }
50
  cs~touch();
51
  while (cs.slice_refs()) {
52
    var mode = cs~load_uint(8);
53
    var msg = cs~load_ref();
54
    send_raw_message(msg, mode);
55
  }
56
  cs.end_parse();
57
  set_data(begin_cell()
58
            .store_uint(stored_seqno + 1, 32)
59
            .store_uint(stored_subwallet, 32)
60
            .store_uint(public_key, 256)
61
            .store_uint(start_at, 32)
62
            .store_dict(rdict)
63
          .end_cell());
64
}
65

66
;; Get methods
67

68
int seqno() method_id {
69
  return get_data().begin_parse().preload_uint(32);
70
}
71

72
int wallet_id() method_id {
73
  var ds = get_data().begin_parse();
74
  ds~load_uint(32);
75
  return ds.preload_uint(32);
76
}
77

78
int get_public_key() method_id {
79
  var ds = get_data().begin_parse();
80
  ds~load_uint(32 + 32);
81
  return ds.preload_uint(256);
82
}
83

84
int compute_balance_at(int utime) inline_ref {
85
  var ds = get_data().begin_parse().skip_bits(32 + 32 + 256);
86
  var (start_at, rdict) = (ds~load_uint(32), ds~load_dict());
87
  ds.end_parse();
88
  var ts = seconds_passed(start_at, utime);
89
  var balance = get_balance().pair_first();
90
  var (_, value, found) = rdict.idict_get_preveq?(32, ts);
91
  if (found) {
92
    balance = max(balance - value~load_grams(), 0);
93
  }
94
  return balance;
95
}
96

97
int balance_at(int utime) method_id {
98
  return compute_balance_at(utime);
99
}
100

101
int balance() method_id {
102
  return compute_balance_at(now());
103
}
104

Использование cookies

Мы используем файлы cookie в соответствии с Политикой конфиденциальности и Политикой использования cookies.

Нажимая кнопку «Принимаю», Вы даете АО «СберТех» согласие на обработку Ваших персональных данных в целях совершенствования нашего веб-сайта и Сервиса GitVerse, а также повышения удобства их использования.

Запретить использование cookies Вы можете самостоятельно в настройках Вашего браузера.