2019-08-21 15:50:20 +02:00
|
|
|
package paypal
|
2017-09-28 00:15:52 +02:00
|
|
|
|
|
|
|
import (
|
2021-01-03 10:28:52 +01:00
|
|
|
"context"
|
2017-09-28 00:15:52 +02:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
2021-01-20 09:28:04 +01:00
|
|
|
// CreateBillingResponse struct
|
|
|
|
CreateBillingResponse struct {
|
2017-09-28 00:15:52 +02:00
|
|
|
ID string `json:"id,omitempty"`
|
|
|
|
State string `json:"state,omitempty"`
|
|
|
|
PaymentDefinitions []PaymentDefinition `json:"payment_definitions,omitempty"`
|
|
|
|
MerchantPreferences MerchantPreferences `json:"merchant_preferences,omitempty"`
|
|
|
|
CreateTime time.Time `json:"create_time,omitempty"`
|
|
|
|
UpdateTime time.Time `json:"update_time,omitempty"`
|
|
|
|
Links []Link `json:"links,omitempty"`
|
|
|
|
}
|
|
|
|
|
2021-07-20 08:37:23 +02:00
|
|
|
// CreateBillingResp.
|
|
|
|
//
|
|
|
|
// Deprecated: use CreateBillingResponse instead.
|
2021-01-20 09:28:04 +01:00
|
|
|
CreateBillingResp = CreateBillingResponse
|
|
|
|
|
|
|
|
// CreateAgreementResponse struct
|
|
|
|
CreateAgreementResponse struct {
|
2017-09-28 00:15:52 +02:00
|
|
|
Name string `json:"name,omitempty"`
|
|
|
|
Description string `json:"description,omitempty"`
|
2021-07-21 07:53:38 +02:00
|
|
|
Plan BillingPlan `json:"plan,omitempty"`
|
2017-09-28 00:15:52 +02:00
|
|
|
Links []Link `json:"links,omitempty"`
|
|
|
|
StartTime time.Time `json:"start_time,omitempty"`
|
|
|
|
}
|
2019-01-14 06:06:05 +01:00
|
|
|
|
2021-07-20 08:37:23 +02:00
|
|
|
// CreateAgreementResp.
|
|
|
|
//
|
|
|
|
// Deprecated: use CreateAgreementResponse instead.
|
2021-07-21 07:53:38 +02:00
|
|
|
CreateAgreementResp = CreateAgreementResponse
|
2021-01-20 09:28:04 +01:00
|
|
|
|
2021-07-20 08:37:23 +02:00
|
|
|
// BillingPlanListParams
|
2019-01-14 06:06:05 +01:00
|
|
|
BillingPlanListParams struct {
|
2020-05-31 07:14:19 +02:00
|
|
|
ListParams
|
|
|
|
Status string `json:"status,omitempty"` //Allowed values: CREATED, ACTIVE, INACTIVE, ALL.
|
2019-01-14 06:06:05 +01:00
|
|
|
}
|
|
|
|
|
2021-07-20 08:37:23 +02:00
|
|
|
// BillingPlanListResponse
|
2021-01-20 09:28:04 +01:00
|
|
|
BillingPlanListResponse struct {
|
2020-05-31 07:14:19 +02:00
|
|
|
SharedListResponse
|
|
|
|
Plans []BillingPlan `json:"plans,omitempty"`
|
2019-01-14 06:06:05 +01:00
|
|
|
}
|
2021-01-20 09:28:04 +01:00
|
|
|
|
2021-07-20 08:37:23 +02:00
|
|
|
// BillingPlanListResp.
|
|
|
|
//
|
|
|
|
// Deprecated: use BillingPlanListResponse instead.
|
2021-01-20 09:28:04 +01:00
|
|
|
BillingPlanListResp = BillingPlanListResponse
|
2017-09-28 00:15:52 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// CreateBillingPlan creates a billing plan in Paypal
|
2020-05-02 17:00:45 +02:00
|
|
|
// Endpoint: POST /v1/payments/billing-plans
|
2021-01-20 09:28:04 +01:00
|
|
|
func (c *Client) CreateBillingPlan(ctx context.Context, plan BillingPlan) (*CreateBillingResponse, error) {
|
2021-01-03 10:28:52 +01:00
|
|
|
req, err := c.NewRequest(ctx, http.MethodPost, fmt.Sprintf("%s%s", c.APIBase, "/v1/payments/billing-plans"), plan)
|
2021-01-20 09:28:04 +01:00
|
|
|
response := &CreateBillingResponse{}
|
2017-09-28 00:15:52 +02:00
|
|
|
if err != nil {
|
|
|
|
return response, err
|
|
|
|
}
|
|
|
|
err = c.SendWithAuth(req, response)
|
|
|
|
return response, err
|
|
|
|
}
|
|
|
|
|
2020-05-31 07:14:19 +02:00
|
|
|
// UpdateBillingPlan updates values inside a billing plan
|
|
|
|
// Endpoint: PATCH /v1/payments/billing-plans
|
2021-01-03 10:28:52 +01:00
|
|
|
func (c *Client) UpdateBillingPlan(ctx context.Context, planId string, pathValues map[string]map[string]interface{}) error {
|
2022-10-07 22:07:27 +02:00
|
|
|
var patchData []Patch
|
2020-05-31 07:14:19 +02:00
|
|
|
for path, data := range pathValues {
|
|
|
|
patchData = append(patchData, Patch{
|
|
|
|
Operation: "replace",
|
|
|
|
Path: path,
|
|
|
|
Value: data,
|
|
|
|
})
|
|
|
|
}
|
2021-01-03 10:28:52 +01:00
|
|
|
|
|
|
|
req, err := c.NewRequest(ctx, http.MethodPatch, fmt.Sprintf("%s%s%s", c.APIBase, "/v1/payments/billing-plans/", planId), patchData)
|
2020-05-31 07:14:19 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = c.SendWithAuth(req, nil)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2017-09-28 00:15:52 +02:00
|
|
|
// ActivatePlan activates a billing plan
|
|
|
|
// By default, a new plan is not activated
|
2020-05-02 17:00:45 +02:00
|
|
|
// Endpoint: PATCH /v1/payments/billing-plans/
|
2021-01-03 10:28:52 +01:00
|
|
|
func (c *Client) ActivatePlan(ctx context.Context, planID string) error {
|
|
|
|
return c.UpdateBillingPlan(ctx, planID, map[string]map[string]interface{}{
|
2020-05-31 07:14:19 +02:00
|
|
|
"/": {"state": BillingPlanStatusActive},
|
|
|
|
})
|
2017-09-28 00:15:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateBillingAgreement creates an agreement for specified plan
|
2020-05-02 17:00:45 +02:00
|
|
|
// Endpoint: POST /v1/payments/billing-agreements
|
2021-07-21 07:53:38 +02:00
|
|
|
// Deprecated: Use POST /v1/billing-agreements/agreements
|
2021-01-20 09:28:04 +01:00
|
|
|
func (c *Client) CreateBillingAgreement(ctx context.Context, a BillingAgreement) (*CreateAgreementResponse, error) {
|
2017-09-28 00:15:52 +02:00
|
|
|
// PayPal needs only ID, so we will remove all fields except Plan ID
|
|
|
|
a.Plan = BillingPlan{
|
|
|
|
ID: a.Plan.ID,
|
|
|
|
}
|
|
|
|
|
2021-01-03 10:28:52 +01:00
|
|
|
req, err := c.NewRequest(ctx, http.MethodPost, fmt.Sprintf("%s%s", c.APIBase, "/v1/payments/billing-agreements"), a)
|
2021-01-20 09:28:04 +01:00
|
|
|
response := &CreateAgreementResponse{}
|
2017-09-28 00:15:52 +02:00
|
|
|
if err != nil {
|
|
|
|
return response, err
|
|
|
|
}
|
|
|
|
err = c.SendWithAuth(req, response)
|
|
|
|
return response, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExecuteApprovedAgreement - Use this call to execute (complete) a PayPal agreement that has been approved by the payer.
|
2020-05-02 17:00:45 +02:00
|
|
|
// Endpoint: POST /v1/payments/billing-agreements/token/agreement-execute
|
2021-01-03 10:28:52 +01:00
|
|
|
func (c *Client) ExecuteApprovedAgreement(ctx context.Context, token string) (*ExecuteAgreementResponse, error) {
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, fmt.Sprintf("%s/v1/payments/billing-agreements/%s/agreement-execute", c.APIBase, token), nil)
|
2019-06-16 04:39:08 +02:00
|
|
|
response := &ExecuteAgreementResponse{}
|
|
|
|
|
2017-09-28 00:15:52 +02:00
|
|
|
if err != nil {
|
2019-06-16 04:39:08 +02:00
|
|
|
return response, err
|
2017-09-28 00:15:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
req.SetBasicAuth(c.ClientID, c.Secret)
|
|
|
|
req.Header.Set("Authorization", "Bearer "+c.Token.Token)
|
|
|
|
|
2019-06-16 04:39:08 +02:00
|
|
|
if err = c.SendWithAuth(req, response); err != nil {
|
|
|
|
return response, err
|
2017-09-28 00:15:52 +02:00
|
|
|
}
|
|
|
|
|
2019-06-16 04:39:08 +02:00
|
|
|
if response.ID == "" {
|
|
|
|
return response, errors.New("Unable to execute agreement with token=" + token)
|
2017-09-28 00:15:52 +02:00
|
|
|
}
|
|
|
|
|
2019-06-16 04:39:08 +02:00
|
|
|
return response, err
|
2017-09-28 00:15:52 +02:00
|
|
|
}
|
2019-01-14 06:06:05 +01:00
|
|
|
|
|
|
|
// ListBillingPlans lists billing-plans
|
2020-05-02 17:00:45 +02:00
|
|
|
// Endpoint: GET /v1/payments/billing-plans
|
2021-01-20 09:28:04 +01:00
|
|
|
func (c *Client) ListBillingPlans(ctx context.Context, bplp BillingPlanListParams) (*BillingPlanListResponse, error) {
|
2021-01-03 10:28:52 +01:00
|
|
|
req, err := c.NewRequest(ctx, "GET", fmt.Sprintf("%s%s", c.APIBase, "/v1/payments/billing-plans"), nil)
|
2021-01-20 09:28:04 +01:00
|
|
|
response := &BillingPlanListResponse{}
|
2020-05-31 07:14:19 +02:00
|
|
|
if err != nil {
|
|
|
|
return response, err
|
|
|
|
}
|
|
|
|
|
2019-01-14 06:06:05 +01:00
|
|
|
q := req.URL.Query()
|
|
|
|
q.Add("page", bplp.Page)
|
|
|
|
q.Add("page_size", bplp.PageSize)
|
|
|
|
q.Add("status", bplp.Status)
|
|
|
|
q.Add("total_required", bplp.TotalRequired)
|
|
|
|
req.URL.RawQuery = q.Encode()
|
2020-05-31 07:14:19 +02:00
|
|
|
|
2019-01-14 06:06:05 +01:00
|
|
|
err = c.SendWithAuth(req, response)
|
|
|
|
return response, err
|
|
|
|
}
|