ad20bf90ff
* refactor: linting. - errcheck - govet - golint - goconst - spellcheck - ... * refactor: migrate from gometalinter to golangci-lint.
395 lines
8.8 KiB
Go
395 lines
8.8 KiB
Go
// Package vegadns implements a DNS provider for solving the DNS-01
|
|
// challenge using VegaDNS.
|
|
package vegadns
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"os"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
const testDomain = "example.com"
|
|
|
|
var ipPort = "127.0.0.1:2112"
|
|
|
|
var jsonMap = map[string]string{
|
|
"token": `
|
|
{
|
|
"access_token":"699dd4ff-e381-46b8-8bf8-5de49dd56c1f",
|
|
"token_type":"bearer",
|
|
"expires_in":3600
|
|
}
|
|
`,
|
|
"domains": `
|
|
{
|
|
"domains":[
|
|
{
|
|
"domain_id":1,
|
|
"domain":"example.com",
|
|
"status":"active",
|
|
"owner_id":0
|
|
}
|
|
]
|
|
}
|
|
`,
|
|
"records": `
|
|
{
|
|
"status":"ok",
|
|
"total_records":2,
|
|
"domain":{
|
|
"status":"active",
|
|
"domain":"example.com",
|
|
"owner_id":0,
|
|
"domain_id":1
|
|
},
|
|
"records":[
|
|
{
|
|
"retry":"2048",
|
|
"minimum":"2560",
|
|
"refresh":"16384",
|
|
"email":"hostmaster.example.com",
|
|
"record_type":"SOA",
|
|
"expire":"1048576",
|
|
"ttl":86400,
|
|
"record_id":1,
|
|
"nameserver":"ns1.example.com",
|
|
"domain_id":1,
|
|
"serial":""
|
|
},
|
|
{
|
|
"name":"example.com",
|
|
"value":"ns1.example.com",
|
|
"record_type":"NS",
|
|
"ttl":3600,
|
|
"record_id":2,
|
|
"location_id":null,
|
|
"domain_id":1
|
|
},
|
|
{
|
|
"name":"_acme-challenge.example.com",
|
|
"value":"my_challenge",
|
|
"record_type":"TXT",
|
|
"ttl":3600,
|
|
"record_id":3,
|
|
"location_id":null,
|
|
"domain_id":1
|
|
}
|
|
]
|
|
}
|
|
`,
|
|
"recordCreated": `
|
|
{
|
|
"status":"ok",
|
|
"record":{
|
|
"name":"_acme-challenge.example.com",
|
|
"value":"my_challenge",
|
|
"record_type":"TXT",
|
|
"ttl":3600,
|
|
"record_id":3,
|
|
"location_id":null,
|
|
"domain_id":1
|
|
}
|
|
}
|
|
`,
|
|
"recordDeleted": `{"status": "ok"}`,
|
|
}
|
|
|
|
type muxCallback func() *http.ServeMux
|
|
|
|
func TestVegaDNSNewDNSProviderFail(t *testing.T) {
|
|
os.Setenv("VEGADNS_URL", "")
|
|
_, err := NewDNSProvider()
|
|
assert.Error(t, err, "VEGADNS_URL env missing")
|
|
}
|
|
|
|
func TestVegaDNSTimeoutSuccess(t *testing.T) {
|
|
ts, err := startTestServer(vegaDNSMuxSuccess)
|
|
require.NoError(t, err)
|
|
|
|
defer ts.Close()
|
|
defer os.Clearenv()
|
|
|
|
provider, err := NewDNSProvider()
|
|
require.NoError(t, err)
|
|
|
|
timeout, interval := provider.Timeout()
|
|
assert.Equal(t, timeout, time.Duration(720000000000))
|
|
assert.Equal(t, interval, time.Duration(60000000000))
|
|
}
|
|
|
|
func TestDNSProvider_Present(t *testing.T) {
|
|
testCases := []struct {
|
|
desc string
|
|
callback muxCallback
|
|
expectedError string
|
|
}{
|
|
{
|
|
desc: "Success",
|
|
callback: vegaDNSMuxSuccess,
|
|
},
|
|
{
|
|
desc: "FailToFindZone",
|
|
callback: vegaDNSMuxFailToFindZone,
|
|
expectedError: "vegadns: can't find Authoritative Zone for _acme-challenge.example.com. in Present: Unable to find auth zone for fqdn _acme-challenge.example.com",
|
|
},
|
|
{
|
|
desc: "FailToCreateTXT",
|
|
callback: vegaDNSMuxFailToCreateTXT,
|
|
expectedError: "vegadns: Got bad answer from VegaDNS on CreateTXT. Code: 400. Message: ",
|
|
},
|
|
}
|
|
|
|
for _, test := range testCases {
|
|
t.Run(test.desc, func(t *testing.T) {
|
|
ts, err := startTestServer(test.callback)
|
|
require.NoError(t, err)
|
|
|
|
defer ts.Close()
|
|
defer os.Clearenv()
|
|
|
|
provider, err := NewDNSProvider()
|
|
require.NoError(t, err)
|
|
|
|
err = provider.Present(testDomain, "token", "keyAuth")
|
|
if len(test.expectedError) == 0 {
|
|
assert.NoError(t, err)
|
|
} else {
|
|
assert.EqualError(t, err, test.expectedError)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestDNSProvider_CleanUp(t *testing.T) {
|
|
testCases := []struct {
|
|
desc string
|
|
callback muxCallback
|
|
expectedError string
|
|
}{
|
|
{
|
|
desc: "Success",
|
|
callback: vegaDNSMuxSuccess,
|
|
},
|
|
{
|
|
desc: "FailToFindZone",
|
|
callback: vegaDNSMuxFailToFindZone,
|
|
expectedError: "vegadns: can't find Authoritative Zone for _acme-challenge.example.com. in CleanUp: Unable to find auth zone for fqdn _acme-challenge.example.com",
|
|
},
|
|
{
|
|
desc: "FailToGetRecordID",
|
|
callback: vegaDNSMuxFailToGetRecordID,
|
|
expectedError: "vegadns: couldn't get Record ID in CleanUp: Got bad answer from VegaDNS on GetRecordID. Code: 404. Message: ",
|
|
},
|
|
}
|
|
|
|
for _, test := range testCases {
|
|
t.Run(test.desc, func(t *testing.T) {
|
|
ts, err := startTestServer(test.callback)
|
|
require.NoError(t, err)
|
|
|
|
defer ts.Close()
|
|
defer os.Clearenv()
|
|
|
|
provider, err := NewDNSProvider()
|
|
require.NoError(t, err)
|
|
|
|
err = provider.CleanUp(testDomain, "token", "keyAuth")
|
|
if len(test.expectedError) == 0 {
|
|
assert.NoError(t, err)
|
|
} else {
|
|
assert.EqualError(t, err, test.expectedError)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func vegaDNSMuxSuccess() *http.ServeMux {
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/1.0/token", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodPost:
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["token"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/domains", func(w http.ResponseWriter, r *http.Request) {
|
|
if r.URL.Query().Get("search") == "example.com" {
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["domains"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusNotFound)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/records", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodGet:
|
|
if r.URL.Query().Get("domain_id") == "1" {
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["records"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusNotFound)
|
|
return
|
|
case http.MethodPost:
|
|
w.WriteHeader(http.StatusCreated)
|
|
fmt.Fprintf(w, jsonMap["recordCreated"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/records/3", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodDelete:
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["recordDeleted"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
fmt.Printf("Not Found for Request: (%+v)\n\n", r)
|
|
})
|
|
|
|
return mux
|
|
}
|
|
|
|
func vegaDNSMuxFailToFindZone() *http.ServeMux {
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/1.0/token", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodPost:
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["token"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/domains", func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
})
|
|
|
|
return mux
|
|
}
|
|
|
|
func vegaDNSMuxFailToCreateTXT() *http.ServeMux {
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/1.0/token", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodPost:
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["token"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/domains", func(w http.ResponseWriter, r *http.Request) {
|
|
if r.URL.Query().Get("search") == testDomain {
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["domains"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusNotFound)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/records", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodGet:
|
|
if r.URL.Query().Get("domain_id") == "1" {
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["records"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusNotFound)
|
|
return
|
|
case http.MethodPost:
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
return mux
|
|
}
|
|
|
|
func vegaDNSMuxFailToGetRecordID() *http.ServeMux {
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/1.0/token", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodPost:
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["token"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/domains", func(w http.ResponseWriter, r *http.Request) {
|
|
if r.URL.Query().Get("search") == testDomain {
|
|
w.WriteHeader(http.StatusOK)
|
|
fmt.Fprintf(w, jsonMap["domains"])
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusNotFound)
|
|
})
|
|
|
|
mux.HandleFunc("/1.0/records", func(w http.ResponseWriter, r *http.Request) {
|
|
switch r.Method {
|
|
case http.MethodGet:
|
|
w.WriteHeader(http.StatusNotFound)
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
})
|
|
|
|
return mux
|
|
}
|
|
|
|
// Starts and returns a test server using a custom ip/port. Defer close() afterwards.
|
|
func startTestServer(callback muxCallback) (*httptest.Server, error) {
|
|
err := os.Setenv("SECRET_VEGADNS_KEY", "key")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = os.Setenv("SECRET_VEGADNS_SECRET", "secret")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = os.Setenv("VEGADNS_URL", "http://"+ipPort)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ts := httptest.NewUnstartedServer(callback())
|
|
|
|
l, err := net.Listen("tcp", ipPort)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ts.Listener = l
|
|
ts.Start()
|
|
|
|
return ts, nil
|
|
}
|