scan.go 24 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007
  1. package dns
  2. import (
  3. "fmt"
  4. "io"
  5. "os"
  6. "path/filepath"
  7. "strconv"
  8. "strings"
  9. )
  10. const maxTok = 2048 // Largest token we can return.
  11. const maxUint16 = 1<<16 - 1
  12. // Tokinize a RFC 1035 zone file. The tokenizer will normalize it:
  13. // * Add ownernames if they are left blank;
  14. // * Suppress sequences of spaces;
  15. // * Make each RR fit on one line (_NEWLINE is send as last)
  16. // * Handle comments: ;
  17. // * Handle braces - anywhere.
  18. const (
  19. // Zonefile
  20. zEOF = iota
  21. zString
  22. zBlank
  23. zQuote
  24. zNewline
  25. zRrtpe
  26. zOwner
  27. zClass
  28. zDirOrigin // $ORIGIN
  29. zDirTTL // $TTL
  30. zDirInclude // $INCLUDE
  31. zDirGenerate // $GENERATE
  32. // Privatekey file
  33. zValue
  34. zKey
  35. zExpectOwnerDir // Ownername
  36. zExpectOwnerBl // Whitespace after the ownername
  37. zExpectAny // Expect rrtype, ttl or class
  38. zExpectAnyNoClass // Expect rrtype or ttl
  39. zExpectAnyNoClassBl // The whitespace after _EXPECT_ANY_NOCLASS
  40. zExpectAnyNoTTL // Expect rrtype or class
  41. zExpectAnyNoTTLBl // Whitespace after _EXPECT_ANY_NOTTL
  42. zExpectRrtype // Expect rrtype
  43. zExpectRrtypeBl // Whitespace BEFORE rrtype
  44. zExpectRdata // The first element of the rdata
  45. zExpectDirTTLBl // Space after directive $TTL
  46. zExpectDirTTL // Directive $TTL
  47. zExpectDirOriginBl // Space after directive $ORIGIN
  48. zExpectDirOrigin // Directive $ORIGIN
  49. zExpectDirIncludeBl // Space after directive $INCLUDE
  50. zExpectDirInclude // Directive $INCLUDE
  51. zExpectDirGenerate // Directive $GENERATE
  52. zExpectDirGenerateBl // Space after directive $GENERATE
  53. )
  54. // ParseError is a parsing error. It contains the parse error and the location in the io.Reader
  55. // where the error occurred.
  56. type ParseError struct {
  57. file string
  58. err string
  59. lex lex
  60. }
  61. func (e *ParseError) Error() (s string) {
  62. if e.file != "" {
  63. s = e.file + ": "
  64. }
  65. s += "dns: " + e.err + ": " + strconv.QuoteToASCII(e.lex.token) + " at line: " +
  66. strconv.Itoa(e.lex.line) + ":" + strconv.Itoa(e.lex.column)
  67. return
  68. }
  69. type lex struct {
  70. token string // text of the token
  71. tokenUpper string // uppercase text of the token
  72. length int // length of the token
  73. err bool // when true, token text has lexer error
  74. value uint8 // value: zString, _BLANK, etc.
  75. line int // line in the file
  76. column int // column in the file
  77. torc uint16 // type or class as parsed in the lexer, we only need to look this up in the grammar
  78. comment string // any comment text seen
  79. }
  80. // Token holds the token that are returned when a zone file is parsed.
  81. type Token struct {
  82. // The scanned resource record when error is not nil.
  83. RR
  84. // When an error occurred, this has the error specifics.
  85. Error *ParseError
  86. // A potential comment positioned after the RR and on the same line.
  87. Comment string
  88. }
  89. // ttlState describes the state necessary to fill in an omitted RR TTL
  90. type ttlState struct {
  91. ttl uint32 // ttl is the current default TTL
  92. isByDirective bool // isByDirective indicates whether ttl was set by a $TTL directive
  93. }
  94. // NewRR reads the RR contained in the string s. Only the first RR is
  95. // returned. If s contains no RR, return nil with no error. The class
  96. // defaults to IN and TTL defaults to 3600. The full zone file syntax
  97. // like $TTL, $ORIGIN, etc. is supported. All fields of the returned
  98. // RR are set, except RR.Header().Rdlength which is set to 0.
  99. func NewRR(s string) (RR, error) {
  100. if len(s) > 0 && s[len(s)-1] != '\n' { // We need a closing newline
  101. return ReadRR(strings.NewReader(s+"\n"), "")
  102. }
  103. return ReadRR(strings.NewReader(s), "")
  104. }
  105. // ReadRR reads the RR contained in q.
  106. // See NewRR for more documentation.
  107. func ReadRR(q io.Reader, filename string) (RR, error) {
  108. defttl := &ttlState{defaultTtl, false}
  109. r := <-parseZoneHelper(q, ".", filename, defttl, 1)
  110. if r == nil {
  111. return nil, nil
  112. }
  113. if r.Error != nil {
  114. return nil, r.Error
  115. }
  116. return r.RR, nil
  117. }
  118. // ParseZone reads a RFC 1035 style zonefile from r. It returns *Tokens on the
  119. // returned channel, each consisting of either a parsed RR and optional comment
  120. // or a nil RR and an error. The string file is only used
  121. // in error reporting. The string origin is used as the initial origin, as
  122. // if the file would start with an $ORIGIN directive.
  123. // The directives $INCLUDE, $ORIGIN, $TTL and $GENERATE are supported.
  124. // The channel t is closed by ParseZone when the end of r is reached.
  125. //
  126. // Basic usage pattern when reading from a string (z) containing the
  127. // zone data:
  128. //
  129. // for x := range dns.ParseZone(strings.NewReader(z), "", "") {
  130. // if x.Error != nil {
  131. // // log.Println(x.Error)
  132. // } else {
  133. // // Do something with x.RR
  134. // }
  135. // }
  136. //
  137. // Comments specified after an RR (and on the same line!) are returned too:
  138. //
  139. // foo. IN A 10.0.0.1 ; this is a comment
  140. //
  141. // The text "; this is comment" is returned in Token.Comment. Comments inside the
  142. // RR are discarded. Comments on a line by themselves are discarded too.
  143. func ParseZone(r io.Reader, origin, file string) chan *Token {
  144. return parseZoneHelper(r, origin, file, nil, 10000)
  145. }
  146. func parseZoneHelper(r io.Reader, origin, file string, defttl *ttlState, chansize int) chan *Token {
  147. t := make(chan *Token, chansize)
  148. go parseZone(r, origin, file, defttl, t, 0)
  149. return t
  150. }
  151. func parseZone(r io.Reader, origin, f string, defttl *ttlState, t chan *Token, include int) {
  152. defer func() {
  153. if include == 0 {
  154. close(t)
  155. }
  156. }()
  157. s, cancel := scanInit(r)
  158. c := make(chan lex)
  159. // Start the lexer
  160. go zlexer(s, c)
  161. defer func() {
  162. cancel()
  163. // zlexer can send up to three tokens, the next one and possibly 2 remainders.
  164. // Do a non-blocking read.
  165. _, ok := <-c
  166. _, ok = <-c
  167. _, ok = <-c
  168. if !ok {
  169. // too bad
  170. }
  171. }()
  172. // 6 possible beginnings of a line, _ is a space
  173. // 0. zRRTYPE -> all omitted until the rrtype
  174. // 1. zOwner _ zRrtype -> class/ttl omitted
  175. // 2. zOwner _ zString _ zRrtype -> class omitted
  176. // 3. zOwner _ zString _ zClass _ zRrtype -> ttl/class
  177. // 4. zOwner _ zClass _ zRrtype -> ttl omitted
  178. // 5. zOwner _ zClass _ zString _ zRrtype -> class/ttl (reversed)
  179. // After detecting these, we know the zRrtype so we can jump to functions
  180. // handling the rdata for each of these types.
  181. if origin != "" {
  182. origin = Fqdn(origin)
  183. if _, ok := IsDomainName(origin); !ok {
  184. t <- &Token{Error: &ParseError{f, "bad initial origin name", lex{}}}
  185. return
  186. }
  187. }
  188. st := zExpectOwnerDir // initial state
  189. var h RR_Header
  190. var prevName string
  191. for l := range c {
  192. // Lexer spotted an error already
  193. if l.err == true {
  194. t <- &Token{Error: &ParseError{f, l.token, l}}
  195. return
  196. }
  197. switch st {
  198. case zExpectOwnerDir:
  199. // We can also expect a directive, like $TTL or $ORIGIN
  200. if defttl != nil {
  201. h.Ttl = defttl.ttl
  202. }
  203. h.Class = ClassINET
  204. switch l.value {
  205. case zNewline:
  206. st = zExpectOwnerDir
  207. case zOwner:
  208. h.Name = l.token
  209. name, ok := toAbsoluteName(l.token, origin)
  210. if !ok {
  211. t <- &Token{Error: &ParseError{f, "bad owner name", l}}
  212. return
  213. }
  214. h.Name = name
  215. prevName = h.Name
  216. st = zExpectOwnerBl
  217. case zDirTTL:
  218. st = zExpectDirTTLBl
  219. case zDirOrigin:
  220. st = zExpectDirOriginBl
  221. case zDirInclude:
  222. st = zExpectDirIncludeBl
  223. case zDirGenerate:
  224. st = zExpectDirGenerateBl
  225. case zRrtpe:
  226. h.Name = prevName
  227. h.Rrtype = l.torc
  228. st = zExpectRdata
  229. case zClass:
  230. h.Name = prevName
  231. h.Class = l.torc
  232. st = zExpectAnyNoClassBl
  233. case zBlank:
  234. // Discard, can happen when there is nothing on the
  235. // line except the RR type
  236. case zString:
  237. ttl, ok := stringToTTL(l.token)
  238. if !ok {
  239. t <- &Token{Error: &ParseError{f, "not a TTL", l}}
  240. return
  241. }
  242. h.Ttl = ttl
  243. if defttl == nil || !defttl.isByDirective {
  244. defttl = &ttlState{ttl, false}
  245. }
  246. st = zExpectAnyNoTTLBl
  247. default:
  248. t <- &Token{Error: &ParseError{f, "syntax error at beginning", l}}
  249. return
  250. }
  251. case zExpectDirIncludeBl:
  252. if l.value != zBlank {
  253. t <- &Token{Error: &ParseError{f, "no blank after $INCLUDE-directive", l}}
  254. return
  255. }
  256. st = zExpectDirInclude
  257. case zExpectDirInclude:
  258. if l.value != zString {
  259. t <- &Token{Error: &ParseError{f, "expecting $INCLUDE value, not this...", l}}
  260. return
  261. }
  262. neworigin := origin // There may be optionally a new origin set after the filename, if not use current one
  263. switch l := <-c; l.value {
  264. case zBlank:
  265. l := <-c
  266. if l.value == zString {
  267. name, ok := toAbsoluteName(l.token, origin)
  268. if !ok {
  269. t <- &Token{Error: &ParseError{f, "bad origin name", l}}
  270. return
  271. }
  272. neworigin = name
  273. }
  274. case zNewline, zEOF:
  275. // Ok
  276. default:
  277. t <- &Token{Error: &ParseError{f, "garbage after $INCLUDE", l}}
  278. return
  279. }
  280. // Start with the new file
  281. includePath := l.token
  282. if !filepath.IsAbs(includePath) {
  283. includePath = filepath.Join(filepath.Dir(f), includePath)
  284. }
  285. r1, e1 := os.Open(includePath)
  286. if e1 != nil {
  287. msg := fmt.Sprintf("failed to open `%s'", l.token)
  288. if !filepath.IsAbs(l.token) {
  289. msg += fmt.Sprintf(" as `%s'", includePath)
  290. }
  291. t <- &Token{Error: &ParseError{f, msg, l}}
  292. return
  293. }
  294. if include+1 > 7 {
  295. t <- &Token{Error: &ParseError{f, "too deeply nested $INCLUDE", l}}
  296. return
  297. }
  298. parseZone(r1, neworigin, includePath, defttl, t, include+1)
  299. st = zExpectOwnerDir
  300. case zExpectDirTTLBl:
  301. if l.value != zBlank {
  302. t <- &Token{Error: &ParseError{f, "no blank after $TTL-directive", l}}
  303. return
  304. }
  305. st = zExpectDirTTL
  306. case zExpectDirTTL:
  307. if l.value != zString {
  308. t <- &Token{Error: &ParseError{f, "expecting $TTL value, not this...", l}}
  309. return
  310. }
  311. if e, _ := slurpRemainder(c, f); e != nil {
  312. t <- &Token{Error: e}
  313. return
  314. }
  315. ttl, ok := stringToTTL(l.token)
  316. if !ok {
  317. t <- &Token{Error: &ParseError{f, "expecting $TTL value, not this...", l}}
  318. return
  319. }
  320. defttl = &ttlState{ttl, true}
  321. st = zExpectOwnerDir
  322. case zExpectDirOriginBl:
  323. if l.value != zBlank {
  324. t <- &Token{Error: &ParseError{f, "no blank after $ORIGIN-directive", l}}
  325. return
  326. }
  327. st = zExpectDirOrigin
  328. case zExpectDirOrigin:
  329. if l.value != zString {
  330. t <- &Token{Error: &ParseError{f, "expecting $ORIGIN value, not this...", l}}
  331. return
  332. }
  333. if e, _ := slurpRemainder(c, f); e != nil {
  334. t <- &Token{Error: e}
  335. }
  336. name, ok := toAbsoluteName(l.token, origin)
  337. if !ok {
  338. t <- &Token{Error: &ParseError{f, "bad origin name", l}}
  339. return
  340. }
  341. origin = name
  342. st = zExpectOwnerDir
  343. case zExpectDirGenerateBl:
  344. if l.value != zBlank {
  345. t <- &Token{Error: &ParseError{f, "no blank after $GENERATE-directive", l}}
  346. return
  347. }
  348. st = zExpectDirGenerate
  349. case zExpectDirGenerate:
  350. if l.value != zString {
  351. t <- &Token{Error: &ParseError{f, "expecting $GENERATE value, not this...", l}}
  352. return
  353. }
  354. if errMsg := generate(l, c, t, origin); errMsg != "" {
  355. t <- &Token{Error: &ParseError{f, errMsg, l}}
  356. return
  357. }
  358. st = zExpectOwnerDir
  359. case zExpectOwnerBl:
  360. if l.value != zBlank {
  361. t <- &Token{Error: &ParseError{f, "no blank after owner", l}}
  362. return
  363. }
  364. st = zExpectAny
  365. case zExpectAny:
  366. switch l.value {
  367. case zRrtpe:
  368. if defttl == nil {
  369. t <- &Token{Error: &ParseError{f, "missing TTL with no previous value", l}}
  370. return
  371. }
  372. h.Rrtype = l.torc
  373. st = zExpectRdata
  374. case zClass:
  375. h.Class = l.torc
  376. st = zExpectAnyNoClassBl
  377. case zString:
  378. ttl, ok := stringToTTL(l.token)
  379. if !ok {
  380. t <- &Token{Error: &ParseError{f, "not a TTL", l}}
  381. return
  382. }
  383. h.Ttl = ttl
  384. if defttl == nil || !defttl.isByDirective {
  385. defttl = &ttlState{ttl, false}
  386. }
  387. st = zExpectAnyNoTTLBl
  388. default:
  389. t <- &Token{Error: &ParseError{f, "expecting RR type, TTL or class, not this...", l}}
  390. return
  391. }
  392. case zExpectAnyNoClassBl:
  393. if l.value != zBlank {
  394. t <- &Token{Error: &ParseError{f, "no blank before class", l}}
  395. return
  396. }
  397. st = zExpectAnyNoClass
  398. case zExpectAnyNoTTLBl:
  399. if l.value != zBlank {
  400. t <- &Token{Error: &ParseError{f, "no blank before TTL", l}}
  401. return
  402. }
  403. st = zExpectAnyNoTTL
  404. case zExpectAnyNoTTL:
  405. switch l.value {
  406. case zClass:
  407. h.Class = l.torc
  408. st = zExpectRrtypeBl
  409. case zRrtpe:
  410. h.Rrtype = l.torc
  411. st = zExpectRdata
  412. default:
  413. t <- &Token{Error: &ParseError{f, "expecting RR type or class, not this...", l}}
  414. return
  415. }
  416. case zExpectAnyNoClass:
  417. switch l.value {
  418. case zString:
  419. ttl, ok := stringToTTL(l.token)
  420. if !ok {
  421. t <- &Token{Error: &ParseError{f, "not a TTL", l}}
  422. return
  423. }
  424. h.Ttl = ttl
  425. if defttl == nil || !defttl.isByDirective {
  426. defttl = &ttlState{ttl, false}
  427. }
  428. st = zExpectRrtypeBl
  429. case zRrtpe:
  430. h.Rrtype = l.torc
  431. st = zExpectRdata
  432. default:
  433. t <- &Token{Error: &ParseError{f, "expecting RR type or TTL, not this...", l}}
  434. return
  435. }
  436. case zExpectRrtypeBl:
  437. if l.value != zBlank {
  438. t <- &Token{Error: &ParseError{f, "no blank before RR type", l}}
  439. return
  440. }
  441. st = zExpectRrtype
  442. case zExpectRrtype:
  443. if l.value != zRrtpe {
  444. t <- &Token{Error: &ParseError{f, "unknown RR type", l}}
  445. return
  446. }
  447. h.Rrtype = l.torc
  448. st = zExpectRdata
  449. case zExpectRdata:
  450. r, e, c1 := setRR(h, c, origin, f)
  451. if e != nil {
  452. // If e.lex is nil than we have encounter a unknown RR type
  453. // in that case we substitute our current lex token
  454. if e.lex.token == "" && e.lex.value == 0 {
  455. e.lex = l // Uh, dirty
  456. }
  457. t <- &Token{Error: e}
  458. return
  459. }
  460. t <- &Token{RR: r, Comment: c1}
  461. st = zExpectOwnerDir
  462. }
  463. }
  464. // If we get here, we and the h.Rrtype is still zero, we haven't parsed anything, this
  465. // is not an error, because an empty zone file is still a zone file.
  466. }
  467. // zlexer scans the sourcefile and returns tokens on the channel c.
  468. func zlexer(s *scan, c chan lex) {
  469. var l lex
  470. str := make([]byte, maxTok) // Should be enough for any token
  471. stri := 0 // Offset in str (0 means empty)
  472. com := make([]byte, maxTok) // Hold comment text
  473. comi := 0
  474. quote := false
  475. escape := false
  476. space := false
  477. commt := false
  478. rrtype := false
  479. owner := true
  480. brace := 0
  481. x, err := s.tokenText()
  482. defer close(c)
  483. for err == nil {
  484. l.column = s.position.Column
  485. l.line = s.position.Line
  486. if stri >= maxTok {
  487. l.token = "token length insufficient for parsing"
  488. l.err = true
  489. c <- l
  490. return
  491. }
  492. if comi >= maxTok {
  493. l.token = "comment length insufficient for parsing"
  494. l.err = true
  495. c <- l
  496. return
  497. }
  498. switch x {
  499. case ' ', '\t':
  500. if escape {
  501. escape = false
  502. str[stri] = x
  503. stri++
  504. break
  505. }
  506. if quote {
  507. // Inside quotes this is legal
  508. str[stri] = x
  509. stri++
  510. break
  511. }
  512. if commt {
  513. com[comi] = x
  514. comi++
  515. break
  516. }
  517. if stri == 0 {
  518. // Space directly in the beginning, handled in the grammar
  519. } else if owner {
  520. // If we have a string and its the first, make it an owner
  521. l.value = zOwner
  522. l.token = string(str[:stri])
  523. l.tokenUpper = strings.ToUpper(l.token)
  524. l.length = stri
  525. // escape $... start with a \ not a $, so this will work
  526. switch l.tokenUpper {
  527. case "$TTL":
  528. l.value = zDirTTL
  529. case "$ORIGIN":
  530. l.value = zDirOrigin
  531. case "$INCLUDE":
  532. l.value = zDirInclude
  533. case "$GENERATE":
  534. l.value = zDirGenerate
  535. }
  536. c <- l
  537. } else {
  538. l.value = zString
  539. l.token = string(str[:stri])
  540. l.tokenUpper = strings.ToUpper(l.token)
  541. l.length = stri
  542. if !rrtype {
  543. if t, ok := StringToType[l.tokenUpper]; ok {
  544. l.value = zRrtpe
  545. l.torc = t
  546. rrtype = true
  547. } else {
  548. if strings.HasPrefix(l.tokenUpper, "TYPE") {
  549. t, ok := typeToInt(l.token)
  550. if !ok {
  551. l.token = "unknown RR type"
  552. l.err = true
  553. c <- l
  554. return
  555. }
  556. l.value = zRrtpe
  557. rrtype = true
  558. l.torc = t
  559. }
  560. }
  561. if t, ok := StringToClass[l.tokenUpper]; ok {
  562. l.value = zClass
  563. l.torc = t
  564. } else {
  565. if strings.HasPrefix(l.tokenUpper, "CLASS") {
  566. t, ok := classToInt(l.token)
  567. if !ok {
  568. l.token = "unknown class"
  569. l.err = true
  570. c <- l
  571. return
  572. }
  573. l.value = zClass
  574. l.torc = t
  575. }
  576. }
  577. }
  578. c <- l
  579. }
  580. stri = 0
  581. if !space && !commt {
  582. l.value = zBlank
  583. l.token = " "
  584. l.length = 1
  585. c <- l
  586. }
  587. owner = false
  588. space = true
  589. case ';':
  590. if escape {
  591. escape = false
  592. str[stri] = x
  593. stri++
  594. break
  595. }
  596. if quote {
  597. // Inside quotes this is legal
  598. str[stri] = x
  599. stri++
  600. break
  601. }
  602. if stri > 0 {
  603. l.value = zString
  604. l.token = string(str[:stri])
  605. l.tokenUpper = strings.ToUpper(l.token)
  606. l.length = stri
  607. c <- l
  608. stri = 0
  609. }
  610. commt = true
  611. com[comi] = ';'
  612. comi++
  613. case '\r':
  614. escape = false
  615. if quote {
  616. str[stri] = x
  617. stri++
  618. break
  619. }
  620. // discard if outside of quotes
  621. case '\n':
  622. escape = false
  623. // Escaped newline
  624. if quote {
  625. str[stri] = x
  626. stri++
  627. break
  628. }
  629. // inside quotes this is legal
  630. if commt {
  631. // Reset a comment
  632. commt = false
  633. rrtype = false
  634. stri = 0
  635. // If not in a brace this ends the comment AND the RR
  636. if brace == 0 {
  637. owner = true
  638. owner = true
  639. l.value = zNewline
  640. l.token = "\n"
  641. l.tokenUpper = l.token
  642. l.length = 1
  643. l.comment = string(com[:comi])
  644. c <- l
  645. l.comment = ""
  646. comi = 0
  647. break
  648. }
  649. com[comi] = ' ' // convert newline to space
  650. comi++
  651. break
  652. }
  653. if brace == 0 {
  654. // If there is previous text, we should output it here
  655. if stri != 0 {
  656. l.value = zString
  657. l.token = string(str[:stri])
  658. l.tokenUpper = strings.ToUpper(l.token)
  659. l.length = stri
  660. if !rrtype {
  661. if t, ok := StringToType[l.tokenUpper]; ok {
  662. l.value = zRrtpe
  663. l.torc = t
  664. rrtype = true
  665. }
  666. }
  667. c <- l
  668. }
  669. l.value = zNewline
  670. l.token = "\n"
  671. l.tokenUpper = l.token
  672. l.length = 1
  673. c <- l
  674. stri = 0
  675. commt = false
  676. rrtype = false
  677. owner = true
  678. comi = 0
  679. }
  680. case '\\':
  681. // comments do not get escaped chars, everything is copied
  682. if commt {
  683. com[comi] = x
  684. comi++
  685. break
  686. }
  687. // something already escaped must be in string
  688. if escape {
  689. str[stri] = x
  690. stri++
  691. escape = false
  692. break
  693. }
  694. // something escaped outside of string gets added to string
  695. str[stri] = x
  696. stri++
  697. escape = true
  698. case '"':
  699. if commt {
  700. com[comi] = x
  701. comi++
  702. break
  703. }
  704. if escape {
  705. str[stri] = x
  706. stri++
  707. escape = false
  708. break
  709. }
  710. space = false
  711. // send previous gathered text and the quote
  712. if stri != 0 {
  713. l.value = zString
  714. l.token = string(str[:stri])
  715. l.tokenUpper = strings.ToUpper(l.token)
  716. l.length = stri
  717. c <- l
  718. stri = 0
  719. }
  720. // send quote itself as separate token
  721. l.value = zQuote
  722. l.token = "\""
  723. l.tokenUpper = l.token
  724. l.length = 1
  725. c <- l
  726. quote = !quote
  727. case '(', ')':
  728. if commt {
  729. com[comi] = x
  730. comi++
  731. break
  732. }
  733. if escape {
  734. str[stri] = x
  735. stri++
  736. escape = false
  737. break
  738. }
  739. if quote {
  740. str[stri] = x
  741. stri++
  742. break
  743. }
  744. switch x {
  745. case ')':
  746. brace--
  747. if brace < 0 {
  748. l.token = "extra closing brace"
  749. l.tokenUpper = l.token
  750. l.err = true
  751. c <- l
  752. return
  753. }
  754. case '(':
  755. brace++
  756. }
  757. default:
  758. escape = false
  759. if commt {
  760. com[comi] = x
  761. comi++
  762. break
  763. }
  764. str[stri] = x
  765. stri++
  766. space = false
  767. }
  768. x, err = s.tokenText()
  769. }
  770. if stri > 0 {
  771. // Send remainder
  772. l.token = string(str[:stri])
  773. l.tokenUpper = strings.ToUpper(l.token)
  774. l.length = stri
  775. l.value = zString
  776. c <- l
  777. }
  778. if brace != 0 {
  779. l.token = "unbalanced brace"
  780. l.tokenUpper = l.token
  781. l.err = true
  782. c <- l
  783. }
  784. }
  785. // Extract the class number from CLASSxx
  786. func classToInt(token string) (uint16, bool) {
  787. offset := 5
  788. if len(token) < offset+1 {
  789. return 0, false
  790. }
  791. class, err := strconv.ParseUint(token[offset:], 10, 16)
  792. if err != nil {
  793. return 0, false
  794. }
  795. return uint16(class), true
  796. }
  797. // Extract the rr number from TYPExxx
  798. func typeToInt(token string) (uint16, bool) {
  799. offset := 4
  800. if len(token) < offset+1 {
  801. return 0, false
  802. }
  803. typ, err := strconv.ParseUint(token[offset:], 10, 16)
  804. if err != nil {
  805. return 0, false
  806. }
  807. return uint16(typ), true
  808. }
  809. // stringToTTL parses things like 2w, 2m, etc, and returns the time in seconds.
  810. func stringToTTL(token string) (uint32, bool) {
  811. s := uint32(0)
  812. i := uint32(0)
  813. for _, c := range token {
  814. switch c {
  815. case 's', 'S':
  816. s += i
  817. i = 0
  818. case 'm', 'M':
  819. s += i * 60
  820. i = 0
  821. case 'h', 'H':
  822. s += i * 60 * 60
  823. i = 0
  824. case 'd', 'D':
  825. s += i * 60 * 60 * 24
  826. i = 0
  827. case 'w', 'W':
  828. s += i * 60 * 60 * 24 * 7
  829. i = 0
  830. case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9':
  831. i *= 10
  832. i += uint32(c) - '0'
  833. default:
  834. return 0, false
  835. }
  836. }
  837. return s + i, true
  838. }
  839. // Parse LOC records' <digits>[.<digits>][mM] into a
  840. // mantissa exponent format. Token should contain the entire
  841. // string (i.e. no spaces allowed)
  842. func stringToCm(token string) (e, m uint8, ok bool) {
  843. if token[len(token)-1] == 'M' || token[len(token)-1] == 'm' {
  844. token = token[0 : len(token)-1]
  845. }
  846. s := strings.SplitN(token, ".", 2)
  847. var meters, cmeters, val int
  848. var err error
  849. switch len(s) {
  850. case 2:
  851. if cmeters, err = strconv.Atoi(s[1]); err != nil {
  852. return
  853. }
  854. fallthrough
  855. case 1:
  856. if meters, err = strconv.Atoi(s[0]); err != nil {
  857. return
  858. }
  859. case 0:
  860. // huh?
  861. return 0, 0, false
  862. }
  863. ok = true
  864. if meters > 0 {
  865. e = 2
  866. val = meters
  867. } else {
  868. e = 0
  869. val = cmeters
  870. }
  871. for val > 10 {
  872. e++
  873. val /= 10
  874. }
  875. if e > 9 {
  876. ok = false
  877. }
  878. m = uint8(val)
  879. return
  880. }
  881. func toAbsoluteName(name, origin string) (absolute string, ok bool) {
  882. // check for an explicit origin reference
  883. if name == "@" {
  884. // require a nonempty origin
  885. if origin == "" {
  886. return "", false
  887. }
  888. return origin, true
  889. }
  890. // require a valid domain name
  891. _, ok = IsDomainName(name)
  892. if !ok || name == "" {
  893. return "", false
  894. }
  895. // check if name is already absolute
  896. if name[len(name)-1] == '.' {
  897. return name, true
  898. }
  899. // require a nonempty origin
  900. if origin == "" {
  901. return "", false
  902. }
  903. return appendOrigin(name, origin), true
  904. }
  905. func appendOrigin(name, origin string) string {
  906. if origin == "." {
  907. return name + origin
  908. }
  909. return name + "." + origin
  910. }
  911. // LOC record helper function
  912. func locCheckNorth(token string, latitude uint32) (uint32, bool) {
  913. switch token {
  914. case "n", "N":
  915. return LOC_EQUATOR + latitude, true
  916. case "s", "S":
  917. return LOC_EQUATOR - latitude, true
  918. }
  919. return latitude, false
  920. }
  921. // LOC record helper function
  922. func locCheckEast(token string, longitude uint32) (uint32, bool) {
  923. switch token {
  924. case "e", "E":
  925. return LOC_EQUATOR + longitude, true
  926. case "w", "W":
  927. return LOC_EQUATOR - longitude, true
  928. }
  929. return longitude, false
  930. }
  931. // "Eat" the rest of the "line". Return potential comments
  932. func slurpRemainder(c chan lex, f string) (*ParseError, string) {
  933. l := <-c
  934. com := ""
  935. switch l.value {
  936. case zBlank:
  937. l = <-c
  938. com = l.comment
  939. if l.value != zNewline && l.value != zEOF {
  940. return &ParseError{f, "garbage after rdata", l}, ""
  941. }
  942. case zNewline:
  943. com = l.comment
  944. case zEOF:
  945. default:
  946. return &ParseError{f, "garbage after rdata", l}, ""
  947. }
  948. return nil, com
  949. }
  950. // Parse a 64 bit-like ipv6 address: "0014:4fff:ff20:ee64"
  951. // Used for NID and L64 record.
  952. func stringToNodeID(l lex) (uint64, *ParseError) {
  953. if len(l.token) < 19 {
  954. return 0, &ParseError{l.token, "bad NID/L64 NodeID/Locator64", l}
  955. }
  956. // There must be three colons at fixes postitions, if not its a parse error
  957. if l.token[4] != ':' && l.token[9] != ':' && l.token[14] != ':' {
  958. return 0, &ParseError{l.token, "bad NID/L64 NodeID/Locator64", l}
  959. }
  960. s := l.token[0:4] + l.token[5:9] + l.token[10:14] + l.token[15:19]
  961. u, err := strconv.ParseUint(s, 16, 64)
  962. if err != nil {
  963. return 0, &ParseError{l.token, "bad NID/L64 NodeID/Locator64", l}
  964. }
  965. return u, nil
  966. }