Contents

网速变慢?你可能需要先设置好 DNS

Question
本文最后更新于 June 18, 2024,文中内容可能已过时,请谨慎使用。

有没有觉得 kiosk007.top 最近访问变快了?没错,我的主站最近接入了 腾讯云的 EO(EdegeOne),当然只是14天体验版,体验过后还是会改会直连模式,毕竟是要收费的。不过能使得访问质量得到加速,第一个重要原因就是我们的请求被指向了就近节点,实现了就近访问。其中就依赖一个重要的底层技术 - DNS


DNS:网络世界的指南针

DNS,英文全称「Domain Name Server」,中文全称「域名服务器」。这种服务器不像网站使用的服务器为用户提供内容,而是将人类易于理解的「域名」转换为机器易于理解的「IP 地址」。

1
2
3
4
# dig kiosk007.top +short
kiosk007.top.eo.dnse1.com.
101.72.224.113 
119.188.123.202

这块的逻辑大家应该有基础网络知识的应该都了解。这里就不多做介绍了。

具体还可以参考: DNS 原理入门


智能解析

DNS(Domain Name System)智能解析是一种更高级的域名解析技术。它可以根据用户的地理位置、网络运营商、访问时间等多种因素,智能地将域名解析到不同的 IP 地址。传统的 DNS 解析通常是将域名固定地解析到一个或几个 IP 地址,而智能解析能够动态地调整解析结果,以优化用户的访问体验。


举个例子,本站 kiosk007.top 接入了 腾讯云 EO。

boce.aliyun.com 上可以看到 kiosk007.top 这个域名从不同地方解析到的结果是不一样的。

  • 在贵阳移动解析到了 221.178.3.6836.147.64.154 ,从 ipinfo.io 可以查看到两个IP都是重庆移动的IP,在地理位置上都属于 西南省份。
  • 在 甘肃移动解析出的 2个IP 分别属于 宁夏移动和甘肃移动的,在地理位置上都属于 西北省份。

用 edns 可以看到使用不同的客户端IP访问,效果是不一样的

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
# dig kiosk007.top @223.5.5.5 +subnet=114.249.170.11 +short
kiosk007.top.eo.dnse1.com.
101.72.224.113
119.188.123.202

# dig kiosk007.top @223.5.5.5 +subnet=111.229.108.83 +short
kiosk007.top.eo.dnse1.com.
116.207.185.121
61.160.209.235



能做到这点,其实是因为 EO 动态加速将域名做到了就近接入。

kiosk007.top 被 CNAME 到了 kiosk007.top.eo.dnse1.com. , 最后再解析出真正的 IP 。

其实 CNAME 是大多数厂家用做调度的核心技术。这里以一个网上找到一个域名的调度说起,从 boce.aliyun.com 可以得出,这个域名是有接入动态加速技术的。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
$ dig mon.snssdk.com

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> mon.snssdk.com
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 40247
;; flags: qr rd ra; QUERY: 1, ANSWER: 8, AUTHORITY: 0, ADDITIONAL: 1

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 65494
;; QUESTION SECTION:
;mon.snssdk.com.			IN	A

;; ANSWER SECTION:
mon.snssdk.com.		227	IN	CNAME	mon.snssdk.com.bytedns1.com.
mon.snssdk.com.bytedns1.com. 33	IN	CNAME	mon.snssdk.com.c.dsa.cdnbuild.net.
mon.snssdk.com.c.dsa.cdnbuild.net. 106 IN CNAME	l7-online-self-max-v6.s.dsa.cdnbuild.net.
l7-online-self-max-v6.s.dsa.cdnbuild.net. 17 IN	A 111.161.204.165
l7-online-self-max-v6.s.dsa.cdnbuild.net. 17 IN	A 111.161.204.160
l7-online-self-max-v6.s.dsa.cdnbuild.net. 17 IN	A 111.161.204.161
l7-online-self-max-v6.s.dsa.cdnbuild.net. 17 IN	A 111.161.204.162
l7-online-self-max-v6.s.dsa.cdnbuild.net. 17 IN	A 111.161.204.164

;; Query time: 10 msec
;; SERVER: 127.0.0.53#53(127.0.0.53) (UDP)
;; WHEN: Wed Nov 27 09:50:27 CST 2024
;; MSG SIZE  rcvd: 246

这里有 三层 CNAME ,分别是

mon.snssdk.com.bytedns1.com. – 接入调度

