Back to Payments & Billing

paypal-integration

paypalpayment gatewaye-commercesubscriptionswebhooksapi integrationcheckoutonline payments
36.8k📄 MIT🕒 2026-06-16Source ↗

Install this skill

npx skills add wshobson/agents

Works across Claude Code, Cursor, Codex, Copilot & Antigravity

The paypal-integration skill provides a structured interface for connecting applications to the PayPal payment ecosystem. It supports direct transaction management, including the initiation of checkout sessions, recurring subscription billing, and automated payout distribution. Developers can deploy client-side Smart Payment Buttons for immediate payment collection or employ server-side REST API calls for custom checkout logic and precise order state verification. The integration manages asynchronous notification handling via IPN to ensure external system updates reflect actual payment status. This toolkit abstracts the complexities of OAuth 2.0 authentication, token management, and order lifecycle states, allowing for the construction of reliable, production-ready payment flows that accommodate both one-time guest purchases and long-term customer subscription agreements.

When to Use This Skill

  • Building an e-commerce checkout page for physical goods
  • Managing recurring monthly software-as-a-service billing cycles
  • Distributing earnings to multiple independent contractors
  • Implementing custom payment UI with server-side validation
  • Handling disputes and automated transaction reversals

How to Invoke This Skill

Example prompts that trigger this skill in Claude Code, Cursor, or Antigravity:

  • Setup PayPal checkout buttons on my website
  • How do I verify a PayPal payment on my server?
  • Configure recurring billing for a monthly subscription
  • Create a script to handle PayPal IPN webhooks
  • Integrate PayPal payouts for my marketplace

Pro Tips

  • 💡Always implement IPN (Instant Payment Notification) verification to prevent fraudulent notifications and ensure data integrity for all transactions.
  • 💡Prioritize the Client-Side JavaScript SDK for simpler integrations, especially for Express Checkout, as it offloads much of the PCI compliance burden.
  • 💡Thoroughly test all payment flows, including edge cases like failed transactions, refunds, and subscription cancellations, in PayPal's sandbox environment before going live to production.

What this skill does

  • One-time order creation and payment capture via REST API
  • Subscription and recurring billing lifecycle management
  • Frontend implementation using PayPal JavaScript SDK
  • Automated mass payouts for marketplaces
  • Asynchronous IPN webhook listener configuration
  • Transaction status tracking and refund execution

When not to use it

  • Processing high-frequency micro-transactions where transaction fees outweigh order value
  • Projects requiring offline-only payment processing without internet connectivity

Example workflow

  1. Initialize OAuth access token using client credentials
  2. Create an order via REST API to receive an order ID
  3. Mount PayPal Smart Buttons on the frontend with the order ID
  4. Capture payment confirmation via the onApprove callback
  5. Validate transaction status on the backend using the order ID

Prerequisites

  • PayPal Developer account
  • Active Sandbox and Live API credentials
  • Publicly accessible callback URL for IPN

Pitfalls & limitations

  • !Forgetting to validate the order status on your server before fulfilling goods
  • !Hardcoding client secrets in client-side JavaScript code
  • !Failing to handle Sandbox mode vs Production endpoints correctly

FAQ

Should I use client-side or server-side integration?
Client-side is faster for simple buttons, but server-side is necessary for secure order validation and complex subscription management.
What is the role of IPN?
IPN acts as an asynchronous notification system to alert your backend server about payment status changes that occur outside the direct browser flow.
Can I test payments without real money?
Yes, PayPal provides a comprehensive Sandbox environment for simulating transactions using test buyer and seller accounts.
Does this skill handle currency conversion?
The API handles transactions in specified currencies, but your backend must define the currency code during the order creation request.

How it compares

This skill provides structured API interaction and authentication management, replacing manual cURL requests and raw HTTP handling with reusable, programmatic client methods.

Source & trust

37k stars📄 MIT🕒 Updated 2026-06-16
📄 Full skill instructions — original source: wshobson/agents
# PayPal Integration

Master PayPal payment integration including Express Checkout, IPN handling, recurring billing, and refund workflows.

## When to Use This Skill

- Integrating PayPal as a payment option
- Implementing express checkout flows
- Setting up recurring billing with PayPal
- Processing refunds and payment disputes
- Handling PayPal webhooks (IPN)
- Supporting international payments
- Implementing PayPal subscriptions

## Core Concepts

### 1. Payment Products

