2017-03-21 02:27:40 +08:00
|
|
|
package main
|
|
|
|
|
|
|
|
/*
|
2018-02-28 06:36:47 +08:00
|
|
|
convertzone: Read and write DNS zone files.
|
2017-03-21 02:27:40 +08:00
|
|
|
|
2018-02-28 06:36:47 +08:00
|
|
|
convertzone [-in=INPUT] [-out=OUTPUT] zonename [filename]
|
2017-03-21 02:27:40 +08:00
|
|
|
|
2018-02-28 06:36:47 +08:00
|
|
|
Input format:
|
|
|
|
-in=bind BIND-style zonefiles (DEFAULT)
|
|
|
|
-in=octodns OctoDNS YAML "config" files.
|
2017-03-21 02:27:40 +08:00
|
|
|
|
2018-02-28 06:36:47 +08:00
|
|
|
Output format:
|
|
|
|
|
|
|
|
-out=dsl DNSControl DSL language (dnsconfig.js) (DEFAULT)
|
|
|
|
-out=tsv TAB-separated values
|
|
|
|
-out=pretty pretty-printed (BIND-style zonefiles)
|
|
|
|
|
|
|
|
zonename The FQDN of the zone name.
|
|
|
|
filename File to read (optional. Defaults to stdin)
|
|
|
|
|
|
|
|
The DSL output format is useful for creating the first
|
|
|
|
draft of your dnsconfig.js when importing zones from
|
|
|
|
other services.
|
|
|
|
|
|
|
|
The TSV format makes it easy to process a zonefile with
|
|
|
|
shell tools. `awk -F"\t" $2 = "A" { print $3 }`
|
|
|
|
|
|
|
|
The PRETTY format is just a nice way to clean up a zonefile.
|
|
|
|
|
|
|
|
If no filename is specified, stdin is assumed.
|
|
|
|
Output is sent to stdout.
|
|
|
|
|
|
|
|
The zonename is required as it can not be guessed automatically from the input.
|
2017-03-21 02:27:40 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/miekg/dns"
|
|
|
|
"github.com/miekg/dns/dnsutil"
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
|
2020-04-15 04:47:30 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/prettyzone"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/providers/octodns/octoyaml"
|
2017-03-21 02:27:40 +08:00
|
|
|
)
|
|
|
|
|
2018-02-28 06:36:47 +08:00
|
|
|
var flagInfmt = flag.String("in", "zone", "zone|octodns")
|
|
|
|
var flagOutfmt = flag.String("out", "dsl", "dsl|tsv|pretty")
|
2017-03-21 02:27:40 +08:00
|
|
|
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")
|
|
|
|
|
2018-02-28 06:36:47 +08:00
|
|
|
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)
|
|
|
|
|
|
|
|
var recs []dns.RR
|
|
|
|
|
|
|
|
// Read it in:
|
|
|
|
|
|
|
|
switch *flagInfmt {
|
|
|
|
case "zone":
|
|
|
|
recs = readZone(zonename, reader, filename)
|
|
|
|
case "oct", "octo", "octodns":
|
|
|
|
recs = readOctodns(zonename, reader, filename)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write it out:
|
|
|
|
|
|
|
|
switch *flagOutfmt {
|
|
|
|
case "pretty":
|
2020-02-24 02:58:49 +08:00
|
|
|
prettyzone.WriteZoneFileRR(os.Stdout, recs, zonename)
|
2018-02-28 06:36:47 +08:00
|
|
|
case "dsl":
|
|
|
|
fmt.Printf(`D("%s", %s, DnsProvider(%s)`, zonename, *flagRegText, *flagProviderText)
|
|
|
|
rrFormat(zonename, filename, recs, defTTL, true)
|
|
|
|
fmt.Println("\n)")
|
|
|
|
case "tsv":
|
|
|
|
rrFormat(zonename, filename, recs, defTTL, false)
|
|
|
|
default:
|
|
|
|
fmt.Println("convertzone [-flags] ZONENAME FILENAME")
|
|
|
|
flag.Usage()
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-03-21 02:27:40 +08:00
|
|
|
// 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.
|
|
|
|
|
2018-01-05 09:35:15 +08:00
|
|
|
if len(args) < 2 {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
return "", "", nil, fmt.Errorf("no command line parameters. Zone name required")
|
2017-03-21 02:27:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
return "", "", nil, fmt.Errorf("Could not open file: %s: %w", filename, err)
|
2017-03-21 02:27:40 +08:00
|
|
|
}
|
|
|
|
} else {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
return "", "", nil, fmt.Errorf("too many command line parameters")
|
2017-03-21 02:27:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return zonename, filename, r, nil
|
|
|
|
}
|
|
|
|
|
2018-02-28 06:36:47 +08:00
|
|
|
func readZone(zonename string, r io.Reader, filename string) []dns.RR {
|
2017-03-21 02:27:40 +08:00
|
|
|
var l []dns.RR
|
|
|
|
for x := range dns.ParseZone(r, zonename, filename) {
|
2018-02-28 06:36:47 +08:00
|
|
|
if x.Error != nil {
|
|
|
|
log.Println(x.Error)
|
|
|
|
} else {
|
2017-03-21 02:27:40 +08:00
|
|
|
l = append(l, x.RR)
|
|
|
|
}
|
|
|
|
}
|
2018-02-28 06:36:47 +08:00
|
|
|
return l
|
|
|
|
}
|
|
|
|
|
|
|
|
func readOctodns(zonename string, r io.Reader, filename string) []dns.RR {
|
|
|
|
var l []dns.RR
|
|
|
|
|
|
|
|
foundRecords, err := octoyaml.ReadYaml(r, zonename)
|
|
|
|
if err != nil {
|
Switch to Go 1.13 error wrapping (#604)
* Replaced errors.Wrap with fmt.Errorf (#589)
* Find: errors\.Wrap\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Replaced errors.Wrapf with fmt.Errorf (#589)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])\)
Replace: fmt.Errorf($2: %w$3, $1)
* Find: errors\.Wrapf\(([^,]+),\s+(["`][^"`]*)(["`])(,[^)]+)\)
* Replace: fmt.Errorf($2: %w$3$4, $1)
* Replaced errors.Errorf with fmt.Errorf (#589)
* Find: errors\.Errorf
Replace: fmt.Errorf
* Cleaned up remaining imports
* Cleanup
* Regenerate provider support matrix
This was broken by #533 ... and it's now the third time this has been missed.
2020-01-29 00:06:56 +08:00
|
|
|
log.Println(fmt.Errorf("can not get corrections: %w", err))
|
2018-02-28 06:36:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, x := range foundRecords {
|
|
|
|
l = append(l, x.ToRR())
|
|
|
|
}
|
|
|
|
return l
|
|
|
|
}
|
|
|
|
|
2017-03-21 02:27:40 +08:00
|
|
|
// rrFormat outputs the zonefile in either DSL or TSV format.
|
2018-02-28 06:36:47 +08:00
|
|
|
func rrFormat(zonename string, filename string, recs []dns.RR, defaultTTL uint32, dsl bool) {
|
2017-03-21 02:27:40 +08:00
|
|
|
zonenamedot := zonename + "."
|
|
|
|
|
2018-02-28 06:36:47 +08:00
|
|
|
for _, x := range recs {
|
2017-03-21 02:27:40 +08:00
|
|
|
|
|
|
|
// 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)
|
|
|
|
}
|
|
|
|
|
2018-05-13 20:31:34 +08:00
|
|
|
var ttlop string
|
|
|
|
if hdr.Ttl == defaultTTL {
|
|
|
|
ttlop = ""
|
|
|
|
} else {
|
|
|
|
ttlop = fmt.Sprintf(", TTL(%d)", hdr.Ttl)
|
|
|
|
}
|
|
|
|
|
2017-08-05 23:59:35 +08:00
|
|
|
// NS records at the apex should be NAMESERVER() records.
|
|
|
|
if hdr.Rrtype == dns.TypeNS && name == "@" {
|
2018-05-13 20:31:34 +08:00
|
|
|
fmt.Printf(",\n\tNAMESERVER('%s'%s)", target, ttlop)
|
|
|
|
continue
|
2017-08-05 23:59:35 +08:00
|
|
|
}
|
|
|
|
|
2017-03-21 02:27:40 +08:00
|
|
|
if !dsl { // TSV format:
|
|
|
|
fmt.Printf("%s\t%s\t%s\t%s\t%s\n", name, ttl, classStr, typeStr, target)
|
|
|
|
} else { // DSL format:
|
2017-08-05 03:26:29 +08:00
|
|
|
switch hdr.Rrtype { // #rtype_variations
|
2017-03-21 02:27:40 +08:00
|
|
|
case dns.TypeMX:
|
|
|
|
m := strings.SplitN(target, "\t", 2)
|
|
|
|
target = m[0] + ", '" + m[1] + "'"
|
|
|
|
case dns.TypeSOA:
|
|
|
|
continue
|
|
|
|
case dns.TypeTXT:
|
2018-02-28 06:36:47 +08:00
|
|
|
if len(x.(*dns.TXT).Txt) == 1 {
|
|
|
|
target = `'` + x.(*dns.TXT).Txt[0] + `'`
|
2018-01-05 09:35:15 +08:00
|
|
|
} else {
|
2018-02-28 06:36:47 +08:00
|
|
|
target = `['` + strings.Join(x.(*dns.TXT).Txt, `', '`) + `']`
|
2018-01-05 09:35:15 +08:00
|
|
|
}
|
2017-03-21 02:27:40 +08:00
|
|
|
default:
|
|
|
|
target = "'" + target + "'"
|
|
|
|
}
|
2018-05-13 20:31:34 +08:00
|
|
|
fmt.Printf(",\n\t%s('%s', %s%s)", typeStr, name, target, ttlop)
|
2017-03-21 02:27:40 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|