mon.snssdk.com.c.dsa.cdnbuild.net. – 内部调度管理

l7-online-self-max-v6.s.dsa.cdnbuild.net. – 调度域


前2层一般是调度的内部管理CNAME,最后一层是 调度域名。动态加速厂商管理的一般是调度域。很多域名可以统一 CNAME 到调度域名。


回到 kiosk007.top ,这里只有一层 CNAME ,我怀疑是 EO 将 调度域隐藏了,这可以通过 CNAME Flattening 来实现, 后面会再说。

1
2
3
4
5
$ dig kiosk007.top +short  
kiosk007.top.eo.dnse1.com.
119.188.123.202
101.72.224.113

下面就借此机会说一下 DNS 都有哪些特性,如果希望网络优化从 DNS 入手,都还有哪些手段。


胶水记录(Glue Record)

胶水记录是一种特殊的记录类型,来对比下它和普通的 DNS 记录有什么差别

  • 普通的 DNS 记录是保存在权威服务器上的
    • 权威服务器一般由域名注册商提供,当然也有由第三方提供的独立解析服务。
    • 用户通过域名解析后台设置解析记录,值是保存在权威服务器上的。
  • 胶水记录保存在注册局的DNS服务器上
    • 注册局(Domain Registry)是管理某个后缀所有域名的机构。
    • 用户通过域名注册商后台设置胶水记录时,值是保存到注册局的服务器上的。

详细介绍之前先回顾一下 DNS 查询过程。

我们查询 DNS 的时候一般是 递归查询 PublicDNS 或者 LocalDNS 。它们会直接输出 DNS解析结果 ,而 LocalDNS 或者 PublicDNS 会做迭代查询,层层迭代直到查到结果,当然有缓存的话直接返回结果。

https://img1.kiosk007.top/static/images/blog/20241201123531-ABUIABAEGAAg94nTjAYo-Ya5kwQwtwU4nQM!600x600.png

ABUIABAEGAAg94nTjAYo-Ya5kwQwtwU4nQM!600x600

LocalDNS 一般由 DHCP 分发的 DNS Server

PublicDNS 是一些公共大厂的DNS Server,常见的有 119.29.29.29 (腾讯云的PublicDNS) 、223.5.5.5 (阿里云的PublicDNS)、180.184.1.1 (火山引擎的 PublicDNS)

以查询 www.qq.com 为例。我们直接查询可以看到有1层 CNAME ,3个IP。

1
2
3
# dig www.qq.com +short
ins-r23tsuuf.ias.tencent-cloud.net.
221.198.70.47

其实我们会以 递归服务器的工作方式来逐级查询 DNS 记录。

一、获取根服务器列表

因为所有域名的 DNS 记录查询入口都在这里,就好像你要进入一个多级的目录,你得一级一级地进入,而根服务器就是入口。

全球共有 13 组公认的 DNS 服务器,可在 IANA 网站上查询到:

https://www.iana.org/domains/root/servers


二、向根服务器发送查询

下面的过程是这样的:

Public DNS:“www.qq.com 的地址你有吗?”

root权威DNS:“没有,我只有 .com 顶级域的 权威地址, 你可以找他们问, .com 顶级域权威DNS是 d.gtld-servers.net. 等 , 对了他们的地址我也有,就是 192.5.6.30 这些”

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
# dig qq.com @m.root-servers.net

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> qq.com @m.root-servers.net
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 50679
;; flags: qr rd; QUERY: 1, ANSWER: 0, AUTHORITY: 13, ADDITIONAL: 27
;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 1232
;; QUESTION SECTION:
;qq.com.				IN	A

;; AUTHORITY SECTION:
com.			172800	IN	NS	h.gtld-servers.net.
com.			172800	IN	NS	l.gtld-servers.net.
com.			172800	IN	NS	j.gtld-servers.net.
com.			172800	IN	NS	f.gtld-servers.net.
com.			172800	IN	NS	b.gtld-servers.net.
com.			172800	IN	NS	e.gtld-servers.net.
com.			172800	IN	NS	m.gtld-servers.net.
com.			172800	IN	NS	d.gtld-servers.net.
com.			172800	IN	NS	k.gtld-servers.net.
com.			172800	IN	NS	a.gtld-servers.net.
com.			172800	IN	NS	c.gtld-servers.net.
com.			172800	IN	NS	i.gtld-servers.net.
com.			172800	IN	NS	g.gtld-servers.net.

