经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 其他 » 区块链 » 查看文章
Ton 区块链的官方 类ERC20-Token 智能合约代码-Transfer部分解析
来源:cnblogs  作者:指尖下的幽灵  时间:2024/6/12 16:14:58  对本文有异议

作者:林冠宏 / 指尖下的幽灵。转载者,请: 务必标明出处。

掘金:https://juejin.im/user/1785262612681997

GitHub : https://github.com/af913337456/

出版的书籍:


Ton 区块链的官方 类ERC20-Token 智能合约代码-Transfer部分解析

最近在学习 Ton 链的智能合约,由于我之前的经验思维主要是集中在以太坊这条链的,即Solidity那套,所以带着长久偏向的思维去阅读 Ton 的合约时发现格格不入,Ton 的合约设计与EVM体系的属于天壤之别。

首先 Ton 的合约是分片的,遵循 Parent-Child 的规则,这里详细了解见:

https://blog.ton.org/how-to-shard-your-ton-smart-contract-and-why-studying-the-anatomy-of-tons-jettons

其次是合约开发的语言,Ton 有三种,用得最多的是 FunC,这是一种完全的非主流语言,在 GitHub 上都没有特点标识的那种。


按照最快了解 Token 智能合约的方式,寻找到官方的合约代码项目。由于Ton 的经济 Token 代码目前还没有类似以太坊的各种模型协议,只能把对应以太坊ERC-20的那部分取下来进行阅读。

下面我将结合Token的转账核心操作的源码来对其整个调用链路 进行细致的分析讲解,所选代码片段也有注释。

先了解合约模式
  • Ton 的合约是分片的,拿 Token 类型的合约做例子,其做法是将一份主合约,被称为 Master 或 Minter 的合约独一份进行部署,再将和 User 的子合约在转账进行时进行新建形式的一一对应部署。
  • 比如说,发布一份名叫 NOT 的 Token 合约,它的 Master 合约将被部署在链上,然后对于后续每一位收到 NOT token 的用户地址,若不存在就都会创建一份与该地址对应的子合约,称为 Wallet 合约。
  • 在 Token 类型的合约中,Master 合约中存储了 Token 的公共信息,比如 Name,Metaurl,Supply 等,而Transfer 转账行为却都发生在各自的 Wallet 合约里面。
  • 为 User 创建 Wallet 合约都要经过 Master 进行。
  • 合约允许各自内部调用,A 合约调用 B 合约的函数。

客户端-发起转账 Token 的流程

例子取于 Golang 客户端项目代码。

  1. func main() {
  2. ...
  3. // 初始化自己的钱包
  4. w, _ := wallet.FromSeed(api, words, wallet.V3R2)
  5. // 根据该 Token 的 Master 合约地址初始化 Token
  6. token := jetton.NewJettonMasterClient(api, address.MustParseAddr("EQD0vdS......"))
  7. // 调用 Master 的合约函数获取转账者的 Wallet 合约
  8. tokenWallet, _ := token.GetJettonWallet(ctx, w.WalletAddress())
  9. tokenBalance, _ := tokenWallet.GetBalance(ctx)
  10. amountTokens := tlb.MustFromDecimal("0.1", 9)
  11. // 转账附带的信息
  12. comment, _ := wallet.CreateCommentCell("Hello from tonutils-go!")
  13. // 初始化收款者的地址,这不是 Wallet 地址
  14. to := address.MustParseAddr("EQCD39VS5jcptHL8vMjEXrzGaRcCVYto7HUn4bpAOg8xqB2N")
  15. // 在 BuildTransferPayloadV2 里指定了 OP = Transfer
  16. transferPayload, _ := tokenWallet.BuildTransferPayloadV2(to, to, amountTokens, tlb.ZeroCoins, comment, nil)
  17. // 构造链上请求的消息结构
  18. msg := wallet.SimpleMessage(tokenWallet.Address(), tlb.MustFromTON("0.05"), transferPayload)
  19. // 发送转账交易,然后结束
  20. tx, _, _ := w.SendWaitTransaction(ctx, msg)
  21. log.Println("transaction confirmed, hash:", base64.StdEncoding.EncodeToString(tx.Hash))
  22. }

上述代码可以看到在发起转账的时候,收款地址并不是 User 的钱包地址,而是其对应的 Wallet 合约地址。这一点就和包括以太坊在内的绝大部分公链都不一样。

合约端对应的转账入口代码解析

