TON DNS 解析器
介绍
TON DNS 是一个强大的工具。它不仅允许将 TON 网站/存储包分配给域名,还可以设置子域名解析。
相关链接
域名合约搜索器
子域名具有实际用途。例如,区块链浏览器目前没有提供通过名称查找域名合约的方法。让我们探索如何创建一个合约,提供查找这类域名的机会。
This contract is deployed at EQDkAbAZNb4uk-6pzTPDO2s0tXZweN-2R08T2Wy6Z3qzH_Zp and linked to resolve-contract.ton
. To test it, you may write <your-domain.ton>.resolve-contract.ton
in the address bar of your favourite TON explorer and get to the page of TON DNS domain contract. Subdomains and .t.me domains are supported as well.
您可以尝试通过访问 resolve-contract.ton.resolve-contract.ton
来查看解析器代码。不幸的是,这将不会显示子解析器(那是不同的智能合约),您将看到域名合约本身的页面。
dnsresolve() 代码
部分重复部分已省略。
(int, cell) dnsresolve(slice subdomain, int category) method_id {
int subdomain_bits = slice_bits(subdomain);
throw_unless(70, (subdomain_bits % 8) == 0);
int starts_with_zero_byte = subdomain.preload_int(8) == 0; ;; assuming that 'subdomain' is not empty
if (starts_with_zero_byte) {
subdomain~load_uint(8);
if (subdomain.slice_bits() == 0) { ;; current contract has no DNS records by itself
return (8, null());
}
}
;; we are loading some subdomain
;; supported subdomains are "ton\0", "me\0t\0" and "address\0"
slice subdomain_sfx = null();
builder domain_nft_address = null();
if (subdomain.starts_with("746F6E00"s)) {
;; we're resolving
;; "ton" \0 <subdomain> \0 [subdomain_sfx]
subdomain~skip_bits(32);
;; reading domain name
subdomain_sfx = subdomain;
while (subdomain_sfx~load_uint(8)) { }
subdomain~skip_last_bits(8 + slice_bits(subdomain_sfx));
domain_nft_address = get_ton_dns_nft_address_by_index(slice_hash(subdomain));
} elseif (subdomain.starts_with("6164647265737300"s)) {
subdomain~skip_bits(64);
domain_nft_address = subdomain~decode_base64_address_to(begin_cell());
subdomain_sfx = subdomain;
if (~ subdomain_sfx.slice_empty?()) {
throw_unless(71, subdomain_sfx~load_uint(8) == 0);
}
} else {
return (0, null());
}
if (slice_empty?(subdomain_sfx)) {
;; example of domain being resolved:
;; [initial, not accessible in this contract] "ton\0resolve-contract\0ton\0ratelance\0"
;; [what is accessible by this contract] "ton\0ratelance\0"
;; subdomain "ratelance"
;; subdomain_sfx ""
;; we want the resolve result to point at contract of 'ratelance.ton', not its owner
;; so we must answer that resolution is complete + "wallet"H is address of 'ratelance.ton' contract
;; dns_smc_address#9fd3 smc_addr:MsgAddressInt flags:(## 8) { flags <= 1 } cap_list:flags . 0?SmcCapList = DNSRecord;
;; _ (HashmapE 256 ^DNSRecord) = DNS_RecordSet;
cell wallet_record = begin_cell().store_uint(0x9fd3, 16).store_builder(domain_nft_address).store_uint(0, 8).end_cell();
if (category == 0) {
cell dns_dict = new_dict();
dns_dict~udict_set_ref(256, "wallet"H, wallet_record);
return (subdomain_bits, dns_dict);
} elseif (category == "wallet"H) {
return (subdomain_bits, wallet_record);
} else {
return (subdomain_bits, null());
}
} else {
;; subdomain "resolve-contract"
;; subdomain_sfx "ton\0ratelance\0"
;; we want to pass \0 further, so that next resolver has opportunity to process only one byte
;; next resolver is contract of 'resolve-contract<.ton>'
;; dns_next_resolver#ba93 resolver:MsgAddressInt = DNSRecord;
cell resolver_record = begin_cell().store_uint(0xba93, 16).store_builder(domain_nft_address).end_cell();
return (subdomain_bits - slice_bits(subdomain_sfx) - 8, resolver_record);
}
}
dnsresolve() 解释
- 用户请求
"stabletimer.ton.resolve-contract.ton"
。 - 应用程序将其转换为
"\0ton\0resolve-contract\0ton\0stabletimer\0"
(第一个零字节是可选的)。 - 根 DNS 解析器将请求定向到 TON DNS 集合,剩余部分为
"\0resolve-contract\0ton\0stabletimer\0"
。 - TON DNS 集合将请求委托给特定域名,留下
"\0ton\0stabletimer\0"
。 - .TON DNS 域名合约将解析传递给编辑器指定的子解析器,子域名为
"ton\0stabletimer\0"
。
这是 dnsresolve() 被调用的点。 分步解释其工作方式:
- 它将子域名和类别作为输入。
- 如果开头有零字节,则跳过。
- 检查子域名是否以
"ton\0"
开头。如果是,- 跳过前32位(子域名 =
"resolve-contract\0"
) - 设置
subdomain_sfx
的值为subdomain
,并读取直到零字节的字节 - (子域名 =
"resolve-contract\0"
,subdomain_sfx =""
) - 从子域名切片的末尾裁剪零字节和 subdomain_sfx(子域名 =
"resolve-contract"
) - 使用 slice_hash 和 get_ton_dns_nft_address_by_index 函数将域名转换为合约地址。您可以在 [[Subresolvers#Appendix 1. resolve-contract.ton 的代码|附录 1]] 中看到它们。
- 跳过前32位(子域名 =
- 否则,dnsresolve() 检查子域名是否以
"address\0"
开头。如果是,它跳过该前缀并读取 base64 地址。 - 如果提供的用于解析的子域名与这些前缀都不匹配,函数通过返回
(0, null())
(零字节前缀解析无 DNS 条目)表示失败。 - 然后检查子域名后缀是否为空。空后缀表示请求已完全满足。如果后缀为空:
- dnsresolve() 为域名的 "wallet" 子部分创建一个 DNS 记录,使用它检索到的 TON 域名合约地址。
- 如果请求类别 0(所有 DNS 条目),则将记录包装在字典中并返回。
- 如果请求类别为 "wallet"H,则按原样返回记录。
- 否则,指定类别没有 DNS 条目,因此函数表示解析成功但未找到任何结果。
- 如果后缀不为空:
- 之前获得的合约地址用作下一个解析器。函数构建指向它的下一个解析器记录。
"\0ton\0stabletimer\0"
被传递给该合约:处理的位是子域名的 位。
总结来说,dnsresolve() 要么:
- 将子域名完全解析为 DNS 记录
- 部分解析为解析器记录,以将解析传递给另一个合约
- 为未知子域名返回“未找到域名”的结果
实际上,base64 地址解析不起作用:如果您尝试输入 <some-address>.address.resolve-contract.ton
,您将收到一个错误,表明域名配置错误或不存在。原因是域名不区分大小写(从真实 DNS 继承的功能),因此会转换为小写,将您带到不存在的工作链的某个地址。
绑定解析器
现在子解析器合约已部署,我们需要将域名指向它,即更改域名的 dns_next_resolver
记录。我们可以通过将以下 TL-B 结构的消息发送到域名合约来实现。
change_dns_record#4eb1f0f9 query_id:uint64 record_key#19f02441ee588fdb26ee24b2568dd035c3c9206e11ab979be62e55558a1d17ff record:^[dns_next_resolver#ba93 resolver:MsgAddressInt]