;; ADDITIONAL SECTION:
m.gtld-servers.net.	172800	IN	A	192.55.83.30
l.gtld-servers.net.	172800	IN	A	192.41.162.30
k.gtld-servers.net.	172800	IN	A	192.52.178.30
j.gtld-servers.net.	172800	IN	A	192.48.79.30
i.gtld-servers.net.	172800	IN	A	192.43.172.30
h.gtld-servers.net.	172800	IN	A	192.54.112.30
g.gtld-servers.net.	172800	IN	A	192.42.93.30
f.gtld-servers.net.	172800	IN	A	192.35.51.30
e.gtld-servers.net.	172800	IN	A	192.12.94.30
d.gtld-servers.net.	172800	IN	A	192.31.80.30
c.gtld-servers.net.	172800	IN	A	192.26.92.30
b.gtld-servers.net.	172800	IN	A	192.33.14.30
a.gtld-servers.net.	172800	IN	A	192.5.6.30
m.gtld-servers.net.	172800	IN	AAAA	2001:501:b1f9::30
l.gtld-servers.net.	172800	IN	AAAA	2001:500:d937::30
k.gtld-servers.net.	172800	IN	AAAA	2001:503:d2d::30
j.gtld-servers.net.	172800	IN	AAAA	2001:502:7094::30
i.gtld-servers.net.	172800	IN	AAAA	2001:503:39c1::30
h.gtld-servers.net.	172800	IN	AAAA	2001:502:8cc::30
g.gtld-servers.net.	172800	IN	AAAA	2001:503:eea3::30
f.gtld-servers.net.	172800	IN	AAAA	2001:503:d414::30
e.gtld-servers.net.	172800	IN	AAAA	2001:502:1ca1::30
d.gtld-servers.net.	172800	IN	AAAA	2001:500:856e::30
c.gtld-servers.net.	172800	IN	AAAA	2001:503:83eb::30
b.gtld-servers.net.	172800	IN	AAAA	2001:503:231d::2:30
a.gtld-servers.net.	172800	IN	AAAA	2001:503:a83e::2:30

;; Query time: 74 msec
;; SERVER: 202.12.27.33#53(m.root-servers.net) (UDP)
;; WHEN: Sun Dec 01 12:23:50 CST 2024
;; MSG SIZE  rcvd: 831

三、向 .com 权威服务器发送查询

这里选了一个 .com 的权威服务器查询。下面的对话过程如下:

PublicDNS :“你有 www.qq.com 的记录吗?”

.com 权威:“没有,但我知道 qq.com 的权威DNS是 ns1.qq.com. 他应该有 www.qq.com 的记录,对了 ns1.qq.com 权威地址的 结尾是 .com ,他的地址,你就不必问我了, 我这里也有,顺便给你哦, 是 101.227.218.144 等”

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
dig www.qq.com @192.55.83.30

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> www.qq.com @a.gtld-servers.net
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 35510
;; flags: qr rd; QUERY: 1, ANSWER: 0, AUTHORITY: 4, ADDITIONAL: 17
;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 4096
;; QUESTION SECTION:
;www.qq.com.			IN	A

;; AUTHORITY SECTION:
qq.com.			172800	IN	NS	ns1.qq.com.
qq.com.			172800	IN	NS	ns2.qq.com.
qq.com.			172800	IN	NS	ns3.qq.com.
qq.com.			172800	IN	NS	ns4.qq.com.

;; ADDITIONAL SECTION:
ns1.qq.com.		172800	IN	A	101.227.218.144
ns1.qq.com.		172800	IN	A	14.116.136.7
ns1.qq.com.		172800	IN	A	157.255.246.101
ns1.qq.com.		172800	IN	A	203.205.220.251
ns1.qq.com.		172800	IN	AAAA	2402:4e00:8030::111
ns2.qq.com.		172800	IN	A	111.230.158.42
ns2.qq.com.		172800	IN	A	211.100.32.218
ns2.qq.com.		172800	IN	AAAA	240e:9f:c600::8
ns2.qq.com.		172800	IN	A	43.129.131.210
ns3.qq.com.		172800	IN	A	112.60.1.69
ns3.qq.com.		172800	IN	A	117.184.232.216
ns3.qq.com.		172800	IN	A	49.51.76.77
ns4.qq.com.		172800	IN	A	170.106.32.66
ns4.qq.com.		172800	IN	A	218.68.91.143
ns4.qq.com.		172800	IN	A	58.144.154.100
ns4.qq.com.		172800	IN	A	59.36.132.142

