mirror of
				https://github.com/coredns/coredns.git
				synced 2025-11-03 18:53:13 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			134 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			134 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", 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
 |