dns/zscan_rr.go

1317 lines
28 KiB
Go
Raw Normal View History

package dns
import (
2012-02-24 05:37:08 +11:00
"encoding/base64"
2011-12-15 01:37:36 +11:00
"net"
"strconv"
"strings"
)
2012-02-12 22:28:08 +11:00
// Parse the rdata of each rrtype.
// All data from the channel c is either _STRING or _BLANK.
// After the rdata there may come 1 _BLANK and then a _NEWLINE
// or immediately a _NEWLINE. If this is not the case we flag
// an *ParseError: garbage after rdata.
func setRR(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
var r RR
2011-12-16 21:20:17 +11:00
e := new(ParseError)
2011-12-15 01:37:36 +11:00
switch h.Rrtype {
case TypeA:
r, e = setA(h, c, f)
2011-12-18 21:22:40 +11:00
goto Slurp
2011-12-15 01:37:36 +11:00
case TypeAAAA:
r, e = setAAAA(h, c, f)
2011-12-18 21:22:40 +11:00
goto Slurp
2011-12-15 01:37:36 +11:00
case TypeNS:
r, e = setNS(h, c, o, f)
2011-12-18 21:22:40 +11:00
goto Slurp
2012-02-12 03:03:09 +11:00
case TypePTR:
r, e = setPTR(h, c, o, f)
goto Slurp
2011-12-15 01:37:36 +11:00
case TypeMX:
r, e = setMX(h, c, o, f)
2011-12-18 21:22:40 +11:00
goto Slurp
2011-12-15 01:37:36 +11:00
case TypeCNAME:
r, e = setCNAME(h, c, o, f)
2011-12-18 21:22:40 +11:00
goto Slurp
2012-02-12 08:40:49 +11:00
case TypeDNAME:
r, e = setDNAME(h, c, o, f)
goto Slurp
2011-12-15 01:37:36 +11:00
case TypeSOA:
r, e = setSOA(h, c, o, f)
2011-12-18 21:22:40 +11:00
goto Slurp
2012-02-12 03:43:05 +11:00
case TypeSSHFP:
r, e = setSSHFP(h, c, f)
2011-12-18 21:22:40 +11:00
goto Slurp
2012-02-12 03:03:09 +11:00
case TypeSRV:
r, e = setSRV(h, c, o, f)
goto Slurp
case TypeNAPTR:
r, e = setNAPTR(h, c, o, f)
goto Slurp
2012-02-24 05:37:08 +11:00
case TypeTALINK:
2012-02-20 02:20:48 +11:00
r, e = setTALINK(h, c, o, f)
goto Slurp
// These types have a variable ending: either chunks of txt or chunks/base64 or hex.
2012-02-12 03:43:05 +11:00
// They need to search for the end of the RR themselves, hence they look for the ending
// newline. Thus there is no need to slurp the remainder, because there is none.
2011-12-18 21:22:40 +11:00
case TypeDNSKEY:
return setDNSKEY(h, c, f)
2011-12-15 01:37:36 +11:00
case TypeRRSIG:
return setRRSIG(h, c, o, f)
2011-12-15 01:37:36 +11:00
case TypeNSEC:
return setNSEC(h, c, o, f)
2011-12-15 01:37:36 +11:00
case TypeNSEC3:
return setNSEC3(h, c, o, f)
2012-02-12 08:47:22 +11:00
case TypeNSEC3PARAM:
return setNSEC3PARAM(h, c, f)
2011-12-16 21:30:29 +11:00
case TypeDS:
return setDS(h, c, f)
2012-02-19 07:19:56 +11:00
case TypeDLV:
return setDLV(h, c, f)
2012-02-19 07:22:58 +11:00
case TypeTA:
return setTA(h, c, f)
2012-02-19 07:04:31 +11:00
case TypeTLSA:
return setTLSA(h, c, f)
2011-12-15 01:37:36 +11:00
case TypeTXT:
return setTXT(h, c, f)
2012-02-19 07:04:31 +11:00
case TypeHIP:
return setHIP(h, c, o, f)
2012-02-19 04:59:19 +11:00
case TypeSPF:
return setSPF(h, c, f)
2012-02-19 07:37:03 +11:00
case TypeDHCID:
return setDHCID(h, c, f)
2012-02-19 07:04:31 +11:00
case TypeIPSECKEY:
return setIPSECKEY(h, c, o, f)
case TypeLOC:
r, e = setLOC(h, c, f)
2011-12-15 01:37:36 +11:00
default:
// RFC3957 RR (Unknown RR handling)
return setRFC3597(h, c, f)
2011-12-15 01:37:36 +11:00
}
2011-12-18 21:22:40 +11:00
Slurp:
if e != nil {
return nil, e
}
if se := slurpRemainder(c, f); se != nil {
2011-12-18 21:22:40 +11:00
return nil, se
}
2011-12-15 01:37:36 +11:00
return r, e
}
func setA(h RR_Header, c chan lex, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_A)
rr.Hdr = h
2011-12-15 01:37:36 +11:00
l := <-c
rr.A = net.ParseIP(l.token)
if rr.A == nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad A A", l}
2011-12-15 01:37:36 +11:00
}
return rr, nil
}
func setAAAA(h RR_Header, c chan lex, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_AAAA)
rr.Hdr = h
2011-12-15 01:37:36 +11:00
l := <-c
rr.AAAA = net.ParseIP(l.token)
if rr.AAAA == nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad AAAA AAAA", l}
2011-12-15 01:37:36 +11:00
}
return rr, nil
}
func setNS(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_NS)
rr.Hdr = h
2011-12-15 01:37:36 +11:00
l := <-c
rr.Ns = l.token
_, ld, ok := IsDomainName(l.token)
2012-02-05 21:20:04 +11:00
if !ok {
return nil, &ParseError{f, "bad NS Ns", l}
2011-12-15 01:37:36 +11:00
}
2012-02-05 21:20:04 +11:00
if rr.Ns[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Ns = appendOrigin(rr.Ns, o)
2012-01-13 08:49:26 +11:00
}
2011-12-15 01:37:36 +11:00
return rr, nil
}
2012-02-12 03:03:09 +11:00
func setPTR(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_PTR)
rr.Hdr = h
l := <-c
rr.Ptr = l.token
_, ld, ok := IsDomainName(l.token)
if !ok {
return nil, &ParseError{f, "bad PTR Ptr", l}
}
if rr.Ptr[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Ptr = appendOrigin(rr.Ptr, o)
2012-02-12 03:03:09 +11:00
}
return rr, nil
}
func setMX(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_MX)
rr.Hdr = h
2011-12-15 01:37:36 +11:00
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad MX Pref", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Pref = uint16(i)
}
<-c // _BLANK
l = <-c // _STRING
rr.Mx = l.token
_, ld, ok := IsDomainName(l.token)
2012-02-05 21:20:04 +11:00
if !ok {
return nil, &ParseError{f, "bad MX Mx", l}
2011-12-15 01:37:36 +11:00
}
if rr.Mx[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Mx = appendOrigin(rr.Mx, o)
2012-01-13 08:49:26 +11:00
}
2011-12-15 01:37:36 +11:00
return rr, nil
}
func setCNAME(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_CNAME)
rr.Hdr = h
2011-12-15 01:37:36 +11:00
l := <-c
rr.Target = l.token
_, ld, ok := IsDomainName(l.token)
2012-02-05 21:20:04 +11:00
if !ok {
return nil, &ParseError{f, "bad CNAME Target", l}
2011-12-15 01:37:36 +11:00
}
if rr.Target[ld-1] != '.' {
rr.Target = appendOrigin(rr.Target, o)
2012-01-13 08:49:26 +11:00
}
2011-12-15 01:37:36 +11:00
return rr, nil
2011-12-14 21:56:12 +11:00
}
2012-02-12 08:40:49 +11:00
func setDNAME(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_DNAME)
rr.Hdr = h
l := <-c
rr.Target = l.token
_, ld, ok := IsDomainName(l.token)
if !ok {
return nil, &ParseError{f, "bad CNAME Target", l}
}
if rr.Target[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Target = appendOrigin(rr.Target, o)
2012-02-12 08:40:49 +11:00
}
return rr, nil
}
func setSOA(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_SOA)
rr.Hdr = h
2011-12-14 21:56:12 +11:00
2011-12-15 01:37:36 +11:00
l := <-c
rr.Ns = l.token
<-c // _BLANK
_, ld, ok := IsDomainName(l.token)
2012-02-05 21:20:04 +11:00
if !ok {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad SOA Ns", l}
2011-12-15 01:37:36 +11:00
}
if rr.Ns[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Ns = appendOrigin(rr.Ns, o)
2012-01-13 08:49:26 +11:00
}
2011-12-14 21:56:12 +11:00
2011-12-15 01:37:36 +11:00
l = <-c
rr.Mbox = l.token
2012-02-05 21:20:04 +11:00
_, ld, ok = IsDomainName(l.token)
if !ok {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad SOA Mbox", l}
2011-12-15 01:37:36 +11:00
}
2012-02-05 21:20:04 +11:00
if rr.Mbox[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Mbox = appendOrigin(rr.Mbox, o)
2012-01-13 08:49:26 +11:00
}
2011-12-15 01:37:36 +11:00
<-c // _BLANK
2011-12-14 21:56:12 +11:00
2012-02-15 19:04:09 +11:00
var v uint32
2011-12-15 01:37:36 +11:00
for i := 0; i < 5; i++ {
l = <-c
2012-02-15 19:04:09 +11:00
if j, e := strconv.Atoi(l.token); e != nil {
if i == 0 {
// Serial should be a number
return nil, &ParseError{f, "bad SOA zone parameter", l}
}
2012-05-01 00:54:02 +10:00
if v, ok = stringToTtl(l.token); !ok {
2012-02-15 19:04:09 +11:00
return nil, &ParseError{f, "bad SOA zone parameter", l}
}
} else {
v = uint32(j)
2011-12-15 01:37:36 +11:00
}
switch i {
case 0:
2012-02-15 19:04:09 +11:00
rr.Serial = v
2011-12-16 21:20:17 +11:00
<-c // _BLANK
2011-12-15 01:37:36 +11:00
case 1:
2012-02-15 19:04:09 +11:00
rr.Refresh = v
2011-12-16 21:20:17 +11:00
<-c // _BLANK
2011-12-15 01:37:36 +11:00
case 2:
2012-02-15 19:04:09 +11:00
rr.Retry = v
2011-12-16 21:20:17 +11:00
<-c // _BLANK
2011-12-15 01:37:36 +11:00
case 3:
2012-02-15 19:04:09 +11:00
rr.Expire = v
2011-12-16 21:20:17 +11:00
<-c // _BLANK
2011-12-15 01:37:36 +11:00
case 4:
2012-02-15 19:04:09 +11:00
rr.Minttl = v
2011-12-15 01:37:36 +11:00
}
}
return rr, nil
2011-12-14 21:56:12 +11:00
}
2012-02-12 03:03:09 +11:00
func setSRV(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_SRV)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad SRV Priority", l}
} else {
rr.Priority = uint16(i)
}
<-c // _BLANK
l = <-c // _STRING
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad SRV Weight", l}
} else {
rr.Weight = uint16(i)
}
<-c // _BLANK
l = <-c // _STRING
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad SRV Port", l}
} else {
rr.Port = uint16(i)
}
<-c // _BLANK
l = <-c // _STRING
2012-02-12 03:43:05 +11:00
rr.Target = l.token
2012-02-12 03:03:09 +11:00
_, ld, ok := IsDomainName(l.token)
if !ok {
return nil, &ParseError{f, "bad SRV Target", l}
}
if rr.Target[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Target = appendOrigin(rr.Target, o)
2012-02-12 03:03:09 +11:00
}
return rr, nil
}
2012-02-12 08:32:45 +11:00
func setNAPTR(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_NAPTR)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad NAPTR Order", l}
} else {
rr.Order = uint16(i)
}
<-c // _BLANK
l = <-c // _STRING
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad NAPTR Preference", l}
} else {
rr.Preference = uint16(i)
}
// Flags
2012-02-12 08:32:45 +11:00
<-c // _BLANK
2012-02-13 09:00:26 +11:00
l = <-c // _QUOTE
if l.value != _QUOTE {
2012-02-13 09:00:26 +11:00
return nil, &ParseError{f, "bad NAPTR Flags", l}
}
l = <-c // Either String or Quote
if l.value == _STRING {
rr.Flags = l.token
l = <-c // _QUOTE
if l.value != _QUOTE {
return nil, &ParseError{f, "bad NAPTR Flags", l}
}
} else if l.value == _QUOTE {
rr.Flags = ""
} else {
return nil, &ParseError{f, "bad NAPTR Flags", l}
}
// Service
2012-02-12 08:32:45 +11:00
<-c // _BLANK
2012-02-13 09:00:26 +11:00
l = <-c // _QUOTE
if l.value != _QUOTE {
2012-02-13 09:00:26 +11:00
return nil, &ParseError{f, "bad NAPTR Service", l}
}
l = <-c // Either String or Quote
if l.value == _STRING {
rr.Service = l.token
l = <-c // _QUOTE
if l.value != _QUOTE {
return nil, &ParseError{f, "bad NAPTR Service", l}
}
} else if l.value == _QUOTE {
rr.Service = ""
} else {
return nil, &ParseError{f, "bad NAPTR Service", l}
}
// Regexp
2012-02-12 08:32:45 +11:00
<-c // _BLANK
2012-02-13 09:00:26 +11:00
l = <-c // _QUOTE
if l.value != _QUOTE {
return nil, &ParseError{f, "bad NAPTR Regexp", l}
}
l = <-c // Either String or Quote
if l.value == _STRING {
rr.Regexp = l.token
l = <-c // _QUOTE
if l.value != _QUOTE {
return nil, &ParseError{f, "bad NAPTR Regexp", l}
}
} else if l.value == _QUOTE {
rr.Regexp = ""
} else {
2012-02-13 09:00:26 +11:00
return nil, &ParseError{f, "bad NAPTR Regexp", l}
}
// After quote no space??
2012-02-12 08:32:45 +11:00
<-c // _BLANK
l = <-c // _STRING
rr.Replacement = l.token
_, ld, ok := IsDomainName(l.token)
if !ok {
return nil, &ParseError{f, "bad NAPTR Replacement", l}
}
if rr.Replacement[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.Replacement = appendOrigin(rr.Replacement, o)
2012-02-12 08:32:45 +11:00
}
return rr, nil
}
2012-02-20 02:20:48 +11:00
func setTALINK(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_TALINK)
rr.Hdr = h
l := <-c
rr.PreviousName = l.token
_, ld, ok := IsDomainName(l.token)
if !ok {
return nil, &ParseError{f, "bad TALINK PreviousName", l}
}
if rr.PreviousName[ld-1] != '.' {
rr.PreviousName = appendOrigin(rr.PreviousName, o)
}
2012-02-24 05:37:08 +11:00
<-c // _BLANK
2012-02-20 02:23:43 +11:00
l = <-c
2012-02-20 02:20:48 +11:00
rr.NextName = l.token
2012-02-20 02:23:43 +11:00
_, ld, ok = IsDomainName(l.token)
2012-02-20 02:20:48 +11:00
if !ok {
return nil, &ParseError{f, "bad TALINK NextName", l}
}
if rr.NextName[ld-1] != '.' {
rr.NextName = appendOrigin(rr.NextName, o)
}
return rr, nil
}
func setLOC(h RR_Header, c chan lex, f string) (RR, *ParseError) {
// Defaults TODO(mg)
rr := new(RR_LOC)
rr.Hdr = h
ok := false
// North
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad LOC Latitude", l}
} else {
rr.Latitude = 1000 * uint32(i)
}
<-c // _BLANK
// Either number, 'N' or 'S'
l = <-c
if rr.Latitude, ok = locCheckNorth(l.token, rr.Latitude); ok {
goto East
}
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad LOC Latitude minutes", l}
} else {
rr.Latitude += 1000 * 60 * uint32(i)
}
<-c // _BLANK
l = <-c
2012-05-01 05:57:42 +10:00
if i, e := strconv.ParseFloat(l.token, 32); e != nil {
return nil, &ParseError{f, "bad LOC Latitude seconds", l}
} else {
2012-05-01 05:57:42 +10:00
rr.Latitude += uint32(1000 * 60 * 60 * i)
}
<-c // _BLANK
// Either number, 'N' or 'S'
l = <-c
if rr.Latitude, ok = locCheckNorth(l.token, rr.Latitude); ok {
goto East
}
// If still alive, flag an error
return nil, &ParseError{f, "bad LOC Latitude North/South", l}
East:
// East
2012-05-01 05:57:42 +10:00
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad LOC Longitude", l}
} else {
rr.Longitude = uint32(1000.0 * i) // +0.0005 in ldns?
}
<-c // _BLANK
// Either number, 'E' or 'W'
l = <-c
if rr.Longitude, ok = locCheckEast(l.token, rr.Longitude); ok {
goto Altitude
}
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad LOC Longitude minutes", l}
} else {
rr.Longitude += 1000 * 60 * uint32(i)
}
<-c // _BLANK
l = <-c
2012-05-01 05:57:42 +10:00
if i, e := strconv.ParseFloat(l.token, 32); e != nil {
return nil, &ParseError{f, "bad LOC Longitude seconds", l}
} else {
2012-05-01 05:57:42 +10:00
rr.Longitude += uint32(1000 * 60 * 60 * i)
}
<-c // _BLANK
// Either number, 'E' or 'W'
l = <-c
if rr.Longitude, ok = locCheckEast(l.token, rr.Longitude); ok {
goto Altitude
}
// If still alive, flag an error
return nil, &ParseError{f, "bad LOC Longitude East/West", l}
Altitude:
<-c // _BLANK
l = <-c
if l.token[len(l.token)-1] == 'M' || l.token[len(l.token)-1] == 'm' {
l.token = l.token[0 : len(l.token)-1]
}
2012-05-01 05:57:42 +10:00
if i, e := strconv.ParseFloat(l.token, 32); e != nil {
return nil, &ParseError{f, "bad LOC Altitude", l}
} else {
2012-05-01 05:57:42 +10:00
rr.Altitude = uint32(i*100.0 + 10000000.0 + 0.5)
}
2012-05-01 05:42:58 +10:00
// And now optionally the other values
l = <-c
count := 0
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
switch count {
case 0: // Size
if e, m, ok := stringToCm(l.token); !ok {
return nil, &ParseError{f, "bad LOC Size", l}
} else {
rr.Size = (e & 0x0f) | (m << 4 & 0xf0)
}
case 1: // HorizPre
if e, m, ok := stringToCm(l.token); !ok {
return nil, &ParseError{f, "bad LOC HorizPre", l}
} else {
rr.HorizPre = (e & 0x0f) | (m << 4 & 0xf0)
}
case 2: // VertPre
if e, m, ok := stringToCm(l.token); !ok {
return nil, &ParseError{f, "bad LOC VertPre", l}
} else {
rr.VertPre = (e & 0x0f) | (m << 4 & 0xf0)
}
}
count++
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad LOC Size, HorizPre or VertPre", l}
}
l = <-c
}
2012-05-01 05:42:58 +10:00
return rr, nil
}
2012-02-19 07:04:31 +11:00
func setHIP(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_HIP)
rr.Hdr = h
// HitLength is not represented
2012-02-19 07:04:31 +11:00
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad HIP PublicKeyAlgorithm", l}
} else {
rr.PublicKeyAlgorithm = uint8(i)
}
<-c // _BLANK
l = <-c // _STRING
2012-02-19 21:27:16 +11:00
rr.Hit = l.token // This can not contain spaces, see RFC 5205 Section 6.
2012-02-24 05:37:08 +11:00
rr.HitLength = uint8(len(rr.Hit)) / 2
2012-02-19 07:04:31 +11:00
<-c // _BLANK
l = <-c // _STRING
rr.PublicKey = l.token // This cannot contain spaces
2012-02-24 05:37:08 +11:00
rr.PublicKeyLength = uint16(base64.StdEncoding.DecodedLen(len(rr.PublicKey)))
2012-02-19 07:04:31 +11:00
// RendezvousServers (if any)
l = <-c
xs := make([]string, 0)
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
_, ld, ok := IsDomainName(l.token)
if !ok {
return nil, &ParseError{f, "bad HIP RendezvousServers", l}
}
if l.token[ld-1] != '.' {
l.token = appendOrigin(l.token, o)
}
xs = append(xs, l.token)
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad HIP RendezvousServers", l}
}
l = <-c
}
rr.RendezvousServers = xs
return rr, nil
}
2012-02-12 08:38:17 +11:00
func setCERT(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_CERT)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad CERT Type", l}
} else {
rr.Type = uint16(i)
}
<-c // _BLANK
l = <-c // _STRING
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad NAPTR KeyTag", l}
} else {
rr.KeyTag = uint16(i)
}
<-c // _BLANK
l = <-c // _STRING
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad NAPTR Algorithm", l}
} else {
rr.Algorithm = uint8(i)
}
// Get the remaining data until we see a NEWLINE
l = <-c
s := ""
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad NAPTR Certificate", l}
}
l = <-c
}
rr.Certificate = s
return rr, nil
}
func setRRSIG(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_RRSIG)
rr.Hdr = h
l := <-c
if t, ok := Str_rr[strings.ToUpper(l.token)]; !ok {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG Typecovered", l}
2011-12-15 01:37:36 +11:00
} else {
rr.TypeCovered = t
}
<-c // _BLANK
l = <-c
if i, err := strconv.Atoi(l.token); err != nil {
return nil, &ParseError{f, "bad RRSIG Algorithm", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Algorithm = uint8(i)
}
<-c // _BLANK
l = <-c
if i, err := strconv.Atoi(l.token); err != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG Labels", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Labels = uint8(i)
}
<-c // _BLANK
l = <-c
if i, err := strconv.Atoi(l.token); err != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG OrigTtl", l}
2011-12-15 01:37:36 +11:00
} else {
rr.OrigTtl = uint32(i)
}
<-c // _BLANK
l = <-c
2012-04-11 23:13:17 +10:00
if i, err := DateToTime(l.token); err != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG Expiration", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Expiration = i
}
<-c // _BLANK
l = <-c
2012-04-11 23:13:17 +10:00
if i, err := DateToTime(l.token); err != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG Inception", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Inception = i
}
<-c // _BLANK
l = <-c
if i, err := strconv.Atoi(l.token); err != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG KeyTag", l}
2011-12-15 01:37:36 +11:00
} else {
rr.KeyTag = uint16(i)
}
<-c // _BLANK
l = <-c
rr.SignerName = l.token
_, ld, ok := IsDomainName(l.token)
2012-02-05 21:20:04 +11:00
if !ok {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG SignerName", l}
2011-12-15 01:37:36 +11:00
}
if rr.SignerName[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.SignerName = appendOrigin(rr.SignerName, o)
2012-01-13 08:49:26 +11:00
}
2011-12-15 01:37:36 +11:00
// Get the remaining data until we see a NEWLINE
l = <-c
2011-12-17 00:48:30 +11:00
s := ""
2011-12-15 01:37:36 +11:00
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad RRSIG Signature", l}
2011-12-15 01:37:36 +11:00
}
l = <-c
}
rr.Signature = s
return rr, nil
}
2011-12-14 21:56:12 +11:00
func setNSEC(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_NSEC)
rr.Hdr = h
2011-12-14 21:56:12 +11:00
2011-12-15 01:37:36 +11:00
l := <-c
rr.NextDomain = l.token
_, ld, ok := IsDomainName(l.token)
2012-02-05 21:20:04 +11:00
if !ok {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad NSEC NextDomain", l}
2011-12-15 01:37:36 +11:00
}
if rr.NextDomain[ld-1] != '.' {
2012-02-15 19:04:09 +11:00
rr.NextDomain = appendOrigin(rr.NextDomain, o)
2012-01-13 08:49:26 +11:00
}
2011-12-14 21:56:12 +11:00
2011-12-15 01:37:36 +11:00
rr.TypeBitMap = make([]uint16, 0)
2012-02-29 06:20:07 +11:00
var k uint16
2011-12-15 01:37:36 +11:00
l = <-c
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _BLANK:
// Ok
case _STRING:
2012-02-29 06:20:07 +11:00
if k, ok = Str_rr[strings.ToUpper(l.token)]; !ok {
if k, ok = typeToInt(l.token); !ok {
return nil, &ParseError{f, "bad NSEC TypeBitMap", l}
}
2011-12-15 01:37:36 +11:00
}
2012-02-29 06:20:07 +11:00
rr.TypeBitMap = append(rr.TypeBitMap, k)
2011-12-15 01:37:36 +11:00
default:
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad NSEC TypeBitMap", l}
2011-12-15 01:37:36 +11:00
}
l = <-c
}
return rr, nil
}
2011-12-14 21:56:12 +11:00
func setNSEC3(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_NSEC3)
rr.Hdr = h
2011-12-14 21:56:12 +11:00
2011-12-15 01:37:36 +11:00
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad NSEC3 Hash", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Hash = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad NSEC3 Flags", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Flags = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad NSEC3 Iterations", l}
2011-12-15 01:37:36 +11:00
} else {
rr.Iterations = uint16(i)
}
<-c
l = <-c
2012-02-26 07:49:57 +11:00
if len(l.token) == 0 {
return nil, &ParseError{f, "bad NSEC3 Salt", l}
}
rr.SaltLength = uint8(len(l.token)) / 2
rr.Salt = l.token
2011-12-15 00:02:55 +11:00
2011-12-15 01:37:36 +11:00
<-c
l = <-c
2012-02-26 07:49:57 +11:00
rr.HashLength = 20 // Fix for NSEC3 (sha1 160 bits)
2011-12-15 01:37:36 +11:00
rr.NextDomain = l.token
2011-12-14 21:56:12 +11:00
2011-12-15 01:37:36 +11:00
rr.TypeBitMap = make([]uint16, 0)
2012-02-29 06:20:07 +11:00
var (
k uint16
2012-02-29 06:20:07 +11:00
ok bool
)
2011-12-15 01:37:36 +11:00
l = <-c
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _BLANK:
// Ok
case _STRING:
2012-02-29 06:20:07 +11:00
if k, ok = Str_rr[strings.ToUpper(l.token)]; !ok {
if k, ok = typeToInt(l.token); !ok {
2012-02-29 06:20:07 +11:00
return nil, &ParseError{f, "bad NSEC3 TypeBitMap", l}
}
2011-12-15 01:37:36 +11:00
}
2012-02-29 06:20:07 +11:00
rr.TypeBitMap = append(rr.TypeBitMap, k)
2011-12-15 01:37:36 +11:00
default:
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad NSEC3 TypeBitMap", l}
2011-12-15 01:37:36 +11:00
}
l = <-c
}
return rr, nil
}
2011-12-14 21:56:12 +11:00
2012-02-12 08:47:22 +11:00
func setNSEC3PARAM(h RR_Header, c chan lex, f string) (RR, *ParseError) {
rr := new(RR_NSEC3PARAM)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad NSEC3PARAM Hash", l}
} else {
rr.Hash = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad NSEC3PARAM Flags", l}
} else {
rr.Flags = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad NSEC3PARAM Iterations", l}
} else {
rr.Iterations = uint16(i)
}
<-c
l = <-c
rr.SaltLength = uint8(len(l.token))
2012-02-12 22:28:08 +11:00
rr.Salt = l.token
return rr, nil
2011-12-16 21:33:30 +11:00
}
func setSSHFP(h RR_Header, c chan lex, f string) (RR, *ParseError) {
2011-12-16 21:20:17 +11:00
rr := new(RR_SSHFP)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad SSHFP Algorithm", l}
2011-12-16 21:20:17 +11:00
} else {
rr.Algorithm = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad SSHFP Type", l}
2011-12-16 21:20:17 +11:00
} else {
rr.Type = uint8(i)
}
<-c // _BLANK
l = <-c
rr.FingerPrint = l.token
return rr, nil
}
func setDNSKEY(h RR_Header, c chan lex, f string) (RR, *ParseError) {
2011-12-16 08:44:09 +11:00
rr := new(RR_DNSKEY)
rr.Hdr = h
2011-12-15 23:15:31 +11:00
2011-12-16 08:44:09 +11:00
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad DNSKEY Flags", l}
2011-12-16 08:44:09 +11:00
} else {
rr.Flags = uint16(i)
}
2011-12-15 23:15:31 +11:00
<-c // _BLANK
l = <-c // _STRING
2011-12-16 08:44:09 +11:00
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad DNSKEY Protocol", l}
2011-12-16 08:44:09 +11:00
} else {
rr.Protocol = uint8(i)
}
2011-12-15 23:15:31 +11:00
<-c // _BLANK
l = <-c // _STRING
2011-12-16 08:44:09 +11:00
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad DNSKEY Algorithm", l}
2011-12-16 08:44:09 +11:00
} else {
rr.Algorithm = uint8(i)
}
l = <-c
var s string
2011-12-15 23:15:31 +11:00
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad DNSKEY PublicKey", l}
2011-12-15 23:15:31 +11:00
}
l = <-c
}
rr.PublicKey = s
return rr, nil
}
func setDS(h RR_Header, c chan lex, f string) (RR, *ParseError) {
2011-12-16 21:30:29 +11:00
rr := new(RR_DS)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad DS KeyTag", l}
2011-12-16 21:30:29 +11:00
} else {
rr.KeyTag = uint16(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
if i, ok := Str_alg[strings.ToUpper(l.token)]; !ok {
return nil, &ParseError{f, "bad DS Algorithm", l}
} else {
rr.Algorithm = i
}
2011-12-16 21:30:29 +11:00
} else {
rr.Algorithm = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad DS DigestType", l}
2011-12-16 21:30:29 +11:00
} else {
rr.DigestType = uint8(i)
}
// There can be spaces here...
l = <-c
2011-12-17 00:48:30 +11:00
s := ""
2011-12-16 21:30:29 +11:00
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
2012-02-12 03:28:36 +11:00
return nil, &ParseError{f, "bad DS Digest", l}
2011-12-16 21:30:29 +11:00
}
l = <-c
}
rr.Digest = s
return rr, nil
}
2012-02-19 07:19:56 +11:00
func setDLV(h RR_Header, c chan lex, f string) (RR, *ParseError) {
rr := new(RR_DLV)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad DLV KeyTag", l}
} else {
rr.KeyTag = uint16(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
if i, ok := Str_alg[strings.ToUpper(l.token)]; !ok {
return nil, &ParseError{f, "bad DLV Algorithm", l}
} else {
rr.Algorithm = i
}
2012-02-19 07:19:56 +11:00
} else {
rr.Algorithm = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad DLV DigestType", l}
} else {
rr.DigestType = uint8(i)
}
// There can be spaces here...
l = <-c
s := ""
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad DLV Digest", l}
}
l = <-c
}
rr.Digest = s
return rr, nil
}
2012-02-19 07:22:58 +11:00
func setTA(h RR_Header, c chan lex, f string) (RR, *ParseError) {
rr := new(RR_TA)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad TA KeyTag", l}
} else {
rr.KeyTag = uint16(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
if i, ok := Str_alg[strings.ToUpper(l.token)]; !ok {
return nil, &ParseError{f, "bad TA Algorithm", l}
} else {
rr.Algorithm = i
}
2012-02-19 07:22:58 +11:00
} else {
rr.Algorithm = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad TA DigestType", l}
} else {
rr.DigestType = uint8(i)
}
// There can be spaces here...
l = <-c
s := ""
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad TA Digest", l}
}
l = <-c
}
rr.Digest = s
return rr, nil
}
2012-02-19 05:08:39 +11:00
func setTLSA(h RR_Header, c chan lex, f string) (RR, *ParseError) {
rr := new(RR_TLSA)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad TLSA Usage", l}
} else {
2012-02-19 05:26:11 +11:00
rr.Usage = uint8(i)
2012-02-19 05:08:39 +11:00
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad TLSA Selector", l}
} else {
rr.Selector = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad TLSA MatchingType", l}
} else {
2012-02-19 05:26:11 +11:00
rr.MatchingType = uint8(i)
2012-02-19 05:08:39 +11:00
}
// There can be spaces here...
l = <-c
s := ""
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad TLSA Certificate", l}
}
l = <-c
}
rr.Certificate = s
return rr, nil
}
func setRFC3597(h RR_Header, c chan lex, f string) (RR, *ParseError) {
rr := new(RR_RFC3597)
rr.Hdr = h
l := <-c
if l.token != "\\#" {
2012-02-16 09:04:46 +11:00
return nil, &ParseError{f, "unkown RR type", l}
}
<-c // _BLANK
l = <-c
rdlength, e := strconv.Atoi(l.token)
if e != nil {
return nil, &ParseError{f, "bad RFC3597 Rdata", l}
}
// There can be spaces here...
l = <-c
s := ""
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad RFC3597 Rdata", l}
}
l = <-c
}
if rdlength*2 != len(s) {
return nil, &ParseError{f, "bad RFC3597 Rdata", l}
}
rr.Rdata = s
return rr, nil
}
2012-02-19 04:59:19 +11:00
func setSPF(h RR_Header, c chan lex, f string) (RR, *ParseError) {
rr := new(RR_SPF)
rr.Hdr = h
// Get the remaining data until we see a NEWLINE
quote := false
l := <-c
var s []string
switch l.value == _QUOTE {
case true: // A number of quoted string
s = make([]string, 0)
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s = append(s, l.token)
case _BLANK:
if quote {
// _BLANK can only be seen in between txt parts.
return nil, &ParseError{f, "bad SPF Txt", l}
}
case _QUOTE:
quote = !quote
default:
return nil, &ParseError{f, "bad SPF Txt", l}
}
l = <-c
}
if quote {
return nil, &ParseError{f, "bad SPF Txt", l}
}
case false: // Unquoted text record
s = make([]string, 1)
for l.value != _NEWLINE && l.value != _EOF {
s[0] += l.token
l = <-c
}
}
rr.Txt = s
return rr, nil
}
func setTXT(h RR_Header, c chan lex, f string) (RR, *ParseError) {
2011-12-15 01:37:36 +11:00
rr := new(RR_TXT)
rr.Hdr = h
2011-12-15 00:02:55 +11:00
2011-12-15 01:37:36 +11:00
// Get the remaining data until we see a NEWLINE
quote := false
2011-12-15 01:37:36 +11:00
l := <-c
var s []string
2012-02-13 08:24:18 +11:00
switch l.value == _QUOTE {
case true: // A number of quoted string
s = make([]string, 0)
2012-02-13 08:24:18 +11:00
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s = append(s, l.token)
case _BLANK:
if quote {
// _BLANK can only be seen in between txt parts.
return nil, &ParseError{f, "bad TXT Txt", l}
}
case _QUOTE:
quote = !quote
default:
return nil, &ParseError{f, "bad TXT Txt", l}
}
2012-02-13 08:24:18 +11:00
l = <-c
}
if quote {
2012-02-16 09:04:46 +11:00
return nil, &ParseError{f, "bad TXT Txt", l}
2012-02-13 08:24:18 +11:00
}
case false: // Unquoted text record
s = make([]string, 1)
2012-02-13 08:24:18 +11:00
for l.value != _NEWLINE && l.value != _EOF {
s[0] += l.token
l = <-c
2011-12-15 01:37:36 +11:00
}
}
rr.Txt = s
return rr, nil
2011-12-15 00:02:55 +11:00
}
2012-02-19 07:49:02 +11:00
func setURI(h RR_Header, c chan lex, f string) (RR, *ParseError) {
rr := new(RR_URI)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad URI Priority", l}
} else {
rr.Priority = uint16(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad URI Weight", l}
} else {
rr.Weight = uint16(i)
}
// _BLANK?
// Get the remaining data until we see a NEWLINE
quote := false
l = <-c
var s string
switch l.value == _QUOTE {
case true:
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
if quote {
// _BLANK can only be seen in between txt parts.
return nil, &ParseError{f, "bad URI Target", l}
}
case _QUOTE:
quote = !quote
default:
return nil, &ParseError{f, "bad URI Target", l}
}
l = <-c
}
if quote {
return nil, &ParseError{f, "bad URI Target", l}
}
case false: // Unquoted
return nil, &ParseError{f, "bad URI Target", l}
}
rr.Target = s
return rr, nil
}
func setIPSECKEY(h RR_Header, c chan lex, o, f string) (RR, *ParseError) {
rr := new(RR_IPSECKEY)
rr.Hdr = h
l := <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad IPSECKEY Precedence", l}
} else {
rr.Precedence = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad IPSECKEY GatewayType", l}
} else {
rr.GatewayType = uint8(i)
}
<-c // _BLANK
l = <-c
if i, e := strconv.Atoi(l.token); e != nil {
return nil, &ParseError{f, "bad IPSECKEY Algorithm", l}
} else {
rr.Algorithm = uint8(i)
}
<-c
l = <-c
rr.Gateway = l.token
l = <-c
var s string
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad IPSECKEY PublicKey", l}
}
l = <-c
}
rr.PublicKey = s
return rr, nil
}
2012-02-19 07:37:03 +11:00
func setDHCID(h RR_Header, c chan lex, f string) (RR, *ParseError) {
2012-02-19 07:49:02 +11:00
// awesome record to parse!
2012-02-19 07:37:03 +11:00
rr := new(RR_DHCID)
rr.Hdr = h
2012-02-19 07:49:02 +11:00
l := <-c // _STRING
2012-02-19 07:37:03 +11:00
var s string
for l.value != _NEWLINE && l.value != _EOF {
switch l.value {
case _STRING:
s += l.token
case _BLANK:
// Ok
default:
return nil, &ParseError{f, "bad DHCID Digest", l}
}
l = <-c
}
rr.Digest = s
return rr, nil
}