;; Query time: 333 msec
;; SERVER: 192.5.6.30#53(a.gtld-servers.net) (UDP)
;; WHEN: Sun Dec 01 12:25:40 CST 2024
;; MSG SIZE  rcvd: 391


.com 权威服务器告诉PublicDNS,qq.comns*.qq.com 这组服务器管理,并且还顺便告诉了他们的 IP 地址。

第四步:向 qq.com 域名权威服务器发送查询

Public 选择了 ns1.qq.com. 这个权威的地址,并使用 101.227.218.144 这个IP访问.

下面是和 qq.com 的权威服务器 ns1.qq.com 进行查询的对话:

PublicDNS: “你这里有 www.qq.com 的地址吗?”

qq.com 的权威:“没有,但我知道 www.qq.com 的权威是 ns-cnc1.qq.com. 对了 ns-cnc1.qq.com 的地址, 我也有,拿去就是 157.255.6.102 "

题外话:为什么这里由出现一层权威??? 原因是 www.qq.com 不是直接有 A 记录的,他有一层 CNAME ,所以这里由有一个 NS 。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
dig www.qq.com @101.227.218.144 

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> www.qq.com @101.227.218.144
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 13872
;; flags: qr rd ad; QUERY: 1, ANSWER: 0, AUTHORITY: 2, ADDITIONAL: 7
;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 4304
; COOKIE: fe39971c695d1887 (echoed)
;; QUESTION SECTION:
;www.qq.com.			IN	A

;; AUTHORITY SECTION:
www.qq.com.		86400	IN	NS	ns-cnc1.qq.com.
www.qq.com.		86400	IN	NS	ns-cnc2.qq.com.

;; ADDITIONAL SECTION:
ns-cnc1.qq.com.		3600	IN	A	157.255.6.102
ns-cnc1.qq.com.		3600	IN	A	218.68.91.139
ns-cnc1.qq.com.		3600	IN	A	140.207.180.96
ns-cnc1.qq.com.		3600	IN	AAAA	240e:e1:aa00:2002::3
ns-cnc2.qq.com.		3600	IN	A	157.255.6.102
ns-cnc2.qq.com.		3600	IN	A	218.68.91.139

;; Query time: 35 msec
;; SERVER: 101.227.218.144#53(101.227.218.144) (UDP)
;; WHEN: Sun Dec 01 12:47:46 CST 2024
;; MSG SIZE  rcvd: 203

余下步骤:

看来解析出一个域名真的是太麻烦了,下面的流程不再详细赘述了,我通过解析操作展示

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
1. 问 ns-cnc1.qq.com 询问结果

# dig www.qq.com @157.255.6.102       

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> www.qq.com @157.255.6.102
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 54329
;; flags: qr aa rd ad; QUERY: 1, ANSWER: 1, AUTHORITY: 0, ADDITIONAL: 1
;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 4304
; COOKIE: cbd86af4a7b28ff0 (echoed)
;; QUESTION SECTION:
;www.qq.com.			IN	A

;; ANSWER SECTION:
www.qq.com.		300	IN	CNAME	ins-r23tsuuf.ias.tencent-cloud.net.

;; Query time: 43 msec
;; SERVER: 157.255.6.102#53(157.255.6.102) (UDP)
;; WHEN: Sun Dec 01 13:01:34 CST 2024
;; MSG SIZE  rcvd: 99

查询 ins-r23tsuuf.ias.tencent-cloud.net. 又要经历上述一层逻辑,这里我们直接给出结果吧。

1
2
dig ins-r23tsuuf.ias.tencent-cloud.net +short
221.198.70.47

CNAME 记录打平 (CNAME Flattening)

CNAME 记录打平最早 CloudFlare 有一篇文章记录 Introducing CNAME Flattening: RFC-Compliant CNAMEs at a Domain’s Root


