paypale/client.go

143 lines
3.3 KiB
Go
Raw Normal View History

2015-10-15 07:43:50 +02:00
package paypalsdk
import (
"bytes"
2015-10-30 08:02:32 +01:00
"encoding/json"
"errors"
"io"
"io/ioutil"
"net/http"
2015-11-20 04:17:42 +01:00
"net/http/httputil"
2015-10-15 07:43:50 +02:00
)
// NewClient returns new Client struct
2015-12-29 10:21:11 +01:00
// APIBase is a base API URL, for testing you can use paypalsdk.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 == "" {
return &Client{}, errors.New("ClientID, Secret and APIBase are required to create a Client")
}
2015-10-15 07:52:16 +02:00
2015-10-30 08:02:32 +01:00
return &Client{
&http.Client{},
clientID,
secret,
APIBase,
nil,
2015-10-30 08:02:32 +01:00
nil,
}, nil
2015-10-15 07:43:50 +02:00
}
// SetLog will set/change the output destination.
// If log file is set paypalsdk will log all requests and responses to this Writer
func (c *Client) SetLog(log io.Writer) error {
c.Log = log
2015-11-20 03:51:18 +01:00
return nil
}
2015-11-19 06:12:42 +01:00
// SetAccessToken sets saved token to current client
func (c *Client) SetAccessToken(token string) error {
c.Token = &TokenResponse{
Token: token,
}
return nil
}
2015-10-15 07:43:50 +02:00
// Send makes a request to the API, the response body will be
// unmarshaled into v, or if v is an io.Writer, the response will
// 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")
}
2015-10-15 07:43:50 +02:00
2016-09-19 06:39:05 +02:00
resp, err = c.client.Do(req)
2015-11-20 03:51:18 +01:00
c.log(req, resp)
2015-10-30 08:02:32 +01:00
if err != nil {
return err
}
defer resp.Body.Close()
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}
2016-09-19 06:39:05 +02:00
data, err = ioutil.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 {
json.Unmarshal(data, errResp)
}
2015-10-15 07:43:50 +02:00
2015-10-30 08:02:32 +01:00
return errResp
}
2015-10-15 07:43:50 +02:00
2015-10-30 08:02:32 +01:00
if v != nil {
if w, ok := v.(io.Writer); ok {
io.Copy(w, resp.Body)
} else {
err = json.NewDecoder(resp.Body).Decode(v)
if err != nil {
return err
}
}
}
2015-10-15 07:43:50 +02:00
2015-10-30 08:02:32 +01:00
return nil
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 {
if c.Token != nil {
2016-09-19 06:39:05 +02:00
if c.Token.ExpiresIn < RequestNewTokenBeforeExpiresIn {
// c.Token willbe updated in GetAccessToken call
_, err := c.GetAccessToken()
if err != nil {
return err
}
}
req.Header.Set("Authorization", "Bearer "+c.Token.Token)
}
2015-11-20 03:57:45 +01:00
return c.Send(req, v)
}
// NewRequest constructs a request
// Convert payload to a JSON
func (c *Client) NewRequest(method, url string, payload interface{}) (*http.Request, error) {
var buf io.Reader
if payload != nil {
var b []byte
b, err := json.Marshal(&payload)
if err != nil {
return nil, err
}
buf = bytes.NewBuffer(b)
}
return http.NewRequest(method, url, buf)
}
2015-12-29 10:21:11 +01:00
// log will dump request and response to the log file
2015-11-20 04:17:42 +01:00
func (c *Client) log(req *http.Request, resp *http.Response) {
if c.Log != nil {
reqDump, _ := httputil.DumpRequestOut(req, true)
respDump, _ := httputil.DumpResponse(resp, true)
2015-11-20 03:51:18 +01:00
c.Log.Write([]byte("Request: " + string(reqDump) + "\nResponse: " + string(respDump) + "\n\n"))
2015-11-20 04:17:42 +01:00
2015-11-20 03:51:18 +01:00
}
}