本文将详细讲解如何用go语言一步一步实现dns域名解析的过程,并简单介绍点dns有关的知识,直接开始正题吧。html
首先咱们要了解dns解析的过程,没有了解的请看这里DNS入门(转)很详细。扫盲结束后,咱们须要了解下dns报文格式,知道了报文的格式是怎样的,才能够写代码构造dns请求包: 服务器
dns请求和应答都是用相同的报文格式,分红5个段(有的报文段在不一样的状况下可能为空),以下: 数据结构
Header段是报文的头部,它定义了报文是请求仍是应答,也定义了其余段是否须要存在,以及是标准查询仍是其余。 dom
Header包含以下字段:函数
各字段分别解释以下:post
ID:请求客户端设置的16位标示,服务器给出应答的时候会带相同的标示字段回来,这样请求客户端就能够区分不一样的请求应答了。ui
QR:1个比特位用来区分是请求(0)仍是应答(1)。编码
OPCODE:4个比特位用来设置查询的种类,应答的时候会带相同值,可用的值以下: 0 标准查询 (QUERY) 1 反向查询 (IQUERY) 2 服务器状态查询 (STATUS) 3-15保留值,暂时未使用url
AA:受权应答(Authoritative Answer) - 这个比特位在应答的时候才有意义,指出给出应答的服务器是查询域名的受权解析服务器。注意由于别名的存在,应答可能存在多个主域名,这个AA位对应请求名,或者应答中的第一个主域名。spa
TC:截断(TrunCation) - 用来指出报文比容许的长度还要长,致使被截断。
RD:指望递归(Recursion Desired) - 这个比特位被请求设置,应答的时候使用的相同的值返回。若是设置了RD,就建议域名服务器进行递归解析,递归查询的支持是可选的。
RA:支持递归(Recursion Available) - 这个比特位在应答中设置或取消,用来表明服务器是否支持递归查询。
Z:保留值,暂时未使用。在全部的请求和应答报文中必须置为0。
RCODE:应答码(Response code) - 这4个比特位在应答报文中设置,表明的含义以下:
0 没有错误。
1 报文格式错误(Format error) - 服务器不能理解请求的报文。
2 服务器失败(Server failure) - 由于服务器的缘由致使没办法处理这个请求。
3 名字错误(Name Error) - 只有对受权域名解析服务器有意义,指出解析的域名不存在。
4 没有实现(Not Implemented) - 域名服务器不支持查询类型。
5 拒绝(Refused) - 服务器因为设置的策略拒绝给出应答。好比,服务器不但愿对某些请求者给出应答,或者服务器不但愿进行某些操做(好比区域传送zone transfer)。
6-15 保留值,暂时未使用。
QDCOUNT 无符号16位整数表示报文请求段中的问题记录数。
ANCOUNT 无符号16位整数表示报文回答段中的回答记录数。
NSCOUNT 无符号16位整数表示报文受权段中的受权记录数。
ARCOUNT 无符号16位整数表示报文附加段中的附加记录数。
根据这些,dns头部的数据结构能够定义以下:
type dnsHeader struct {
Id uint16
Bits uint16
Qdcount, Ancount, Nscount, Arcount uint16
}
构造头部信息咱们主要处理Bits,能够直接根据需求对相应位置值,也能够定义好每个字段,经过移位的方式而后相加构造请求的头部各个字段。推荐后一种方法,这样就有:
header.Bits = QR<<15 + OperationCode<<11 + AuthoritativeAnswer<<10 + Truncation<<9 + RecursionDesired<<8 + RecursionAvailable<<7 + ResponseCode
其余的头部信息就比较简单了:
requestHeader := dnsHeader{
Id: 0x0010,
Qdcount: 1,
Ancount: 0,
Nscount: 0,
Arcount: 0,
}
报文头搞定后,接下来就是查询问题Question:
Question段描述了查询的问题,包括查询类型(QTYPE),查询类(QCLASS),以及查询的域名(QNAME)。字段含义以下 QNAME:域名被编码为一些labels序列,每一个labels包含一个字节表示后续字符串长度,以及这个字符串,以0长度和空字符串来表示域名结束。注意这个字段可能为奇数字节,不须要进行边界填充对齐。 QTYPE:2个字节表示查询类型,.取值能够为任何可用的类型值,以及通配码来表示全部的资源记录。 QCLASS:2个字节表示查询的协议类,好比,IN表明Internet。因此咱们直接定义查询的结构体以下:
type dnsQuery struct {
QuestionType uint16
QuestionClass uint16
}
查询的域名不定义在查询的结构体中,由函数接收参数的方式接收。
剩下的3个段包含相同的格式:一系列可能为空的资源记录(RRs)。Answer段包含回答问题的RRs;受权段包含受权域名服务器的RRs;附加段包含和请求相关的,可是不是必须回答的RRs。而在发送请求的时候,咱们是发起请求方,因此这些字段放空就好。
完整代码:
// 002 project main.go package main import ( "bytes" "encoding/binary" "fmt" "net" "strings" "time" ) type dnsHeader struct { Id uint16 Bits uint16 Qdcount, Ancount, Nscount, Arcount uint16 } func (header *dnsHeader) SetFlag(QR uint16, OperationCode uint16, AuthoritativeAnswer uint16, Truncation uint16, RecursionDesired uint16, RecursionAvailable uint16, ResponseCode uint16) { header.Bits = QR<<15 + OperationCode<<11 + AuthoritativeAnswer<<10 + Truncation<<9 + RecursionDesired<<8 + RecursionAvailable<<7 + ResponseCode } type dnsQuery struct { QuestionType uint16 QuestionClass uint16 } func ParseDomainName(domain string) []byte { var ( buffer bytes.Buffer segments []string = strings.Split(domain, ".") ) for _, seg := range segments { binary.Write(&buffer, binary.BigEndian, byte(len(seg))) binary.Write(&buffer, binary.BigEndian, []byte(seg)) } binary.Write(&buffer, binary.BigEndian, byte(0x00)) return buffer.Bytes() } func Send(dnsServer, domain string) ([]byte, int, time.Duration) { requestHeader := dnsHeader{ Id: 0x0010, Qdcount: 1, Ancount: 0, Nscount: 0, Arcount: 0, } requestHeader.SetFlag(0, 0, 0, 0, 1, 0, 0) requestQuery := dnsQuery{ QuestionType: 1, QuestionClass: 1, } var ( conn net.Conn err error buffer bytes.Buffer ) if conn, err = net.Dial("udp", dnsServer); err != nil { fmt.Println(err.Error()) return make([]byte, 0), 0, 0 } defer conn.Close() binary.Write(&buffer, binary.BigEndian, requestHeader) binary.Write(&buffer, binary.BigEndian, ParseDomainName(domain)) binary.Write(&buffer, binary.BigEndian, requestQuery) buf := make([]byte, 1024) t1 := time.Now() if _, err := conn.Write(buffer.Bytes()); err != nil { fmt.Println(err.Error()) return make([]byte, 0), 0, 0 } length, err := conn.Read(buf) t := time.Now().Sub(t1) return buf, length, t } func main() { remsg, n, _ := Send("114.114.114.114:53", "www.baidu.com") fmt.Println(remsg, n) }
抓个包看看:
这是发出去的,看看详细的Questions信息:
咱们设置的请求类型是1,class是1,意味着是请求A记录,class IN。下一节咱们在来讨论下如何处理服务器端响应的内容。