这篇文章介绍了Cloudflare推出的一项新功能——CNAME Flattening。以下是文章的主要内容总结:

  1. 背景和问题

    • 传统上,域名的根记录(即没有子域名的顶级域名,如example.com)需要指向一个IP地址(使用A记录)。这限制了使用基于云的服务,因为将服务绑定到IP地址可能会导致资源共享和性能问题。
    • CNAME记录允许一个域名指向另一个域名(最终会解析为A记录和IP地址),这为子域名(如www.example.com)提供了灵活性,但在根记录中使用CNAME存在问题,因为DNS规范不允许根记录使用CNAME。
  2. CNAME的优势

    • CNAME作为别名,提供了灵活性,允许服务在不改变DNS设置的情况下动态调整IP地址。
    • 通过使用CNAME,可以避免因资源过载而导致的性能问题,并且可以轻松地将流量从一个IP地址转移到另一个IP地址。
  3. DNS规范和问题

    • DNS规范规定根记录不能是CNAME,因为根记录通常需要与其他记录(如MX记录用于邮件,NS记录用于域名服务器)关联。

    • 遵循这一规范的大多数权威DNS服务器不允许在根记录中包含CNAME记录。

  4. CNAME Flattening的解决方案

    • Cloudflare通过扩展其权威DNS基础设施,实现了一种在遵循RFC的同时支持根记录使用CNAME的方法。当根记录中有CNAME时,Cloudflare会递归地解析CNAME链,直到找到A记录,然后返回与A记录关联的IP地址,这个过程称为“CNAME Flattening”。

    • 这种方法遵循DNS规范,对任何与Cloudflare DNS交互的服务都是透明的,并且解决了之前遇到的边缘情况问题,如Microsoft Exchange邮件服务器的问题。

  5. 好处

    • CNAME Flattening允许在不违反DNS规范的情况下在根记录中使用CNAME,提供了灵活性。

    • 通过缓存CNAME响应,Cloudflare减少了CNAME解析时间,平均减少了约30%。

    • Cloudflare已经开始在更广泛的范围内测试CNAME Flattening,包括内部使用的CNAME。


当然… dnspod 是做了特殊支持的,支持 MX 记录和 CNAME 记录共存。

ANAME

和 CNAME Flattening 相似。

假设本站的域名 img1.kiosk007.top . 有 2层CNAME ,而且通过CNAME 一眼就看出用的是 七牛云。

1
2
3
4
5
6
7
# dig img1.kiosk007.top +short
img1-kiosk007-top-idvf0pq.qiniudns.com.
allcdn.china.qiniu.qnydns.com.
allcdn.lv2.qnydns.com.
111.45.28.78
112.16.229.53

能不能隐藏 我用了 七牛云 这个事实呢?也就是说需要消除掉 中间的CNAME 从而直接给出 A 记录,从原理上是可以实现的,这就需要引入 ANAME 了,

这篇文章分享了什么是 ANAME



  1. ANAME记录的定义
    • ANAME记录是一种能够在域名根上使用的记录,类似于CNAME,但没有CNAME的限制。它可以被视为一个转变为CNAME的A记录。
  2. ANAME记录的用途
    • ANAME记录允许你将裸域名(无子域名的顶级域名)指向一个主机名或完全限定域名(FQDN)。
    • 它们适用于需要频繁更新和/或最精确的GeoDNS解析的关键系统,尤其是对于内容分发网络(CDN)来说非常有用。
  3. ANAME记录的工作原理
    • 当查询ANAME记录时,权威名称服务器实际上会解析存储在ANAME记录中的主机或FQDN。
    • 如果ANAME记录无法解析,服务器将返回最近缓存的IP地址,确保用户仍然可以接收内容,避免不必要的停机时间。
  4. ANAME记录的历史
    • DNS Made Easy的创始人之一是ANAME技术的创新者之一。在ANAME存在之前,他们使用API调用来实现相同的结果,后来开发了自己的记录,并将其命名为“ANAME记录”。
  5. ANAME记录示例
    • 文章提供了一个ANAME记录的示例,展示了如何将dnsmadeeasy.com的ANAME记录指向一个FQDN(例如127.1.1.1)。
    • 与A记录和CNAME记录相比,ANAME记录结合了两者的功能,并消除了CNAME记录的限制。
  6. ANAME记录的优势
    • ANAME记录可以以多种方式集成到组织的DNS策略中,允许你绕过某些记录限制,定制DNS配置以适应你的域名。

DNS 解析优化调优

全球加速DNS解析优化

众所周知的原因,海外用户访问国内的网络质量是非常差的,所以如果一个域名希望是全球访问时,那么这个域名的权威NS 应该设置在哪里呢?

如果设置在国内,那么海外访问会比较慢,如果设置在海外,那么国内访问会比较慢。

对于一个云厂商,想要加速DNS阶段的解析速度,其中重要一点就是需要做的就是优化其NS访问速度。

