// Package dnspod implements a client for the dnspod API. // // In order to use this package you will need a dnspod account and your API Token. package dnspod import ( // "bytes" "encoding/json" "fmt" "io" "net" "net/http" "net/url" "strings" "time" ) const ( libraryVersion = "0.1" baseURL = "https://dnsapi.cn/" userAgent = "dnspod-go/" + libraryVersion apiVersion = "v1" timeout = 5 keepAlive = 30 ) // dnspod API docs: https://www.dnspod.cn/docs/info.html type CommonParams struct { LoginToken string Format string Lang string ErrorOnEmpty string UserID string Timeout int KeepAlive int } func newPayLoad(params CommonParams) url.Values { p := url.Values{} if params.LoginToken != "" { p.Set("login_token", params.LoginToken) } if params.Format != "" { p.Set("format", params.Format) } if params.Lang != "" { p.Set("lang", params.Lang) } if params.ErrorOnEmpty != "" { p.Set("error_on_empty", params.ErrorOnEmpty) } if params.UserID != "" { p.Set("user_id", params.UserID) } return p } type Status struct { Code string `json:"code,omitempty"` Message string `json:"message,omitempty"` CreatedAt string `json:"created_at,omitempty"` } type Client struct { // HTTP client used to communicate with the API. HttpClient *http.Client // CommonParams used communicating with the dnspod API. CommonParams CommonParams // Base URL for API requests. // Defaults to the public dnspod API, but can be set to a different endpoint (e.g. the sandbox). // BaseURL should always be specified with a trailing slash. BaseURL string // User agent used when communicating with the dnspod API. UserAgent string // Services used for talking to different parts of the dnspod API. Domains *DomainsService } // NewClient returns a new dnspod API client. func NewClient(CommonParams CommonParams) *Client { var _timeout, _keepalive int _timeout = timeout _keepalive = keepAlive if CommonParams.Timeout != 0 { _timeout = CommonParams.Timeout } if CommonParams.KeepAlive != 0 { _keepalive = CommonParams.KeepAlive } cli := http.Client{ Transport: &http.Transport{ Dial: (&net.Dialer{ Timeout: time.Duration(_timeout) * time.Second, KeepAlive: time.Duration(_keepalive) * time.Second, }).Dial, }, } c := &Client{HttpClient: &cli, CommonParams: CommonParams, BaseURL: baseURL, UserAgent: userAgent} c.Domains = &DomainsService{client: c} return c } // NewRequest creates an API request. // The path is expected to be a relative path and will be resolved // according to the BaseURL of the Client. Paths should always be specified without a preceding slash. func (client *Client) NewRequest(method, path string, payload url.Values) (*http.Request, error) { url := client.BaseURL + fmt.Sprintf("%s", path) req, err := http.NewRequest(method, url, strings.NewReader(payload.Encode())) if err != nil { return nil, err } req.Header.Add("Content-Type", "application/x-www-form-urlencoded") req.Header.Add("Accept", "application/json") req.Header.Add("User-Agent", client.UserAgent) return req, nil } func (c *Client) get(path string, v interface{}) (*Response, error) { return c.Do("GET", path, nil, v) } func (c *Client) post(path string, payload url.Values, v interface{}) (*Response, error) { return c.Do("POST", path, payload, v) } func (c *Client) put(path string, payload url.Values, v interface{}) (*Response, error) { return c.Do("PUT", path, payload, v) } func (c *Client) delete(path string, payload url.Values) (*Response, error) { return c.Do("DELETE", path, payload, nil) } // Do sends an API request and returns the API response. // The API response is JSON decoded and stored in the value pointed by v, // or returned as an error if an API error has occurred. // If v implements the io.Writer interface, the raw response body will be written to v, // without attempting to decode it. func (c *Client) Do(method, path string, payload url.Values, v interface{}) (*Response, error) { req, err := c.NewRequest(method, path, payload) if err != nil { return nil, err } res, err := c.HttpClient.Do(req) if err != nil { return nil, err } defer res.Body.Close() response := &Response{Response: res} err = CheckResponse(res) if err != nil { return response, err } if v != nil { if w, ok := v.(io.Writer); ok { io.Copy(w, res.Body) } else { err = json.NewDecoder(res.Body).Decode(v) } } return response, err } // A Response represents an API response. type Response struct { *http.Response } // An ErrorResponse represents an error caused by an API request. type ErrorResponse struct { Response *http.Response // HTTP response that caused this error Message string `json:"message"` // human-readable message } // Error implements the error interface. func (r *ErrorResponse) Error() string { return fmt.Sprintf("%v %v: %d %v", r.Response.Request.Method, r.Response.Request.URL, r.Response.StatusCode, r.Message) } // CheckResponse checks the API response for errors, and returns them if present. // A response is considered an error if the status code is different than 2xx. Specific requests // may have additional requirements, but this is sufficient in most of the cases. func CheckResponse(r *http.Response) error { if code := r.StatusCode; 200 <= code && code <= 299 { return nil } errorResponse := &ErrorResponse{Response: r} err := json.NewDecoder(r.Body).Decode(errorResponse) if err != nil { return err } return errorResponse } // Date custom type. type Date struct { time.Time } // UnmarshalJSON handles the deserialization of the custom Date type. func (d *Date) UnmarshalJSON(data []byte) error { var s string if err := json.Unmarshal(data, &s); err != nil { return fmt.Errorf("date should be a string, got %s", data) } t, err := time.Parse("2006-01-02", s) if err != nil { return fmt.Errorf("invalid date: %v", err) } d.Time = t return nil }