Skip to content

Commit a1ef7a6

Browse files
authored
Add RFC 9540 oblivious services via service binding records (miekg#1567)
1 parent c3301c3 commit a1ef7a6

File tree

3 files changed

+62
-4
lines changed

3 files changed

+62
-4
lines changed

parse_test.go

+11
Original file line numberDiff line numberDiff line change
@@ -1609,7 +1609,18 @@ func TestParseSVCB(t *testing.T) {
16091609
// From draft-ietf-add-ddr-06
16101610
`_dns.example.net. SVCB 1 example.net. alpn=h2 dohpath=/dns-query{?dns}`: `_dns.example.net. 3600 IN SVCB 1 example.net. alpn="h2" dohpath="/dns-query{?dns}"`,
16111611
`_dns.example.net. SVCB 1 example.net. alpn=h2 dohpath=/dns\045query{\?dns}`: `_dns.example.net. 3600 IN SVCB 1 example.net. alpn="h2" dohpath="/dns-query{?dns}"`,
1612+
// From RFC9461 Section 7 (https://datatracker.ietf.org/doc/html/rfc9461#section-7)
1613+
`_dns.simple.example. 7200 IN SVCB 1 simple.example. alpn=dot`: `_dns.simple.example. 7200 IN SVCB 1 simple.example. alpn="dot"`,
1614+
`_dns.doh.example. 7200 IN SVCB 1 doh.example. alpn=h2 dohpath=/dns-query{?dns}`: `_dns.doh.example. 7200 IN SVCB 1 doh.example. alpn="h2" dohpath="/dns-query{?dns}"`,
1615+
`_dns.resolver.example. 7200 IN SVCB 1 resolver.example. alpn=dot,doq,h2,h3 dohpath=/q{?dns}`: `_dns.resolver.example. 7200 IN SVCB 1 resolver.example. alpn="dot,doq,h2,h3" dohpath="/q{?dns}"`,
1616+
`_dns.resolver.example. 7200 IN SVCB 2 resolver.example. alpn=dot port=8530`: `_dns.resolver.example. 7200 IN SVCB 2 resolver.example. alpn="dot" port="8530"`,
1617+
// From RFC 9540 Section 4.2.1 (https://www.rfc-editor.org/rfc/rfc9540.html#name-the-ohttp-svcparamkey)
1618+
`_dns.resolver.arpa 7200 IN SVCB 1 doh.example.net alpn=h2 dohpath=/dns-query{?dns} ohttp`: `_dns.resolver.arpa. 7200 IN SVCB 1 doh.example.net. alpn="h2" dohpath="/dns-query{?dns}" ohttp=""`,
1619+
// From RFC 9540 Section 4.1 (HTTPS RR) (https://www.rfc-editor.org/rfc/rfc9540.html#name-use-in-https-service-rrs)
1620+
`svc.example.com. 7200 IN HTTPS 1 . alpn=h2 ohttp`: `svc.example.com. 7200 IN HTTPS 1 . alpn="h2" ohttp=""`,
1621+
`svc.example.com. 7200 IN HTTPS 1 . mandatory=ohttp ohttp`: `svc.example.com. 7200 IN HTTPS 1 . mandatory="ohttp" ohttp=""`,
16121622
}
1623+
16131624
for s, o := range svcbs {
16141625
rr, err := NewRR(s)
16151626
if err != nil {

svcb.go

+46-4
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ import (
1414
// SVCBKey is the type of the keys used in the SVCB RR.
1515
type SVCBKey uint16
1616

17-
// Keys defined in draft-ietf-dnsop-svcb-https-08 Section 14.3.2.
17+
// Keys defined in rfc9460
1818
const (
1919
SVCB_MANDATORY SVCBKey = iota
2020
SVCB_ALPN
@@ -23,7 +23,8 @@ const (
2323
SVCB_IPV4HINT
2424
SVCB_ECHCONFIG
2525
SVCB_IPV6HINT
26-
SVCB_DOHPATH // draft-ietf-add-svcb-dns-02 Section 9
26+
SVCB_DOHPATH // rfc9461 Section 5
27+
SVCB_OHTTP // rfc9540 Section 8
2728

2829
svcb_RESERVED SVCBKey = 65535
2930
)
@@ -37,6 +38,7 @@ var svcbKeyToStringMap = map[SVCBKey]string{
3738
SVCB_ECHCONFIG: "ech",
3839
SVCB_IPV6HINT: "ipv6hint",
3940
SVCB_DOHPATH: "dohpath",
41+
SVCB_OHTTP: "ohttp",
4042
}
4143

4244
var svcbStringToKeyMap = reverseSVCBKeyMap(svcbKeyToStringMap)
@@ -201,6 +203,8 @@ func makeSVCBKeyValue(key SVCBKey) SVCBKeyValue {
201203
return new(SVCBIPv6Hint)
202204
case SVCB_DOHPATH:
203205
return new(SVCBDoHPath)
206+
case SVCB_OHTTP:
207+
return new(SVCBOhttp)
204208
case svcb_RESERVED:
205209
return nil
206210
default:
@@ -771,8 +775,8 @@ func (s *SVCBIPv6Hint) copy() SVCBKeyValue {
771775
// SVCBDoHPath pair is used to indicate the URI template that the
772776
// clients may use to construct a DNS over HTTPS URI.
773777
//
774-
// See RFC xxxx (https://datatracker.ietf.org/doc/html/draft-ietf-add-svcb-dns-02)
775-
// and RFC yyyy (https://datatracker.ietf.org/doc/html/draft-ietf-add-ddr-06).
778+
// See RFC 9461 (https://datatracker.ietf.org/doc/html/rfc9461)
779+
// and RFC 9462 (https://datatracker.ietf.org/doc/html/rfc9462).
776780
//
777781
// A basic example of using the dohpath option together with the alpn
778782
// option to indicate support for DNS over HTTPS on a certain path:
@@ -816,6 +820,44 @@ func (s *SVCBDoHPath) copy() SVCBKeyValue {
816820
}
817821
}
818822

823+
// The "ohttp" SvcParamKey is used to indicate that a service described in a SVCB RR
824+
// can be accessed as a target using an associated gateway.
825+
// Both the presentation and wire-format values for the "ohttp" parameter MUST be empty.
826+
//
827+
// See RFC 9460 (https://datatracker.ietf.org/doc/html/rfc9460/)
828+
// and RFC 9230 (https://datatracker.ietf.org/doc/html/rfc9230/)
829+
//
830+
// A basic example of using the dohpath option together with the alpn
831+
// option to indicate support for DNS over HTTPS on a certain path:
832+
//
833+
// s := new(dns.SVCB)
834+
// s.Hdr = dns.RR_Header{Name: ".", Rrtype: dns.TypeSVCB, Class: dns.ClassINET}
835+
// e := new(dns.SVCBAlpn)
836+
// e.Alpn = []string{"h2", "h3"}
837+
// p := new(dns.SVCBOhttp)
838+
// s.Value = append(s.Value, e, p)
839+
type SVCBOhttp struct{}
840+
841+
func (*SVCBOhttp) Key() SVCBKey { return SVCB_OHTTP }
842+
func (*SVCBOhttp) copy() SVCBKeyValue { return &SVCBOhttp{} }
843+
func (*SVCBOhttp) pack() ([]byte, error) { return []byte{}, nil }
844+
func (*SVCBOhttp) String() string { return "" }
845+
func (*SVCBOhttp) len() int { return 0 }
846+
847+
func (*SVCBOhttp) unpack(b []byte) error {
848+
if len(b) != 0 {
849+
return errors.New("dns: svcbotthp: svcbotthp must have no value")
850+
}
851+
return nil
852+
}
853+
854+
func (*SVCBOhttp) parse(b string) error {
855+
if b != "" {
856+
return errors.New("dns: svcbotthp: svcbotthp must have no value")
857+
}
858+
return nil
859+
}
860+
819861
// SVCBLocal pair is intended for experimental/private use. The key is recommended
820862
// to be in the range [SVCB_PRIVATE_LOWER, SVCB_PRIVATE_UPPER].
821863
// Basic use pattern for creating a keyNNNNN option:

svcb_test.go

+5
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ func TestSVCB(t *testing.T) {
2424
{`key65002`, ``},
2525
{`key65003`, `=\"\"`},
2626
{`key65004`, `\254\ \ \030\000`},
27+
{`ohttp`, ``},
2728
}
2829

2930
for _, o := range svcbs {
@@ -86,6 +87,10 @@ func TestDecodeBadSVCB(t *testing.T) {
8687
key: SVCB_IPV6HINT,
8788
data: []byte{0, 0, 0},
8889
},
90+
{
91+
key: SVCB_OHTTP,
92+
data: []byte{0},
93+
},
8994
}
9095
for _, o := range svcbs {
9196
err := makeSVCBKeyValue(SVCBKey(o.key)).unpack(o.data)

0 commit comments

Comments
 (0)