2017-12-19 22:55:09 +00:00
package dns
2016-05-16 06:09:28 +00:00
import (
2017-01-26 19:46:44 +00:00
"context"
2016-05-16 06:09:28 +00:00
"sync"
"time"
2017-01-26 19:46:44 +00:00
"github.com/miekg/dns"
2016-08-20 18:55:45 +00:00
"v2ray.com/core/app/dispatcher"
2017-11-14 23:36:14 +00:00
"v2ray.com/core/common"
2016-12-09 10:35:27 +00:00
"v2ray.com/core/common/buf"
2016-08-20 18:55:45 +00:00
"v2ray.com/core/common/dice"
2017-08-29 10:56:57 +00:00
"v2ray.com/core/common/net"
2016-08-20 18:55:45 +00:00
"v2ray.com/core/transport/internet/udp"
2016-05-16 06:09:28 +00:00
)
const (
2016-05-17 00:30:00 +00:00
CleanupInterval = time . Second * 120
CleanupThreshold = 512
2016-05-16 06:09:28 +00:00
)
2016-06-01 20:09:24 +00:00
var (
2017-11-19 20:43:20 +00:00
multiQuestionDNS = map [ net . Address ] bool {
2017-11-19 19:42:34 +00:00
net . IPAddress ( [ ] byte { 8 , 8 , 8 , 8 } ) : true ,
net . IPAddress ( [ ] byte { 8 , 8 , 4 , 4 } ) : true ,
net . IPAddress ( [ ] byte { 9 , 9 , 9 , 9 } ) : true ,
}
2016-06-01 20:09:24 +00:00
)
2016-05-16 06:09:28 +00:00
type ARecord struct {
IPs [ ] net . IP
Expire time . Time
}
type NameServer interface {
QueryA ( domain string ) <- chan * ARecord
}
type PendingRequest struct {
expire time . Time
response chan <- * ARecord
}
type UDPNameServer struct {
sync . Mutex
2017-08-29 10:56:57 +00:00
address net . Destination
2016-05-17 00:30:00 +00:00
requests map [ uint16 ] * PendingRequest
2017-01-27 13:45:16 +00:00
udpServer * udp . Dispatcher
2016-05-17 00:30:00 +00:00
nextCleanup time . Time
2016-05-16 06:09:28 +00:00
}
2017-08-29 10:56:57 +00:00
func NewUDPNameServer ( address net . Destination , dispatcher dispatcher . Interface ) * UDPNameServer {
2016-05-16 06:09:28 +00:00
s := & UDPNameServer {
2016-11-13 13:33:00 +00:00
address : address ,
requests : make ( map [ uint16 ] * PendingRequest ) ,
2017-01-27 13:45:16 +00:00
udpServer : udp . NewDispatcher ( dispatcher ) ,
2016-05-16 06:09:28 +00:00
}
return s
}
2017-11-19 20:43:20 +00:00
func ( s * UDPNameServer ) Cleanup ( ) {
2016-05-17 00:30:00 +00:00
expiredRequests := make ( [ ] uint16 , 0 , 16 )
now := time . Now ( )
2017-11-19 20:43:20 +00:00
s . Lock ( )
for id , r := range s . requests {
2016-05-17 00:30:00 +00:00
if r . expire . Before ( now ) {
expiredRequests = append ( expiredRequests , id )
close ( r . response )
2016-05-16 06:09:28 +00:00
}
}
2016-05-17 00:30:00 +00:00
for _ , id := range expiredRequests {
2017-11-19 20:43:20 +00:00
delete ( s . requests , id )
2016-05-17 00:30:00 +00:00
}
2017-11-19 20:43:20 +00:00
s . Unlock ( )
2016-05-16 06:09:28 +00:00
}
2017-11-19 20:43:20 +00:00
func ( s * UDPNameServer ) AssignUnusedID ( response chan <- * ARecord ) uint16 {
2016-05-16 06:09:28 +00:00
var id uint16
2017-11-19 20:43:20 +00:00
s . Lock ( )
if len ( s . requests ) > CleanupThreshold && s . nextCleanup . Before ( time . Now ( ) ) {
s . nextCleanup = time . Now ( ) . Add ( CleanupInterval )
go s . Cleanup ( )
2016-05-17 00:30:00 +00:00
}
2016-05-16 06:09:28 +00:00
for {
2017-04-27 09:54:15 +00:00
id = dice . RollUint16 ( )
2017-11-19 20:43:20 +00:00
if _ , found := s . requests [ id ] ; found {
2016-05-16 06:09:28 +00:00
continue
}
2017-12-19 20:28:12 +00:00
newError ( "add pending request id " , id ) . AtDebug ( ) . WriteToLog ( )
2017-11-19 20:43:20 +00:00
s . requests [ id ] = & PendingRequest {
2016-06-01 20:09:24 +00:00
expire : time . Now ( ) . Add ( time . Second * 8 ) ,
2016-05-16 06:09:28 +00:00
response : response ,
}
break
}
2017-11-19 20:43:20 +00:00
s . Unlock ( )
2016-05-16 06:09:28 +00:00
return id
}
2017-11-19 20:43:20 +00:00
func ( s * UDPNameServer ) HandleResponse ( payload * buf . Buffer ) {
2016-05-16 06:09:28 +00:00
msg := new ( dns . Msg )
2016-12-05 14:19:14 +00:00
err := msg . Unpack ( payload . Bytes ( ) )
2017-11-18 19:00:09 +00:00
if err == dns . ErrTruncated {
2017-12-19 20:28:12 +00:00
newError ( "truncated message received. DNS server should still work. If you see anything abnormal, please submit an issue to v2ray-core." ) . AtWarning ( ) . WriteToLog ( )
2017-11-18 19:00:09 +00:00
} else if err != nil {
2017-12-19 20:28:12 +00:00
newError ( "failed to parse DNS response" ) . Base ( err ) . AtWarning ( ) . WriteToLog ( )
2016-05-16 06:09:28 +00:00
return
}
record := & ARecord {
IPs : make ( [ ] net . IP , 0 , 16 ) ,
}
id := msg . Id
2017-11-14 23:36:14 +00:00
ttl := uint32 ( 3600 ) // an hour
2017-12-19 20:28:12 +00:00
newError ( "handling response for id " , id , " content: " , msg ) . AtDebug ( ) . WriteToLog ( )
2016-05-16 06:09:28 +00:00
2017-11-19 20:43:20 +00:00
s . Lock ( )
request , found := s . requests [ id ]
2016-05-16 06:09:28 +00:00
if ! found {
2017-11-19 20:43:20 +00:00
s . Unlock ( )
2016-05-16 06:09:28 +00:00
return
}
2017-11-19 20:43:20 +00:00
delete ( s . requests , id )
s . Unlock ( )
2016-05-16 06:09:28 +00:00
for _ , rr := range msg . Answer {
2016-05-16 18:53:18 +00:00
switch rr := rr . ( type ) {
case * dns . A :
record . IPs = append ( record . IPs , rr . A )
if rr . Hdr . Ttl < ttl {
ttl = rr . Hdr . Ttl
}
case * dns . AAAA :
record . IPs = append ( record . IPs , rr . AAAA )
if rr . Hdr . Ttl < ttl {
ttl = rr . Hdr . Ttl
2016-05-16 06:09:28 +00:00
}
}
}
record . Expire = time . Now ( ) . Add ( time . Second * time . Duration ( ttl ) )
request . response <- record
close ( request . response )
}
2017-11-19 20:43:20 +00:00
func ( s * UDPNameServer ) BuildQueryA ( domain string , id uint16 ) * buf . Buffer {
2016-05-16 06:09:28 +00:00
msg := new ( dns . Msg )
2016-06-01 20:09:24 +00:00
msg . Id = id
2016-05-16 06:09:28 +00:00
msg . RecursionDesired = true
msg . Question = [ ] dns . Question {
2016-07-24 11:44:29 +00:00
{
2016-05-16 06:09:28 +00:00
Name : dns . Fqdn ( domain ) ,
Qtype : dns . TypeA ,
Qclass : dns . ClassINET ,
2017-11-19 19:42:34 +00:00
} }
2017-11-19 20:43:20 +00:00
if multiQuestionDNS [ s . address . Address ] {
2017-11-19 19:42:34 +00:00
msg . Question = append ( msg . Question , dns . Question {
2017-11-14 23:36:14 +00:00
Name : dns . Fqdn ( domain ) ,
Qtype : dns . TypeAAAA ,
Qclass : dns . ClassINET ,
2017-11-19 19:42:34 +00:00
} )
}
2016-05-16 06:09:28 +00:00
2016-12-11 20:43:16 +00:00
buffer := buf . New ( )
2017-11-14 23:36:14 +00:00
common . Must ( buffer . Reset ( func ( b [ ] byte ) ( int , error ) {
2016-12-11 20:43:16 +00:00
writtenBuffer , err := msg . PackBuffer ( b )
return len ( writtenBuffer ) , err
2017-11-14 23:36:14 +00:00
} ) )
2016-05-16 06:09:28 +00:00
2016-06-01 20:09:24 +00:00
return buffer
}
2017-11-19 20:43:20 +00:00
func ( s * UDPNameServer ) QueryA ( domain string ) <- chan * ARecord {
2016-06-01 20:09:24 +00:00
response := make ( chan * ARecord , 1 )
2017-11-19 20:43:20 +00:00
id := s . AssignUnusedID ( response )
2016-06-01 20:09:24 +00:00
2017-11-14 23:36:14 +00:00
ctx , cancel := context . WithCancel ( context . Background ( ) )
2017-11-19 20:43:20 +00:00
s . udpServer . Dispatch ( ctx , s . address , s . BuildQueryA ( domain , id ) , s . HandleResponse )
2016-06-01 20:09:24 +00:00
go func ( ) {
for i := 0 ; i < 2 ; i ++ {
time . Sleep ( time . Second )
2017-11-19 20:43:20 +00:00
s . Lock ( )
_ , found := s . requests [ id ]
s . Unlock ( )
2017-11-14 23:36:14 +00:00
if ! found {
2016-06-01 20:09:24 +00:00
break
}
2017-11-19 20:43:20 +00:00
s . udpServer . Dispatch ( ctx , s . address , s . BuildQueryA ( domain , id ) , s . HandleResponse )
2016-06-01 20:09:24 +00:00
}
2017-01-26 19:46:44 +00:00
cancel ( )
2016-06-01 20:09:24 +00:00
} ( )
2016-05-16 06:09:28 +00:00
return response
}
2016-05-16 16:05:01 +00:00
type LocalNameServer struct {
}
2017-11-19 20:43:20 +00:00
func ( * LocalNameServer ) QueryA ( domain string ) <- chan * ARecord {
2016-05-16 18:53:18 +00:00
response := make ( chan * ARecord , 1 )
2016-05-16 16:05:01 +00:00
go func ( ) {
defer close ( response )
2017-12-19 22:55:09 +00:00
resolver := net . SystemIPResolver ( )
ips , err := resolver . LookupIP ( domain )
2016-05-16 16:05:01 +00:00
if err != nil {
2017-12-19 20:28:12 +00:00
newError ( "failed to lookup IPs for domain " , domain ) . Base ( err ) . WriteToLog ( )
2016-05-16 16:05:01 +00:00
return
}
response <- & ARecord {
IPs : ips ,
2017-11-14 23:36:14 +00:00
Expire : time . Now ( ) . Add ( time . Hour ) ,
2016-05-16 16:05:01 +00:00
}
} ( )
return response
}