This change avoids the hard coding of HTTPS, allowing flexibility in whether HTTP or HTTPS is used. Signed-off-by: Sebastian Dahlgren <sebdah@fb.com>
133 lines
2.9 KiB
Go
133 lines
2.9 KiB
Go
package doh
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/base64"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
// MimeType is the DoH mimetype that should be used.
|
|
const MimeType = "application/dns-message"
|
|
|
|
// Path is the URL path that should be used.
|
|
const Path = "/dns-query"
|
|
|
|
// NewRequest returns a new DoH request given a HTTP method, URL and dns.Msg.
|
|
//
|
|
// The URL should not have a path, so please exclude /dns-query. The URL will
|
|
// be prefixed with https:// by default, unless it's already prefixed with
|
|
// either http:// or https://.
|
|
func NewRequest(method, url string, m *dns.Msg) (*http.Request, error) {
|
|
buf, err := m.Pack()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !strings.HasPrefix(url, "http://") && !strings.HasPrefix(url, "https://") {
|
|
url = fmt.Sprintf("https://%s", url)
|
|
}
|
|
|
|
switch method {
|
|
case http.MethodGet:
|
|
b64 := base64.RawURLEncoding.EncodeToString(buf)
|
|
|
|
req, err := http.NewRequest(
|
|
http.MethodGet,
|
|
fmt.Sprintf("%s%s?dns=%s", url, Path, b64),
|
|
nil,
|
|
)
|
|
if err != nil {
|
|
return req, err
|
|
}
|
|
|
|
req.Header.Set("content-type", MimeType)
|
|
req.Header.Set("accept", MimeType)
|
|
return req, nil
|
|
|
|
case http.MethodPost:
|
|
req, err := http.NewRequest(
|
|
http.MethodPost,
|
|
fmt.Sprintf("%s%s?bla=foo:443", url, Path),
|
|
bytes.NewReader(buf),
|
|
)
|
|
if err != nil {
|
|
return req, err
|
|
}
|
|
|
|
req.Header.Set("content-type", MimeType)
|
|
req.Header.Set("accept", MimeType)
|
|
return req, nil
|
|
|
|
default:
|
|
return nil, fmt.Errorf("method not allowed: %s", method)
|
|
}
|
|
}
|
|
|
|
// ResponseToMsg converts a http.Response to a dns message.
|
|
func ResponseToMsg(resp *http.Response) (*dns.Msg, error) {
|
|
defer resp.Body.Close()
|
|
|
|
return toMsg(resp.Body)
|
|
}
|
|
|
|
// RequestToMsg converts a http.Request to a dns message.
|
|
func RequestToMsg(req *http.Request) (*dns.Msg, error) {
|
|
switch req.Method {
|
|
case http.MethodGet:
|
|
return requestToMsgGet(req)
|
|
|
|
case http.MethodPost:
|
|
return requestToMsgPost(req)
|
|
|
|
default:
|
|
return nil, fmt.Errorf("method not allowed: %s", req.Method)
|
|
}
|
|
}
|
|
|
|
// requestToMsgPost extracts the dns message from the request body.
|
|
func requestToMsgPost(req *http.Request) (*dns.Msg, error) {
|
|
defer req.Body.Close()
|
|
return toMsg(req.Body)
|
|
}
|
|
|
|
// requestToMsgGet extract the dns message from the GET request.
|
|
func requestToMsgGet(req *http.Request) (*dns.Msg, error) {
|
|
values := req.URL.Query()
|
|
b64, ok := values["dns"]
|
|
if !ok {
|
|
return nil, fmt.Errorf("no 'dns' query parameter found")
|
|
}
|
|
if len(b64) != 1 {
|
|
return nil, fmt.Errorf("multiple 'dns' query values found")
|
|
}
|
|
return base64ToMsg(b64[0])
|
|
}
|
|
|
|
func toMsg(r io.ReadCloser) (*dns.Msg, error) {
|
|
buf, err := io.ReadAll(http.MaxBytesReader(nil, r, 65536))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m := new(dns.Msg)
|
|
err = m.Unpack(buf)
|
|
return m, err
|
|
}
|
|
|
|
func base64ToMsg(b64 string) (*dns.Msg, error) {
|
|
buf, err := b64Enc.DecodeString(b64)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
m := new(dns.Msg)
|
|
err = m.Unpack(buf)
|
|
|
|
return m, err
|
|
}
|
|
|
|
var b64Enc = base64.RawURLEncoding
|