generate.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. package dns
  2. import (
  3. "bytes"
  4. "errors"
  5. "fmt"
  6. "strconv"
  7. "strings"
  8. )
  9. // Parse the $GENERATE statement as used in BIND9 zones.
  10. // See http://www.zytrax.com/books/dns/ch8/generate.html for instance.
  11. // We are called after '$GENERATE '. After which we expect:
  12. // * the range (12-24/2)
  13. // * lhs (ownername)
  14. // * [[ttl][class]]
  15. // * type
  16. // * rhs (rdata)
  17. // But we are lazy here, only the range is parsed *all* occurrences
  18. // of $ after that are interpreted.
  19. // Any error are returned as a string value, the empty string signals
  20. // "no error".
  21. func generate(l lex, c chan lex, t chan *Token, o string) string {
  22. step := 1
  23. if i := strings.IndexAny(l.token, "/"); i != -1 {
  24. if i+1 == len(l.token) {
  25. return "bad step in $GENERATE range"
  26. }
  27. if s, err := strconv.Atoi(l.token[i+1:]); err == nil {
  28. if s < 0 {
  29. return "bad step in $GENERATE range"
  30. }
  31. step = s
  32. } else {
  33. return "bad step in $GENERATE range"
  34. }
  35. l.token = l.token[:i]
  36. }
  37. sx := strings.SplitN(l.token, "-", 2)
  38. if len(sx) != 2 {
  39. return "bad start-stop in $GENERATE range"
  40. }
  41. start, err := strconv.Atoi(sx[0])
  42. if err != nil {
  43. return "bad start in $GENERATE range"
  44. }
  45. end, err := strconv.Atoi(sx[1])
  46. if err != nil {
  47. return "bad stop in $GENERATE range"
  48. }
  49. if end < 0 || start < 0 || end < start {
  50. return "bad range in $GENERATE range"
  51. }
  52. <-c // _BLANK
  53. // Create a complete new string, which we then parse again.
  54. s := ""
  55. BuildRR:
  56. l = <-c
  57. if l.value != zNewline && l.value != zEOF {
  58. s += l.token
  59. goto BuildRR
  60. }
  61. for i := start; i <= end; i += step {
  62. var (
  63. escape bool
  64. dom bytes.Buffer
  65. mod string
  66. err error
  67. offset int
  68. )
  69. for j := 0; j < len(s); j++ { // No 'range' because we need to jump around
  70. switch s[j] {
  71. case '\\':
  72. if escape {
  73. dom.WriteByte('\\')
  74. escape = false
  75. continue
  76. }
  77. escape = true
  78. case '$':
  79. mod = "%d"
  80. offset = 0
  81. if escape {
  82. dom.WriteByte('$')
  83. escape = false
  84. continue
  85. }
  86. escape = false
  87. if j+1 >= len(s) { // End of the string
  88. dom.WriteString(fmt.Sprintf(mod, i+offset))
  89. continue
  90. } else {
  91. if s[j+1] == '$' {
  92. dom.WriteByte('$')
  93. j++
  94. continue
  95. }
  96. }
  97. // Search for { and }
  98. if s[j+1] == '{' { // Modifier block
  99. sep := strings.Index(s[j+2:], "}")
  100. if sep == -1 {
  101. return "bad modifier in $GENERATE"
  102. }
  103. mod, offset, err = modToPrintf(s[j+2 : j+2+sep])
  104. if err != nil {
  105. return err.Error()
  106. }
  107. j += 2 + sep // Jump to it
  108. }
  109. dom.WriteString(fmt.Sprintf(mod, i+offset))
  110. default:
  111. if escape { // Pretty useless here
  112. escape = false
  113. continue
  114. }
  115. dom.WriteByte(s[j])
  116. }
  117. }
  118. // Re-parse the RR and send it on the current channel t
  119. rx, err := NewRR("$ORIGIN " + o + "\n" + dom.String())
  120. if err != nil {
  121. return err.Error()
  122. }
  123. t <- &Token{RR: rx}
  124. // Its more efficient to first built the rrlist and then parse it in
  125. // one go! But is this a problem?
  126. }
  127. return ""
  128. }
  129. // Convert a $GENERATE modifier 0,0,d to something Printf can deal with.
  130. func modToPrintf(s string) (string, int, error) {
  131. xs := strings.SplitN(s, ",", 3)
  132. if len(xs) != 3 {
  133. return "", 0, errors.New("bad modifier in $GENERATE")
  134. }
  135. // xs[0] is offset, xs[1] is width, xs[2] is base
  136. if xs[2] != "o" && xs[2] != "d" && xs[2] != "x" && xs[2] != "X" {
  137. return "", 0, errors.New("bad base in $GENERATE")
  138. }
  139. offset, err := strconv.Atoi(xs[0])
  140. if err != nil || offset > 255 {
  141. return "", 0, errors.New("bad offset in $GENERATE")
  142. }
  143. width, err := strconv.Atoi(xs[1])
  144. if err != nil || width > 255 {
  145. return "", offset, errors.New("bad width in $GENERATE")
  146. }
  147. switch {
  148. case width < 0:
  149. return "", offset, errors.New("bad width in $GENERATE")
  150. case width == 0:
  151. return "%" + xs[1] + xs[2], offset, nil
  152. }
  153. return "%0" + xs[1] + xs[2], offset, nil
  154. }