DNS是网络里面很常用的服务,有一整套的设计 策略和方法,算是很成熟的技术了。作者本人最近刚好用到了DNS,便顺便整理的了笔记,希望对大家有些帮助。本文笔者主要想通过dig工具来讲解下DNS是怎么一回事。
DNS:(Domain Name System的缩写),作用就是根据域名查找到对应的IP地址 ,就像是一个存储了很多条电话记录的电话本一样。详细参考维基百科:https://zh.wikipedia.org/wiki/%E5%9F%9F%E5%90%8D%E7%B3%BB%E7%BB%9F
域名:是服务器的别名,例如:www.baidu.com,互联网中每个地址都需要有一个IP,才能通讯,但是IP地址太难记住了,于是便有了域名作为该服务器的别名。
域名的引入除了能让人更好的记住之外,还有另外一个好处,那就是一个域名可以对应1个或者多个IP地址,用户在访问一个网站的时候,只需要输入它的域名就好了,但是实际上到底访问的是那台服务器,却是不一定的。
Dig:是一个在类Unix命令行模式下查询DNS包括NS记录,A记录,MX记录等相关信息的工具。
我们先看一个dig的例子。
$dig baidu.com
1.显示Dig的版本号以及查询参数。
2.显示查询的结果,status表示的是查询的结果,NOERROR表示查询成功。 3.QuestionSection表示的是查询的域名信息,上图显示的是baidu.com。 4.Answer Section 表示的是查询到的结果,其中baidu.com对应两个A记录,分别是220.181.38.148和39.156.69.79。
5.表示的是查询baidu.com,到返回查询结果,花费的时间,查询的服务器,查询的时间。
紧接着,我们来看下整个dig的详细步骤,因为如此以来,我们便可以看到DNS的详细解析步骤了。
$ dig +trace baidu.com
从上面dig的显示可以看到,www.baidu.com是如何一步步被DNS服务器解析的,下面我们来分步骤介绍下:
1.根域名. ,需要到DNS的根域名服务器进行查询,全球总共有13个根域名服务器,它们分别从a.root-servers.net.到m.root-servers.net. 。它们的IP地址可以参看:https://www.iana.org/domains/root/servers
2.顶级域名TLD(top-level domain).com. ,根域名服务器会告诉对应的顶级域名服务器是那一些。查找顶级域名.com.则需要到DNS的顶级域名服务器来查询,这里显示的顶级域名服务器是a.gtld-servers.net.。
3.次级域名SLD(second-level domain)baidu.com.,顶级域名服务器会告知对应的次级域名服务器,baidu.com对应的次级域名服务器是ns1.baidu.com.,看起来是百度自己的DNS服务器。
4.三级域名www.baidu.com.,通过次级域名服务器的查询,我们知道了三级域名对应的CName(全称:Canonical Name 别名解析的意思)为www.a.shifen.com.,到了这里我们就知道了www.baidu.com最终对应的记录是CName,值是www.a.shifen.com.。
5.别名解析a.shifen.com.对应的NS服务器是ns5.a.shifen.com.,我们可以继续去查看该CName的解析过程,这个与上面的baidu.com是类似的。
从上面的例子中,我们可以看到域名的层次关系,笔者刚开始看的时候只是知道个大概,想着大部分人应该也是,于是便顺便整理下这部分基础知识,让大家熟悉下。
从上面的例子可以看出来,www.baidu.com会显示为www.baidu.com.,这里多了一个.,而这一个.表示的就是根域名,并且所有的域名后面实际上都有这么一个.root,一般用.来代替。
根域名的下一级叫做顶级域名TLD,例如这里的.com,详细列表参考:https://www.iana.org/domains/root/db
根域名的下一级叫做次级域名SLD,例如www.baidu.com.中的.baidu,用户可以注册这一季的域名,例如dnspod上面就是支持的这一层的域名记录。
次级域名的下一级叫做三级域名,例如www.baidu.com.中的www就是一个三级域名,这一层域名往往表示的是主机(Host)的域名,等同与Host的IP地址。
Go中的基础库里面也提供了一个简单的pkg, net, 实现例子如下所示:
package main
import (
"fmt"
"net"
)
func main() {
ipRecords, _ := net.LookupIP("baidu.com")
fmt.Println("net.LookupIp baidu.com:")
for _, ip := range ipRecords {
fmt.Println(ip)
}
fmt.Println("net.LookupCname baidu.com:")
cname, _ := net.LookupCNAME("m.baidu.com")
fmt.Println(cname)
nameServer, _ := net.LookupNS("baidu.com")
fmt.Println("net.LookupNS baidu.com:")
for _, ns := range nameServer {
fmt.Println(ns)
}
fmt.Println("net.LookupMX baidu.com:")
mxRecords, _ := net.LookupMX("baidu.com")
for _, mx := range mxRecords {
fmt.Println(mx.Host, mx.Pref)
}
}
Output:
net.LookupIp baidu.com:
220.181.38.148
39.156.69.79
net.LookupCname baidu.com:
wap.n.shifen.com.
net.LookupNS baidu.com:
&{ns4.baidu.com.}
&{ns3.baidu.com.}
&{ns7.baidu.com.}
&{dns.baidu.com.}
&{ns2.baidu.com.}
net.LookupMX baidu.com:
mx.maillb.baidu.com. 10
mx.n.shifen.com. 15
mx50.baidu.com. 20
mx1.baidu.com. 20
jpmx.baidu.com. 20
备注:除此之外,Go有一套使用很广,功能很强的的dns开源项目"github.com/miekg/dns"。