2017-04-18 01:02:51 +00:00
|
|
|
package dns
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
// PrivateRdata is an interface used for implementing "Private Use" RR types, see
|
|
|
|
// RFC 6895. This allows one to experiment with new RR types, without requesting an
|
|
|
|
// official type code. Also see dns.PrivateHandle and dns.PrivateHandleRemove.
|
|
|
|
type PrivateRdata interface {
|
|
|
|
// String returns the text presentaton of the Rdata of the Private RR.
|
|
|
|
String() string
|
|
|
|
// Parse parses the Rdata of the private RR.
|
|
|
|
Parse([]string) error
|
|
|
|
// Pack is used when packing a private RR into a buffer.
|
|
|
|
Pack([]byte) (int, error)
|
|
|
|
// Unpack is used when unpacking a private RR from a buffer.
|
|
|
|
// TODO(miek): diff. signature than Pack, see edns0.go for instance.
|
|
|
|
Unpack([]byte) (int, error)
|
|
|
|
// Copy copies the Rdata.
|
|
|
|
Copy(PrivateRdata) error
|
|
|
|
// Len returns the length in octets of the Rdata.
|
|
|
|
Len() int
|
|
|
|
}
|
|
|
|
|
|
|
|
// PrivateRR represents an RR that uses a PrivateRdata user-defined type.
|
|
|
|
// It mocks normal RRs and implements dns.RR interface.
|
|
|
|
type PrivateRR struct {
|
|
|
|
Hdr RR_Header
|
|
|
|
Data PrivateRdata
|
|
|
|
}
|
|
|
|
|
|
|
|
func mkPrivateRR(rrtype uint16) *PrivateRR {
|
|
|
|
// Panics if RR is not an instance of PrivateRR.
|
|
|
|
rrfunc, ok := TypeToRR[rrtype]
|
|
|
|
if !ok {
|
|
|
|
panic(fmt.Sprintf("dns: invalid operation with Private RR type %d", rrtype))
|
|
|
|
}
|
|
|
|
|
|
|
|
anyrr := rrfunc()
|
|
|
|
switch rr := anyrr.(type) {
|
|
|
|
case *PrivateRR:
|
|
|
|
return rr
|
|
|
|
}
|
|
|
|
panic(fmt.Sprintf("dns: RR is not a PrivateRR, TypeToRR[%d] generator returned %T", rrtype, anyrr))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Header return the RR header of r.
|
|
|
|
func (r *PrivateRR) Header() *RR_Header { return &r.Hdr }
|
|
|
|
|
|
|
|
func (r *PrivateRR) String() string { return r.Hdr.String() + r.Data.String() }
|
|
|
|
|
|
|
|
// Private len and copy parts to satisfy RR interface.
|
2018-12-16 11:19:16 +00:00
|
|
|
func (r *PrivateRR) len(off int, compression map[string]struct{}) int {
|
|
|
|
l := r.Hdr.len(off, compression)
|
|
|
|
l += r.Data.Len()
|
|
|
|
return l
|
|
|
|
}
|
|
|
|
|
2017-04-18 01:02:51 +00:00
|
|
|
func (r *PrivateRR) copy() RR {
|
|
|
|
// make new RR like this:
|
|
|
|
rr := mkPrivateRR(r.Hdr.Rrtype)
|
2018-08-31 01:57:19 +00:00
|
|
|
rr.Hdr = r.Hdr
|
2017-04-18 01:02:51 +00:00
|
|
|
|
|
|
|
err := r.Data.Copy(rr.Data)
|
|
|
|
if err != nil {
|
|
|
|
panic("dns: got value that could not be used to copy Private rdata")
|
|
|
|
}
|
|
|
|
return rr
|
|
|
|
}
|
2018-12-16 11:19:16 +00:00
|
|
|
|
|
|
|
func (r *PrivateRR) pack(msg []byte, off int, compression compressionMap, compress bool) (int, int, error) {
|
|
|
|
headerEnd, off, err := r.Hdr.pack(msg, off, compression, compress)
|
2017-04-18 01:02:51 +00:00
|
|
|
if err != nil {
|
2018-12-16 11:19:16 +00:00
|
|
|
return off, off, err
|
2017-04-18 01:02:51 +00:00
|
|
|
}
|
|
|
|
n, err := r.Data.Pack(msg[off:])
|
|
|
|
if err != nil {
|
2018-12-16 11:19:16 +00:00
|
|
|
return headerEnd, len(msg), err
|
2017-04-18 01:02:51 +00:00
|
|
|
}
|
|
|
|
off += n
|
2018-12-16 11:19:16 +00:00
|
|
|
return headerEnd, off, nil
|
2017-04-18 01:02:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// PrivateHandle registers a private resource record type. It requires
|
|
|
|
// string and numeric representation of private RR type and generator function as argument.
|
|
|
|
func PrivateHandle(rtypestr string, rtype uint16, generator func() PrivateRdata) {
|
|
|
|
rtypestr = strings.ToUpper(rtypestr)
|
|
|
|
|
|
|
|
TypeToRR[rtype] = func() RR { return &PrivateRR{RR_Header{}, generator()} }
|
|
|
|
TypeToString[rtype] = rtypestr
|
|
|
|
StringToType[rtypestr] = rtype
|
|
|
|
|
|
|
|
typeToUnpack[rtype] = func(h RR_Header, msg []byte, off int) (RR, int, error) {
|
|
|
|
if noRdata(h) {
|
|
|
|
return &h, off, nil
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
|
|
|
|
rr := mkPrivateRR(h.Rrtype)
|
|
|
|
rr.Hdr = h
|
|
|
|
|
|
|
|
off1, err := rr.Data.Unpack(msg[off:])
|
|
|
|
off += off1
|
|
|
|
if err != nil {
|
|
|
|
return rr, off, err
|
|
|
|
}
|
|
|
|
return rr, off, err
|
|
|
|
}
|
|
|
|
|
2018-11-10 07:04:36 +00:00
|
|
|
setPrivateRR := func(h RR_Header, c *zlexer, o, f string) (RR, *ParseError, string) {
|
2017-04-18 01:02:51 +00:00
|
|
|
rr := mkPrivateRR(h.Rrtype)
|
|
|
|
rr.Hdr = h
|
|
|
|
|
|
|
|
var l lex
|
|
|
|
text := make([]string, 0, 2) // could be 0..N elements, median is probably 1
|
|
|
|
Fetch:
|
|
|
|
for {
|
|
|
|
// TODO(miek): we could also be returning _QUOTE, this might or might not
|
|
|
|
// be an issue (basically parsing TXT becomes hard)
|
2018-11-10 07:04:36 +00:00
|
|
|
switch l, _ = c.Next(); l.value {
|
2017-04-18 01:02:51 +00:00
|
|
|
case zNewline, zEOF:
|
|
|
|
break Fetch
|
|
|
|
case zString:
|
|
|
|
text = append(text, l.token)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err := rr.Data.Parse(text)
|
|
|
|
if err != nil {
|
|
|
|
return nil, &ParseError{f, err.Error(), l}, ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return rr, nil, ""
|
|
|
|
}
|
|
|
|
|
|
|
|
typeToparserFunc[rtype] = parserFunc{setPrivateRR, true}
|
|
|
|
}
|
|
|
|
|
2018-10-06 06:09:29 +00:00
|
|
|
// PrivateHandleRemove removes definitions required to support private RR type.
|
2017-04-18 01:02:51 +00:00
|
|
|
func PrivateHandleRemove(rtype uint16) {
|
|
|
|
rtypestr, ok := TypeToString[rtype]
|
|
|
|
if ok {
|
|
|
|
delete(TypeToRR, rtype)
|
|
|
|
delete(TypeToString, rtype)
|
|
|
|
delete(typeToparserFunc, rtype)
|
|
|
|
delete(StringToType, rtypestr)
|
|
|
|
delete(typeToUnpack, rtype)
|
|
|
|
}
|
|
|
|
}
|