**PayPal Checkout**

- One-time payments
- Express checkout experience
- Guest and PayPal account payments

**PayPal Subscriptions**

- Recurring billing
- Subscription plans
- Automatic renewals

**PayPal Payouts**

- Send money to multiple recipients
- Marketplace and platform payments

### 2. Integration Methods

**Client-Side (JavaScript SDK)**

- Smart Payment Buttons
- Hosted payment flow
- Minimal backend code

**Server-Side (REST API)**

- Full control over payment flow
- Custom checkout UI
- Advanced features

### 3. IPN (Instant Payment Notification)

- Webhook-like payment notifications
- Asynchronous payment updates
- Verification required

## Quick Start

// Frontend - PayPal Smart Buttons
<div id="paypal-button-container"></div>

<script src="https://www.paypal.com/sdk/js?client-id=YOUR_CLIENT_ID&currency=USD"></script>
<script>
paypal.Buttons({
createOrder: function(data, actions) {
return actions.order.create({
purchase_units: [{
amount: {
value: '25.00'
}
}]
});
},
onApprove: function(data, actions) {
return actions.order.capture().then(function(details) {
// Payment successful
console.log('Transaction completed by ' + details.payer.name.given_name);

// Send to backend for verification
fetch('/api/paypal/capture', {
method: 'POST',
headers: {'Content-Type': 'application/json'},
body: JSON.stringify({orderID: data.orderID})
});
});
}
}).render('#paypal-button-container');
</script>


# Backend - Verify and capture order
from paypalrestsdk import Payment
import paypalrestsdk

paypalrestsdk.configure({
"mode": "sandbox", # or "live"
"client_id": "YOUR_CLIENT_ID",
"client_secret": "YOUR_CLIENT_SECRET"
})

def capture_paypal_order(order_id):
"""Capture a PayPal order."""
payment = Payment.find(order_id)

if payment.execute({"payer_id": payment.payer.payer_info.payer_id}):
# Payment successful
return {
'status': 'success',
'transaction_id': payment.id,
'amount': payment.transactions[0].amount.total
}
else:
# Payment failed
return {
'status': 'failed',
'error': payment.error
}


## Express Checkout Implementation

### Server-Side Order Creation

import requests
import json

class PayPalClient:
def __init__(self, client_id, client_secret, mode='sandbox'):
self.client_id = client_id
self.client_secret = client_secret
self.base_url = 'https://api-m.sandbox.paypal.com' if mode == 'sandbox' else 'https://api-m.paypal.com'
self.access_token = self.get_access_token()

def get_access_token(self):
"""Get OAuth access token."""
url = f"{self.base_url}/v1/oauth2/token"
headers = {"Accept": "application/json", "Accept-Language": "en_US"}

response = requests.post(
url,
headers=headers,
data={"grant_type": "client_credentials"},
auth=(self.client_id, self.client_secret)
)

return response.json()['access_token']

def create_order(self, amount, currency='USD'):
"""Create a PayPal order."""
url = f"{self.base_url}/v2/checkout/orders"
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {self.access_token}"
}

payload = {
"intent": "CAPTURE",
"purchase_units": [{
"amount": {
"currency_code": currency,
"value": str(amount)
}
}]
}

response = requests.post(url, headers=headers, json=payload)
return response.json()

def capture_order(self, order_id):
"""Capture payment for an order."""
url = f"{self.base_url}/v2/checkout/orders/{order_id}/capture"
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {self.access_token}"
}

response = requests.post(url, headers=headers)
return response.json()

def get_order_details(self, order_id):
"""Get order details."""
url = f"{self.base_url}/v2/checkout/orders/{order_id}"
headers = {
"Authorization": f"Bearer {self.access_token}"
}

response = requests.get(url, headers=headers)
return response.json()


## IPN (Instant Payment Notification) Handling

### IPN Verification and Processing

from flask import Flask, request
import requests
from urllib.parse import parse_qs

app = Flask(__name__)

@app.route('/ipn', methods=['POST'])
def handle_ipn():
"""Handle PayPal IPN notifications."""
# Get IPN message
ipn_data = request.form.to_dict()

# Verify IPN with PayPal
if not verify_ipn(ipn_data):
return 'IPN verification failed', 400

# Process IPN based on transaction type
payment_status = ipn_data.get('payment_status')
txn_type = ipn_data.get('txn_type')