内部消息的入口函数,根据 op 参数指定调用入口。

  1. () recv_internal(int my_balance, int msg_value, cell in_msg_full, slice in_msg_body) impure {
  2. if (in_msg_body.slice_empty?()) { ;; ignore empty messages
  3. return ();
  4. }
  5. slice cs = in_msg_full.begin_parse();
  6. int flags = cs~load_uint(4);
  7. if (flags & 1) {
  8. on_bounce(in_msg_body);
  9. return ();
  10. }
  11. slice sender_address = cs~load_msg_addr();
  12. cs~load_msg_addr(); ;; skip dst
  13. cs~load_coins(); ;; skip value
  14. cs~skip_bits(1); ;; skip extracurrency collection
  15. cs~load_coins(); ;; skip ihr_fee
  16. int fwd_fee = muldiv(cs~load_coins(), 3, 2); ;; we use message fwd_fee for estimation of forward_payload costs
  17. int op = in_msg_body~load_uint(32);
  18. if (op == op::transfer()) { ;; outgoing transfer
  19. ;; sender_address 是一开始的转账者
  20. ;; msg_value 是改次转账中的 Ton 数额
  21. send_tokens(in_msg_body, sender_address, msg_value, fwd_fee);
  22. return ();
  23. }
  24. if (op == op::internal_transfer()) { ;; incoming transfer
  25. ;; my_balance 是当前所执行的合约所有者的 Ton 余额
  26. receive_tokens(in_msg_body, sender_address, my_balance, fwd_fee, msg_value);
  27. return ();
  28. }
  29. if (op == op::burn()) { ;; burn
  30. burn_tokens(in_msg_body, sender_address, msg_value, fwd_fee);
  31. return ();
  32. }
  33. throw(0xffff);
  34. }
  • recv_internal 是系统内置的函数入口,相当于 main;
  • 系统函数还有:load_datasave_data,加载的是当前合约的数据,存储也是存储到当前合约。代码中的变量 jetton_master_address 地址永远是父合约地址
  • 转账发起时,指定 op 是 transfer,走到代码处理点 op == op::transfer,进入到 send_tokens
  • send_tokens函数源码及其解析注释内容见下??
  1. () send_tokens (slice in_msg_body, slice sender_address, int msg_value, int fwd_fee) impure {
  2. int query_id = in_msg_body~load_uint(64);
  3. int jetton_amount = in_msg_body~load_coins();
  4. slice to_owner_address = in_msg_body~load_msg_addr(); ;; 收款人
  5. force_chain(to_owner_address);
  6. ;; owner_address 转账者,jetton_master_address Token主地址
  7. (int balance, slice owner_address, slice jetton_master_address, cell jetton_wallet_code) = load_data();
  8. ;; msg_value 是改次转账中的 Ton 数额,不是 token 余额
  9. balance -= jetton_amount; ;; balance 是转账者余额,jetton_amount 是要转的数额
  10. throw_unless(705, equal_slices(owner_address, sender_address)); ;; 要求发起转账的人一致
  11. throw_unless(706, balance >= 0); ;; 要求减去 jetton_amount 余额大于 0,防止超出
  12. cell state_init = calculate_jetton_wallet_state_init(to_owner_address, jetton_master_address, jetton_wallet_code);
  13. slice to_wallet_address = calculate_jetton_wallet_address(state_init);
  14. slice response_address = in_msg_body~load_msg_addr(); ;; 转账结束后要被通知到的地址
  15. cell custom_payload = in_msg_body~load_dict();
  16. int forward_ton_amount = in_msg_body~load_coins(); ;; 附属的要转的 Ton 的数额,可以是 0,客户端赋值
  17. throw_unless(708, slice_bits(in_msg_body) >= 1);
  18. slice either_forward_payload = in_msg_body;
  19. var msg = begin_cell()
  20. .store_uint(0x18, 6)
  21. .store_slice(to_wallet_address) ;; 走到收款人的合约处
  22. .store_coins(0)
  23. .store_uint(4 + 2 + 1, 1 + 4 + 4 + 64 + 32 + 1 + 1 + 1)
  24. .store_ref(state_init); ;; 如果对方没被部署过 wallet 合约,那么这个消息会触发部署
  25. var msg_body = begin_cell()
  26. .store_uint(op::internal_transfer(), 32) ;; internal_transfer 引导到下一个 op
  27. .store_uint(query_id, 64)
  28. .store_coins(jetton_amount)
  29. .store_slice(owner_address) ;; owner_address 转账者
  30. .store_slice(response_address) ;; 转账结束后要被通知到的地址
  31. .store_coins(forward_ton_amount)
  32. .store_slice(either_forward_payload)
  33. .end_cell();
  34. msg = msg.store_ref(msg_body);
  35. int fwd_count = forward_ton_amount ? 2 : 1;
  36. throw_unless(709, msg_value >
  37. forward_ton_amount +
  38. ;; 3 messages: wal1->wal2, wal2->owner, wal2->response
  39. ;; but last one is optional (it is ok if it fails)
  40. fwd_count * fwd_fee +
  41. (2 * gas_consumption() + min_tons_for_storage()));
  42. ;; universal message send fee calculation may be activated here
  43. ;; by using this instead of fwd_fee
  44. ;; msg_fwd_fee(to_wallet, msg_body, state_init, 15)
  45. send_raw_message(msg.end_cell(), 64); ;; revert on errors
  46. ;; 如果 send_raw_message 没出错,那么下面就会完成最后一步
  47. save_data(balance, owner_address, jetton_master_address, jetton_wallet_code); ;; 存储转账者的 token 余额
  48. }
  • send_tokens 里面,还会进行一次内部合约调用,调用到收款人的 Wallet 合约,对应到 op 是 internal_transfer,而 internal_transfer 的处理函数是 receive_tokens
  • receive_tokens函数源码及其解析注释内容见下??
  1. () receive_tokens (slice in_msg_body, slice sender_address, int my_ton_balance, int fwd_fee, int msg_value) impure {
  2. ;; NOTE we can not allow fails in action phase since in that case there will be
  3. ;; no bounce. Thus check and throw in computation phase.
  4. (int balance, slice owner_address, slice jetton_master_address, cell jetton_wallet_code) = load_data();
  5. int query_id = in_msg_body~load_uint(64);
  6. int jetton_amount = in_msg_body~load_coins(); ;; token 代币余额
  7. balance += jetton_amount; ;; token 代币余额累加
  8. slice from_address = in_msg_body~load_msg_addr(); ;; 原始的 Token 转账者
  9. slice response_address = in_msg_body~load_msg_addr(); ;; 转账结束后要被通知到的地址
  10. ;; sender_address 系统地址,意味着这个函数只能由系统内部调用,排除了外部调用
  11. throw_unless(707,
  12. equal_slices(jetton_master_address, sender_address)
  13. |
  14. equal_slices(calculate_user_jetton_wallet_address(from_address, jetton_master_address, jetton_wallet_code), sender_address)
  15. );
  16. int forward_ton_amount = in_msg_body~load_coins(); ;; 附属要转账 ton 数值
  17. int ton_balance_before_msg = my_ton_balance - msg_value;
  18. int storage_fee = min_tons_for_storage() - min(ton_balance_before_msg, min_tons_for_storage());
  19. msg_value -= (storage_fee + gas_consumption());
  20. if(forward_ton_amount) { ;; 附属要转账 ton,如果不是 0
  21. msg_value -= (forward_ton_amount + fwd_fee);
  22. slice either_forward_payload = in_msg_body;
  23. var msg_body = begin_cell()
  24. .store_uint(op::transfer_notification(), 32)
  25. .store_uint(query_id, 64)
  26. .store_coins(jetton_amount)
  27. .store_slice(from_address)
  28. .store_slice(either_forward_payload)
  29. .end_cell();
  30. var msg = begin_cell()
  31. .store_uint(0x10, 6) ;; we should not bounce here cause receiver can have uninitialized contract
  32. .store_slice(owner_address) ;; 当前 Token 收款人地址
  33. .store_coins(forward_ton_amount) ;; 收款人加上这部分附属的 Ton。付款人减去
  34. .store_uint(1, 1 + 4 + 4 + 64 + 32 + 1 + 1)
  35. .store_ref(msg_body);
  36. send_raw_message(msg.end_cell(), 1);
  37. }
  38. if ((response_address.preload_uint(2) != 0) & (msg_value > 0)) {
  39. var msg = begin_cell()
  40. .store_uint(0x10, 6) ;; nobounce - int_msg_info$0 ihr_disabled:Bool bounce:Bool bounced:Bool src:MsgAddress -> 010000
  41. .store_slice(response_address)
  42. .store_coins(msg_value) ;; 超过的 Ton 手续费退款到这个地址
  43. .store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1)
  44. .store_uint(op::excesses(), 32)
  45. .store_uint(query_id, 64);
  46. send_raw_message(msg.end_cell(), 2);
  47. }
  48. ;; 下面为收款人加上 Token
  49. save_data(balance, owner_address, jetton_master_address, jetton_wallet_code);
  50. }
  • 走完 receive_tokens 之后,整个转账行为就在链上闭环了。
注意:上面的源码实现部分比较绕,务必结合注释阅读。

原文链接:https://www.cnblogs.com/linguanh/p/18243703

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号