lego/providers/dns/joker/internal/dmapi/client.go

245 lines
5.3 KiB
Go
Raw Normal View History

2020-10-08 14:52:50 +00:00
// Package dmapi Client for DMAPI joker.com.
// https://joker.com/faq/category/39/22-dmapi.html
package dmapi
2019-04-28 12:33:50 +00:00
import (
2020-02-27 18:14:46 +00:00
"errors"
2019-04-28 12:33:50 +00:00
"fmt"
2021-08-25 09:44:11 +00:00
"io"
2019-04-28 12:33:50 +00:00
"net/http"
"net/url"
2020-10-08 14:52:50 +00:00
"path"
2019-04-28 12:33:50 +00:00
"strconv"
"strings"
2020-09-02 01:20:01 +00:00
"github.com/go-acme/lego/v4/challenge/dns01"
"github.com/go-acme/lego/v4/log"
2019-04-28 12:33:50 +00:00
)
const defaultBaseURL = "https://dmapi.joker.com/request/"
2020-10-08 14:52:50 +00:00
// Response Joker DMAPI Response.
type Response struct {
2019-04-28 12:33:50 +00:00
Headers url.Values
Body string
StatusCode int
StatusText string
AuthSid string
}
2020-10-08 14:52:50 +00:00
type AuthInfo struct {
APIKey string
Username string
Password string
authSid string
}
2019-04-28 12:33:50 +00:00
2020-10-08 14:52:50 +00:00
// Client a DMAPI Client.
type Client struct {
HTTPClient *http.Client
BaseURL string
2019-04-28 12:33:50 +00:00
2020-10-08 14:52:50 +00:00
Debug bool
2019-04-28 12:33:50 +00:00
2020-10-08 14:52:50 +00:00
auth AuthInfo
}
2019-04-28 12:33:50 +00:00
2020-10-08 14:52:50 +00:00
// NewClient creates a new DMAPI Client.
func NewClient(auth AuthInfo) *Client {
return &Client{
HTTPClient: http.DefaultClient,
BaseURL: defaultBaseURL,
Debug: false,
auth: auth,
2019-04-28 12:33:50 +00:00
}
}
2020-10-08 14:52:50 +00:00
// Login performs a login to Joker's DMAPI.
func (c *Client) Login() (*Response, error) {
if c.auth.authSid != "" {
2019-04-28 12:33:50 +00:00
// already logged in
return nil, nil
}
var values url.Values
switch {
2020-10-08 14:52:50 +00:00
case c.auth.Username != "" && c.auth.Password != "":
values = url.Values{
2020-10-08 14:52:50 +00:00
"username": {c.auth.Username},
"password": {c.auth.Password},
}
2020-10-08 14:52:50 +00:00
case c.auth.APIKey != "":
values = url.Values{"api-key": {c.auth.APIKey}}
default:
2020-02-27 18:14:46 +00:00
return nil, errors.New("no username and password or api-key")
}
2020-10-08 14:52:50 +00:00
response, err := c.postRequest("login", values)
2019-04-28 12:33:50 +00:00
if err != nil {
return response, err
}
if response == nil {
2020-02-27 18:14:46 +00:00
return nil, errors.New("login returned nil response")
2019-04-28 12:33:50 +00:00
}
if response.AuthSid == "" {
2020-02-27 18:14:46 +00:00
return response, errors.New("login did not return valid Auth-Sid")
2019-04-28 12:33:50 +00:00
}
2020-10-08 14:52:50 +00:00
c.auth.authSid = response.AuthSid
2019-04-28 12:33:50 +00:00
return response, nil
}
2020-10-08 14:52:50 +00:00
// Logout closes authenticated session with Joker's DMAPI.
func (c *Client) Logout() (*Response, error) {
if c.auth.authSid == "" {
2020-02-27 18:14:46 +00:00
return nil, errors.New("already logged out")
2019-04-28 12:33:50 +00:00
}
2020-10-08 14:52:50 +00:00
response, err := c.postRequest("logout", url.Values{})
2019-04-28 12:33:50 +00:00
if err == nil {
2020-10-08 14:52:50 +00:00
c.auth.authSid = ""
2019-04-28 12:33:50 +00:00
}
return response, err
}
2020-10-08 14:52:50 +00:00
// GetZone returns content of DNS zone for domain.
func (c *Client) GetZone(domain string) (*Response, error) {
if c.auth.authSid == "" {
2020-02-27 18:14:46 +00:00
return nil, errors.New("must be logged in to get zone")
2019-04-28 12:33:50 +00:00
}
2020-10-08 14:52:50 +00:00
return c.postRequest("dns-zone-get", url.Values{"domain": {dns01.UnFqdn(domain)}})
2019-04-28 12:33:50 +00:00
}
2020-10-08 14:52:50 +00:00
// PutZone uploads DNS zone to Joker DMAPI.
func (c *Client) PutZone(domain, zone string) (*Response, error) {
if c.auth.authSid == "" {
2020-02-27 18:14:46 +00:00
return nil, errors.New("must be logged in to put zone")
2019-04-28 12:33:50 +00:00
}
2020-10-08 14:52:50 +00:00
return c.postRequest("dns-zone-put", url.Values{"domain": {dns01.UnFqdn(domain)}, "zone": {strings.TrimSpace(zone)}})
2019-04-28 12:33:50 +00:00
}
2020-05-08 17:35:25 +00:00
// postRequest performs actual HTTP request.
2020-10-08 14:52:50 +00:00
func (c *Client) postRequest(cmd string, data url.Values) (*Response, error) {
baseURL, err := url.Parse(c.BaseURL)
if err != nil {
return nil, err
}
endpoint, err := baseURL.Parse(path.Join(baseURL.Path, cmd))
if err != nil {
return nil, err
}
2019-04-28 12:33:50 +00:00
2020-10-08 14:52:50 +00:00
if c.auth.authSid != "" {
data.Set("auth-sid", c.auth.authSid)
2019-04-28 12:33:50 +00:00
}
2020-10-08 14:52:50 +00:00
if c.Debug {
log.Infof("postRequest:\n\tURL: %q\n\tData: %v", endpoint.String(), data)
2019-04-28 12:33:50 +00:00
}
2020-10-08 14:52:50 +00:00
resp, err := c.HTTPClient.PostForm(endpoint.String(), data)
2019-04-28 12:33:50 +00:00
if err != nil {
return nil, err
}
defer resp.Body.Close()
2021-08-25 09:44:11 +00:00
body, err := io.ReadAll(resp.Body)
2019-04-28 12:33:50 +00:00
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
}
2020-10-08 14:52:50 +00:00
// parseResponse parses HTTP response body.
func parseResponse(message string) *Response {
r := &Response{Headers: url.Values{}, StatusCode: -1}
2022-08-22 15:05:31 +00:00
lines, body, _ := strings.Cut(message, "\n\n")
2020-10-08 14:52:50 +00:00
2022-08-22 15:05:31 +00:00
for _, line := range strings.Split(lines, "\n") {
2020-10-08 14:52:50 +00:00
if strings.TrimSpace(line) == "" {
continue
}
2022-08-22 15:05:31 +00:00
k, v, _ := strings.Cut(line, ":")
2020-10-08 14:52:50 +00:00
2022-08-22 15:05:31 +00:00
val := strings.TrimSpace(v)
2020-10-08 14:52:50 +00:00
2022-08-22 15:05:31 +00:00
r.Headers.Add(k, val)
2020-10-08 14:52:50 +00:00
2022-08-22 15:05:31 +00:00
switch k {
2020-10-08 14:52:50 +00:00
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
}
}
2022-08-22 15:05:31 +00:00
r.Body = body
2020-10-08 14:52:50 +00:00
return r
}
2020-05-08 17:35:25 +00:00
// Temporary workaround, until it get fixed on API side.
2019-04-28 12:33:50 +00:00
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, " ")
}
2020-10-08 14:52:50 +00:00
// RemoveTxtEntryFromZone clean-ups all TXT records with given name.
func RemoveTxtEntryFromZone(zone, relative string) (string, bool) {
2019-04-28 12:33:50 +00:00
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
}
2020-10-08 14:52:50 +00:00
// AddTxtEntryToZone returns DNS zone with added TXT record.
func AddTxtEntryToZone(zone, relative, value string, ttl int) string {
2019-04-28 12:33:50 +00:00
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"))
}