mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-12 18:37:54 +08:00
be50e1a6a3
* Check for the right number of command line args * Handle TXT records with multiple strings
170 lines
4.3 KiB
Go
170 lines
4.3 KiB
Go
package main
|
|
|
|
/*
|
|
convertzone: Read BIND-style zonefile and output.
|
|
|
|
convertzone [-mode=MODE] zonename [filename]
|
|
|
|
-mode=tsv TAB-separated values (default)
|
|
-mode=dsl DNSControl DSL
|
|
-mode=pretty Sort and pretty-print records.
|
|
|
|
zonename The FQDN of the zone name.
|
|
filename File to read (default: stdin)
|
|
*/
|
|
|
|
import (
|
|
"bufio"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/StackExchange/dnscontrol/providers/bind"
|
|
"github.com/miekg/dns"
|
|
"github.com/miekg/dns/dnsutil"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var flagMode = flag.String("mode", "tsv", "tsv|dsl|pretty")
|
|
var flagDefaultTTL = flag.Uint("ttl", 300, "Default TTL")
|
|
var flagRegText = flag.String("registrar", "REG_FILL_IN", "registrar text")
|
|
var flagProviderText = flag.String("provider", "DNS_FILL_IN", "provider text")
|
|
|
|
// parseargs parses the non-flag arguments.
|
|
func parseargs(args []string) (zonename string, filename string, r io.Reader, err error) {
|
|
// 1 args: first arg is the zonename. Read stdin.
|
|
// 2 args: first arg is the zonename. 2nd is the filename.
|
|
// Anything else returns an error.
|
|
|
|
if len(args) < 2 {
|
|
return "", "", nil, fmt.Errorf("no command line parameters. Zone name required")
|
|
}
|
|
|
|
zonename = args[0]
|
|
|
|
if len(args) == 1 {
|
|
filename = "stdin"
|
|
r = bufio.NewReader(os.Stdin)
|
|
} else if len(args) == 2 {
|
|
filename = flag.Arg(1)
|
|
r, err = os.Open(filename)
|
|
if err != nil {
|
|
return "", "", nil, errors.Wrapf(err, "Could not open file: %s", filename)
|
|
}
|
|
} else {
|
|
return "", "", nil, fmt.Errorf("too many command line parameters")
|
|
}
|
|
|
|
return zonename, filename, r, nil
|
|
}
|
|
|
|
// pretty outputs the zonefile using the prettyprinter.
|
|
func pretty(zonename string, filename string, r io.Reader, defaultTTL uint32) {
|
|
var l []dns.RR
|
|
for x := range dns.ParseZone(r, zonename, filename) {
|
|
if x.Error == nil {
|
|
l = append(l, x.RR)
|
|
}
|
|
}
|
|
bind.WriteZoneFile(os.Stdout, l, zonename)
|
|
}
|
|
|
|
// rrFormat outputs the zonefile in either DSL or TSV format.
|
|
func rrFormat(zonename string, filename string, r io.Reader, defaultTTL uint32, dsl bool) {
|
|
zonenamedot := zonename + "."
|
|
|
|
for x := range dns.ParseZone(r, zonename, filename) {
|
|
if x.Error != nil {
|
|
log.Println(x.Error)
|
|
continue
|
|
}
|
|
|
|
// Skip comments. Parse the formatted version.
|
|
line := x.String()
|
|
if line[0] == ';' {
|
|
continue
|
|
}
|
|
items := strings.SplitN(line, "\t", 5)
|
|
if len(items) < 5 {
|
|
log.Fatalf("Too few items in: %v", line)
|
|
}
|
|
|
|
target := items[4]
|
|
|
|
hdr := x.Header()
|
|
nameFqdn := hdr.Name
|
|
name := dnsutil.TrimDomainName(nameFqdn, zonenamedot)
|
|
ttl := strconv.FormatUint(uint64(hdr.Ttl), 10)
|
|
classStr := dns.ClassToString[hdr.Class]
|
|
typeStr := dns.TypeToString[hdr.Rrtype]
|
|
|
|
// MX records should split out the prio vs. target.
|
|
if hdr.Rrtype == dns.TypeMX {
|
|
target = strings.Replace(target, " ", "\t", 1)
|
|
}
|
|
|
|
// NS records at the apex should be NAMESERVER() records.
|
|
if hdr.Rrtype == dns.TypeNS && name == "@" {
|
|
typeStr = "NAMESERVER"
|
|
}
|
|
|
|
if !dsl { // TSV format:
|
|
fmt.Printf("%s\t%s\t%s\t%s\t%s\n", name, ttl, classStr, typeStr, target)
|
|
} else { // DSL format:
|
|
switch hdr.Rrtype { // #rtype_variations
|
|
case dns.TypeMX:
|
|
m := strings.SplitN(target, "\t", 2)
|
|
target = m[0] + ", '" + m[1] + "'"
|
|
case dns.TypeSOA:
|
|
continue
|
|
case dns.TypeTXT:
|
|
if len(x.RR.(*dns.TXT).Txt) == 1 {
|
|
target = `'` + x.RR.(*dns.TXT).Txt[0] + `'`
|
|
} else {
|
|
target = `['` + strings.Join(x.RR.(*dns.TXT).Txt, `', '`) + `']`
|
|
}
|
|
default:
|
|
target = "'" + target + "'"
|
|
}
|
|
if hdr.Ttl == defaultTTL {
|
|
ttl = ""
|
|
} else {
|
|
ttl = fmt.Sprintf(", TTL(%d)", hdr.Ttl)
|
|
}
|
|
fmt.Printf(",\n\t%s('%s', %s%s)", typeStr, name, target, ttl)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
func main() {
|
|
flag.Parse()
|
|
zonename, filename, reader, err := parseargs(flag.Args())
|
|
if err != nil {
|
|
fmt.Printf("ERROR: %v\n\n", err)
|
|
fmt.Println("convertzone [-flags] ZONENAME FILENAME")
|
|
flag.Usage()
|
|
os.Exit(1)
|
|
}
|
|
|
|
defTTL := uint32(*flagDefaultTTL)
|
|
|
|
switch *flagMode {
|
|
case "pretty":
|
|
pretty(zonename, filename, reader, defTTL)
|
|
case "dsl":
|
|
fmt.Printf(`D("%s", %s, DnsProvider(%s)`, zonename, *flagRegText, *flagProviderText)
|
|
rrFormat(zonename, filename, reader, defTTL, true)
|
|
fmt.Println("\n)")
|
|
case "tsv":
|
|
rrFormat(zonename, filename, reader, defTTL, false)
|
|
default:
|
|
fmt.Println("convertzone [-flags] ZONENAME FILENAME")
|
|
flag.Usage()
|
|
}
|
|
|
|
}
|