if payment_status == 'Completed':
handle_payment_completed(ipn_data)
elif payment_status == 'Refunded':
handle_refund(ipn_data)
elif payment_status == 'Reversed':
handle_chargeback(ipn_data)

return 'IPN processed', 200

def verify_ipn(ipn_data):
"""Verify IPN message authenticity."""
# Add 'cmd' parameter
verify_data = ipn_data.copy()
verify_data['cmd'] = '_notify-validate'

# Send back to PayPal for verification
paypal_url = 'https://ipnpb.sandbox.paypal.com/cgi-bin/webscr' # or production URL

response = requests.post(paypal_url, data=verify_data)

return response.text == 'VERIFIED'

def handle_payment_completed(ipn_data):
"""Process completed payment."""
txn_id = ipn_data.get('txn_id')
payer_email = ipn_data.get('payer_email')
mc_gross = ipn_data.get('mc_gross')
item_name = ipn_data.get('item_name')

# Check if already processed (prevent duplicates)
if is_transaction_processed(txn_id):
return

# Update database
# Send confirmation email
# Fulfill order
print(f"Payment completed: {txn_id}, Amount: ${mc_gross}")

def handle_refund(ipn_data):
"""Handle refund."""
parent_txn_id = ipn_data.get('parent_txn_id')
mc_gross = ipn_data.get('mc_gross')

# Process refund in your system
print(f"Refund processed: {parent_txn_id}, Amount: ${mc_gross}")

def handle_chargeback(ipn_data):
"""Handle payment reversal/chargeback."""
txn_id = ipn_data.get('txn_id')
reason_code = ipn_data.get('reason_code')

# Handle chargeback
print(f"Chargeback: {txn_id}, Reason: {reason_code}")


## Subscription/Recurring Billing

### Create Subscription Plan

def create_subscription_plan(name, amount, interval='MONTH'):
"""Create a subscription plan."""
client = PayPalClient(CLIENT_ID, CLIENT_SECRET)

url = f"{client.base_url}/v1/billing/plans"
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {client.access_token}"
}

payload = {
"product_id": "PRODUCT_ID", # Create product first
"name": name,
"billing_cycles": [{
"frequency": {
"interval_unit": interval,
"interval_count": 1
},
"tenure_type": "REGULAR",
"sequence": 1,
"total_cycles": 0, # Infinite
"pricing_scheme": {
"fixed_price": {
"value": str(amount),
"currency_code": "USD"
}
}
}],
"payment_preferences": {
"auto_bill_outstanding": True,
"setup_fee": {
"value": "0",
"currency_code": "USD"
},
"setup_fee_failure_action": "CONTINUE",
"payment_failure_threshold": 3
}
}

response = requests.post(url, headers=headers, json=payload)
return response.json()

def create_subscription(plan_id, subscriber_email):
"""Create a subscription for a customer."""
client = PayPalClient(CLIENT_ID, CLIENT_SECRET)

url = f"{client.base_url}/v1/billing/subscriptions"
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {client.access_token}"
}

payload = {
"plan_id": plan_id,
"subscriber": {
"email_address": subscriber_email
},
"application_context": {
"return_url": "https://yourdomain.com/subscription/success",
"cancel_url": "https://yourdomain.com/subscription/cancel"
}
}

response = requests.post(url, headers=headers, json=payload)
subscription = response.json()

# Get approval URL
for link in subscription.get('links', []):
if link['rel'] == 'approve':
return {
'subscription_id': subscription['id'],
'approval_url': link['href']
}


## Refund Workflows

def create_refund(capture_id, amount=None, note=None):
"""Create a refund for a captured payment."""
client = PayPalClient(CLIENT_ID, CLIENT_SECRET)

url = f"{client.base_url}/v2/payments/captures/{capture_id}/refund"
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {client.access_token}"
}

payload = {}
if amount:
payload["amount"] = {
"value": str(amount),
"currency_code": "USD"
}

if note:
payload["note_to_payer"] = note

response = requests.post(url, headers=headers, json=payload)
return response.json()

def get_refund_details(refund_id):
"""Get refund details."""
client = PayPalClient(CLIENT_ID, CLIENT_SECRET)

url = f"{client.base_url}/v2/payments/refunds/{refund_id}"
headers = {
"Authorization": f"Bearer {client.access_token}"
}

response = requests.get(url, headers=headers)
return response.json()


## Error Handling

class PayPalError(Exception):
"""Custom PayPal error."""
pass