聪明的做法应该是给权威DNS做一次智能解析。国内用国内的权威DNS地址,海外用海外的权威DNS地址。


知识补充:

我们知道想要使用一家云厂商的 加速域名,需要将我们自己的域名 CNAME 过去,

阿里云的常见 CNAME 的二级域名常见的有:

  • queniurc.com.
  • queniuaa.com.

七牛云的二级域:

  • qiniudns.com.

不错,阿里云的CNAME 的权威就是这么做的~

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
# dig xxxxx.queniuaa.com @m.gtld-servers.net.

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> xxxxx.queniuaa.com @m.gtld-servers.net.
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 32001
;; flags: qr rd; QUERY: 1, ANSWER: 0, AUTHORITY: 2, ADDITIONAL: 1
;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 4096
;; QUESTION SECTION:
;xxxxx.queniuaa.com.		IN	A

;; AUTHORITY SECTION:
queniuaa.com.		172800	IN	NS	vipns6.queniudns.net.
queniuaa.com.		172800	IN	NS	vipns7.queniudns.net.

;; Query time: 251 msec
;; SERVER: 192.55.83.30#53(m.gtld-servers.net.) (UDP)
;; WHEN: Sat Dec 07 15:08:50 CST 2024
;; MSG SIZE  rcvd: 102

我们看下 阿里云的 DNS 智能解析效果。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
# dig vipns6.queniudns.net. +subnet=114.114.11.1 +short   # (114.114.11.1 是一个国内IP)
47.240.195.220      (这些结果解析出的都是 国内的IP)
121.40.6.145      
47.108.110.113
203.107.13.55
47.88.250.116
47.254.165.60
47.120.230.155

# dig vipns6.queniudns.net. +subnet=1.1.1.1 +short    # (1.1.1.1 是一个海外IP)
47.89.91.91			(解析的结果是美国 Aliyun)
205.204.111.111

但实际效果不然,如果我们自己建权威的话会发现,智能解析没有效果?国内的权威还是会收到大量的海外解析请求。为什么?智能解析失效了吗?不是的,这是因为胶水记录。


这里再举个阿里云动态加速的反例

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
# dig cdngslb.com.  @i.gtld-servers.net.

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> cdngslb.com. @i.gtld-servers.net.
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 43709
;; flags: qr rd; QUERY: 1, ANSWER: 0, AUTHORITY: 2, ADDITIONAL: 15
;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 4096
;; QUESTION SECTION:
;cdngslb.com.			IN	A

;; AUTHORITY SECTION:
cdngslb.com.		172800	IN	NS	ns1.cdngslb.com.
cdngslb.com.		172800	IN	NS	ns2.cdngslb.com.

;; ADDITIONAL SECTION:
ns1.cdngslb.com.	172800	IN	A	106.11.35.19
ns1.cdngslb.com.	172800	IN	A	106.11.41.157
ns1.cdngslb.com.	172800	IN	A	140.205.103.192
ns1.cdngslb.com.	172800	IN	A	140.205.122.66
ns1.cdngslb.com.	172800	IN	AAAA	2401:b180:4100::14
ns1.cdngslb.com.	172800	IN	A	47.241.207.18
ns1.cdngslb.com.	172800	IN	A	47.88.74.38
ns2.cdngslb.com.	172800	IN	A	106.11.35.18
ns2.cdngslb.com.	172800	IN	A	106.11.41.158
ns2.cdngslb.com.	172800	IN	A	140.205.103.194
ns2.cdngslb.com.	172800	IN	A	140.205.122.77
ns2.cdngslb.com.	172800	IN	AAAA	2401:b180:4100::15
ns2.cdngslb.com.	172800	IN	A	47.241.207.16
ns2.cdngslb.com.	172800	IN	A	47.88.74.36

;; Query time: 201 msec
;; SERVER: 192.43.172.30#53(i.gtld-servers.net.) (UDP)
;; WHEN: Sat Dec 07 16:22:44 CST 2024
;; MSG SIZE  rcvd: 324

看到了吗?胶水记录将 Aliyun 全球加速CNAME 权威的所有A记录全部返回了,也就是虽然 cdngslb.com 是全球加速的,且做了智能解析,国内的用国内的权威,海外的用海外的,但是!!!! .com 权威服务器将 cdngslb.com 的全部NS服务器的A记录记录都返回了。这会导致 智能解析失效…


所以 !!! 胶水记录用不好的话在全球加速这种场景下就是个副作用。

