forked from TrueCloudLab/lego
211 lines
4.8 KiB
Go
211 lines
4.8 KiB
Go
package joker
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/url"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/go-acme/lego/v3/challenge/dns01"
|
|
"github.com/go-acme/lego/v3/log"
|
|
)
|
|
|
|
const defaultBaseURL = "https://dmapi.joker.com/request/"
|
|
|
|
// Joker DMAPI Response.
|
|
type response struct {
|
|
Headers url.Values
|
|
Body string
|
|
StatusCode int
|
|
StatusText string
|
|
AuthSid string
|
|
}
|
|
|
|
// parseResponse parses HTTP response body.
|
|
func parseResponse(message string) *response {
|
|
r := &response{Headers: url.Values{}, StatusCode: -1}
|
|
|
|
parts := strings.SplitN(message, "\n\n", 2)
|
|
|
|
for _, line := range strings.Split(parts[0], "\n") {
|
|
if strings.TrimSpace(line) == "" {
|
|
continue
|
|
}
|
|
|
|
kv := strings.SplitN(line, ":", 2)
|
|
|
|
val := ""
|
|
if len(kv) == 2 {
|
|
val = strings.TrimSpace(kv[1])
|
|
}
|
|
|
|
r.Headers.Add(kv[0], val)
|
|
|
|
switch kv[0] {
|
|
case "Status-Code":
|
|
i, err := strconv.Atoi(val)
|
|
if err == nil {
|
|
r.StatusCode = i
|
|
}
|
|
case "Status-Text":
|
|
r.StatusText = val
|
|
case "Auth-Sid":
|
|
r.AuthSid = val
|
|
}
|
|
}
|
|
|
|
if len(parts) > 1 {
|
|
r.Body = parts[1]
|
|
}
|
|
|
|
return r
|
|
}
|
|
|
|
// login performs a login to Joker's DMAPI.
|
|
func (d *DNSProvider) login() (*response, error) {
|
|
if d.config.AuthSid != "" {
|
|
// already logged in
|
|
return nil, nil
|
|
}
|
|
|
|
var values url.Values
|
|
switch {
|
|
case d.config.Username != "" && d.config.Password != "":
|
|
values = url.Values{
|
|
"username": {d.config.Username},
|
|
"password": {d.config.Password},
|
|
}
|
|
case d.config.APIKey != "":
|
|
values = url.Values{"api-key": {d.config.APIKey}}
|
|
default:
|
|
return nil, errors.New("no username and password or api-key")
|
|
}
|
|
|
|
response, err := d.postRequest("login", values)
|
|
if err != nil {
|
|
return response, err
|
|
}
|
|
|
|
if response == nil {
|
|
return nil, errors.New("login returned nil response")
|
|
}
|
|
|
|
if response.AuthSid == "" {
|
|
return response, errors.New("login did not return valid Auth-Sid")
|
|
}
|
|
|
|
d.config.AuthSid = response.AuthSid
|
|
|
|
return response, nil
|
|
}
|
|
|
|
// logout closes authenticated session with Joker's DMAPI.
|
|
func (d *DNSProvider) logout() (*response, error) {
|
|
if d.config.AuthSid == "" {
|
|
return nil, errors.New("already logged out")
|
|
}
|
|
|
|
response, err := d.postRequest("logout", url.Values{})
|
|
if err == nil {
|
|
d.config.AuthSid = ""
|
|
}
|
|
return response, err
|
|
}
|
|
|
|
// getZone returns content of DNS zone for domain.
|
|
func (d *DNSProvider) getZone(domain string) (*response, error) {
|
|
if d.config.AuthSid == "" {
|
|
return nil, errors.New("must be logged in to get zone")
|
|
}
|
|
|
|
return d.postRequest("dns-zone-get", url.Values{"domain": {dns01.UnFqdn(domain)}})
|
|
}
|
|
|
|
// putZone uploads DNS zone to Joker DMAPI.
|
|
func (d *DNSProvider) putZone(domain, zone string) (*response, error) {
|
|
if d.config.AuthSid == "" {
|
|
return nil, errors.New("must be logged in to put zone")
|
|
}
|
|
|
|
return d.postRequest("dns-zone-put", url.Values{"domain": {dns01.UnFqdn(domain)}, "zone": {strings.TrimSpace(zone)}})
|
|
}
|
|
|
|
// postRequest performs actual HTTP request.
|
|
func (d *DNSProvider) postRequest(cmd string, data url.Values) (*response, error) {
|
|
uri := d.config.BaseURL + cmd
|
|
|
|
if d.config.AuthSid != "" {
|
|
data.Set("auth-sid", d.config.AuthSid)
|
|
}
|
|
|
|
if d.config.Debug {
|
|
log.Infof("postRequest:\n\tURL: %q\n\tData: %v", uri, data)
|
|
}
|
|
|
|
resp, err := d.config.HTTPClient.PostForm(uri, data)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if resp.StatusCode != 200 {
|
|
return nil, fmt.Errorf("HTTP error %d [%s]: %v", resp.StatusCode, http.StatusText(resp.StatusCode), string(body))
|
|
}
|
|
|
|
return parseResponse(string(body)), nil
|
|
}
|
|
|
|
// Temporary workaround, until it get fixed on API side.
|
|
func fixTxtLines(line string) string {
|
|
fields := strings.Fields(line)
|
|
|
|
if len(fields) < 6 || fields[1] != "TXT" {
|
|
return line
|
|
}
|
|
|
|
if fields[3][0] == '"' && fields[4] == `"` {
|
|
fields[3] = strings.TrimSpace(fields[3]) + `"`
|
|
fields = append(fields[:4], fields[5:]...)
|
|
}
|
|
|
|
return strings.Join(fields, " ")
|
|
}
|
|
|
|
// removeTxtEntryFromZone clean-ups all TXT records with given name.
|
|
func removeTxtEntryFromZone(zone, relative string) (string, bool) {
|
|
prefix := fmt.Sprintf("%s TXT 0 ", relative)
|
|
|
|
modified := false
|
|
var zoneEntries []string
|
|
for _, line := range strings.Split(zone, "\n") {
|
|
if strings.HasPrefix(line, prefix) {
|
|
modified = true
|
|
continue
|
|
}
|
|
zoneEntries = append(zoneEntries, line)
|
|
}
|
|
|
|
return strings.TrimSpace(strings.Join(zoneEntries, "\n")), modified
|
|
}
|
|
|
|
// addTxtEntryToZone returns DNS zone with added TXT record.
|
|
func addTxtEntryToZone(zone, relative, value string, ttl int) string {
|
|
var zoneEntries []string
|
|
|
|
for _, line := range strings.Split(zone, "\n") {
|
|
zoneEntries = append(zoneEntries, fixTxtLines(line))
|
|
}
|
|
|
|
newZoneEntry := fmt.Sprintf("%s TXT 0 %q %d", relative, value, ttl)
|
|
zoneEntries = append(zoneEntries, newZoneEntry)
|
|
|
|
return strings.TrimSpace(strings.Join(zoneEntries, "\n"))
|
|
}
|