DNS support edns-client-subnet

转自:http://noops.me/?p=653&utm_source=tuicool&utm_medium=referral

作者: wilbur   7,104 浏览  |  2013/06/26   4:07 下午

看了2天RFC,终于让DNS支持edns-client-subnet协议,通过google dns resolver的请求,可以获取用户的ip地址。
国内很多CDN和DNS提供商都已经实现了,但网上的中文资料比较少,所以在这里分享一下,能力有限,错误之处还请谅解。

问题

  • CDN使用DNS获取查询IP,根据IP对用户进行地域调度。但这里获取的IP地址是DNS地址,而不是用户真实的IP地址。
  • 大多数情况下,我们假设用户通过会使用离自己网络最近的DNS resolver,CDN调度基本还是准确的。
  • 但也有很多nameserver设置错误,或者用户使用google public dns(nameserver 8.8.8.8/8.8.4.4)或opendns进行DNS resolver

比如:

  1. 国内用户设置nameserver 8.8.8.8 (dig xxx.com @8.8.8.8)
  2. 我们得到的DNS query IP是74.125.16.208,判断IP属于美国,,,加利福尼亚州山景市谷歌公司
  3. 这个时候,我们的DNS会返回离美国加州最近的CDN节点IP给用户。
  4. 国内用户错误的调度到美国节点…… :(

edns-client-subnet

  • google提交了一份DNS扩展协议,允许DNS resolver传递用户的ip地址给authoritative DNS server.
  • CDN的DNS支持该协议,就可以获取用户真实的IP地址,进行准确的调度。
    图片1
  • OpenDNS和Google Public DNS已经支持了该协议,如果希望他们的query中带有用户IP,需要联系他们添加白名单。提供nameserver的hostname、ip以及可以用来测试解析的域名即可,一般几天就可以搞定。(注:我是晚上22:l00提交的申请,第二天10:00就已经生效了)

实现

一. 支持发送和接收edns-client-subnet的dig

  1. 先下载bind,下载地址
  2. 下载edns-client-subnet dig patch,下载地址
    下载上述2个包,将patch打进bind,编译出dig进行测试:

注意上面的OPT PSEUDOSECTION,已经可以发送和接收edns-client-subnet请求了

二. 协议

  • DNS协议
  • DNS query会包含header和RR 2部分,这里只介绍我们关注地方,网上可以搜到很多协议的介绍,比如这个http://archercai.blog.sohu.com/60779796.html
  • header会描述本次请求中Questions、Answer RRs、Authority RRs和Additional RRs的数量,RR部分会详细描述每个资源的内容,所有的RR格式是相同的,如下:

  • 个人理解edns-client-subnet是对edns协议的扩展,附加在一个DNS请求的Additional RRs区域,这里重点描述edns-client-subnet的结构
    • EDNS协议 Extension mechanisms for DNS (EDNS0):http://tools.ietf.org/html/draft-ietf-dnsind-edns0-01
  • EDNS0每个字段的结构和描述如下:

  • OPT 的值41,详细的协议值如下:

  • RDLENGTH描述RDATAD的长度,edns-client-subnet的详细格式存在RDATA中,如下:

  • OPTION-CODE: 2个字节
  • OPTION-LENGTH: 2个字节,描述它之后的内容长度(BYTE)
  • FAMILY: 2个字节,1表示ipv4, 2表示ipv6
  • ADDRESS: 实际存放IP地址的地方,ipv4长度为4,google发送过来的长度一般为3,隐藏了ip地址最后一位

三. 开发

完成前2个步骤,就可以开搞了,逻辑很简单:
1. 判断dns query是否包含Additional RRs,读取NAME部分
2. 读取10个字节(byte),判断TYPE是否为41,rdlength > 8
3. 如果rdlength > 8,再读取8个字节,对应OPTION-CODE(2)–>OPTION-LENGTH(2)–>FAMILY(2)–>SOURCE NETMASK(1)–>SCOPE NETMASK(1)
4. 读取剩下的address,长度 rdlength – 8 或者 option-length – 4都行
注:读取到的地址长度为4,可以用socket.inet_ntoa变成ip地址,如果不够4个字节,需要后面补\x00
5. 获取到的IP地址就可以用来进行判断调度了
6. respond时也需要增加一个Additional RRs区域,直接把请求的Additional内容发过去就可以(如果支持source netmask,将请求中的source netmask复制到scope netmask中,OpenDNS要求必须支持scope netmask)

四. 抓包

  1. 发包
    • 发送dns query请求时,可以看到Questions:1, Additional RRs: 1
    • Additional RRs中,type: 41(OPT), rdlength: 12 (google发过来的包,长度为11,没有IP地址最后一位)
    • 12 – OPTION-CODE(2) – OPTION-LENGTH(2) – FAMILY(2) – SOURCE NETMASK(1) – SCOPE NETMASK(1) = 4,IPV4 地址的大小
      图片2
  2. 回包
    • 发送dns query请求时,可以看到Questions:1, Answer RRs:1, Additional RRs: 1
      图片3

EDNS Client Subnet (ECS) 是DNS协议的一个扩展,它允许客户端在其DNS查询中包含源IP地址的子网信息。这样做是为了提高DNS缓存的效率,特别是对于大型网络环境,可以减少递归查找的时间。在C语言中实现EDNS ECS通常涉及到以下几个步骤: 1. **结构体定义**:首先,你需要定义一个表示DNS请求的结构体,其中会包含一个用于存储子网信息的字段,例如`struct dns_message`或自定义的数据结构。 ```c typedef struct { uint8_t ecs_data[DNS_MAX_EDNS_SUBNET_LEN]; size_t ecs_length; } DnsEdnsSubnetData; ``` 2. **解析和构建**:当你从用户输入获取到源IP地址和子网掩码时,需要将其转换成二进制并放入`ecs_data`数组中。你可以使用标准库函数处理这部分工作。 3. **设置DNS查询**:在发送DNS查询时,你需要在DNS消息头中添加EDNS扩展,指定EDNS版本和支持ECS标志,并将`DnsEdnsSubnetData`附加上。 ```c void add_ecs_to_dns(DnsMessage *msg, const char* src_ip, const char* subnet_mask) { // ... 实现细节,比如计算子网ID和填充数据 msg->edns.edns_version = DNS_EDNS_VERSION; msg->edns.flags |= DNS_RCODE_NOERROR; msg->edns.options[DNS_ECNDS_OFFSET] = htons(sizeof(DnsEdnsSubnetData)); msg->ecs_data = ...; // 填充子网信息 } ``` 4. **发送和解析响应**:然后通过网络发送查询,并在接收到DNS服务器的响应时检查是否有ECS信息返回。如果有的话,可以根据该信息优化缓存策略。 注意:在实际应用中,上述代码仅提供了一个基本的框架,具体的实现可能还需要依赖于特定的DNS库(如libunbound、dnsmasq等)提供的API。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值