那如何避免胶水记录呢?????

就像 case 1 中的一样,将 CNAME 和 CNAME 权威的根域设置不一样就可以了。

比如 CNAME 是 .com 结尾的,CNAME 的权威是 .net 结尾的。这样就可以防止出现胶水记录了,在全球加速的场景下是有优势的。


多层CNAME 解析加速

假设你的域名 CNAME 套了好多层,那么就意味着需要很多层的递归才能解析出最终的 A 记录。

CNAME 是 为了满足 复杂的调度和管理需求的,但是其实可以做到在 多层 CNAME 的基础之上,保证解析的效率。

比如 DNS Pod 推出的 CNAME 加速。

文章连接: https://docs.dnspod.com/dns/cname-flattening-zh/


假设 a.comb.comc.com 都是在 DNSPod 解析的域名:

域名记录类型记录值
www.a.comCNAMEwww.b.com
www.b.comCNAMEwww.c.com
www.c.comA1.2.3.4

一般情况下,递归需要到授权服务器请求三次才能得到 www.a.com 的 IP 地址,如下图所示:


启用 CNAME 加速功能,授权服务器会把 CNAME 记录和最终的 A 记录一次返回给递归,递归服务器由请求三次授权服务器,减小到请求一次,如下图所示:

这样就极大地减少了请求和应答中网络通信消耗的时间,让解析变得更快,特别是在设置多条 CNAME 解析记录的情况下,加速效果更明显。


CoreDNS 验证

CoreDNS是一个灵活、可扩展的开源DNS服务器,它支持多种协议和后端存储,使其成为构建现代、可定制的DNS基础设施的理想选择。下面将介绍如何安装和配置CoreDNS,以便你可以快速搭建一个强大而稳定的DNS服务。

CroeDNS 部署


  1. 下载 CoreDNS 的二进制文件,放置到 bin 文件夹下。
1
2
3
4
5
wget https://github.com/coredns/coredns/releases/download/v1.12.0/coredns_1.12.0_linux_amd64.tgz

tar -xvf coredns_1.12.0_linux_amd64.tgz -C .

sudo cp coredns /usr/local/bin/coredns

  1. 创建system文件,使用 systemd 管理
1
vi /etc/systemd/system/coredns.service

内容如下:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
[Unit]
Description=CoreDNS DNS server
Documentation=https://coredns.io
After=network.target

[Service]
PermissionsStartOnly=true
LimitNOFILE=1048576
LimitNPROC=512
CapabilityBoundingSet=CAP_NET_BIND_SERVICE
AmbientCapabilities=CAP_NET_BIND_SERVICE
NoNewPrivileges=true
User=root
WorkingDirectory=~
ExecStart=/usr/local/bin/coredns -conf=/etc/coredns/Corefile
ExecReload=/bin/kill -SIGUSR1 
Restart=on-failure

[Install]
WantedBy=multi-user.target

保存完成后重载 systemd 配置:systemctl daemon-reload


  1. 创建配置文件

首先创建文件夹用于存放配置文件

1
mkdir /etc/coredns

新建配置文件

1
vim /etc/coredns/Corefile

这是一个简单的配置:

.:53 {
	bind 127.0.0.1 ::1
        errors
        health {
            lameduck 5s
        }
        ready
	forward . 223.5.5.5 129.29.29.29
        cache 30
        reload
        loadbalance
    }   

  1. 确保 53 端口未被占用
lsof -i :53

systemd-resolved 占用53端口(Ubuntu Server常见) 首先修改 /etc/systemd/resolved.conf,内容如下:

[Resolve]
DNS=114.114.114.114
DNSStubListener=no

DNS 是配置本机指向的DNS服务器,建议配置运营商或者公共DNS。 DNSStubListener=no 是指不监听53端口。 配置完成后执行命令重启系统服务: sudo systemctl daemon-reload sudo systemctl restart systemd-resolved 查看/etc/resolv.conf 软链位置,执行命令 ll /etc/resolv.conf,如果软链显示如下,则不需要改动: /etc/resolv.conf -> /run/systemd/resolve/resolv.conf,否则设置软链 sudo ln -s -f /run/systemd/resolve/resolv.conf /etc/resolv.conf


  1. 设置开机自启动并启动 CoreDNS
systemctl enable coredns
systemctl start coredns

配置 example.io 域名解析

example.io 这个域名并不存在,我们在本地给他一个解析结果。