def handle_paypal_api_call(api_function):
"""Wrapper for PayPal API calls with error handling."""
try:
result = api_function()
return result
except requests.exceptions.RequestException as e:
# Network error
raise PayPalError(f"Network error: {str(e)}")
except Exception as e:
# Other errors
raise PayPalError(f"PayPal API error: {str(e)}")

# Usage
try:
order = handle_paypal_api_call(lambda: client.create_order(25.00))
except PayPalError as e:
# Handle error appropriately
log_error(e)


## Testing

# Use sandbox credentials
SANDBOX_CLIENT_ID = "..."
SANDBOX_SECRET = "..."

# Test accounts
# Create test buyer and seller accounts at developer.paypal.com

def test_payment_flow():
"""Test complete payment flow."""
client = PayPalClient(SANDBOX_CLIENT_ID, SANDBOX_SECRET, mode='sandbox')

# Create order
order = client.create_order(10.00)
assert 'id' in order

# Get approval URL
approval_url = next((link['href'] for link in order['links'] if link['rel'] == 'approve'), None)
assert approval_url is not None

# After approval (manual step with test account)
# Capture order
# captured = client.capture_order(order['id'])
# assert captured['status'] == 'COMPLETED'


## Resources

- **references/express-checkout.md**: Express Checkout implementation guide
- **references/ipn-handling.md**: IPN verification and processing
- **references/refund-workflows.md**: Refund handling patterns
- **references/billing-agreements.md**: Recurring billing setup
- **assets/paypal-client.py**: Production PayPal client
- **assets/ipn-processor.py**: IPN webhook processor
- **assets/recurring-billing.py**: Subscription management

## Best Practices

1. **Always Verify IPN**: Never trust IPN without verification
2. **Idempotent Processing**: Handle duplicate IPN notifications
3. **Error Handling**: Implement robust error handling
4. **Logging**: Log all transactions and errors
5. **Test Thoroughly**: Use sandbox extensively
6. **Webhook Backup**: Don't rely solely on client-side callbacks
7. **Currency Handling**: Always specify currency explicitly

## Common Pitfalls

- **Not Verifying IPN**: Accepting IPN without verification
- **Duplicate Processing**: Not checking for duplicate transactions
- **Wrong Environment**: Mixing sandbox and production URLs/credentials
- **Missing Webhooks**: Not handling all payment states
- **Hardcoded Values**: Not making configurable for different environments

How to Use This Skill Unit

Option A: Project-Specific (Recommended)

  1. Click "Download" above
  2. In your project, create the directory: .agent/skills/paypal-integration/
  3. Save the file as SKILL.md
  4. The agent will automatically discover the skill based on its description.

Option B: Global Installation (All Agents)

Save the file to these locations to make it available across all projects:

  • Claude Code: ~/.claude/skills/wshobson/agents/paypal-integration/SKILL.md
  • Cursor: ~/.cursor/skills/wshobson/agents/paypal-integration/SKILL.md
  • Antigravity: ~/.gemini/antigravity/skills/wshobson/agents/paypal-integration/SKILL.md

🚀 Install with CLI:
npx skills add wshobson/agents

Read the Master Guide: Mastering Agent Skills

Related Skill Units

Recommended Rules

View more rules

Recommended Workflows

View more workflows

Recommended MCP Servers

View more MCP servers

Take It Further

Maximize your productivity with these powerful resources

📋

Define Your Standards

Set up coding standards to ensure this workflow produces consistent, high-quality results.

Browse Rules Library
📖

Master Workflows

Learn how to create custom workflows, use Turbo Mode, and build your automation library.

Complete Guide

How to use this Skill in Claude Code & Cursor

For Claude Code (CLI)

To use this skill in Claude Code, copy the rule content into your project's custom instructions or follow our Add-Skill CLI guide. This ensures Claude follows your standards during every code generation.

For Cursor & Windsurf

For Cursor or Windsurf, individual skills are best used in the "Rules for AI" section. This specific unit helps the agent avoid payments & billing issues, leading to cleaner, more efficient code.

Why the skill format matters: the standardized Agent Skills format lets your AI agent load detailed instructions only when they are relevant, keeping your prompt clean while improving results.

Source & attribution

This skill is categorized under Payments & Billing and is published by W. Shobson, maintained in wshobson/agents.

← Browse All Agent Skills
Sponsored AI assistant. Recommendations may be paid.