paypale/client.go

193 lines
5.2 KiB
Go
Raw Normal View History

2019-08-21 15:50:20 +02:00
package paypal
2015-10-15 07:43:50 +02:00
import (
"bytes"
2021-01-03 10:28:52 +01:00
"context"
2015-10-30 08:02:32 +01:00
"encoding/json"
"errors"
2016-12-01 05:32:21 +01:00
"fmt"
2015-10-30 08:02:32 +01:00
"io"
"net/http"
2015-11-20 04:17:42 +01:00
"net/http/httputil"
2017-07-07 00:52:39 +02:00
"time"
2015-10-15 07:43:50 +02:00
)
// NewClient returns new Client struct
2019-08-21 15:50:20 +02:00
// APIBase is a base API URL, for testing you can use paypal.APIBaseSandBox
2015-10-15 07:43:50 +02:00
func NewClient(clientID string, secret string, APIBase string) (*Client, error) {
2015-10-30 08:02:32 +01:00
if clientID == "" || secret == "" || APIBase == "" {
2017-03-02 04:43:23 +01:00
return nil, errors.New("ClientID, Secret and APIBase are required to create a Client")
2015-10-30 08:02:32 +01:00
}
2015-10-15 07:52:16 +02:00
2015-10-30 08:02:32 +01:00
return &Client{
Client: &http.Client{},
2017-03-02 04:43:23 +01:00
ClientID: clientID,
Secret: secret,
APIBase: APIBase,
2015-10-30 08:02:32 +01:00
}, nil
2015-10-15 07:43:50 +02:00
}
2017-03-02 04:43:23 +01:00
// GetAccessToken returns struct of TokenResponse
// No need to call SetAccessToken to apply new access token for current Client
// Endpoint: POST /v1/oauth2/token
2021-01-03 10:28:52 +01:00
func (c *Client) GetAccessToken(ctx context.Context) (*TokenResponse, error) {
2017-03-02 04:43:23 +01:00
buf := bytes.NewBuffer([]byte("grant_type=client_credentials"))
2021-01-03 10:28:52 +01:00
req, err := http.NewRequestWithContext(ctx, "POST", fmt.Sprintf("%s%s", c.APIBase, "/v1/oauth2/token"), buf)
2017-03-02 04:43:23 +01:00
if err != nil {
return &TokenResponse{}, err
}
req.Header.Set("Content-type", "application/x-www-form-urlencoded")
2019-06-16 04:39:08 +02:00
response := &TokenResponse{}
err = c.SendWithBasicAuth(req, response)
2017-03-02 04:43:23 +01:00
// Set Token fur current Client
2019-06-16 04:39:08 +02:00
if response.Token != "" {
c.Token = response
c.tokenExpiresAt = time.Now().Add(time.Duration(response.ExpiresIn) * time.Second)
2017-03-02 04:43:23 +01:00
}
2019-06-16 04:39:08 +02:00
return response, err
2015-11-20 03:51:18 +01:00
}
// SetHTTPClient sets *http.Client to current client
2018-08-29 09:52:31 +02:00
func (c *Client) SetHTTPClient(client *http.Client) {
c.Client = client
}
2015-11-19 06:12:42 +01:00
// SetAccessToken sets saved token to current client
2018-08-29 09:52:31 +02:00
func (c *Client) SetAccessToken(token string) {
2015-11-19 06:12:42 +01:00
c.Token = &TokenResponse{
Token: token,
}
c.tokenExpiresAt = time.Time{}
2015-11-19 06:12:42 +01:00
}
2017-03-02 04:43:23 +01:00
// SetLog will set/change the output destination.
2019-08-21 15:50:20 +02:00
// If log file is set paypal will log all requests and responses to this Writer
2018-08-29 09:52:31 +02:00
func (c *Client) SetLog(log io.Writer) {
2017-03-02 04:43:23 +01:00
c.Log = log
}
// SetReturnRepresentation enables verbose response
// Verbose response: https://developer.paypal.com/docs/api/orders/v2/#orders-authorize-header-parameters
func (c *Client) SetReturnRepresentation() {
c.returnRepresentation = true
}
2015-10-15 07:43:50 +02:00
// Send makes a request to the API, the response body will be
2021-09-12 20:53:49 +02:00
// unmarshalled into v, or if v is an io.Writer, the response will
2015-10-15 07:43:50 +02:00
// be written to it without decoding
func (c *Client) Send(req *http.Request, v interface{}) error {
2016-09-19 06:39:05 +02:00
var (
err error
resp *http.Response
data []byte
)
2015-10-30 08:02:32 +01:00
// Set default headers
req.Header.Set("Accept", "application/json")
req.Header.Set("Accept-Language", "en_US")
2015-10-15 07:43:50 +02:00
2015-10-30 08:02:32 +01:00
// Default values for headers
if req.Header.Get("Content-type") == "" {
req.Header.Set("Content-type", "application/json")
}
if c.returnRepresentation {
req.Header.Set("Prefer", "return=representation")
}
2024-08-19 13:24:58 +02:00
if c.Log != nil {
if reqDump, err := httputil.DumpRequestOut(req, true); err == nil {
c.Log.Write([]byte(fmt.Sprintf("Request: %s\n", reqDump)))
}
}
2015-10-15 07:43:50 +02:00
resp, err = c.Client.Do(req)
2015-10-30 08:02:32 +01:00
if err != nil {
return err
}
2024-08-19 13:24:58 +02:00
if c.Log != nil {
if respDump, err := httputil.DumpResponse(resp, true); err == nil {
c.Log.Write([]byte(fmt.Sprintf("Response from %s: %s\n", req.URL, respDump)))
}
}
defer func(Body io.ReadCloser) error {
return Body.Close()
}(resp.Body)
2015-10-15 07:43:50 +02:00
2015-10-30 08:02:32 +01:00
if resp.StatusCode < 200 || resp.StatusCode > 299 {
errResp := &ErrorResponse{Response: resp}
data, err = io.ReadAll(resp.Body)
2015-10-15 07:43:50 +02:00
2015-10-30 08:02:32 +01:00
if err == nil && len(data) > 0 {
err := json.Unmarshal(data, errResp)
if err != nil {
return err
}
2015-10-30 08:02:32 +01:00
}
2015-10-15 07:43:50 +02:00
2015-10-30 08:02:32 +01:00
return errResp
}
2018-08-29 09:52:31 +02:00
if v == nil {
return nil
}
if w, ok := v.(io.Writer); ok {
_, err := io.Copy(w, resp.Body)
return err
2015-10-30 08:02:32 +01:00
}
2015-10-15 07:43:50 +02:00
2018-08-29 09:52:31 +02:00
return json.NewDecoder(resp.Body).Decode(v)
2015-10-15 07:43:50 +02:00
}
2015-11-20 03:51:18 +01:00
2015-11-20 03:57:45 +01:00
// SendWithAuth makes a request to the API and apply OAuth2 header automatically.
2016-09-19 06:39:05 +02:00
// If the access token soon to be expired or already expired, it will try to get a new one before
2015-11-20 03:57:45 +01:00
// making the main request
2016-09-19 06:39:05 +02:00
// client.Token will be updated when changed
2015-11-20 03:57:45 +01:00
func (c *Client) SendWithAuth(req *http.Request, v interface{}) error {
// c.Lock()
c.mu.Lock()
2019-03-01 05:17:03 +01:00
// Note: Here we do not want to `defer c.Unlock()` because we need `c.Send(...)`
// to happen outside of the locked section.
if c.Token == nil || (!c.tokenExpiresAt.IsZero() && time.Until(c.tokenExpiresAt) < RequestNewTokenBeforeExpiresIn) {
// c.Token will be updated in GetAccessToken call
if _, err := c.GetAccessToken(req.Context()); err != nil {
// c.Unlock()
c.mu.Unlock()
return err
2016-09-19 06:39:05 +02:00
}
}
req.Header.Set("Authorization", "Bearer "+c.Token.Token)
2019-03-01 05:17:03 +01:00
// Unlock the client mutex before sending the request, this allows multiple requests
// to be in progress at the same time.
// c.Unlock()
c.mu.Unlock()
2015-11-20 03:57:45 +01:00
return c.Send(req, v)
}
// SendWithBasicAuth makes a request to the API using clientID:secret basic auth
func (c *Client) SendWithBasicAuth(req *http.Request, v interface{}) error {
req.SetBasicAuth(c.ClientID, c.Secret)
return c.Send(req, v)
}
// NewRequest constructs a request
// Convert payload to a JSON
2021-01-03 10:28:52 +01:00
func (c *Client) NewRequest(ctx context.Context, method, url string, payload interface{}) (*http.Request, error) {
var buf io.Reader
if payload != nil {
b, err := json.Marshal(&payload)
if err != nil {
return nil, err
}
buf = bytes.NewBuffer(b)
}
2021-01-03 10:28:52 +01:00
return http.NewRequestWithContext(ctx, method, url, buf)
}