mirror of
https://github.com/fumiama/terasu.git
synced 2026-06-05 01:00:23 +08:00
161 lines
3.4 KiB
Go
161 lines
3.4 KiB
Go
package dns
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"encoding/json"
|
|
"errors"
|
|
"net"
|
|
"net/http"
|
|
"net/url"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"golang.org/x/net/http2"
|
|
|
|
"github.com/fumiama/terasu"
|
|
"github.com/fumiama/terasu/ip"
|
|
)
|
|
|
|
var (
|
|
ErrEmptyHostAddress = errors.New("empty host addr")
|
|
)
|
|
|
|
type recordType uint16
|
|
|
|
const (
|
|
recordTypeNone recordType = 0
|
|
recordTypeA recordType = 1
|
|
recordTypeAAAA recordType = 28
|
|
)
|
|
|
|
type dohjsonresponse struct {
|
|
Status uint32
|
|
TC bool
|
|
RD bool
|
|
RA bool
|
|
AD bool
|
|
CD bool
|
|
Question []struct {
|
|
Name string `json:"name"`
|
|
Type recordType `json:"type"`
|
|
}
|
|
Answer []struct {
|
|
Name string `json:"name"`
|
|
Type recordType `json:"type"`
|
|
TTL uint16
|
|
Data string `json:"data"`
|
|
}
|
|
EdnsClientSubnet string `json:"edns_client_subnet"`
|
|
Comment string
|
|
}
|
|
|
|
func (jr *dohjsonresponse) hosts() []string {
|
|
if len(jr.Answer) == 0 {
|
|
return nil
|
|
}
|
|
hosts := make([]string, 0, len(jr.Answer))
|
|
for _, ans := range jr.Answer {
|
|
if ans.Type == recordTypeA || ans.Type == recordTypeAAAA {
|
|
hosts = append(hosts, ans.Data)
|
|
}
|
|
}
|
|
return hosts
|
|
}
|
|
|
|
var trsHTTP2ClientWithSystemDNS = http.Client{
|
|
Transport: &http2.Transport{
|
|
DialTLSContext: func(ctx context.Context, network, addr string, cfg *tls.Config) (net.Conn, error) {
|
|
host, port, err := net.SplitHostPort(addr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
addrs := lookupTable.Get(host)
|
|
if len(addrs) == 0 {
|
|
addrs, err = net.DefaultResolver.LookupHost(ctx, host)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
lookupTable.Set(host, addrs)
|
|
}
|
|
if len(addr) == 0 {
|
|
return nil, ErrEmptyHostAddress
|
|
}
|
|
var conn net.Conn
|
|
var tlsConn *tls.Conn
|
|
for _, a := range addrs {
|
|
conn, err = dnsDialer.DialContext(ctx, network, net.JoinHostPort(a, port))
|
|
if err != nil {
|
|
continue
|
|
}
|
|
tlsConn = tls.Client(conn, cfg)
|
|
err = terasu.Use(tlsConn).HandshakeContext(ctx, terasu.DefaultFirstFragmentLen)
|
|
if err == nil {
|
|
break
|
|
}
|
|
_ = tlsConn.Close()
|
|
tlsConn = nil
|
|
conn, err = dnsDialer.DialContext(ctx, network, net.JoinHostPort(a, port))
|
|
if err != nil {
|
|
continue
|
|
}
|
|
tlsConn = tls.Client(conn, cfg)
|
|
err = tlsConn.HandshakeContext(ctx)
|
|
if err == nil {
|
|
break
|
|
}
|
|
_ = tlsConn.Close()
|
|
tlsConn = nil
|
|
}
|
|
return tlsConn, err
|
|
},
|
|
},
|
|
}
|
|
|
|
func lookupdoh(ctx context.Context, server, u string) (jr dohjsonresponse, err error) {
|
|
jr, err = lookupdohwithtype(ctx, server, u, preferreddohtype())
|
|
if err == nil {
|
|
return
|
|
}
|
|
if ip.IsIPv6Available.Get() {
|
|
jr, err = lookupdohwithtype(ctx, server, u, recordTypeA)
|
|
}
|
|
return
|
|
}
|
|
|
|
func lookupdohwithtype(ctx context.Context, server, u string, typ recordType) (jr dohjsonresponse, err error) {
|
|
sb := strings.Builder{}
|
|
sb.WriteString(server)
|
|
sb.WriteString("?name=")
|
|
sb.WriteString(url.QueryEscape(u))
|
|
if typ != recordTypeNone {
|
|
sb.WriteString("&type=")
|
|
sb.WriteString(strconv.Itoa(int(typ)))
|
|
}
|
|
req, err := http.NewRequestWithContext(ctx, "GET", sb.String(), nil)
|
|
if err != nil {
|
|
return
|
|
}
|
|
req.Header.Add("accept", "application/dns-json")
|
|
resp, err := trsHTTP2ClientWithSystemDNS.Do(req)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer resp.Body.Close()
|
|
err = json.NewDecoder(resp.Body).Decode(&jr)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if jr.Status != 0 {
|
|
err = errors.New("comment: " + jr.Comment)
|
|
}
|
|
return
|
|
}
|
|
|
|
func preferreddohtype() recordType {
|
|
if ip.IsIPv6Available.Get() {
|
|
return recordTypeAAAA
|
|
}
|
|
return recordTypeA
|
|
}
|