使用 file 插件自定义域名的解析. file 插件

.:53 {
	bind 127.0.0.1 ::1
        errors
        health {
            lameduck 5s
        }
        ready
        cache 30
        reload
        loadbalance
	forward . 223.5.5.5 119.29.29.29
}

example.io:53 {
	bind 127.0.0.1
	file /etc/coredns/example.io.db example.io
}

创建 /etc/coredns/example.io.db 文件。内容如下

$TTL 3600 ; 记录超时时间
$ORIGIN example.io. ; 指定 origin,下面的@符号可以作为他的别名,注意后面的.
; SOA 格式 [domain_name] IN SOA [域主服务器或主DNS服务器名] [管理员email] (时间信息)
@           IN  SOA  ns1.example.io. admin.example.io. (
                     2019071601 ; Serial
                     4H         ; Refresh
                     1H         ; Retry
                     7D         ; Expire
                     4H )       ; Negative Cache TTL

; 配置 DNS 记录,指向 ns1.example.io
@           IN  NS   ns1
; 配置 ns1.example.io 的 A 记录, 指向coredns所在的机器
ns1         IN  A    127.0.0.1
; 配置 example.io 的 A 记录,指向网站或其他用途的机器
@           IN  A    1.1.1.1
; 配置泛域名,没有准确的三级子域名的域名全部指向此IPV4地址
*   	    IN  A    5.5.5.5
cdn 	    IN  CNAME cname1
cname1      IN  CNAME cname2
cname2      IN  A   1.2.3.4

尝试解析一下,看看效果

1
2
3
4
# dig cdn.example.io  @127.0.0.1 +short
cname1.example.io.
cname2.example.io.
1.2.3.4

尝试实现 CNAME Flattening

这里使用 CoreDNS 的 Flatten插件

这里需要自己手动重新编译一个 coredns ,coredns 是golang 实现的,而本人的 golang 环境是现成的,就不介绍 golang 环境的搭建了。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
git clone https://github.com/coredns/coredns.git

# 在 plugin.cfg 文件中添加
vim plugin.cfg

...   # 添加一条
flatten:github.com/litobro/flatten


# 开始编译
go generate
go build

sudo cp coredns /usr/local/bin

对配置文件做如下的修改

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
.:53 {
        bind 127.0.0.1 ::1
        errors
        health {
            lameduck 5s
        }
        ready
        cache 30
        reload
        loadbalance
        forward . 223.5.5.5 119.29.29.29
}

kiosk.io:53 {
        bind 127.0.0.1
        log
        flatten kiosk.io cdn.example.io 127.0.0.1:53
}

example.io:53 {
        bind 127.0.0.1
        log
        file /etc/coredns/example.io.db example.io
}

启动 coreDNS:

1
2
3
4
5
6
7
8
9
# /usr/local/bin/coredns -conf=/etc/coredns/Corefile 
.:53 on 127.0.0.1
example.io.:53 on 127.0.0.1
kiosk.io.:53 on 127.0.0.1
.:53 on ::1
[INFO] plugin/reload: Running configuration SHA512 = be58f721ab98e4affc5464a859492153ee35d2aaa3b558eb2c623258f9bbc3d7f3ba998af7b1973673179cb625550c0aa065111fbcab4f7aa49896cc8d1fc5d0
CoreDNS-1.11.3
linux/amd64, go1.23.0, 

验证效果:

kiosk.io 这个域名被 转移映射到 example.io 上

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
# dig kiosk.io @127.0.0.1 

; <<>> DiG 9.18.28-0ubuntu0.24.04.1-Ubuntu <<>> kiosk.io @127.0.0.1
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 38076
;; flags: qr aa rd; QUERY: 1, ANSWER: 3, AUTHORITY: 0, ADDITIONAL: 1
;; WARNING: recursion requested but not available

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 1232
; COOKIE: a4084560500f58f9 (echoed)
;; QUESTION SECTION:
;kiosk.io.			IN	A

;; ANSWER SECTION:
kiosk.io.		3600	IN	CNAME	cname1.example.io.
kiosk.io.		3600	IN	CNAME	cname2.example.io.
kiosk.io.		3600	IN	A	1.2.3.4

;; Query time: 0 msec
;; SERVER: 127.0.0.1#53(127.0.0.1) (UDP)
;; WHEN: Sat Dec 07 19:51:05 CST 2024
;; MSG SIZE  rcvd: 151