| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | package test
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import (
 | 
					
						
							|  |  |  | 	"bufio"
 | 
					
						
							|  |  |  | 	"io/ioutil"
 | 
					
						
							|  |  |  | 	"log"
 | 
					
						
							|  |  |  | 	"os"
 | 
					
						
							|  |  |  | 	"path/filepath"
 | 
					
						
							| 
									
										
										
										
											2017-09-10 20:29:38 +01:00
										 |  |  | 	"strconv"
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | 	"testing"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/core/dnsserver"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/mholt/caddy"
 | 
					
						
							|  |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-31 07:14:49 +00:00
										 |  |  | // As we use the filesystem as-is, these files need to exist ON DISK for the readme test to work. This is especially
 | 
					
						
							|  |  |  | // useful for the *file* and *dnssec* plugins as their Corefiles are now tested as well. We create files in the
 | 
					
						
							|  |  |  | // current dir for all these, meaning the example READMEs MUST use relative path in their READMEs.
 | 
					
						
							|  |  |  | var contents = map[string]string{
 | 
					
						
							|  |  |  | 	"Kexample.org.+013+45330.key":     examplePub,
 | 
					
						
							|  |  |  | 	"Kexample.org.+013+45330.private": examplePriv,
 | 
					
						
							|  |  |  | 	"example.org.signed":              exampleOrg, // not signed, but does not matter for this test.
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const (
 | 
					
						
							|  |  |  | 	examplePub = `example.org. IN DNSKEY 256 3 13 eNMYFZYb6e0oJOV47IPo5f/UHy7wY9aBebotvcKakIYLyyGscBmXJQhbKLt/LhrMNDE2Q96hQnI5PdTBeOLzhQ==
 | 
					
						
							|  |  |  | `
 | 
					
						
							|  |  |  | 	examplePriv = `Private-key-format: v1.3
 | 
					
						
							|  |  |  | Algorithm: 13 (ECDSAP256SHA256)
 | 
					
						
							|  |  |  | PrivateKey: f03VplaIEA+KHI9uizlemUSbUJH86hPBPjmcUninPoM=
 | 
					
						
							|  |  |  | `
 | 
					
						
							|  |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TestReadme parses all README.mds of the plugins and checks if every example Corefile
 | 
					
						
							| 
									
										
										
										
											2017-10-18 17:20:27 +01:00
										 |  |  | // actually works. Each corefile snippet is only used if the language is set to 'corefile':
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | //
 | 
					
						
							|  |  |  | // ~~~ corefile
 | 
					
						
							|  |  |  | // . {
 | 
					
						
							|  |  |  | //	# check-this-please
 | 
					
						
							|  |  |  | // }
 | 
					
						
							|  |  |  | // ~~~
 | 
					
						
							|  |  |  | func TestReadme(t *testing.T) {
 | 
					
						
							| 
									
										
										
										
											2017-09-10 20:29:38 +01:00
										 |  |  | 	port := 30053
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | 	caddy.Quiet = true
 | 
					
						
							|  |  |  | 	dnsserver.Quiet = true
 | 
					
						
							| 
									
										
										
										
											2017-09-10 20:29:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-31 07:14:49 +00:00
										 |  |  | 	create(contents)
 | 
					
						
							|  |  |  | 	defer remove(contents)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | 	log.SetOutput(ioutil.Discard)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 09:36:06 +01:00
										 |  |  | 	middle := filepath.Join("..", "plugin")
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | 	dirs, err := ioutil.ReadDir(middle)
 | 
					
						
							|  |  |  | 	if err != nil {
 | 
					
						
							|  |  |  | 		t.Fatalf("Could not read %s: %q", middle, err)
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	for _, d := range dirs {
 | 
					
						
							|  |  |  | 		if !d.IsDir() {
 | 
					
						
							|  |  |  | 			continue
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		readme := filepath.Join(middle, d.Name())
 | 
					
						
							|  |  |  | 		readme = filepath.Join(readme, "README.md")
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		inputs, err := corefileFromReadme(readme)
 | 
					
						
							|  |  |  | 		if err != nil {
 | 
					
						
							|  |  |  | 			continue
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Test each snippet.
 | 
					
						
							|  |  |  | 		for _, in := range inputs {
 | 
					
						
							| 
									
										
										
										
											2017-09-10 20:29:38 +01:00
										 |  |  | 			dnsserver.Port = strconv.Itoa(port)
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | 			server, err := caddy.Start(in)
 | 
					
						
							|  |  |  | 			if err != nil {
 | 
					
						
							| 
									
										
										
										
											2017-09-28 09:46:41 +01:00
										 |  |  | 				t.Errorf("Failed to start server with %s, for input %q:\n%s", readme, err, in.Body())
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | 			}
 | 
					
						
							|  |  |  | 			server.Stop()
 | 
					
						
							| 
									
										
										
										
											2017-09-10 20:29:38 +01:00
										 |  |  | 			port++
 | 
					
						
							| 
									
										
										
										
											2017-09-10 19:52:15 +01:00
										 |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // corefileFromReadme parses a readme and returns all fragments that
 | 
					
						
							|  |  |  | // have ~~~ corefile (or ``` corefile).
 | 
					
						
							|  |  |  | func corefileFromReadme(readme string) ([]*Input, error) {
 | 
					
						
							|  |  |  | 	f, err := os.Open(readme)
 | 
					
						
							|  |  |  | 	if err != nil {
 | 
					
						
							|  |  |  | 		return nil, err
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	defer f.Close()
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s := bufio.NewScanner(f)
 | 
					
						
							|  |  |  | 	input := []*Input{}
 | 
					
						
							|  |  |  | 	corefile := false
 | 
					
						
							|  |  |  | 	temp := ""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for s.Scan() {
 | 
					
						
							|  |  |  | 		line := s.Text()
 | 
					
						
							|  |  |  | 		if line == "~~~ corefile" || line == "``` corefile" {
 | 
					
						
							|  |  |  | 			corefile = true
 | 
					
						
							|  |  |  | 			continue
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if corefile && (line == "~~~" || line == "```") {
 | 
					
						
							|  |  |  | 			// last line
 | 
					
						
							|  |  |  | 			input = append(input, NewInput(temp))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			temp = ""
 | 
					
						
							|  |  |  | 			corefile = false
 | 
					
						
							|  |  |  | 			continue
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if corefile {
 | 
					
						
							|  |  |  | 			temp += line + "\n" // readd newline stripped by s.Text()
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err := s.Err(); err != nil {
 | 
					
						
							|  |  |  | 		return nil, err
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return input, nil
 | 
					
						
							|  |  |  | }
 | 
					
						
							| 
									
										
										
										
											2017-10-31 07:14:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | func create(c map[string]string) {
 | 
					
						
							|  |  |  | 	for name, content := range c {
 | 
					
						
							|  |  |  | 		ioutil.WriteFile(name, []byte(content), 0644)
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func remove(c map[string]string) {
 | 
					
						
							|  |  |  | 	for name := range c {
 | 
					
						
							|  |  |  | 		os.Remove(name)
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | }
 |