| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | // Copyright 2018 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of go-ethereum. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // go-ethereum is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // go-ethereum is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  | // along with go-ethereum. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"crypto/ecdsa" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/accounts/keystore" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2020-05-19 10:44:46 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/console/prompt" | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/p2p/dnsdisc" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/p2p/enode" | 
					
						
							| 
									
										
										
										
											2020-11-25 21:00:23 +01:00
										 |  |  | 	"gopkg.in/urfave/cli.v1" | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	dnsCommand = cli.Command{ | 
					
						
							|  |  |  | 		Name:  "dns", | 
					
						
							|  |  |  | 		Usage: "DNS Discovery Commands", | 
					
						
							|  |  |  | 		Subcommands: []cli.Command{ | 
					
						
							|  |  |  | 			dnsSyncCommand, | 
					
						
							|  |  |  | 			dnsSignCommand, | 
					
						
							|  |  |  | 			dnsTXTCommand, | 
					
						
							|  |  |  | 			dnsCloudflareCommand, | 
					
						
							| 
									
										
										
										
											2019-12-12 22:25:12 +01:00
										 |  |  | 			dnsRoute53Command, | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | 			dnsRoute53NukeCommand, | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dnsSyncCommand = cli.Command{ | 
					
						
							|  |  |  | 		Name:      "sync", | 
					
						
							|  |  |  | 		Usage:     "Download a DNS discovery tree", | 
					
						
							|  |  |  | 		ArgsUsage: "<url> [ <directory> ]", | 
					
						
							|  |  |  | 		Action:    dnsSync, | 
					
						
							|  |  |  | 		Flags:     []cli.Flag{dnsTimeoutFlag}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dnsSignCommand = cli.Command{ | 
					
						
							|  |  |  | 		Name:      "sign", | 
					
						
							|  |  |  | 		Usage:     "Sign a DNS discovery tree", | 
					
						
							|  |  |  | 		ArgsUsage: "<tree-directory> <key-file>", | 
					
						
							|  |  |  | 		Action:    dnsSign, | 
					
						
							|  |  |  | 		Flags:     []cli.Flag{dnsDomainFlag, dnsSeqFlag}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dnsTXTCommand = cli.Command{ | 
					
						
							|  |  |  | 		Name:      "to-txt", | 
					
						
							|  |  |  | 		Usage:     "Create a DNS TXT records for a discovery tree", | 
					
						
							|  |  |  | 		ArgsUsage: "<tree-directory> <output-file>", | 
					
						
							|  |  |  | 		Action:    dnsToTXT, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dnsCloudflareCommand = cli.Command{ | 
					
						
							|  |  |  | 		Name:      "to-cloudflare", | 
					
						
							| 
									
										
										
										
											2019-12-12 22:25:12 +01:00
										 |  |  | 		Usage:     "Deploy DNS TXT records to CloudFlare", | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 		ArgsUsage: "<tree-directory>", | 
					
						
							|  |  |  | 		Action:    dnsToCloudflare, | 
					
						
							|  |  |  | 		Flags:     []cli.Flag{cloudflareTokenFlag, cloudflareZoneIDFlag}, | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-12 22:25:12 +01:00
										 |  |  | 	dnsRoute53Command = cli.Command{ | 
					
						
							|  |  |  | 		Name:      "to-route53", | 
					
						
							|  |  |  | 		Usage:     "Deploy DNS TXT records to Amazon Route53", | 
					
						
							|  |  |  | 		ArgsUsage: "<tree-directory>", | 
					
						
							|  |  |  | 		Action:    dnsToRoute53, | 
					
						
							| 
									
										
										
										
											2021-03-20 00:22:24 +01:00
										 |  |  | 		Flags: []cli.Flag{ | 
					
						
							|  |  |  | 			route53AccessKeyFlag, | 
					
						
							|  |  |  | 			route53AccessSecretFlag, | 
					
						
							|  |  |  | 			route53ZoneIDFlag, | 
					
						
							|  |  |  | 			route53RegionFlag, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-12-12 22:25:12 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | 	dnsRoute53NukeCommand = cli.Command{ | 
					
						
							|  |  |  | 		Name:      "nuke-route53", | 
					
						
							|  |  |  | 		Usage:     "Deletes DNS TXT records of a subdomain on Amazon Route53", | 
					
						
							|  |  |  | 		ArgsUsage: "<domain>", | 
					
						
							|  |  |  | 		Action:    dnsNukeRoute53, | 
					
						
							|  |  |  | 		Flags: []cli.Flag{ | 
					
						
							|  |  |  | 			route53AccessKeyFlag, | 
					
						
							|  |  |  | 			route53AccessSecretFlag, | 
					
						
							|  |  |  | 			route53ZoneIDFlag, | 
					
						
							|  |  |  | 			route53RegionFlag, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	dnsTimeoutFlag = cli.DurationFlag{ | 
					
						
							|  |  |  | 		Name:  "timeout", | 
					
						
							|  |  |  | 		Usage: "Timeout for DNS lookups", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dnsDomainFlag = cli.StringFlag{ | 
					
						
							|  |  |  | 		Name:  "domain", | 
					
						
							|  |  |  | 		Usage: "Domain name of the tree", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	dnsSeqFlag = cli.UintFlag{ | 
					
						
							|  |  |  | 		Name:  "seq", | 
					
						
							|  |  |  | 		Usage: "New sequence number of the tree", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-17 11:32:29 +01:00
										 |  |  | const ( | 
					
						
							| 
									
										
										
										
											2021-11-11 08:07:11 -08:00
										 |  |  | 	rootTTL               = 30 * 60              // 30 min | 
					
						
							|  |  |  | 	treeNodeTTL           = 4 * 7 * 24 * 60 * 60 // 4 weeks | 
					
						
							|  |  |  | 	treeNodeTTLCloudflare = 24 * 60 * 60         // 1 day | 
					
						
							| 
									
										
										
										
											2020-01-17 11:32:29 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | // dnsSync performs dnsSyncCommand. | 
					
						
							|  |  |  | func dnsSync(ctx *cli.Context) error { | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		c      = dnsClient(ctx) | 
					
						
							|  |  |  | 		url    = ctx.Args().Get(0) | 
					
						
							|  |  |  | 		outdir = ctx.Args().Get(1) | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	domain, _, err := dnsdisc.ParseURL(url) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if outdir == "" { | 
					
						
							|  |  |  | 		outdir = domain | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t, err := c.SyncTree(url) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	def := treeToDefinition(url, t) | 
					
						
							|  |  |  | 	def.Meta.LastModified = time.Now() | 
					
						
							| 
									
										
										
										
											2019-10-29 16:08:57 +01:00
										 |  |  | 	writeTreeMetadata(outdir, def) | 
					
						
							|  |  |  | 	writeTreeNodes(outdir, def) | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func dnsSign(ctx *cli.Context) error { | 
					
						
							|  |  |  | 	if ctx.NArg() < 2 { | 
					
						
							|  |  |  | 		return fmt.Errorf("need tree definition directory and key file as arguments") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		defdir  = ctx.Args().Get(0) | 
					
						
							|  |  |  | 		keyfile = ctx.Args().Get(1) | 
					
						
							|  |  |  | 		def     = loadTreeDefinition(defdir) | 
					
						
							|  |  |  | 		domain  = directoryName(defdir) | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	if def.Meta.URL != "" { | 
					
						
							|  |  |  | 		d, _, err := dnsdisc.ParseURL(def.Meta.URL) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return fmt.Errorf("invalid 'url' field: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		domain = d | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if ctx.IsSet(dnsDomainFlag.Name) { | 
					
						
							|  |  |  | 		domain = ctx.String(dnsDomainFlag.Name) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if ctx.IsSet(dnsSeqFlag.Name) { | 
					
						
							|  |  |  | 		def.Meta.Seq = ctx.Uint(dnsSeqFlag.Name) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		def.Meta.Seq++ // Auto-bump sequence number if not supplied via flag. | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	t, err := dnsdisc.MakeTree(def.Meta.Seq, def.Nodes, def.Meta.Links) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	key := loadSigningKey(keyfile) | 
					
						
							|  |  |  | 	url, err := t.Sign(key, domain) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return fmt.Errorf("can't sign: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def = treeToDefinition(url, t) | 
					
						
							|  |  |  | 	def.Meta.LastModified = time.Now() | 
					
						
							| 
									
										
										
										
											2019-10-29 16:08:57 +01:00
										 |  |  | 	writeTreeMetadata(defdir, def) | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | // directoryName returns the directory name of the given path. | 
					
						
							|  |  |  | // For example, when dir is "foo/bar", it returns "bar". | 
					
						
							|  |  |  | // When dir is ".", and the working directory is "example/foo", it returns "foo". | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | func directoryName(dir string) string { | 
					
						
							|  |  |  | 	abs, err := filepath.Abs(dir) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return filepath.Base(abs) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | // dnsToTXT performs dnsTXTCommand. | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | func dnsToTXT(ctx *cli.Context) error { | 
					
						
							|  |  |  | 	if ctx.NArg() < 1 { | 
					
						
							|  |  |  | 		return fmt.Errorf("need tree definition directory as argument") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	output := ctx.Args().Get(1) | 
					
						
							|  |  |  | 	if output == "" { | 
					
						
							|  |  |  | 		output = "-" // default to stdout | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	domain, t, err := loadTreeDefinitionForExport(ctx.Args().Get(0)) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	writeTXTJSON(output, t.ToTXT(domain)) | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | // dnsToCloudflare performs dnsCloudflareCommand. | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | func dnsToCloudflare(ctx *cli.Context) error { | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | 	if ctx.NArg() != 1 { | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 		return fmt.Errorf("need tree definition directory as argument") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	domain, t, err := loadTreeDefinitionForExport(ctx.Args().Get(0)) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	client := newCloudflareClient(ctx) | 
					
						
							| 
									
										
										
										
											2019-12-12 22:25:12 +01:00
										 |  |  | 	return client.deploy(domain, t) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | // dnsToRoute53 performs dnsRoute53Command. | 
					
						
							| 
									
										
										
										
											2019-12-12 22:25:12 +01:00
										 |  |  | func dnsToRoute53(ctx *cli.Context) error { | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | 	if ctx.NArg() != 1 { | 
					
						
							| 
									
										
										
										
											2019-12-12 22:25:12 +01:00
										 |  |  | 		return fmt.Errorf("need tree definition directory as argument") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	domain, t, err := loadTreeDefinitionForExport(ctx.Args().Get(0)) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	client := newRoute53Client(ctx) | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 	return client.deploy(domain, t) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-19 14:54:55 +02:00
										 |  |  | // dnsNukeRoute53 performs dnsRoute53NukeCommand. | 
					
						
							|  |  |  | func dnsNukeRoute53(ctx *cli.Context) error { | 
					
						
							|  |  |  | 	if ctx.NArg() != 1 { | 
					
						
							|  |  |  | 		return fmt.Errorf("need domain name as argument") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	client := newRoute53Client(ctx) | 
					
						
							|  |  |  | 	return client.deleteDomain(ctx.Args().First()) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | // loadSigningKey loads a private key in Ethereum keystore format. | 
					
						
							|  |  |  | func loadSigningKey(keyfile string) *ecdsa.PrivateKey { | 
					
						
							|  |  |  | 	keyjson, err := ioutil.ReadFile(keyfile) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		exit(fmt.Errorf("failed to read the keyfile at '%s': %v", keyfile, err)) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-05-19 10:44:46 +02:00
										 |  |  | 	password, _ := prompt.Stdin.PromptPassword("Please enter the password for '" + keyfile + "': ") | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 	key, err := keystore.DecryptKey(keyjson, password) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		exit(fmt.Errorf("error decrypting key: %v", err)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return key.PrivateKey | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // dnsClient configures the DNS discovery client from command line flags. | 
					
						
							|  |  |  | func dnsClient(ctx *cli.Context) *dnsdisc.Client { | 
					
						
							|  |  |  | 	var cfg dnsdisc.Config | 
					
						
							|  |  |  | 	if commandHasFlag(ctx, dnsTimeoutFlag) { | 
					
						
							|  |  |  | 		cfg.Timeout = ctx.Duration(dnsTimeoutFlag.Name) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-12 10:15:36 +01:00
										 |  |  | 	return dnsdisc.NewClient(cfg) | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // There are two file formats for DNS node trees on disk: | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The 'TXT' format is a single JSON file containing DNS TXT records | 
					
						
							|  |  |  | // as a JSON object where the keys are names and the values are objects | 
					
						
							|  |  |  | // containing the value of the record. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The 'definition' format is a directory containing two files: | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //      enrtree-info.json    -- contains sequence number & links to other trees | 
					
						
							|  |  |  | //      nodes.json           -- contains the nodes as a JSON array. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This format exists because it's convenient to edit. nodes.json can be generated | 
					
						
							|  |  |  | // in multiple ways: it may be written by a DHT crawler or compiled by a human. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type dnsDefinition struct { | 
					
						
							|  |  |  | 	Meta  dnsMetaJSON | 
					
						
							|  |  |  | 	Nodes []*enode.Node | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type dnsMetaJSON struct { | 
					
						
							|  |  |  | 	URL          string    `json:"url,omitempty"` | 
					
						
							|  |  |  | 	Seq          uint      `json:"seq"` | 
					
						
							|  |  |  | 	Sig          string    `json:"signature,omitempty"` | 
					
						
							|  |  |  | 	Links        []string  `json:"links"` | 
					
						
							|  |  |  | 	LastModified time.Time `json:"lastModified"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func treeToDefinition(url string, t *dnsdisc.Tree) *dnsDefinition { | 
					
						
							|  |  |  | 	meta := dnsMetaJSON{ | 
					
						
							|  |  |  | 		URL:   url, | 
					
						
							|  |  |  | 		Seq:   t.Seq(), | 
					
						
							|  |  |  | 		Sig:   t.Signature(), | 
					
						
							|  |  |  | 		Links: t.Links(), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if meta.Links == nil { | 
					
						
							|  |  |  | 		meta.Links = []string{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return &dnsDefinition{Meta: meta, Nodes: t.Nodes()} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // loadTreeDefinition loads a directory in 'definition' format. | 
					
						
							|  |  |  | func loadTreeDefinition(directory string) *dnsDefinition { | 
					
						
							|  |  |  | 	metaFile, nodesFile := treeDefinitionFiles(directory) | 
					
						
							|  |  |  | 	var def dnsDefinition | 
					
						
							|  |  |  | 	err := common.LoadJSON(metaFile, &def.Meta) | 
					
						
							|  |  |  | 	if err != nil && !os.IsNotExist(err) { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if def.Meta.Links == nil { | 
					
						
							|  |  |  | 		def.Meta.Links = []string{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Check link syntax. | 
					
						
							|  |  |  | 	for _, link := range def.Meta.Links { | 
					
						
							|  |  |  | 		if _, _, err := dnsdisc.ParseURL(link); err != nil { | 
					
						
							|  |  |  | 			exit(fmt.Errorf("invalid link %q: %v", link, err)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Check/convert nodes. | 
					
						
							|  |  |  | 	nodes := loadNodesJSON(nodesFile) | 
					
						
							|  |  |  | 	if err := nodes.verify(); err != nil { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	def.Nodes = nodes.nodes() | 
					
						
							|  |  |  | 	return &def | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // loadTreeDefinitionForExport loads a DNS tree and ensures it is signed. | 
					
						
							|  |  |  | func loadTreeDefinitionForExport(dir string) (domain string, t *dnsdisc.Tree, err error) { | 
					
						
							|  |  |  | 	metaFile, _ := treeDefinitionFiles(dir) | 
					
						
							|  |  |  | 	def := loadTreeDefinition(dir) | 
					
						
							|  |  |  | 	if def.Meta.URL == "" { | 
					
						
							|  |  |  | 		return "", nil, fmt.Errorf("missing 'url' field in %v", metaFile) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	domain, pubkey, err := dnsdisc.ParseURL(def.Meta.URL) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return "", nil, fmt.Errorf("invalid 'url' field in %v: %v", metaFile, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if t, err = dnsdisc.MakeTree(def.Meta.Seq, def.Nodes, def.Meta.Links); err != nil { | 
					
						
							|  |  |  | 		return "", nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err := ensureValidTreeSignature(t, pubkey, def.Meta.Sig); err != nil { | 
					
						
							|  |  |  | 		return "", nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return domain, t, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ensureValidTreeSignature checks that sig is valid for tree and assigns it as the | 
					
						
							|  |  |  | // tree's signature if valid. | 
					
						
							|  |  |  | func ensureValidTreeSignature(t *dnsdisc.Tree, pubkey *ecdsa.PublicKey, sig string) error { | 
					
						
							|  |  |  | 	if sig == "" { | 
					
						
							|  |  |  | 		return fmt.Errorf("missing signature, run 'devp2p dns sign' first") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err := t.SetSignature(pubkey, sig); err != nil { | 
					
						
							|  |  |  | 		return fmt.Errorf("invalid signature on tree, run 'devp2p dns sign' to update it") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-29 16:08:57 +01:00
										 |  |  | // writeTreeMetadata writes a DNS node tree metadata file to the given directory. | 
					
						
							|  |  |  | func writeTreeMetadata(directory string, def *dnsDefinition) { | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 	metaJSON, err := json.MarshalIndent(&def.Meta, "", jsonIndent) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err := os.Mkdir(directory, 0744); err != nil && !os.IsExist(err) { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-10-29 16:08:57 +01:00
										 |  |  | 	metaFile, _ := treeDefinitionFiles(directory) | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | 	if err := ioutil.WriteFile(metaFile, metaJSON, 0644); err != nil { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-29 16:08:57 +01:00
										 |  |  | func writeTreeNodes(directory string, def *dnsDefinition) { | 
					
						
							|  |  |  | 	ns := make(nodeSet, len(def.Nodes)) | 
					
						
							|  |  |  | 	ns.add(def.Nodes...) | 
					
						
							|  |  |  | 	_, nodesFile := treeDefinitionFiles(directory) | 
					
						
							|  |  |  | 	writeNodesJSON(nodesFile, ns) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-25 11:38:13 +02:00
										 |  |  | func treeDefinitionFiles(directory string) (string, string) { | 
					
						
							|  |  |  | 	meta := filepath.Join(directory, "enrtree-info.json") | 
					
						
							|  |  |  | 	nodes := filepath.Join(directory, "nodes.json") | 
					
						
							|  |  |  | 	return meta, nodes | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // writeTXTJSON writes TXT records in JSON format. | 
					
						
							|  |  |  | func writeTXTJSON(file string, txt map[string]string) { | 
					
						
							|  |  |  | 	txtJSON, err := json.MarshalIndent(txt, "", jsonIndent) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if file == "-" { | 
					
						
							|  |  |  | 		os.Stdout.Write(txtJSON) | 
					
						
							|  |  |  | 		fmt.Println() | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err := ioutil.WriteFile(file, txtJSON, 0644); err != nil { | 
					
						
							|  |  |  | 		exit(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |