mirror of
https://github.com/go-acme/lego
synced 2026-03-14 14:35:48 +01:00
135 lines
3.1 KiB
Go
135 lines
3.1 KiB
Go
package internal
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"net/url"
|
|
"time"
|
|
|
|
"github.com/go-acme/lego/v4/providers/dns/internal/errutils"
|
|
)
|
|
|
|
const defaultBaseURL = "https://api.beget.com/api/"
|
|
|
|
// Client the beget.com client.
|
|
type Client struct {
|
|
login string
|
|
password string
|
|
|
|
BaseURL *url.URL
|
|
HTTPClient *http.Client
|
|
}
|
|
|
|
// NewClient Creates a beget.com client.
|
|
func NewClient(login, password string) *Client {
|
|
baseURL, _ := url.Parse(defaultBaseURL)
|
|
|
|
return &Client{
|
|
login: login,
|
|
password: password,
|
|
BaseURL: baseURL,
|
|
HTTPClient: &http.Client{Timeout: 5 * time.Second},
|
|
}
|
|
}
|
|
|
|
// GetTXTRecords returns TXT records.
|
|
// https://beget.com/ru/kb/api/funkczii-upravleniya-dns#getdata
|
|
func (c *Client) GetTXTRecords(ctx context.Context, domain string) ([]Record, error) {
|
|
request := GetRecordsRequest{Fqdn: domain}
|
|
|
|
resp, err := c.doRequest(ctx, request, "dns", "getData")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = resp.HasError()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
result := GetRecordsResult{}
|
|
|
|
err = json.Unmarshal(resp.Answer.Result, &result)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unmarshal result: %s: %w", string(resp.Answer.Result), err)
|
|
}
|
|
|
|
return result.Records.TXT, nil
|
|
}
|
|
|
|
// ChangeTXTRecord changes TXT records.
|
|
// https://beget.com/ru/kb/api/funkczii-upravleniya-dns#changerecords
|
|
func (c *Client) ChangeTXTRecord(ctx context.Context, domain string, records []Record) error {
|
|
request := ChangeRecordsRequest{
|
|
Fqdn: domain,
|
|
Records: RecordList{TXT: records},
|
|
}
|
|
|
|
resp, err := c.doRequest(ctx, request, "dns", "changeRecords")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return resp.HasError()
|
|
}
|
|
|
|
func (c *Client) doRequest(ctx context.Context, data any, fragments ...string) (*APIResponse, error) {
|
|
endpoint := c.BaseURL.JoinPath(fragments...)
|
|
|
|
inputData, err := json.Marshal(data)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to mashall input data: %w", err)
|
|
}
|
|
|
|
query := endpoint.Query()
|
|
query.Add("input_data", string(inputData))
|
|
query.Add("login", c.login)
|
|
query.Add("passwd", c.password)
|
|
query.Add("input_format", "json")
|
|
query.Add("output_format", "json")
|
|
endpoint.RawQuery = query.Encode()
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, endpoint.String(), http.NoBody)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to create request: %w", err)
|
|
}
|
|
|
|
resp, err := c.HTTPClient.Do(req)
|
|
if err != nil {
|
|
return nil, errutils.NewHTTPDoError(req, err)
|
|
}
|
|
|
|
defer func() { _ = resp.Body.Close() }()
|
|
|
|
if resp.StatusCode/100 != 2 {
|
|
return nil, parseError(req, resp)
|
|
}
|
|
|
|
raw, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, errutils.NewReadResponseError(req, resp.StatusCode, err)
|
|
}
|
|
|
|
var apiResp APIResponse
|
|
err = json.Unmarshal(raw, &apiResp)
|
|
if err != nil {
|
|
return nil, errutils.NewUnmarshalError(req, resp.StatusCode, raw, err)
|
|
}
|
|
|
|
return &apiResp, nil
|
|
}
|
|
|
|
func parseError(req *http.Request, resp *http.Response) error {
|
|
raw, _ := io.ReadAll(resp.Body)
|
|
|
|
var apiResp APIResponse
|
|
err := json.Unmarshal(raw, &apiResp)
|
|
if err != nil {
|
|
return errutils.NewUnexpectedStatusCodeError(req, resp.StatusCode, raw)
|
|
}
|
|
|
|
return fmt.Errorf("[status code %d] %w", resp.StatusCode, apiResp)
|
|
}
|