-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathresolver_doh.go
117 lines (96 loc) · 2.61 KB
/
resolver_doh.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
package main
import (
"bytes"
"context"
"encoding/binary"
"errors"
"io"
"net"
"net/http"
"time"
)
var _ Dialer = (*DoHResolverDialer)(nil)
type DoHResolverDialer struct {
EndPoint string
UserAgent string
Transport http.RoundTripper
}
func (d *DoHResolverDialer) DialContext(ctx context.Context, network, addr string) (net.Conn, error) {
return &dohConn{d, nil}, nil
}
type dohConn struct {
dialer *DoHResolverDialer
buffer *bytes.Buffer
}
func (c *dohConn) Read(b []byte) (n int, err error) {
if c.buffer == nil {
err = io.ErrUnexpectedEOF
return
}
n, err = c.buffer.Read(b)
return
}
func (c *dohConn) Write(b []byte) (n int, err error) {
if len(b) < 2 {
return 0, errors.New("dns message too short")
}
if int(binary.BigEndian.Uint16(b))+2 != len(b) {
return 0, errors.New("dns message head size mismath")
}
req, err := http.NewRequest(http.MethodPost, c.dialer.EndPoint, bytes.NewReader(b[2:]))
if err != nil {
return 0, err
}
req.Header.Set("content-type", "application/dns-message")
if c.dialer.UserAgent != "" {
req.Header.Set("user-agent", c.dialer.UserAgent)
}
var tr = c.dialer.Transport
if tr == nil {
tr = http.DefaultTransport
}
ctx, cancel := context.WithTimeout(req.Context(), 5*time.Second)
defer cancel()
resp, err := tr.RoundTrip(req.WithContext(ctx))
if err != nil {
return 0, err
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK || resp.ContentLength <= 0 {
var errmsg string
if resp.Body != nil {
data := make([]byte, 1024)
if n, err := resp.Body.Read(data); err != nil {
errmsg = err.Error()
} else {
errmsg = string(data[:n])
}
}
return 0, errors.New("proxy: read from " + c.dialer.EndPoint + " error: " + resp.Status + ": " + errmsg)
}
c.buffer = new(bytes.Buffer)
binary.Write(c.buffer, binary.BigEndian, uint16(resp.ContentLength))
_, err = io.Copy(c.buffer, resp.Body)
if err != nil {
return 0, err
}
return len(b), nil
}
func (c *dohConn) Close() (err error) {
return
}
func (c *dohConn) LocalAddr() net.Addr {
return &net.TCPAddr{}
}
func (c *dohConn) RemoteAddr() (addr net.Addr) {
return &net.TCPAddr{}
}
func (c *dohConn) SetDeadline(t time.Time) error {
return &net.OpError{Op: "set", Net: "http2", Source: nil, Addr: nil, Err: errors.New("deadline not supported")}
}
func (c *dohConn) SetReadDeadline(t time.Time) error {
return &net.OpError{Op: "set", Net: "http2", Source: nil, Addr: nil, Err: errors.New("deadline not supported")}
}
func (c *dohConn) SetWriteDeadline(t time.Time) error {
return &net.OpError{Op: "set", Net: "http2", Source: nil, Addr: nil, Err: errors.New("deadline not supported")}
}